/* Formatting library for C++ Copyright (c) 2012 - 2016, Victor Zverovich All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef FMT_FORMAT_H_ #define FMT_FORMAT_H_ #include #include #include #include #include #include #include #include "fmt/core.h" #ifdef _SECURE_SCL # define FMT_SECURE_SCL _SECURE_SCL #else # define FMT_SECURE_SCL 0 #endif #if FMT_SECURE_SCL # include #endif #ifdef __has_builtin # define FMT_HAS_BUILTIN(x) __has_builtin(x) #else # define FMT_HAS_BUILTIN(x) 0 #endif #ifdef __GNUC__ # if FMT_GCC_VERSION >= 406 # pragma GCC diagnostic push // Disable the warning about declaration shadowing because it affects too // many valid cases. # pragma GCC diagnostic ignored "-Wshadow" // Disable the warning about implicit conversions that may change the sign of // an integer; silencing it otherwise would require many explicit casts. # pragma GCC diagnostic ignored "-Wsign-conversion" # endif #endif #ifdef __clang__ # define FMT_CLANG_VERSION (__clang_major__ * 100 + __clang_minor__) #endif #if defined(__INTEL_COMPILER) # define FMT_ICC_VERSION __INTEL_COMPILER #elif defined(__ICL) # define FMT_ICC_VERSION __ICL #endif #if defined(__clang__) && !defined(FMT_ICC_VERSION) # pragma clang diagnostic push # pragma clang diagnostic ignored "-Wdocumentation-unknown-command" # pragma clang diagnostic ignored "-Wgnu-string-literal-operator-template" # pragma clang diagnostic ignored "-Wpadded" #endif #ifdef __GNUC_LIBSTD__ # define FMT_GNUC_LIBSTD_VERSION (__GNUC_LIBSTD__ * 100 + __GNUC_LIBSTD_MINOR__) #endif #ifdef __has_cpp_attribute # define FMT_HAS_CPP_ATTRIBUTE(x) __has_cpp_attribute(x) #else # define FMT_HAS_CPP_ATTRIBUTE(x) 0 #endif #ifndef FMT_THROW # if FMT_EXCEPTIONS # define FMT_THROW(x) throw x # else # define FMT_THROW(x) assert(false) # endif #endif // Use the compiler's attribute noreturn. #if defined(__MINGW32__) || defined(__MINGW64__) # define FMT_NORETURN __attribute__((noreturn)) #elif FMT_HAS_CPP_ATTRIBUTE(noreturn) # define FMT_NORETURN [[noreturn]] #else # define FMT_NORETURN #endif #ifndef FMT_USE_USER_DEFINED_LITERALS // For Intel's compiler both it and the system gcc/msc must support UDLs. # if (FMT_HAS_FEATURE(cxx_user_literals) || \ FMT_GCC_VERSION >= 407 || FMT_MSC_VER >= 1900) && \ (!defined(FMT_ICC_VERSION) || FMT_ICC_VERSION >= 1500) # define FMT_USE_USER_DEFINED_LITERALS 1 # else # define FMT_USE_USER_DEFINED_LITERALS 0 # endif #endif #if FMT_USE_USER_DEFINED_LITERALS && \ (FMT_GCC_VERSION >= 600 || FMT_CLANG_VERSION >= 304) # define FMT_UDL_TEMPLATE 1 #else # define FMT_UDL_TEMPLATE 0 #endif #if FMT_GCC_VERSION >= 400 || FMT_HAS_BUILTIN(__builtin_clz) # define FMT_BUILTIN_CLZ(n) __builtin_clz(n) #endif #if FMT_GCC_VERSION >= 400 || FMT_HAS_BUILTIN(__builtin_clzll) # define FMT_BUILTIN_CLZLL(n) __builtin_clzll(n) #endif // Some compilers masquerade as both MSVC and GCC-likes or otherwise support // __builtin_clz and __builtin_clzll, so only define FMT_BUILTIN_CLZ using the // MSVC intrinsics if the clz and clzll builtins are not available. #if FMT_MSC_VER && !defined(FMT_BUILTIN_CLZLL) # include // _BitScanReverse, _BitScanReverse64 namespace fmt { namespace internal { # pragma intrinsic(_BitScanReverse) inline uint32_t clz(uint32_t x) { unsigned long r = 0; _BitScanReverse(&r, x); assert(x != 0); // Static analysis complains about using uninitialized data // "r", but the only way that can happen is if "x" is 0, // which the callers guarantee to not happen. # pragma warning(suppress: 6102) return 31 - r; } # define FMT_BUILTIN_CLZ(n) fmt::internal::clz(n) # ifdef _WIN64 # pragma intrinsic(_BitScanReverse64) # endif inline uint32_t clzll(uint64_t x) { unsigned long r = 0; # ifdef _WIN64 _BitScanReverse64(&r, x); # else // Scan the high 32 bits. if (_BitScanReverse(&r, static_cast(x >> 32))) return 63 - (r + 32); // Scan the low 32 bits. _BitScanReverse(&r, static_cast(x)); # endif assert(x != 0); // Static analysis complains about using uninitialized data // "r", but the only way that can happen is if "x" is 0, // which the callers guarantee to not happen. # pragma warning(suppress: 6102) return 63 - r; } # define FMT_BUILTIN_CLZLL(n) fmt::internal::clzll(n) } } #endif namespace fmt { namespace internal { struct dummy_int { int data[2]; operator int() const { return 0; } }; typedef std::numeric_limits fputil; // Dummy implementations of system functions such as signbit and ecvt called // if the latter are not available. inline dummy_int signbit(...) { return dummy_int(); } inline dummy_int _ecvt_s(...) { return dummy_int(); } inline dummy_int isinf(...) { return dummy_int(); } inline dummy_int _finite(...) { return dummy_int(); } inline dummy_int isnan(...) { return dummy_int(); } inline dummy_int _isnan(...) { return dummy_int(); } } } // namespace fmt namespace std { // Standard permits specialization of std::numeric_limits. This specialization // is used to resolve ambiguity between isinf and std::isinf in glibc: // https://gcc.gnu.org/bugzilla/show_bug.cgi?id=48891 // and the same for isnan and signbit. template <> class numeric_limits : public std::numeric_limits { public: // Portable version of isinf. template static bool isinfinity(T x) { using namespace fmt::internal; // The resolution "priority" is: // isinf macro > std::isinf > ::isinf > fmt::internal::isinf if (const_check(sizeof(isinf(x)) == sizeof(bool) || sizeof(isinf(x)) == sizeof(int))) { return isinf(x) != 0; } return !_finite(static_cast(x)); } // Portable version of isnan. template static bool isnotanumber(T x) { using namespace fmt::internal; if (const_check(sizeof(isnan(x)) == sizeof(bool) || sizeof(isnan(x)) == sizeof(int))) { return isnan(x) != 0; } return _isnan(static_cast(x)) != 0; } // Portable version of signbit. static bool isnegative(double x) { using namespace fmt::internal; if (const_check(sizeof(signbit(x)) == sizeof(int))) return signbit(x) != 0; if (x < 0) return true; if (!isnotanumber(x)) return false; int dec = 0, sign = 0; char buffer[2]; // The buffer size must be >= 2 or _ecvt_s will fail. _ecvt_s(buffer, sizeof(buffer), x, 0, &dec, &sign); return sign != 0; } }; } // namespace std namespace fmt { template class basic_writer; /** A formatting error such as invalid format string. */ class format_error : public std::runtime_error { public: explicit format_error(const char *message) : std::runtime_error(message) {} explicit format_error(const std::string &message) : std::runtime_error(message) {} ~format_error() throw(); }; namespace internal { // Casts nonnegative integer to unsigned. template constexpr typename std::make_unsigned::type to_unsigned(Int value) { FMT_ASSERT(value >= 0, "negative value"); return static_cast::type>(value); } // The number of characters to store in the basic_memory_buffer object itself // to avoid dynamic memory allocation. enum { INLINE_BUFFER_SIZE = 500 }; #if FMT_SECURE_SCL // Use checked iterator to avoid warnings on MSVC. template inline stdext::checked_array_iterator make_ptr(T *ptr, std::size_t size) { return stdext::checked_array_iterator(ptr, size); } #else template inline T *make_ptr(T *ptr, std::size_t) { return ptr; } #endif } // namespace internal // A wrapper around std::locale used to reduce compile times since // is very heavy. class locale; class locale_provider { public: virtual ~locale_provider() {} virtual fmt::locale locale(); }; template template void basic_buffer::append(const U *begin, const U *end) { std::size_t new_size = size_ + internal::to_unsigned(end - begin); reserve(new_size); std::uninitialized_copy(begin, end, internal::make_ptr(ptr_, capacity_) + size_); size_ = new_size; } template inline std::basic_string to_string(const basic_buffer& buffer) { return std::basic_string(buffer.data(), buffer.size()); } /** \rst A dynamically growing memory buffer for trivially copyable/constructible types with the first SIZE elements stored in the object itself. You can use one of the following typedefs for common character types: +----------------+------------------------------+ | Type | Definition | +================+==============================+ | memory_buffer | basic_memory_buffer | +----------------+------------------------------+ | wmemory_buffer | basic_memory_buffer | +----------------+------------------------------+ **Example**:: memory_buffer out; format_to(out, "The answer is {}.", 42); This will write the following output to the ``out`` object: .. code-block:: none The answer is 42. The output can be converted to an ``std::string`` with ``to_string(out)``. \endrst */ template > class basic_memory_buffer : private Allocator, public basic_buffer { private: T store_[SIZE]; // Deallocate memory allocated by the buffer. void deallocate() { T* data = this->data(); if (data != store_) Allocator::deallocate(data, this->capacity()); } protected: void grow(std::size_t size); public: explicit basic_memory_buffer(const Allocator &alloc = Allocator()) : Allocator(alloc) { this->set(store_, SIZE); } ~basic_memory_buffer() { deallocate(); } private: // Move data from other to this buffer. void move(basic_memory_buffer &other) { Allocator &this_alloc = *this, &other_alloc = other; this_alloc = std::move(other_alloc); T* data = other.data(); std::size_t size = other.size(), capacity = other.capacity(); if (data == other.store_) { this->set(store_, capacity); std::uninitialized_copy(other.store_, other.store_ + size, internal::make_ptr(store_, capacity)); } else { this->set(data, capacity); // Set pointer to the inline array so that delete is not called // when deallocating. other.set(other.store_, 0); } this->resize(size); } public: /** \rst Constructs a :class:`fmt::basic_memory_buffer` object moving the content of the other object to it. \endrst */ basic_memory_buffer(basic_memory_buffer &&other) { move(other); } /** \rst Moves the content of the other ``basic_memory_buffer`` object to this one. \endrst */ basic_memory_buffer &operator=(basic_memory_buffer &&other) { assert(this != &other); deallocate(); move(other); return *this; } // Returns a copy of the allocator associated with this buffer. Allocator get_allocator() const { return *this; } }; template void basic_memory_buffer::grow(std::size_t size) { std::size_t old_capacity = this->capacity(); std::size_t new_capacity = old_capacity + old_capacity / 2; if (size > new_capacity) new_capacity = size; T *old_data = this->data(); T *new_data = this->allocate(new_capacity); // The following code doesn't throw, so the raw pointer above doesn't leak. std::uninitialized_copy(old_data, old_data + this->size(), internal::make_ptr(new_data, new_capacity)); this->set(new_data, new_capacity); // deallocate must not throw according to the standard, but even if it does, // the buffer already uses the new storage and will deallocate it in // destructor. if (old_data != store_) Allocator::deallocate(old_data, old_capacity); } typedef basic_memory_buffer memory_buffer; typedef basic_memory_buffer wmemory_buffer; /** \rst A fixed-size memory buffer. For a dynamically growing buffer use :class:`fmt::basic_memory_buffer`. Trying to increase the buffer size past the initial capacity will throw ``std::runtime_error``. \endrst */ template class basic_fixed_buffer : public basic_buffer { public: /** \rst Constructs a :class:`fmt::basic_fixed_buffer` object for *array* of the given size. \endrst */ basic_fixed_buffer(Char *array, std::size_t size) { this->set(array, size); } /** \rst Constructs a :class:`fmt::basic_fixed_buffer` object for *array* of the size known at compile time. \endrst */ template explicit basic_fixed_buffer(Char (&array)[SIZE]) { this->set(array, SIZE); } protected: FMT_API void grow(std::size_t size); }; namespace internal { template class basic_char_traits { public: static Char cast(int value) { return static_cast(value); } }; template class char_traits; template <> class char_traits : public basic_char_traits { private: // Conversion from wchar_t to char is not allowed. static char convert(wchar_t); public: static char convert(char value) { return value; } // Formats a floating-point number. template FMT_API static int format_float(char *buffer, std::size_t size, const char *format, unsigned width, int precision, T value); }; template <> class char_traits : public basic_char_traits { public: static wchar_t convert(char value) { return value; } static wchar_t convert(wchar_t value) { return value; } template FMT_API static int format_float(wchar_t *buffer, std::size_t size, const wchar_t *format, unsigned width, int precision, T value); }; template class null_terminating_iterator; template constexpr const Char *pointer_from(null_terminating_iterator it); // An iterator that produces a null terminator on *end. This simplifies parsing // and allows comparing the performance of processing a null-terminated string // vs string_view. template class null_terminating_iterator { public: using difference_type = std::ptrdiff_t; using value_type = Char; using pointer = const Char*; using reference = const Char&; using iterator_category = std::random_access_iterator_tag; null_terminating_iterator() : ptr_(0), end_(0) {} constexpr null_terminating_iterator(const Char *ptr, const Char *end) : ptr_(ptr), end_(end) {} template constexpr explicit null_terminating_iterator(const Range &r) : ptr_(r.begin()), end_(r.end()) {} null_terminating_iterator &operator=(const Char *ptr) { assert(ptr <= end_); ptr_ = ptr; return *this; } constexpr Char operator*() const { return ptr_ != end_ ? *ptr_ : 0; } constexpr null_terminating_iterator operator++() { ++ptr_; return *this; } constexpr null_terminating_iterator operator++(int) { null_terminating_iterator result(*this); ++ptr_; return result; } constexpr null_terminating_iterator operator--() { --ptr_; return *this; } constexpr null_terminating_iterator operator+(difference_type n) { return null_terminating_iterator(ptr_ + n, end_); } constexpr null_terminating_iterator operator-(difference_type n) { return null_terminating_iterator(ptr_ - n, end_); } constexpr null_terminating_iterator operator+=(difference_type n) { ptr_ += n; return *this; } constexpr difference_type operator-(null_terminating_iterator other) const { return ptr_ - other.ptr_; } constexpr bool operator!=(null_terminating_iterator other) const { return ptr_ != other.ptr_; } bool operator>=(null_terminating_iterator other) const { return ptr_ >= other.ptr_; } friend constexpr const Char *pointer_from(null_terminating_iterator it); private: const Char *ptr_; const Char *end_; }; template constexpr const T *pointer_from(const T *p) { return p; } template constexpr const Char *pointer_from(null_terminating_iterator it) { return it.ptr_; } // A range that can grow dynamically. template class dynamic_range { private: Container &container_; public: using iterator = decltype(container_.begin()); using value_type = typename Container::value_type; struct sentinel { friend bool operator!=(sentinel, iterator) { return false; } friend bool operator!=(iterator, sentinel) { return false; } }; explicit dynamic_range(Container &c) : container_(c) {} iterator begin() const { return container_.begin(); } sentinel end() const { return sentinel(); } friend iterator grow(dynamic_range r, size_t n) { auto size = r.container_.size(); r.container_.resize(size + n); return r.container_.begin() + size; } }; // Returns true if value is negative, false otherwise. // Same as (value < 0) but doesn't produce warnings if T is an unsigned type. template constexpr typename std::enable_if< std::numeric_limits::is_signed, bool>::type is_negative(T value) { return value < 0; } template constexpr typename std::enable_if< !std::numeric_limits::is_signed, bool>::type is_negative(T) { return false; } template struct int_traits { // Smallest of uint32_t and uint64_t that is large enough to represent // all values of T. typedef typename std::conditional< std::numeric_limits::digits <= 32, uint32_t, uint64_t>::type main_type; }; // Static data is placed in this class template to allow header-only // configuration. template struct FMT_API basic_data { static const uint32_t POWERS_OF_10_32[]; static const uint64_t POWERS_OF_10_64[]; static const char DIGITS[]; }; #ifndef FMT_USE_EXTERN_TEMPLATES // Clang doesn't have a feature check for extern templates so we check // for variadic templates which were introduced in the same version. # define FMT_USE_EXTERN_TEMPLATES (__clang__) #endif #if FMT_USE_EXTERN_TEMPLATES && !defined(FMT_HEADER_ONLY) extern template struct basic_data; #endif typedef basic_data<> data; #ifdef FMT_BUILTIN_CLZLL // Returns the number of decimal digits in n. Leading zeros are not counted // except for n == 0 in which case count_digits returns 1. inline unsigned count_digits(uint64_t n) { // Based on http://graphics.stanford.edu/~seander/bithacks.html#IntegerLog10 // and the benchmark https://github.com/localvoid/cxx-benchmark-count-digits. int t = (64 - FMT_BUILTIN_CLZLL(n | 1)) * 1233 >> 12; return to_unsigned(t) - (n < data::POWERS_OF_10_64[t]) + 1; } #else // Fallback version of count_digits used when __builtin_clz is not available. inline unsigned count_digits(uint64_t n) { unsigned count = 1; for (;;) { // Integer division is slow so do it for a group of four digits instead // of for every digit. The idea comes from the talk by Alexandrescu // "Three Optimization Tips for C++". See speed-test for a comparison. if (n < 10) return count; if (n < 100) return count + 1; if (n < 1000) return count + 2; if (n < 10000) return count + 3; n /= 10000u; count += 4; } } #endif #if FMT_HAS_CPP_ATTRIBUTE(always_inline) # define FMT_ALWAYS_INLINE __attribute__((always_inline)) #else # define FMT_ALWAYS_INLINE #endif template inline char *lg(uint32_t n, Handler h) FMT_ALWAYS_INLINE; // Computes g = floor(log10(n)) and calls h.on(n); template inline char *lg(uint32_t n, Handler h) { return n < 100 ? n < 10 ? h.template on<0>(n) : h.template on<1>(n) : n < 1000000 ? n < 10000 ? n < 1000 ? h.template on<2>(n) : h.template on<3>(n) : n < 100000 ? h.template on<4>(n) : h.template on<5>(n) : n < 100000000 ? n < 10000000 ? h.template on<6>(n) : h.template on<7>(n) : n < 1000000000 ? h.template on<8>(n) : h.template on<9>(n); } // An lg handler that formats a decimal number. // Usage: lg(n, decimal_formatter(buffer)); class decimal_formatter { private: char *buffer_; void write_pair(unsigned N, uint32_t index) { std::memcpy(buffer_ + N, data::DIGITS + index * 2, 2); } public: explicit decimal_formatter(char *buf) : buffer_(buf) {} template char *on(uint32_t u) { if (N == 0) { *buffer_ = static_cast(u) + '0'; } else if (N == 1) { write_pair(0, u); } else { // The idea of using 4.32 fixed-point numbers is based on // https://github.com/jeaiii/itoa unsigned n = N - 1; unsigned a = n / 5 * n * 53 / 16; uint64_t t = ((1ULL << (32 + a)) / data::POWERS_OF_10_32[n] + 1 - n / 9); t = ((t * u) >> a) + n / 5 * 4; write_pair(0, t >> 32); for (int i = 2; i < N; i += 2) { t = 100ULL * static_cast(t); write_pair(i, t >> 32); } if (N % 2 == 0) { buffer_[N] = static_cast( (10ULL * static_cast(t)) >> 32) + '0'; } } return buffer_ += N + 1; } }; // An lg handler that formats a decimal number with a terminating null. class decimal_formatter_null : public decimal_formatter { public: explicit decimal_formatter_null(char *buf) : decimal_formatter(buf) {} template char *on(uint32_t u) { char *buf = decimal_formatter::on(u); *buf = '\0'; return buf; } }; #ifdef FMT_BUILTIN_CLZ // Optional version of count_digits for better performance on 32-bit platforms. inline unsigned count_digits(uint32_t n) { int t = (32 - FMT_BUILTIN_CLZ(n | 1)) * 1233 >> 12; return to_unsigned(t) - (n < data::POWERS_OF_10_32[t]) + 1; } #endif // A functor that doesn't add a thousands separator. struct no_thousands_sep { template void operator()(Char *) {} }; // A functor that adds a thousands separator. template class add_thousands_sep { private: basic_string_view sep_; // Index of a decimal digit with the least significant digit having index 0. unsigned digit_index_; public: explicit add_thousands_sep(basic_string_view sep) : sep_(sep), digit_index_(0) {} void operator()(Char *&buffer) { if (++digit_index_ % 3 != 0) return; buffer -= sep_.size(); std::uninitialized_copy(sep_.data(), sep_.data() + sep_.size(), internal::make_ptr(buffer, sep_.size())); } }; template Char thousands_sep(locale_provider *lp); // Formats a decimal unsigned integer value writing into buffer. // thousands_sep is a functor that is called after writing each char to // add a thousands separator if necessary. template inline void format_decimal(Char *buffer, UInt value, unsigned num_digits, ThousandsSep thousands_sep) { buffer += num_digits; while (value >= 100) { // Integer division is slow so do it for a group of two digits instead // of for every digit. The idea comes from the talk by Alexandrescu // "Three Optimization Tips for C++". See speed-test for a comparison. unsigned index = static_cast((value % 100) * 2); value /= 100; *--buffer = data::DIGITS[index + 1]; thousands_sep(buffer); *--buffer = data::DIGITS[index]; thousands_sep(buffer); } if (value < 10) { *--buffer = static_cast('0' + value); return; } unsigned index = static_cast(value * 2); *--buffer = data::DIGITS[index + 1]; thousands_sep(buffer); *--buffer = data::DIGITS[index]; } template inline void format_decimal(Char *buffer, UInt value, unsigned num_digits) { return format_decimal(buffer, value, num_digits, no_thousands_sep()); } #ifndef _WIN32 # define FMT_USE_WINDOWS_H 0 #elif !defined(FMT_USE_WINDOWS_H) # define FMT_USE_WINDOWS_H 1 #endif // Define FMT_USE_WINDOWS_H to 0 to disable use of windows.h. // All the functionality that relies on it will be disabled too. #if FMT_USE_WINDOWS_H // A converter from UTF-8 to UTF-16. // It is only provided for Windows since other systems support UTF-8 natively. class utf8_to_utf16 { private: wmemory_buffer buffer_; public: FMT_API explicit utf8_to_utf16(string_view s); operator wstring_view() const { return wstring_view(&buffer_[0], size()); } size_t size() const { return buffer_.size() - 1; } const wchar_t *c_str() const { return &buffer_[0]; } std::wstring str() const { return std::wstring(&buffer_[0], size()); } }; // A converter from UTF-16 to UTF-8. // It is only provided for Windows since other systems support UTF-8 natively. class utf16_to_utf8 { private: memory_buffer buffer_; public: utf16_to_utf8() {} FMT_API explicit utf16_to_utf8(wstring_view s); operator string_view() const { return string_view(&buffer_[0], size()); } size_t size() const { return buffer_.size() - 1; } const char *c_str() const { return &buffer_[0]; } std::string str() const { return std::string(&buffer_[0], size()); } // Performs conversion returning a system error code instead of // throwing exception on conversion error. This method may still throw // in case of memory allocation error. FMT_API int convert(wstring_view s); }; FMT_API void format_windows_error(fmt::buffer &out, int error_code, fmt::string_view message) FMT_NOEXCEPT; #endif template struct null {}; } // namespace internal struct monostate {}; /** \rst Visits an argument dispatching to the appropriate visit method based on the argument type. For example, if the argument type is ``double`` then ``vis(value)`` will be called with the value of type ``double``. \endrst */ template constexpr typename std::result_of::type visit(Visitor &&vis, basic_arg arg) { using char_type = typename Context::char_type; switch (arg.type_) { case internal::NONE: return vis(monostate()); case internal::NAMED_ARG: FMT_ASSERT(false, "invalid argument type"); break; case internal::INT: return vis(arg.value_.int_value); case internal::UINT: return vis(arg.value_.uint_value); case internal::LONG_LONG: return vis(arg.value_.long_long_value); case internal::ULONG_LONG: return vis(arg.value_.ulong_long_value); case internal::BOOL: return vis(arg.value_.int_value != 0); case internal::CHAR: return vis(static_cast(arg.value_.int_value)); case internal::DOUBLE: return vis(arg.value_.double_value); case internal::LONG_DOUBLE: return vis(arg.value_.long_double_value); case internal::CSTRING: return vis(arg.value_.string.value); case internal::STRING: return vis(basic_string_view( arg.value_.string.value, arg.value_.string.size)); case internal::POINTER: return vis(arg.value_.pointer); case internal::CUSTOM: return vis(typename basic_arg::handle(arg.value_.custom)); } return typename std::result_of::type(); } enum alignment { ALIGN_DEFAULT, ALIGN_LEFT, ALIGN_RIGHT, ALIGN_CENTER, ALIGN_NUMERIC }; // Flags. enum {SIGN_FLAG = 1, PLUS_FLAG = 2, MINUS_FLAG = 4, HASH_FLAG = 8}; enum format_spec_tag {fill_tag, align_tag, width_tag, type_tag}; // Format specifier. template class format_spec { private: T value_; public: typedef T value_type; explicit format_spec(T value) : value_(value) {} T value() const { return value_; } }; // template // using fill_spec = format_spec; template class fill_spec : public format_spec { public: explicit fill_spec(Char value) : format_spec(value) {} }; typedef format_spec width_spec; typedef format_spec type_spec; // An empty format specifier. struct empty_spec {}; // An alignment specifier. struct align_spec : empty_spec { unsigned width_; // Fill is always wchar_t and cast to char if necessary to avoid having // two specialization of AlignSpec and its subclasses. wchar_t fill_; alignment align_; constexpr align_spec( unsigned width, wchar_t fill, alignment align = ALIGN_DEFAULT) : width_(width), fill_(fill), align_(align) {} constexpr unsigned width() const { return width_; } constexpr wchar_t fill() const { return fill_; } constexpr alignment align() const { return align_; } int precision() const { return -1; } }; // Format specifiers. template class basic_format_specs : public align_spec { private: template typename std::enable_if::value || std::is_same::value, void>::type set(fill_spec fill) { fill_ = fill.value(); } void set(width_spec width) { width_ = width.value(); } void set(type_spec type) { type_ = type.value(); } template void set(Spec spec, Specs... tail) { set(spec); set(tail...); } public: unsigned flags_; int precision_; Char type_; constexpr basic_format_specs( unsigned width = 0, char type = 0, wchar_t fill = ' ') : align_spec(width, fill), flags_(0), precision_(-1), type_(type) {} template explicit basic_format_specs(FormatSpecs... specs) : align_spec(0, ' '), flags_(0), precision_(-1), type_(0) { set(specs...); } constexpr bool flag(unsigned f) const { return (flags_ & f) != 0; } constexpr int precision() const { return precision_; } constexpr Char type() const { return type_; } }; typedef basic_format_specs format_specs; template constexpr unsigned basic_parse_context::next_arg_id() { if (next_arg_id_ >= 0) return internal::to_unsigned(next_arg_id_++); on_error("cannot switch from manual to automatic argument indexing"); return 0; } namespace internal { template constexpr void handle_int_type_spec(char spec, Handler &&handler) { switch (spec) { case 0: case 'd': handler.on_dec(); break; case 'x': case 'X': handler.on_hex(); break; case 'b': case 'B': handler.on_bin(); break; case 'o': handler.on_oct(); break; case 'n': handler.on_num(); break; default: handler.on_error(); } } template constexpr void handle_float_type_spec(char spec, Handler &&handler) { switch (spec) { case 0: case 'g': case 'G': handler.on_general(); break; case 'e': case 'E': handler.on_exp(); break; case 'f': case 'F': handler.on_fixed(); break; case 'a': case 'A': handler.on_hex(); break; default: handler.on_error(); break; } } template constexpr void handle_char_specs( const basic_format_specs &specs, Handler &&handler) { if (specs.type() && specs.type() != 'c') { handler.on_int(); return; } if (specs.align() == ALIGN_NUMERIC || specs.flag(~0u) != 0) handler.on_error("invalid format specifier for char"); handler.on_char(); } template constexpr void handle_cstring_type_spec(char spec, Handler &&handler) { if (spec == 0 || spec == 's') handler.on_string(); else if (spec == 'p') handler.on_pointer(); else handler.on_error("invalid type specifier"); } template constexpr void check_string_type_spec(char spec, ErrorHandler &&eh) { if (spec != 0 && spec != 's') eh.on_error("invalid type specifier"); } template constexpr void check_pointer_type_spec(char spec, ErrorHandler &&eh) { if (spec != 0 && spec != 'p') eh.on_error("invalid type specifier"); } template class int_type_checker : private ErrorHandler { public: constexpr explicit int_type_checker(ErrorHandler eh) : ErrorHandler(eh) {} constexpr void on_dec() {} constexpr void on_hex() {} constexpr void on_bin() {} constexpr void on_oct() {} constexpr void on_num() {} constexpr void on_error() { ErrorHandler::on_error("invalid type specifier"); } }; template class float_type_checker : private ErrorHandler { public: constexpr explicit float_type_checker(ErrorHandler eh) : ErrorHandler(eh) {} constexpr void on_general() {} constexpr void on_exp() {} constexpr void on_fixed() {} constexpr void on_hex() {} constexpr void on_error() { ErrorHandler::on_error("invalid type specifier"); } }; template class char_specs_checker : public ErrorHandler { private: char type_; public: constexpr char_specs_checker(char type, ErrorHandler eh) : ErrorHandler(eh), type_(type) {} constexpr void on_int() { handle_int_type_spec(type_, int_type_checker(*this)); } constexpr void on_char() {} }; template class cstring_type_checker : public ErrorHandler { public: constexpr explicit cstring_type_checker(ErrorHandler eh) : ErrorHandler(eh) {} constexpr void on_string() {} constexpr void on_pointer() {} }; template void arg_map::init(const basic_format_args &args) { if (map_) return; map_ = new entry[args.max_size()]; bool use_values = args.type(MAX_PACKED_ARGS - 1) == internal::NONE; if (use_values) { for (unsigned i = 0;/*nothing*/; ++i) { internal::type arg_type = args.type(i); switch (arg_type) { case internal::NONE: return; case internal::NAMED_ARG: push_back(args.values_[i]); break; default: break; // Do nothing. } } return; } for (unsigned i = 0; i != MAX_PACKED_ARGS; ++i) { if (args.type(i) == internal::NAMED_ARG) push_back(args.args_[i].value_); } for (unsigned i = MAX_PACKED_ARGS; ; ++i) { switch (args.args_[i].type_) { case internal::NONE: return; case internal::NAMED_ARG: push_back(args.args_[i].value_); break; default: break; // Do nothing. } } } template class arg_formatter_base { public: typedef basic_format_specs format_specs; private: using writer_type = basic_writer>; writer_type writer_; format_specs &specs_; FMT_DISALLOW_COPY_AND_ASSIGN(arg_formatter_base); void write_char(Char value) { writer_.write_padded(1, specs_, [value](auto &it) { *it++ = internal::char_traits::cast(value); }); } void write_pointer(const void *p) { specs_.flags_ = HASH_FLAG; specs_.type_ = 'x'; writer_.write_int(reinterpret_cast(p), specs_); } protected: writer_type &writer() { return writer_; } format_specs &spec() { return specs_; } void write(bool value) { writer_.write_str(string_view(value ? "true" : "false"), specs_); } void write(const Char *value) { writer_.write_str(basic_string_view( value, value != 0 ? std::char_traits::length(value) : 0), specs_); } public: typedef Char char_type; arg_formatter_base(basic_buffer &b, format_specs &s) : writer_(b), specs_(s) {} void operator()(monostate) { FMT_ASSERT(false, "invalid argument type"); } template typename std::enable_if::value>::type operator()(T value) { writer_.write_int(value, specs_); } template typename std::enable_if::value>::type operator()(T value) { writer_.write_double(value, specs_); } void operator()(bool value) { if (specs_.type_) return (*this)(value ? 1 : 0); write(value); } void operator()(Char value) { struct spec_handler : internal::error_handler { arg_formatter_base &formatter; Char value; spec_handler(arg_formatter_base& f, Char val): formatter(f), value(val) {} void on_int() { formatter.writer_.write_int(value, formatter.specs_); } void on_char() { formatter.write_char(value); } }; internal::handle_char_specs(specs_, spec_handler(*this, value)); } void operator()(const Char *value) { struct spec_handler : internal::error_handler { arg_formatter_base &formatter; const Char *value; spec_handler(arg_formatter_base &f, const Char *val) : formatter(f), value(val) {} void on_string() { formatter.write(value); } void on_pointer() { formatter.write_pointer(value); } }; internal::handle_cstring_type_spec( specs_.type_, spec_handler(*this, value)); } void operator()(basic_string_view value) { internal::check_string_type_spec(specs_.type_, internal::error_handler()); writer_.write_str(value, specs_); } void operator()(const void *value) { check_pointer_type_spec(specs_.type_, internal::error_handler()); write_pointer(value); } }; struct format_string {}; template constexpr bool is_name_start(Char c) { return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || '_' == c; } // Parses the input as an unsigned integer. This function assumes that the // first character is a digit and presence of a non-digit character at the end. // it: an iterator pointing to the beginning of the input range. template constexpr unsigned parse_nonnegative_int(Iterator &it, ErrorHandler &&eh) { assert('0' <= *it && *it <= '9'); unsigned value = 0; // Convert to unsigned to prevent a warning. unsigned max_int = (std::numeric_limits::max)(); unsigned big = max_int / 10; do { // Check for overflow. if (value > big) { value = max_int + 1; break; } value = value * 10 + (*it - '0'); // Workaround for MSVC "setup_exception stack overflow" error: auto next = it; ++next; it = next; } while ('0' <= *it && *it <= '9'); if (value > max_int) eh.on_error("number is too big"); return value; } template class custom_formatter { private: Context &ctx_; public: explicit custom_formatter(Context &ctx): ctx_(ctx) {} bool operator()(typename basic_arg::handle h) { h.format(ctx_); return true; } template bool operator()(T) { return false; } }; template struct is_integer { enum { value = std::is_integral::value && !std::is_same::value && !std::is_same::value && !std::is_same::value }; }; template class width_checker { public: explicit constexpr width_checker(ErrorHandler &eh) : handler_(eh) {} template constexpr typename std::enable_if< is_integer::value, unsigned long long>::type operator()(T value) { if (is_negative(value)) handler_.on_error("negative width"); return value; } template constexpr typename std::enable_if< !is_integer::value, unsigned long long>::type operator()(T) { handler_.on_error("width is not integer"); return 0; } private: ErrorHandler &handler_; }; template class precision_checker { public: explicit constexpr precision_checker(ErrorHandler &eh) : handler_(eh) {} template constexpr typename std::enable_if< is_integer::value, unsigned long long>::type operator()(T value) { if (is_negative(value)) handler_.on_error("negative precision"); return value; } template constexpr typename std::enable_if< !is_integer::value, unsigned long long>::type operator()(T) { handler_.on_error("precision is not integer"); return 0; } private: ErrorHandler &handler_; }; // A format specifier handler that sets fields in basic_format_specs. template class specs_setter { public: explicit constexpr specs_setter(basic_format_specs &specs): specs_(specs) {} constexpr specs_setter(const specs_setter &other) : specs_(other.specs_) {} constexpr void on_align(alignment align) { specs_.align_ = align; } constexpr void on_fill(Char fill) { specs_.fill_ = fill; } constexpr void on_plus() { specs_.flags_ |= SIGN_FLAG | PLUS_FLAG; } constexpr void on_minus() { specs_.flags_ |= MINUS_FLAG; } constexpr void on_space() { specs_.flags_ |= SIGN_FLAG; } constexpr void on_hash() { specs_.flags_ |= HASH_FLAG; } constexpr void on_zero() { specs_.align_ = ALIGN_NUMERIC; specs_.fill_ = '0'; } constexpr void on_width(unsigned width) { specs_.width_ = width; } constexpr void on_precision(unsigned precision) { specs_.precision_ = precision; } constexpr void end_precision() {} constexpr void on_type(Char type) { specs_.type_ = type; } protected: basic_format_specs &specs_; }; // A format specifier handler that checks if specifiers are consistent with the // argument type. template class specs_checker : public Handler { public: constexpr specs_checker(const Handler& handler, internal::type arg_type) : Handler(handler), arg_type_(arg_type) {} constexpr specs_checker(const specs_checker &other) : Handler(other), arg_type_(other.arg_type_) {} constexpr void on_align(alignment align) { if (align == ALIGN_NUMERIC) require_numeric_argument(); Handler::on_align(align); } constexpr void on_plus() { check_sign(); Handler::on_plus(); } constexpr void on_minus() { check_sign(); Handler::on_minus(); } constexpr void on_space() { check_sign(); Handler::on_space(); } constexpr void on_hash() { require_numeric_argument(); Handler::on_hash(); } constexpr void on_zero() { require_numeric_argument(); Handler::on_zero(); } constexpr void end_precision() { if (is_integral(arg_type_) || arg_type_ == POINTER) this->on_error("precision not allowed for this argument type"); } private: constexpr void require_numeric_argument() { if (!is_arithmetic(arg_type_)) this->on_error("format specifier requires numeric argument"); } constexpr void check_sign() { require_numeric_argument(); if (is_integral(arg_type_) && arg_type_ != INT && arg_type_ != LONG_LONG && arg_type_ != CHAR) { this->on_error("format specifier requires signed argument"); } } internal::type arg_type_; }; template