|
|
|
@ -18,7 +18,7 @@
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
// The fmt library version in the form major * 10000 + minor * 100 + patch.
|
|
|
|
|
#define FMT_VERSION 70003
|
|
|
|
|
#define FMT_VERSION 70103
|
|
|
|
|
|
|
|
|
|
#ifdef __clang__
|
|
|
|
|
# define FMT_CLANG_VERSION (__clang_major__ * 100 + __clang_minor__)
|
|
|
|
@ -57,6 +57,7 @@
|
|
|
|
|
# define FMT_MSC_VER 0
|
|
|
|
|
# define FMT_SUPPRESS_MSC_WARNING(n)
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef __has_feature
|
|
|
|
|
# define FMT_HAS_FEATURE(x) __has_feature(x)
|
|
|
|
|
#else
|
|
|
|
@ -64,7 +65,7 @@
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#if defined(__has_include) && !defined(__INTELLISENSE__) && \
|
|
|
|
|
!(FMT_ICC_VERSION && FMT_ICC_VERSION < 1600)
|
|
|
|
|
(!FMT_ICC_VERSION || FMT_ICC_VERSION >= 1600)
|
|
|
|
|
# define FMT_HAS_INCLUDE(x) __has_include(x)
|
|
|
|
|
#else
|
|
|
|
|
# define FMT_HAS_INCLUDE(x) 0
|
|
|
|
@ -99,7 +100,7 @@
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef FMT_OVERRIDE
|
|
|
|
|
# if FMT_HAS_FEATURE(cxx_override) || \
|
|
|
|
|
# if FMT_HAS_FEATURE(cxx_override_control) || \
|
|
|
|
|
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1900
|
|
|
|
|
# define FMT_OVERRIDE override
|
|
|
|
|
# else
|
|
|
|
@ -152,7 +153,7 @@
|
|
|
|
|
# if FMT_HAS_CPP14_ATTRIBUTE(deprecated) || FMT_MSC_VER >= 1900
|
|
|
|
|
# define FMT_DEPRECATED [[deprecated]]
|
|
|
|
|
# else
|
|
|
|
|
# if defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
# if (defined(__GNUC__) && !defined(__LCC__)) || defined(__clang__)
|
|
|
|
|
# define FMT_DEPRECATED __attribute__((deprecated))
|
|
|
|
|
# elif FMT_MSC_VER
|
|
|
|
|
# define FMT_DEPRECATED __declspec(deprecated)
|
|
|
|
@ -177,9 +178,17 @@
|
|
|
|
|
# endif
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef FMT_BEGIN_NAMESPACE
|
|
|
|
|
#ifndef FMT_USE_INLINE_NAMESPACES
|
|
|
|
|
# if FMT_HAS_FEATURE(cxx_inline_namespaces) || FMT_GCC_VERSION >= 404 || \
|
|
|
|
|
FMT_MSC_VER >= 1900
|
|
|
|
|
(FMT_MSC_VER >= 1900 && !_MANAGED)
|
|
|
|
|
# define FMT_USE_INLINE_NAMESPACES 1
|
|
|
|
|
# else
|
|
|
|
|
# define FMT_USE_INLINE_NAMESPACES 0
|
|
|
|
|
# endif
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef FMT_BEGIN_NAMESPACE
|
|
|
|
|
# if FMT_USE_INLINE_NAMESPACES
|
|
|
|
|
# define FMT_INLINE_NAMESPACE inline namespace
|
|
|
|
|
# define FMT_END_NAMESPACE \
|
|
|
|
|
} \
|
|
|
|
@ -269,8 +278,7 @@ struct monostate {};
|
|
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
|
|
// A helper function to suppress bogus "conditional expression is constant"
|
|
|
|
|
// warnings.
|
|
|
|
|
// A helper function to suppress "conditional expression is constant" warnings.
|
|
|
|
|
template <typename T> constexpr T const_check(T value) { return value; }
|
|
|
|
|
|
|
|
|
|
FMT_NORETURN FMT_API void assert_fail(const char* file, int line,
|
|
|
|
@ -299,7 +307,8 @@ template <typename T> struct std_string_view {};
|
|
|
|
|
|
|
|
|
|
#ifdef FMT_USE_INT128
|
|
|
|
|
// Do nothing.
|
|
|
|
|
#elif defined(__SIZEOF_INT128__) && !FMT_NVCC && !(FMT_CLANG_VERSION && FMT_MSC_VER)
|
|
|
|
|
#elif defined(__SIZEOF_INT128__) && !FMT_NVCC && \
|
|
|
|
|
!(FMT_CLANG_VERSION && FMT_MSC_VER)
|
|
|
|
|
# define FMT_USE_INT128 1
|
|
|
|
|
using int128_t = __int128_t;
|
|
|
|
|
using uint128_t = __uint128_t;
|
|
|
|
@ -506,6 +515,18 @@ template <typename S> struct char_t_impl<S, enable_if_t<is_string<S>::value>> {
|
|
|
|
|
using type = typename result::value_type;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Reports a compile-time error if S is not a valid format string.
|
|
|
|
|
template <typename..., typename S, FMT_ENABLE_IF(!is_compile_string<S>::value)>
|
|
|
|
|
FMT_INLINE void check_format_string(const S&) {
|
|
|
|
|
#ifdef FMT_ENFORCE_COMPILE_STRING
|
|
|
|
|
static_assert(is_compile_string<S>::value,
|
|
|
|
|
"FMT_ENFORCE_COMPILE_STRING requires all format strings to use "
|
|
|
|
|
"FMT_STRING.");
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
template <typename..., typename S, FMT_ENABLE_IF(is_compile_string<S>::value)>
|
|
|
|
|
void check_format_string(S);
|
|
|
|
|
|
|
|
|
|
struct error_handler {
|
|
|
|
|
constexpr error_handler() = default;
|
|
|
|
|
constexpr error_handler(const error_handler&) = default;
|
|
|
|
@ -545,8 +566,9 @@ class basic_format_parse_context : private ErrorHandler {
|
|
|
|
|
using iterator = typename basic_string_view<Char>::iterator;
|
|
|
|
|
|
|
|
|
|
explicit constexpr basic_format_parse_context(
|
|
|
|
|
basic_string_view<Char> format_str, ErrorHandler eh = {})
|
|
|
|
|
: ErrorHandler(eh), format_str_(format_str), next_arg_id_(0) {}
|
|
|
|
|
basic_string_view<Char> format_str, ErrorHandler eh = {},
|
|
|
|
|
int next_arg_id = 0)
|
|
|
|
|
: ErrorHandler(eh), format_str_(format_str), next_arg_id_(next_arg_id) {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
Returns an iterator to the beginning of the format string range being
|
|
|
|
@ -616,8 +638,24 @@ template <typename T, typename Context>
|
|
|
|
|
using has_formatter =
|
|
|
|
|
std::is_constructible<typename Context::template formatter_type<T>>;
|
|
|
|
|
|
|
|
|
|
// Checks whether T is a container with contiguous storage.
|
|
|
|
|
template <typename T> struct is_contiguous : std::false_type {};
|
|
|
|
|
template <typename Char>
|
|
|
|
|
struct is_contiguous<std::basic_string<Char>> : std::true_type {};
|
|
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
|
|
// Extracts a reference to the container from back_insert_iterator.
|
|
|
|
|
template <typename Container>
|
|
|
|
|
inline Container& get_container(std::back_insert_iterator<Container> it) {
|
|
|
|
|
using bi_iterator = std::back_insert_iterator<Container>;
|
|
|
|
|
struct accessor : bi_iterator {
|
|
|
|
|
accessor(bi_iterator iter) : bi_iterator(iter) {}
|
|
|
|
|
using bi_iterator::container;
|
|
|
|
|
};
|
|
|
|
|
return *accessor(it).container;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
A contiguous memory buffer with an optional growing ability. It is an internal
|
|
|
|
@ -640,6 +678,8 @@ template <typename T> class buffer {
|
|
|
|
|
size_(sz),
|
|
|
|
|
capacity_(cap) {}
|
|
|
|
|
|
|
|
|
|
~buffer() = default;
|
|
|
|
|
|
|
|
|
|
/** Sets the buffer data and capacity. */
|
|
|
|
|
void set(T* buf_data, size_t buf_capacity) FMT_NOEXCEPT {
|
|
|
|
|
ptr_ = buf_data;
|
|
|
|
@ -655,7 +695,6 @@ template <typename T> class buffer {
|
|
|
|
|
|
|
|
|
|
buffer(const buffer&) = delete;
|
|
|
|
|
void operator=(const buffer&) = delete;
|
|
|
|
|
virtual ~buffer() = default;
|
|
|
|
|
|
|
|
|
|
T* begin() FMT_NOEXCEPT { return ptr_; }
|
|
|
|
|
T* end() FMT_NOEXCEPT { return ptr_ + size_; }
|
|
|
|
@ -675,24 +714,26 @@ template <typename T> class buffer {
|
|
|
|
|
/** Returns a pointer to the buffer data. */
|
|
|
|
|
const T* data() const FMT_NOEXCEPT { return ptr_; }
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
Resizes the buffer. If T is a POD type new elements may not be initialized.
|
|
|
|
|
*/
|
|
|
|
|
void resize(size_t new_size) {
|
|
|
|
|
reserve(new_size);
|
|
|
|
|
size_ = new_size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Clears this buffer. */
|
|
|
|
|
void clear() { size_ = 0; }
|
|
|
|
|
|
|
|
|
|
/** Reserves space to store at least *capacity* elements. */
|
|
|
|
|
void reserve(size_t new_capacity) {
|
|
|
|
|
// Tries resizing the buffer to contain *count* elements. If T is a POD type
|
|
|
|
|
// the new elements may not be initialized.
|
|
|
|
|
void try_resize(size_t count) {
|
|
|
|
|
try_reserve(count);
|
|
|
|
|
size_ = count <= capacity_ ? count : capacity_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Tries increasing the buffer capacity to *new_capacity*. It can increase the
|
|
|
|
|
// capacity by a smaller amount than requested but guarantees there is space
|
|
|
|
|
// for at least one additional element either by increasing the capacity or by
|
|
|
|
|
// flushing the buffer if it is full.
|
|
|
|
|
void try_reserve(size_t new_capacity) {
|
|
|
|
|
if (new_capacity > capacity_) grow(new_capacity);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void push_back(const T& value) {
|
|
|
|
|
reserve(size_ + 1);
|
|
|
|
|
try_reserve(size_ + 1);
|
|
|
|
|
ptr_[size_++] = value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -705,32 +746,150 @@ template <typename T> class buffer {
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// A container-backed buffer.
|
|
|
|
|
struct buffer_traits {
|
|
|
|
|
explicit buffer_traits(size_t) {}
|
|
|
|
|
size_t count() const { return 0; }
|
|
|
|
|
size_t limit(size_t size) { return size; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class fixed_buffer_traits {
|
|
|
|
|
private:
|
|
|
|
|
size_t count_ = 0;
|
|
|
|
|
size_t limit_;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
explicit fixed_buffer_traits(size_t limit) : limit_(limit) {}
|
|
|
|
|
size_t count() const { return count_; }
|
|
|
|
|
size_t limit(size_t size) {
|
|
|
|
|
size_t n = limit_ > count_ ? limit_ - count_ : 0;
|
|
|
|
|
count_ += size;
|
|
|
|
|
return size < n ? size : n;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// A buffer that writes to an output iterator when flushed.
|
|
|
|
|
template <typename OutputIt, typename T, typename Traits = buffer_traits>
|
|
|
|
|
class iterator_buffer final : public Traits, public buffer<T> {
|
|
|
|
|
private:
|
|
|
|
|
OutputIt out_;
|
|
|
|
|
enum { buffer_size = 256 };
|
|
|
|
|
T data_[buffer_size];
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void grow(size_t) final FMT_OVERRIDE {
|
|
|
|
|
if (this->size() == buffer_size) flush();
|
|
|
|
|
}
|
|
|
|
|
void flush();
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
explicit iterator_buffer(OutputIt out, size_t n = buffer_size)
|
|
|
|
|
: Traits(n),
|
|
|
|
|
buffer<T>(data_, 0, buffer_size),
|
|
|
|
|
out_(out) {}
|
|
|
|
|
~iterator_buffer() { flush(); }
|
|
|
|
|
|
|
|
|
|
OutputIt out() {
|
|
|
|
|
flush();
|
|
|
|
|
return out_;
|
|
|
|
|
}
|
|
|
|
|
size_t count() const { return Traits::count() + this->size(); }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T> class iterator_buffer<T*, T> final : public buffer<T> {
|
|
|
|
|
protected:
|
|
|
|
|
void grow(size_t) final FMT_OVERRIDE {}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
explicit iterator_buffer(T* out, size_t = 0) : buffer<T>(out, 0, ~size_t()) {}
|
|
|
|
|
|
|
|
|
|
T* out() { return &*this->end(); }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// A buffer that writes to a container with the contiguous storage.
|
|
|
|
|
template <typename Container>
|
|
|
|
|
class container_buffer : public buffer<typename Container::value_type> {
|
|
|
|
|
class iterator_buffer<std::back_insert_iterator<Container>,
|
|
|
|
|
enable_if_t<is_contiguous<Container>::value,
|
|
|
|
|
typename Container::value_type>>
|
|
|
|
|
final : public buffer<typename Container::value_type> {
|
|
|
|
|
private:
|
|
|
|
|
Container& container_;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void grow(size_t capacity) FMT_OVERRIDE {
|
|
|
|
|
void grow(size_t capacity) final FMT_OVERRIDE {
|
|
|
|
|
container_.resize(capacity);
|
|
|
|
|
this->set(&container_[0], capacity);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
explicit container_buffer(Container& c)
|
|
|
|
|
explicit iterator_buffer(Container& c)
|
|
|
|
|
: buffer<typename Container::value_type>(c.size()), container_(c) {}
|
|
|
|
|
explicit iterator_buffer(std::back_insert_iterator<Container> out, size_t = 0)
|
|
|
|
|
: iterator_buffer(get_container(out)) {}
|
|
|
|
|
std::back_insert_iterator<Container> out() {
|
|
|
|
|
return std::back_inserter(container_);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Extracts a reference to the container from back_insert_iterator.
|
|
|
|
|
template <typename Container>
|
|
|
|
|
inline Container& get_container(std::back_insert_iterator<Container> it) {
|
|
|
|
|
using bi_iterator = std::back_insert_iterator<Container>;
|
|
|
|
|
struct accessor : bi_iterator {
|
|
|
|
|
accessor(bi_iterator iter) : bi_iterator(iter) {}
|
|
|
|
|
using bi_iterator::container;
|
|
|
|
|
};
|
|
|
|
|
return *accessor(it).container;
|
|
|
|
|
// A buffer that counts the number of code units written discarding the output.
|
|
|
|
|
template <typename T = char> class counting_buffer final : public buffer<T> {
|
|
|
|
|
private:
|
|
|
|
|
enum { buffer_size = 256 };
|
|
|
|
|
T data_[buffer_size];
|
|
|
|
|
size_t count_ = 0;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void grow(size_t) final FMT_OVERRIDE {
|
|
|
|
|
if (this->size() != buffer_size) return;
|
|
|
|
|
count_ += this->size();
|
|
|
|
|
this->clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
counting_buffer() : buffer<T>(data_, 0, buffer_size) {}
|
|
|
|
|
|
|
|
|
|
size_t count() { return count_ + this->size(); }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// An output iterator that appends to the buffer.
|
|
|
|
|
// It is used to reduce symbol sizes for the common case.
|
|
|
|
|
template <typename T>
|
|
|
|
|
class buffer_appender : public std::back_insert_iterator<buffer<T>> {
|
|
|
|
|
using base = std::back_insert_iterator<buffer<T>>;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
explicit buffer_appender(buffer<T>& buf) : base(buf) {}
|
|
|
|
|
buffer_appender(base it) : base(it) {}
|
|
|
|
|
|
|
|
|
|
buffer_appender& operator++() {
|
|
|
|
|
base::operator++();
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
buffer_appender operator++(int) {
|
|
|
|
|
buffer_appender tmp = *this;
|
|
|
|
|
++*this;
|
|
|
|
|
return tmp;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Maps an output iterator into a buffer.
|
|
|
|
|
template <typename T, typename OutputIt>
|
|
|
|
|
iterator_buffer<OutputIt, T> get_buffer(OutputIt);
|
|
|
|
|
template <typename T> buffer<T>& get_buffer(buffer_appender<T>);
|
|
|
|
|
|
|
|
|
|
template <typename OutputIt> OutputIt get_buffer_init(OutputIt out) {
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
template <typename T> buffer<T>& get_buffer_init(buffer_appender<T> out) {
|
|
|
|
|
return get_container(out);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Buffer>
|
|
|
|
|
auto get_iterator(Buffer& buf) -> decltype(buf.out()) {
|
|
|
|
|
return buf.out();
|
|
|
|
|
}
|
|
|
|
|
template <typename T> buffer_appender<T> get_iterator(buffer<T>& buf) {
|
|
|
|
|
return buffer_appender<T>(buf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename Char = char, typename Enable = void>
|
|
|
|
@ -759,7 +918,8 @@ template <typename Char> struct named_arg_info {
|
|
|
|
|
template <typename T, typename Char, size_t NUM_ARGS, size_t NUM_NAMED_ARGS>
|
|
|
|
|
struct arg_data {
|
|
|
|
|
// args_[0].named_args points to named_args_ to avoid bloating format_args.
|
|
|
|
|
T args_[1 + (NUM_ARGS != 0 ? NUM_ARGS : 1)];
|
|
|
|
|
// +1 to workaround a bug in gcc 7.5 that causes duplicated-branches warning.
|
|
|
|
|
T args_[1 + (NUM_ARGS != 0 ? NUM_ARGS : +1)];
|
|
|
|
|
named_arg_info<Char> named_args_[NUM_NAMED_ARGS];
|
|
|
|
|
|
|
|
|
|
template <typename... U>
|
|
|
|
@ -771,7 +931,8 @@ struct arg_data {
|
|
|
|
|
|
|
|
|
|
template <typename T, typename Char, size_t NUM_ARGS>
|
|
|
|
|
struct arg_data<T, Char, NUM_ARGS, 0> {
|
|
|
|
|
T args_[NUM_ARGS != 0 ? NUM_ARGS : 1];
|
|
|
|
|
// +1 to workaround a bug in gcc 7.5 that causes duplicated-branches warning.
|
|
|
|
|
T args_[NUM_ARGS != 0 ? NUM_ARGS : +1];
|
|
|
|
|
|
|
|
|
|
template <typename... U>
|
|
|
|
|
FMT_INLINE arg_data(const U&... init) : args_{init...} {}
|
|
|
|
@ -959,6 +1120,8 @@ enum { long_short = sizeof(long) == sizeof(int) };
|
|
|
|
|
using long_type = conditional_t<long_short, int, long long>;
|
|
|
|
|
using ulong_type = conditional_t<long_short, unsigned, unsigned long long>;
|
|
|
|
|
|
|
|
|
|
struct unformattable {};
|
|
|
|
|
|
|
|
|
|
// Maps formatting arguments to core types.
|
|
|
|
|
template <typename Context> struct arg_mapper {
|
|
|
|
|
using char_type = typename Context::char_type;
|
|
|
|
@ -1067,15 +1230,7 @@ template <typename Context> struct arg_mapper {
|
|
|
|
|
return map(val.value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int map(...) {
|
|
|
|
|
constexpr bool formattable = sizeof(Context) == 0;
|
|
|
|
|
static_assert(
|
|
|
|
|
formattable,
|
|
|
|
|
"Cannot format argument. To make type T formattable provide a "
|
|
|
|
|
"formatter<T> specialization: "
|
|
|
|
|
"https://fmt.dev/latest/api.html#formatting-user-defined-types");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
unformattable map(...) { return {}; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// A type constant after applying arg_mapper<Context>.
|
|
|
|
@ -1199,15 +1354,25 @@ FMT_CONSTEXPR_DECL FMT_INLINE auto visit_format_arg(
|
|
|
|
|
return vis(monostate());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Checks whether T is a container with contiguous storage.
|
|
|
|
|
template <typename T> struct is_contiguous : std::false_type {};
|
|
|
|
|
template <typename Char>
|
|
|
|
|
struct is_contiguous<std::basic_string<Char>> : std::true_type {};
|
|
|
|
|
template <typename Char>
|
|
|
|
|
struct is_contiguous<detail::buffer<Char>> : std::true_type {};
|
|
|
|
|
template <typename T> struct formattable : std::false_type {};
|
|
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
|
|
// A workaround for gcc 4.8 to make void_t work in a SFINAE context.
|
|
|
|
|
template <typename... Ts> struct void_t_impl { using type = void; };
|
|
|
|
|
template <typename... Ts>
|
|
|
|
|
using void_t = typename detail::void_t_impl<Ts...>::type;
|
|
|
|
|
|
|
|
|
|
template <typename It, typename T, typename Enable = void>
|
|
|
|
|
struct is_output_iterator : std::false_type {};
|
|
|
|
|
|
|
|
|
|
template <typename It, typename T>
|
|
|
|
|
struct is_output_iterator<
|
|
|
|
|
It, T,
|
|
|
|
|
void_t<typename std::iterator_traits<It>::iterator_category,
|
|
|
|
|
decltype(*std::declval<It>() = std::declval<T>())>>
|
|
|
|
|
: std::true_type {};
|
|
|
|
|
|
|
|
|
|
template <typename OutputIt>
|
|
|
|
|
struct is_back_insert_iterator : std::false_type {};
|
|
|
|
|
template <typename Container>
|
|
|
|
@ -1219,6 +1384,9 @@ struct is_contiguous_back_insert_iterator : std::false_type {};
|
|
|
|
|
template <typename Container>
|
|
|
|
|
struct is_contiguous_back_insert_iterator<std::back_insert_iterator<Container>>
|
|
|
|
|
: is_contiguous<Container> {};
|
|
|
|
|
template <typename Char>
|
|
|
|
|
struct is_contiguous_back_insert_iterator<buffer_appender<Char>>
|
|
|
|
|
: std::true_type {};
|
|
|
|
|
|
|
|
|
|
// A type-erased reference to an std::locale to avoid heavy <locale> include.
|
|
|
|
|
class locale_ref {
|
|
|
|
@ -1250,13 +1418,24 @@ FMT_CONSTEXPR basic_format_arg<Context> make_arg(const T& value) {
|
|
|
|
|
return arg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T> int check(unformattable) {
|
|
|
|
|
static_assert(
|
|
|
|
|
formattable<T>(),
|
|
|
|
|
"Cannot format an argument. To make type T formattable provide a "
|
|
|
|
|
"formatter<T> specialization: https://fmt.dev/latest/api.html#udt");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename U> inline const U& check(const U& val) {
|
|
|
|
|
return val;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// The type template parameter is there to avoid an ODR violation when using
|
|
|
|
|
// a fallback formatter in one translation unit and an implicit conversion in
|
|
|
|
|
// another (not recommended).
|
|
|
|
|
template <bool IS_PACKED, typename Context, type, typename T,
|
|
|
|
|
FMT_ENABLE_IF(IS_PACKED)>
|
|
|
|
|
inline value<Context> make_arg(const T& val) {
|
|
|
|
|
return arg_mapper<Context>().map(val);
|
|
|
|
|
return check<T>(arg_mapper<Context>().map(val));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <bool IS_PACKED, typename Context, type, typename T,
|
|
|
|
@ -1356,13 +1535,13 @@ template <typename OutputIt, typename Char> class basic_format_context {
|
|
|
|
|
|
|
|
|
|
template <typename Char>
|
|
|
|
|
using buffer_context =
|
|
|
|
|
basic_format_context<std::back_insert_iterator<detail::buffer<Char>>, Char>;
|
|
|
|
|
basic_format_context<detail::buffer_appender<Char>, Char>;
|
|
|
|
|
using format_context = buffer_context<char>;
|
|
|
|
|
using wformat_context = buffer_context<wchar_t>;
|
|
|
|
|
|
|
|
|
|
// Workaround a bug in gcc: https://stackoverflow.com/q/62767544/471164.
|
|
|
|
|
// Workaround an alias issue: https://stackoverflow.com/q/62767544/471164.
|
|
|
|
|
#define FMT_BUFFER_CONTEXT(Char) \
|
|
|
|
|
basic_format_context<std::back_insert_iterator<detail::buffer<Char>>, Char>
|
|
|
|
|
basic_format_context<detail::buffer_appender<Char>, Char>
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
@ -1414,7 +1593,7 @@ class format_arg_store
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
Constructs an `~fmt::format_arg_store` object that contains references to
|
|
|
|
|
Constructs a `~fmt::format_arg_store` object that contains references to
|
|
|
|
|
arguments and can be implicitly converted to `~fmt::format_args`. `Context`
|
|
|
|
|
can be omitted in which case it defaults to `~fmt::context`.
|
|
|
|
|
See `~fmt::arg` for lifetime considerations.
|
|
|
|
@ -1426,6 +1605,27 @@ inline format_arg_store<Context, Args...> make_format_args(
|
|
|
|
|
return {args...};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
Constructs a `~fmt::format_arg_store` object that contains references
|
|
|
|
|
to arguments and can be implicitly converted to `~fmt::format_args`.
|
|
|
|
|
If ``format_str`` is a compile-time string then `make_args_checked` checks
|
|
|
|
|
its validity at compile time.
|
|
|
|
|
\endrst
|
|
|
|
|
*/
|
|
|
|
|
template <typename... Args, typename S, typename Char = char_t<S>>
|
|
|
|
|
inline auto make_args_checked(const S& format_str,
|
|
|
|
|
const remove_reference_t<Args>&... args)
|
|
|
|
|
-> format_arg_store<buffer_context<Char>, remove_reference_t<Args>...> {
|
|
|
|
|
static_assert(
|
|
|
|
|
detail::count<(
|
|
|
|
|
std::is_base_of<detail::view, remove_reference_t<Args>>::value &&
|
|
|
|
|
std::is_reference<Args>::value)...>() == 0,
|
|
|
|
|
"passing views as lvalues is disallowed");
|
|
|
|
|
detail::check_format_string<Args...>(format_str);
|
|
|
|
|
return {args...};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
Returns a named argument to be used in a formatting function. It should only
|
|
|
|
@ -1729,7 +1929,14 @@ template <typename Context> class basic_format_args {
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** An alias to ``basic_format_args<context>``. */
|
|
|
|
|
#ifdef FMT_ARM_ABI_COMPATIBILITY
|
|
|
|
|
/** An alias to ``basic_format_args<format_context>``. */
|
|
|
|
|
// Separate types would result in shorter symbols but break ABI compatibility
|
|
|
|
|
// between clang and gcc on ARM (#1919).
|
|
|
|
|
using format_args = basic_format_args<format_context>;
|
|
|
|
|
using wformat_args = basic_format_args<wformat_context>;
|
|
|
|
|
#else
|
|
|
|
|
// DEPRECATED! These are kept for ABI compatibility.
|
|
|
|
|
// It is a separate type rather than an alias to make symbols readable.
|
|
|
|
|
struct format_args : basic_format_args<format_context> {
|
|
|
|
|
template <typename... Args>
|
|
|
|
@ -1738,31 +1945,9 @@ struct format_args : basic_format_args<format_context> {
|
|
|
|
|
struct wformat_args : basic_format_args<wformat_context> {
|
|
|
|
|
using basic_format_args::basic_format_args;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
|
|
// Reports a compile-time error if S is not a valid format string.
|
|
|
|
|
template <typename..., typename S, FMT_ENABLE_IF(!is_compile_string<S>::value)>
|
|
|
|
|
FMT_INLINE void check_format_string(const S&) {
|
|
|
|
|
#ifdef FMT_ENFORCE_COMPILE_STRING
|
|
|
|
|
static_assert(is_compile_string<S>::value,
|
|
|
|
|
"FMT_ENFORCE_COMPILE_STRING requires all format strings to use "
|
|
|
|
|
"FMT_STRING.");
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
template <typename..., typename S, FMT_ENABLE_IF(is_compile_string<S>::value)>
|
|
|
|
|
void check_format_string(S);
|
|
|
|
|
|
|
|
|
|
template <typename... Args, typename S, typename Char = char_t<S>>
|
|
|
|
|
inline format_arg_store<buffer_context<Char>, remove_reference_t<Args>...>
|
|
|
|
|
make_args_checked(const S& format_str,
|
|
|
|
|
const remove_reference_t<Args>&... args) {
|
|
|
|
|
static_assert(count<(std::is_base_of<view, remove_reference_t<Args>>::value &&
|
|
|
|
|
std::is_reference<Args>::value)...>() == 0,
|
|
|
|
|
"passing views as lvalues is disallowed");
|
|
|
|
|
check_format_string<Args...>(format_str);
|
|
|
|
|
return {args...};
|
|
|
|
|
}
|
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
|
|
template <typename Char, FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
|
|
|
|
|
std::basic_string<Char> vformat(
|
|
|
|
@ -1772,9 +1957,10 @@ std::basic_string<Char> vformat(
|
|
|
|
|
FMT_API std::string vformat(string_view format_str, format_args args);
|
|
|
|
|
|
|
|
|
|
template <typename Char>
|
|
|
|
|
typename FMT_BUFFER_CONTEXT(Char)::iterator vformat_to(
|
|
|
|
|
void vformat_to(
|
|
|
|
|
buffer<Char>& buf, basic_string_view<Char> format_str,
|
|
|
|
|
basic_format_args<FMT_BUFFER_CONTEXT(type_identity_t<Char>)> args);
|
|
|
|
|
basic_format_args<FMT_BUFFER_CONTEXT(type_identity_t<Char>)> args,
|
|
|
|
|
detail::locale_ref loc = {});
|
|
|
|
|
|
|
|
|
|
template <typename Char, typename Args,
|
|
|
|
|
FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
|
|
|
|
@ -1789,26 +1975,80 @@ inline void vprint_mojibake(std::FILE*, string_view, format_args) {}
|
|
|
|
|
/** Formats a string and writes the output to ``out``. */
|
|
|
|
|
// GCC 8 and earlier cannot handle std::back_insert_iterator<Container> with
|
|
|
|
|
// vformat_to<ArgFormatter>(...) overload, so SFINAE on iterator type instead.
|
|
|
|
|
template <
|
|
|
|
|
typename OutputIt, typename S, typename Char = char_t<S>,
|
|
|
|
|
FMT_ENABLE_IF(detail::is_contiguous_back_insert_iterator<OutputIt>::value)>
|
|
|
|
|
OutputIt vformat_to(
|
|
|
|
|
OutputIt out, const S& format_str,
|
|
|
|
|
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
|
|
|
|
auto& c = detail::get_container(out);
|
|
|
|
|
detail::container_buffer<remove_reference_t<decltype(c)>> buf(c);
|
|
|
|
|
template <typename OutputIt, typename S, typename Char = char_t<S>,
|
|
|
|
|
bool enable = detail::is_output_iterator<OutputIt, Char>::value>
|
|
|
|
|
auto vformat_to(OutputIt out, const S& format_str,
|
|
|
|
|
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
|
|
|
|
-> typename std::enable_if<enable, OutputIt>::type {
|
|
|
|
|
decltype(detail::get_buffer<Char>(out)) buf(detail::get_buffer_init(out));
|
|
|
|
|
detail::vformat_to(buf, to_string_view(format_str), args);
|
|
|
|
|
return out;
|
|
|
|
|
return detail::get_iterator(buf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Container, typename S, typename... Args,
|
|
|
|
|
FMT_ENABLE_IF(
|
|
|
|
|
is_contiguous<Container>::value&& detail::is_string<S>::value)>
|
|
|
|
|
inline std::back_insert_iterator<Container> format_to(
|
|
|
|
|
std::back_insert_iterator<Container> out, const S& format_str,
|
|
|
|
|
Args&&... args) {
|
|
|
|
|
return vformat_to(out, to_string_view(format_str),
|
|
|
|
|
detail::make_args_checked<Args...>(format_str, args...));
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
Formats arguments, writes the result to the output iterator ``out`` and returns
|
|
|
|
|
the iterator past the end of the output range.
|
|
|
|
|
|
|
|
|
|
**Example**::
|
|
|
|
|
|
|
|
|
|
std::vector<char> out;
|
|
|
|
|
fmt::format_to(std::back_inserter(out), "{}", 42);
|
|
|
|
|
\endrst
|
|
|
|
|
*/
|
|
|
|
|
// We cannot use FMT_ENABLE_IF because of a bug in gcc 8.3.
|
|
|
|
|
template <typename OutputIt, typename S, typename... Args,
|
|
|
|
|
bool enable = detail::is_output_iterator<OutputIt, char_t<S>>::value>
|
|
|
|
|
inline auto format_to(OutputIt out, const S& format_str, Args&&... args) ->
|
|
|
|
|
typename std::enable_if<enable, OutputIt>::type {
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
return vformat_to(out, to_string_view(format_str), vargs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename OutputIt> struct format_to_n_result {
|
|
|
|
|
/** Iterator past the end of the output range. */
|
|
|
|
|
OutputIt out;
|
|
|
|
|
/** Total (not truncated) output size. */
|
|
|
|
|
size_t size;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename OutputIt, typename Char, typename... Args,
|
|
|
|
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value)>
|
|
|
|
|
inline format_to_n_result<OutputIt> vformat_to_n(
|
|
|
|
|
OutputIt out, size_t n, basic_string_view<Char> format_str,
|
|
|
|
|
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
|
|
|
|
detail::iterator_buffer<OutputIt, Char, detail::fixed_buffer_traits> buf(out,
|
|
|
|
|
n);
|
|
|
|
|
detail::vformat_to(buf, format_str, args);
|
|
|
|
|
return {buf.out(), buf.count()};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
\rst
|
|
|
|
|
Formats arguments, writes up to ``n`` characters of the result to the output
|
|
|
|
|
iterator ``out`` and returns the total output size and the iterator past the
|
|
|
|
|
end of the output range.
|
|
|
|
|
\endrst
|
|
|
|
|
*/
|
|
|
|
|
template <typename OutputIt, typename S, typename... Args,
|
|
|
|
|
bool enable = detail::is_output_iterator<OutputIt, char_t<S>>::value>
|
|
|
|
|
inline auto format_to_n(OutputIt out, size_t n, const S& format_str,
|
|
|
|
|
const Args&... args) ->
|
|
|
|
|
typename std::enable_if<enable, format_to_n_result<OutputIt>>::type {
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
return vformat_to_n(out, n, to_string_view(format_str), vargs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
Returns the number of characters in the output of
|
|
|
|
|
``format(format_str, args...)``.
|
|
|
|
|
*/
|
|
|
|
|
template <typename... Args>
|
|
|
|
|
inline size_t formatted_size(string_view format_str, Args&&... args) {
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
detail::counting_buffer<> buf;
|
|
|
|
|
detail::vformat_to(buf, format_str, vargs);
|
|
|
|
|
return buf.count();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename S, typename Char = char_t<S>>
|
|
|
|
@ -1832,7 +2072,7 @@ FMT_INLINE std::basic_string<Char> vformat(
|
|
|
|
|
// std::basic_string<char_t<S>> to reduce the symbol size.
|
|
|
|
|
template <typename S, typename... Args, typename Char = char_t<S>>
|
|
|
|
|
FMT_INLINE std::basic_string<Char> format(const S& format_str, Args&&... args) {
|
|
|
|
|
const auto& vargs = detail::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
return detail::vformat(to_string_view(format_str), vargs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -1852,7 +2092,7 @@ FMT_API void vprint(std::FILE*, string_view, format_args);
|
|
|
|
|
*/
|
|
|
|
|
template <typename S, typename... Args, typename Char = char_t<S>>
|
|
|
|
|
inline void print(std::FILE* f, const S& format_str, Args&&... args) {
|
|
|
|
|
const auto& vargs = detail::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
return detail::is_unicode<Char>()
|
|
|
|
|
? vprint(f, to_string_view(format_str), vargs)
|
|
|
|
|
: detail::vprint_mojibake(f, to_string_view(format_str), vargs);
|
|
|
|
@ -1871,7 +2111,7 @@ inline void print(std::FILE* f, const S& format_str, Args&&... args) {
|
|
|
|
|
*/
|
|
|
|
|
template <typename S, typename... Args, typename Char = char_t<S>>
|
|
|
|
|
inline void print(const S& format_str, Args&&... args) {
|
|
|
|
|
const auto& vargs = detail::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
const auto& vargs = fmt::make_args_checked<Args...>(format_str, args...);
|
|
|
|
|
return detail::is_unicode<Char>()
|
|
|
|
|
? vprint(to_string_view(format_str), vargs)
|
|
|
|
|
: detail::vprint_mojibake(stdout, to_string_view(format_str),
|
|
|
|
|