d4c6fa3122
cd4af11ef Update version 1ebc2f7cc Bump version f4c997062 Fix changelog 72920ba30 Update changelog 0907c08ae Fix handling of default alignmment with locale (#1801) 37c8f4eaf Don't use 128 bit integers with clang-cl (#1800) eaaaec999 Workaround a bug in msvc ccf8561cb Workaround broken numeric_limites, part 2 (#1787) 0cc73ebf7 Report error on missing named argument (#1796) 33efc3c94 Fix handling of iterators in locale-specific formatting (#1782) b9d749095 Update version 86b63bb71 Bump version cbf6be960 Update changelog 229ee9b46 Workaround broken numeric_limits (#1725) 2b7a146fa Fix a regression in handling digit separators (#1782) 89d0c7124 Fix compatibility with CMake 3.4 (#1779) f19b1a521 Update version 5c67fefb2 Fix a changelog entry 1d2a556e1 Fix undefined reference error 04c9b62fb Merge release branch 6be6762e5 Fix date f1dd2eb3c Bump version fbf3b943c Workaround a bug in gcc a29a01d30 Fix docs 9f0b3afb7 Bump version in namespace 86b2f99f8 Fix the docs c472ff12d Update version 5173a76ba Update version 1614af352 Minor corrections in the changelog 569a9b3a7 Bump version 4e7e3c65a Update docs 0f7a6bfa1 Add a section on std::format compatibility 4faec5a5e Update README.rst 7dbc8ac71 Update changelog c87dd746f Update changelog 372175caf Revert changelog changes 904754876 Add ClickHouse to the list of projects (#1751) d30bca64e Revert changelog conversion since GFM is not supported there d6047cdc4 Update changelog 810241b36 Convert changlog to markdown 661c47473 Rename changelog 7c33059fa Update ChangeLog.rst 9e20883ab Update README.rst 41899d522 Update changelog f42f45908 Update changelog 2381df654 Update readme 7ae816563 Update README.rst c56cf3d07 Update changelog and readme 01309a34a Deprecate arg_formatter a62d06055 Update changelog 23e3a2eee Update changelog d8e0554b9 Disable numeric formatting by default 1e8eea4f4 Update changelog 44bd5384a Fix formatting 20e19387a Update changelog 56fed7814 FMT_NUMERIC_ALIGN -> FMT_DEPRECATED_NUMERIC_ALIGN 56e63078f Make the n specifier an opt-in 31ce6bc70 Fix a conversion warning with Clang10 on Windows (#1750) c9c5b90da Fix a typo. Thanks Tracy Chapman from TripleChecker 1f3f84631 Fix a typo 5de62af60 Fix possible infinite recursion in FMT_ASSERT (#1744) cbddab2fe Use consistent include style f69b6eaab Add a simple buffered stream with no sync ba363b3a2 Use digit pairs as in unrolledlut a6f8e7d86 Update changelog e753244ab Update changelog 98a7a8b40 Update changelog and disable internal 3135d95fd Don't use non-portable attribute 8630a8f5f Tweak the docs cc3a88e6b Extract docs from compile.h 79c4b6bd7 Apply clang-format d130ee070 Document format string compilation d0f90b5be Spelling fixes 6e080660d Update README.rst 31c3a2426 Spelling fixes 613b3b459 Spelling fixes 978521bb8 Fix a compile error introduced in #1738 4e94c649f Deprecate compile 1a83443e6 Add user-defined type support to compilation 8bef1c3b3 Tweaks for EDG based compilers (Intel, nVidia, MCST/Elbrus, etc). b287c37c6 Do not use -Wl,--as-needed with emscripten. 2cac8a9d2 Reintroduce UDT support to fmt::to_string and test ADL 9a4cc8842 Add FMT_COMPILE support to format_to 5ddf9ee1b Streamline default FP formatting 0b3a83f7f Update README.rst 5aa5c9873 Added #define WIN32_LEAN_AND_MEAN before including windows.h (#1729) 397ad1bec Optimize common case 7431165f3 Make to_string bypass format ee4d4c7fd Inline compiled format ab2f8484e Finish text::format e900d735b Re-enable assert in format_decimal f4de7b684 Fix ambiguity 1f8f5450b Reuse format_decimal d702a68df Fix formatting of bool with FMT_COMPILE and add more tests e956a14e9 Use write instead of format_int in to_string 98dcc251e Undo branching reduction 5b8641ddd Undo branching reduction 8c88abde6 Fix sign handling in 'L' 23b976a61 Reduce branching 9edee0e72 Optimize small string parsing a909d42b7 Fix a warning 16637341b Enable compilation for all types 2d71d7e03 Add a simple format string compilation API d259fcfb0 Tweak comments 704ed557a Move project in order to solve a CMake warning 8603bd20d Update README.rst 547f12ae6 Fix a warning (#1722) f904e8a1b c++11 use formatting user-defined types (#1721) 100e8af08 Update README.rst c11d0f056 Update README.rst 2453ee576 Improve default formatting 47ae52155 MINGW cross compiler fixes 936a1833c Add default_arg_formatter f2c9cb624 Fix a UB d3107f855 Cleanup arg_formatter_base 5e7c70e20 Simplify arg_formatter_base 38cc68b3e Inline visitor 6732ea500 Make symbols readable 57ddc77ce Make advance_to a noop for back_insert_iterator 50bad7d62 Optimize format string parsing 8f7a824e4 Inline visit f11e96870 Optimize format string parsing 09737dd83 Optimize format handler d9e3d6e6e Move format_handler to detail 795b47a7b Fix a warning (#1712) 95c6ac0cc fix typo which caused the loss of the counting information when using a printf context with a truncating_iterator 21409cfdd Fix warnings 88c8d534e Move digits10 to where they belong and add comments 0f3eaeac0 Fix a warning 344218510 Ignore /doc/node_modules directory 16aec0617 Cleanup arg_formatter_base 1e1193590 Fix format_decimal overloads 0893c9c2e Inline parse_format_string 3245145a4 Remove undocumented buffer_range and output_range 57fc44907 Increase VM disk size 7d22bebb6 Remove uses of buffer_range 8f2b5fe74 Don't install sphinx cache files f095c67b6 Remove uses of buffer_range 5aabf1f71 Simplify copy_str 19c5b5d15 Simplify arg_formatter 519571ede Simplify arg_formatter_base ac8dfd841 Improve handling of separators 2c6165a22 Reduce the number of comparisons 28639969e Use memcpy for copying digits f5fa1dee5 Support custom FMT_INC_DIR in pkgconfig and cmake configs (#1702) 51bf9cfac Fix Mingw support 1a716caf5 Optimize common case 98d4bbf81 Update README.rst 8c8f74a87 fix zero flag for char types and make zero flag ignored if a precision is specified bc1b89da2 Temporarily revert parsing changes a7fb321ac Remove a redundant branch 8cadb9650 fix max/min macro (#1697) 297c3b2ed Fix an example (thanks Alexey Kuzmenko) 943532fec Make ostream formatter work with compile-time format strings (#1692) bd8804019 Update README.rst f230300ac Knuth is using fmt library (#1691) a265e25b7 Optimize small string parsing 2aa2526f6 Optimize small string concatenation 8d78045e7 Move void_t to where it's used 7aafa6bc6 Update analytics c66aae165 Adding sentinel support to fmt::join(). (#1689) 6d66de380 Add c specifier support to integral types (#1652) 6b219a58d fix interaction of space flag and '+' flag, as well as '-' flag and '0' flag (#1687) eee2023c2 Update signatures c5ed73aab Add fmt::detail::buffer to the docs (#704) ea1cd9638 Fix apidoc d3964d7b1 Merge branch 'master' of github.com:fmtlib/fmt d18c6723a Update docs 96c18b26c make plus flag for printf not be ignored for char argument (#1683) ba25baeb9 Apply doc patch to 6.2.1 981b517cc nested replacement fields may omit arg_id (#1681) 922ea924b Make dynamic_format_arg_store reusable and add reserve() (#1677) e0d98923c Update version 806926537 internal -> detail (#1538) 963ee0831 Simplify named arguments 02a6fe59f Named arguments go brrr de290f5c4 Ditch internal::arg_map d0623de51 Bump version 73e335ed3 Make implicit capture explicit for C++20 (#1669) b4d46e398 Update changelog a182f7341 Update changelog 68201831a Support named args in dynamic_format_arg_store (#1655). (#1663) 7f723fbcb Consistently namespace qualify size_t c06851456 Purge basic_writer 2f05054dd Purge basic_writer f0ce21164 Revert enum change 44639b11f Fix some warnings (#1667) 1c86a99e8 Purge basic_writer 8f511fc12 Make copyfmt not throw (#1666) 59fe455f3 Remove compatibility stubs b0f47a13e Separate nonfinite formatting d6cea50d0 Remove deprecated APIs 40bc7163f Move FMT_MAYBE_UNUSED to where it's actually used 080e44d0b Fix inconsistent type detection (#1662) 7e57cace5 Exclude std::abort from compilation when compiling CUDA with Clang (#1661) 7b66e2f21 Inherit arg_formatter_base from basic_writer bab3f5800 Refactor pointer formatting 9cc7edfdd Move int_writer to the namespace scope 8d9d528bf Improve handling of alignment 8efd1a8ef Improve handling of alignment a71bc9c82 Use '0' fill with numeric align for consistency with std::format 60d85d598 Suppress ubsan warning c3099beb6 Cleanup cbb4cb899 Remove undocumented deprecated APIs b85e9ac38 Simplify vformat_to e3710ab97 FMT_CONSTEXPR -> constexpr d59751f0f Update date formatting example to use threadsafe localtime d6abb2fa0 Reduce library size e9fdea90b Update README.rst 44b6584f2 Update README.rst 78f041ab5 build: Fix installation paths 7ca89bf87 Reduce template bloat in write_int 3c114d091 Fix a shadowing warning (#1658) e2ef12a8c Allow to avoid inclusion of os.cc in fmt target bca82719a Pass iterator by value 99da38962 Make write_padded non-members f19d66794 Bump fuzzer allocation limit 3e6984761 Reduce branching in write_padded 9ac1eebd4 Reduce library size e2ff91067 Replace FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION with fmt-specific macro (#1650) f2ed03b91 Fix a warning (#1649) 9dde9f013 Reduce library size b1af642d1 Reduce library size 4a617f25c Clarify encoding conversion in chrono 6f435f55c Improve compile time by using extern template (#1452) cb475cb88 Clarify why we don't check argument id 1e1ac6e96 Check dynamic width/precision id at compile time (#1614) e51c449fe Revert "Check dynamic widht/precision id at compile time (#1614)" 0463665ef Don't access a C string past precision in printf (#1595) 7d748a6f8 Check dynamic widht/precision id at compile time (#1614) 2b75bd7ce Get rid of do_check_format_string 4a1d5931c Simplify udl_formatter with FMT_STRING 811b0f905 Enable compile-time error tests 450e8eed9 Fix markup b8fbcec1b Clarify formatter reuse 56bc86ffa Suppress bogus MSVC analysis warnings 3f79357ef Fix a recent regression in handling max packed arguments 8a11148f9 Add Facebook Folly to the list of projects e371e8b68 Tweak readme 813732fed Improve readme formatting 3670d5b3f README: add vectorized.io/redpanda in the list of users 9e2ad7cf6 Add windows terminal to the projects using {fmt} 63479c851 Use a delegating ctor and add inlines 5944fcad3 Remove remaining wchar_t instantiation e253b371b Don't generate RTTI for allocator 0c86f467b Fix build on ancient gcc 1929df4bc Simplify format_args a13822181 Always inline arg_data functions 04e0dfd4b Always inline value ctors 04cde756b Simplify checks c9a57b9a8 Fix incorrect assumptions about nul termination f46f5ecaf Reenable constexpr _compile on GCC 9 6e8d7e277 Don't use constexpr on Intel compiler (#1628) 567ed03f8 Merge arg overloads and cleanup c3fa33314 Remove warning in core.h with when compiling with gcc and -Wshadow 84898b462 Remove warning in format.h when compiling with gcc and -Wshadow 538d83fd0 Cleanup named arguments 8a4630686 Improve handling of named arguments a9d62d3f3 Add check for CompiledFormat to avoid ambiguous call fdcf7870a Add stack-based named argument storage 5899267c4 Fix a clang-tidy warning 07b4c246e Fix a typo e99809f29 Fix ostream support in sprintf (#1631) 3cd5179f3 Fixed clang tidy warning -multiple declarations in a single statement reduces readability 7404e33a7 Fix clang warning about explicit ctor 3aab2171e Clean up basic_format_args 7645ca072 Clean up printf e30d8391e Suppress an MSVC warning (#1622) 8cd8ef03e Simplify warning suppression bbb6b357c Add floating-point L specifier (#1624) 36ea32640 Suppress a bogus MSVC warning 141a00d64 Define FMT_EXTERN_TEMPLATE_API on export 3860edc5d Bump version 7d01859ef Fix handling of unsigned char strings in printf 63b23e786 Merge branch 'master' of github.com:fmtlib/fmt 4999796c1 Fix the docs 34b3f7b7a Avoid windows issue with min() max() macros 27e3c0fe9 Update signature in the docs git-subtree-dir: externals/fmt git-subtree-split: cd4af11efc9c622896a3e4cb599fa28668ca3d05
1123 lines
35 KiB
C++
1123 lines
35 KiB
C++
// Formatting library for C++ - chrono support
|
|
//
|
|
// Copyright (c) 2012 - present, Victor Zverovich
|
|
// All rights reserved.
|
|
//
|
|
// For the license information refer to format.h.
|
|
|
|
#ifndef FMT_CHRONO_H_
|
|
#define FMT_CHRONO_H_
|
|
|
|
#include <chrono>
|
|
#include <ctime>
|
|
#include <locale>
|
|
#include <sstream>
|
|
|
|
#include "format.h"
|
|
#include "locale.h"
|
|
|
|
FMT_BEGIN_NAMESPACE
|
|
|
|
// Enable safe chrono durations, unless explicitly disabled.
|
|
#ifndef FMT_SAFE_DURATION_CAST
|
|
# define FMT_SAFE_DURATION_CAST 1
|
|
#endif
|
|
#if FMT_SAFE_DURATION_CAST
|
|
|
|
// For conversion between std::chrono::durations without undefined
|
|
// behaviour or erroneous results.
|
|
// This is a stripped down version of duration_cast, for inclusion in fmt.
|
|
// See https://github.com/pauldreik/safe_duration_cast
|
|
//
|
|
// Copyright Paul Dreik 2019
|
|
namespace safe_duration_cast {
|
|
|
|
template <typename To, typename From,
|
|
FMT_ENABLE_IF(!std::is_same<From, To>::value &&
|
|
std::numeric_limits<From>::is_signed ==
|
|
std::numeric_limits<To>::is_signed)>
|
|
FMT_CONSTEXPR To lossless_integral_conversion(const From from, int& ec) {
|
|
ec = 0;
|
|
using F = std::numeric_limits<From>;
|
|
using T = std::numeric_limits<To>;
|
|
static_assert(F::is_integer, "From must be integral");
|
|
static_assert(T::is_integer, "To must be integral");
|
|
|
|
// A and B are both signed, or both unsigned.
|
|
if (F::digits <= T::digits) {
|
|
// From fits in To without any problem.
|
|
} else {
|
|
// From does not always fit in To, resort to a dynamic check.
|
|
if (from < (T::min)() || from > (T::max)()) {
|
|
// outside range.
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
}
|
|
return static_cast<To>(from);
|
|
}
|
|
|
|
/**
|
|
* converts From to To, without loss. If the dynamic value of from
|
|
* can't be converted to To without loss, ec is set.
|
|
*/
|
|
template <typename To, typename From,
|
|
FMT_ENABLE_IF(!std::is_same<From, To>::value &&
|
|
std::numeric_limits<From>::is_signed !=
|
|
std::numeric_limits<To>::is_signed)>
|
|
FMT_CONSTEXPR To lossless_integral_conversion(const From from, int& ec) {
|
|
ec = 0;
|
|
using F = std::numeric_limits<From>;
|
|
using T = std::numeric_limits<To>;
|
|
static_assert(F::is_integer, "From must be integral");
|
|
static_assert(T::is_integer, "To must be integral");
|
|
|
|
if (F::is_signed && !T::is_signed) {
|
|
// From may be negative, not allowed!
|
|
if (fmt::detail::is_negative(from)) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
|
|
// From is positive. Can it always fit in To?
|
|
if (F::digits <= T::digits) {
|
|
// yes, From always fits in To.
|
|
} else {
|
|
// from may not fit in To, we have to do a dynamic check
|
|
if (from > static_cast<From>((T::max)())) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!F::is_signed && T::is_signed) {
|
|
// can from be held in To?
|
|
if (F::digits < T::digits) {
|
|
// yes, From always fits in To.
|
|
} else {
|
|
// from may not fit in To, we have to do a dynamic check
|
|
if (from > static_cast<From>((T::max)())) {
|
|
// outside range.
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
}
|
|
}
|
|
|
|
// reaching here means all is ok for lossless conversion.
|
|
return static_cast<To>(from);
|
|
|
|
} // function
|
|
|
|
template <typename To, typename From,
|
|
FMT_ENABLE_IF(std::is_same<From, To>::value)>
|
|
FMT_CONSTEXPR To lossless_integral_conversion(const From from, int& ec) {
|
|
ec = 0;
|
|
return from;
|
|
} // function
|
|
|
|
// clang-format off
|
|
/**
|
|
* converts From to To if possible, otherwise ec is set.
|
|
*
|
|
* input | output
|
|
* ---------------------------------|---------------
|
|
* NaN | NaN
|
|
* Inf | Inf
|
|
* normal, fits in output | converted (possibly lossy)
|
|
* normal, does not fit in output | ec is set
|
|
* subnormal | best effort
|
|
* -Inf | -Inf
|
|
*/
|
|
// clang-format on
|
|
template <typename To, typename From,
|
|
FMT_ENABLE_IF(!std::is_same<From, To>::value)>
|
|
FMT_CONSTEXPR To safe_float_conversion(const From from, int& ec) {
|
|
ec = 0;
|
|
using T = std::numeric_limits<To>;
|
|
static_assert(std::is_floating_point<From>::value, "From must be floating");
|
|
static_assert(std::is_floating_point<To>::value, "To must be floating");
|
|
|
|
// catch the only happy case
|
|
if (std::isfinite(from)) {
|
|
if (from >= T::lowest() && from <= (T::max)()) {
|
|
return static_cast<To>(from);
|
|
}
|
|
// not within range.
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
|
|
// nan and inf will be preserved
|
|
return static_cast<To>(from);
|
|
} // function
|
|
|
|
template <typename To, typename From,
|
|
FMT_ENABLE_IF(std::is_same<From, To>::value)>
|
|
FMT_CONSTEXPR To safe_float_conversion(const From from, int& ec) {
|
|
ec = 0;
|
|
static_assert(std::is_floating_point<From>::value, "From must be floating");
|
|
return from;
|
|
}
|
|
|
|
/**
|
|
* safe duration cast between integral durations
|
|
*/
|
|
template <typename To, typename FromRep, typename FromPeriod,
|
|
FMT_ENABLE_IF(std::is_integral<FromRep>::value),
|
|
FMT_ENABLE_IF(std::is_integral<typename To::rep>::value)>
|
|
To safe_duration_cast(std::chrono::duration<FromRep, FromPeriod> from,
|
|
int& ec) {
|
|
using From = std::chrono::duration<FromRep, FromPeriod>;
|
|
ec = 0;
|
|
// the basic idea is that we need to convert from count() in the from type
|
|
// to count() in the To type, by multiplying it with this:
|
|
struct Factor
|
|
: std::ratio_divide<typename From::period, typename To::period> {};
|
|
|
|
static_assert(Factor::num > 0, "num must be positive");
|
|
static_assert(Factor::den > 0, "den must be positive");
|
|
|
|
// the conversion is like this: multiply from.count() with Factor::num
|
|
// /Factor::den and convert it to To::rep, all this without
|
|
// overflow/underflow. let's start by finding a suitable type that can hold
|
|
// both To, From and Factor::num
|
|
using IntermediateRep =
|
|
typename std::common_type<typename From::rep, typename To::rep,
|
|
decltype(Factor::num)>::type;
|
|
|
|
// safe conversion to IntermediateRep
|
|
IntermediateRep count =
|
|
lossless_integral_conversion<IntermediateRep>(from.count(), ec);
|
|
if (ec) {
|
|
return {};
|
|
}
|
|
// multiply with Factor::num without overflow or underflow
|
|
if (Factor::num != 1) {
|
|
const auto max1 = detail::max_value<IntermediateRep>() / Factor::num;
|
|
if (count > max1) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
const auto min1 =
|
|
(std::numeric_limits<IntermediateRep>::min)() / Factor::num;
|
|
if (count < min1) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
count *= Factor::num;
|
|
}
|
|
|
|
// this can't go wrong, right? den>0 is checked earlier.
|
|
if (Factor::den != 1) {
|
|
count /= Factor::den;
|
|
}
|
|
// convert to the to type, safely
|
|
using ToRep = typename To::rep;
|
|
const ToRep tocount = lossless_integral_conversion<ToRep>(count, ec);
|
|
if (ec) {
|
|
return {};
|
|
}
|
|
return To{tocount};
|
|
}
|
|
|
|
/**
|
|
* safe duration_cast between floating point durations
|
|
*/
|
|
template <typename To, typename FromRep, typename FromPeriod,
|
|
FMT_ENABLE_IF(std::is_floating_point<FromRep>::value),
|
|
FMT_ENABLE_IF(std::is_floating_point<typename To::rep>::value)>
|
|
To safe_duration_cast(std::chrono::duration<FromRep, FromPeriod> from,
|
|
int& ec) {
|
|
using From = std::chrono::duration<FromRep, FromPeriod>;
|
|
ec = 0;
|
|
if (std::isnan(from.count())) {
|
|
// nan in, gives nan out. easy.
|
|
return To{std::numeric_limits<typename To::rep>::quiet_NaN()};
|
|
}
|
|
// maybe we should also check if from is denormal, and decide what to do about
|
|
// it.
|
|
|
|
// +-inf should be preserved.
|
|
if (std::isinf(from.count())) {
|
|
return To{from.count()};
|
|
}
|
|
|
|
// the basic idea is that we need to convert from count() in the from type
|
|
// to count() in the To type, by multiplying it with this:
|
|
struct Factor
|
|
: std::ratio_divide<typename From::period, typename To::period> {};
|
|
|
|
static_assert(Factor::num > 0, "num must be positive");
|
|
static_assert(Factor::den > 0, "den must be positive");
|
|
|
|
// the conversion is like this: multiply from.count() with Factor::num
|
|
// /Factor::den and convert it to To::rep, all this without
|
|
// overflow/underflow. let's start by finding a suitable type that can hold
|
|
// both To, From and Factor::num
|
|
using IntermediateRep =
|
|
typename std::common_type<typename From::rep, typename To::rep,
|
|
decltype(Factor::num)>::type;
|
|
|
|
// force conversion of From::rep -> IntermediateRep to be safe,
|
|
// even if it will never happen be narrowing in this context.
|
|
IntermediateRep count =
|
|
safe_float_conversion<IntermediateRep>(from.count(), ec);
|
|
if (ec) {
|
|
return {};
|
|
}
|
|
|
|
// multiply with Factor::num without overflow or underflow
|
|
if (Factor::num != 1) {
|
|
constexpr auto max1 = detail::max_value<IntermediateRep>() /
|
|
static_cast<IntermediateRep>(Factor::num);
|
|
if (count > max1) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
constexpr auto min1 = std::numeric_limits<IntermediateRep>::lowest() /
|
|
static_cast<IntermediateRep>(Factor::num);
|
|
if (count < min1) {
|
|
ec = 1;
|
|
return {};
|
|
}
|
|
count *= static_cast<IntermediateRep>(Factor::num);
|
|
}
|
|
|
|
// this can't go wrong, right? den>0 is checked earlier.
|
|
if (Factor::den != 1) {
|
|
using common_t = typename std::common_type<IntermediateRep, intmax_t>::type;
|
|
count /= static_cast<common_t>(Factor::den);
|
|
}
|
|
|
|
// convert to the to type, safely
|
|
using ToRep = typename To::rep;
|
|
|
|
const ToRep tocount = safe_float_conversion<ToRep>(count, ec);
|
|
if (ec) {
|
|
return {};
|
|
}
|
|
return To{tocount};
|
|
}
|
|
} // namespace safe_duration_cast
|
|
#endif
|
|
|
|
// Prevents expansion of a preceding token as a function-style macro.
|
|
// Usage: f FMT_NOMACRO()
|
|
#define FMT_NOMACRO
|
|
|
|
namespace detail {
|
|
inline null<> localtime_r FMT_NOMACRO(...) { return null<>(); }
|
|
inline null<> localtime_s(...) { return null<>(); }
|
|
inline null<> gmtime_r(...) { return null<>(); }
|
|
inline null<> gmtime_s(...) { return null<>(); }
|
|
} // namespace detail
|
|
|
|
// Thread-safe replacement for std::localtime
|
|
inline std::tm localtime(std::time_t time) {
|
|
struct dispatcher {
|
|
std::time_t time_;
|
|
std::tm tm_;
|
|
|
|
dispatcher(std::time_t t) : time_(t) {}
|
|
|
|
bool run() {
|
|
using namespace fmt::detail;
|
|
return handle(localtime_r(&time_, &tm_));
|
|
}
|
|
|
|
bool handle(std::tm* tm) { return tm != nullptr; }
|
|
|
|
bool handle(detail::null<>) {
|
|
using namespace fmt::detail;
|
|
return fallback(localtime_s(&tm_, &time_));
|
|
}
|
|
|
|
bool fallback(int res) { return res == 0; }
|
|
|
|
#if !FMT_MSC_VER
|
|
bool fallback(detail::null<>) {
|
|
using namespace fmt::detail;
|
|
std::tm* tm = std::localtime(&time_);
|
|
if (tm) tm_ = *tm;
|
|
return tm != nullptr;
|
|
}
|
|
#endif
|
|
};
|
|
dispatcher lt(time);
|
|
// Too big time values may be unsupported.
|
|
if (!lt.run()) FMT_THROW(format_error("time_t value out of range"));
|
|
return lt.tm_;
|
|
}
|
|
|
|
// Thread-safe replacement for std::gmtime
|
|
inline std::tm gmtime(std::time_t time) {
|
|
struct dispatcher {
|
|
std::time_t time_;
|
|
std::tm tm_;
|
|
|
|
dispatcher(std::time_t t) : time_(t) {}
|
|
|
|
bool run() {
|
|
using namespace fmt::detail;
|
|
return handle(gmtime_r(&time_, &tm_));
|
|
}
|
|
|
|
bool handle(std::tm* tm) { return tm != nullptr; }
|
|
|
|
bool handle(detail::null<>) {
|
|
using namespace fmt::detail;
|
|
return fallback(gmtime_s(&tm_, &time_));
|
|
}
|
|
|
|
bool fallback(int res) { return res == 0; }
|
|
|
|
#if !FMT_MSC_VER
|
|
bool fallback(detail::null<>) {
|
|
std::tm* tm = std::gmtime(&time_);
|
|
if (tm) tm_ = *tm;
|
|
return tm != nullptr;
|
|
}
|
|
#endif
|
|
};
|
|
dispatcher gt(time);
|
|
// Too big time values may be unsupported.
|
|
if (!gt.run()) FMT_THROW(format_error("time_t value out of range"));
|
|
return gt.tm_;
|
|
}
|
|
|
|
namespace detail {
|
|
inline size_t strftime(char* str, size_t count, const char* format,
|
|
const std::tm* time) {
|
|
return std::strftime(str, count, format, time);
|
|
}
|
|
|
|
inline size_t strftime(wchar_t* str, size_t count, const wchar_t* format,
|
|
const std::tm* time) {
|
|
return std::wcsftime(str, count, format, time);
|
|
}
|
|
} // namespace detail
|
|
|
|
template <typename Char> struct formatter<std::tm, Char> {
|
|
template <typename ParseContext>
|
|
auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
auto it = ctx.begin();
|
|
if (it != ctx.end() && *it == ':') ++it;
|
|
auto end = it;
|
|
while (end != ctx.end() && *end != '}') ++end;
|
|
tm_format.reserve(detail::to_unsigned(end - it + 1));
|
|
tm_format.append(it, end);
|
|
tm_format.push_back('\0');
|
|
return end;
|
|
}
|
|
|
|
template <typename FormatContext>
|
|
auto format(const std::tm& tm, FormatContext& ctx) -> decltype(ctx.out()) {
|
|
basic_memory_buffer<Char> buf;
|
|
size_t start = buf.size();
|
|
for (;;) {
|
|
size_t size = buf.capacity() - start;
|
|
size_t count = detail::strftime(&buf[start], size, &tm_format[0], &tm);
|
|
if (count != 0) {
|
|
buf.resize(start + count);
|
|
break;
|
|
}
|
|
if (size >= tm_format.size() * 256) {
|
|
// If the buffer is 256 times larger than the format string, assume
|
|
// that `strftime` gives an empty result. There doesn't seem to be a
|
|
// better way to distinguish the two cases:
|
|
// https://github.com/fmtlib/fmt/issues/367
|
|
break;
|
|
}
|
|
const size_t MIN_GROWTH = 10;
|
|
buf.reserve(buf.capacity() + (size > MIN_GROWTH ? size : MIN_GROWTH));
|
|
}
|
|
return std::copy(buf.begin(), buf.end(), ctx.out());
|
|
}
|
|
|
|
basic_memory_buffer<Char> tm_format;
|
|
};
|
|
|
|
namespace detail {
|
|
template <typename Period> FMT_CONSTEXPR const char* get_units() {
|
|
return nullptr;
|
|
}
|
|
template <> FMT_CONSTEXPR const char* get_units<std::atto>() { return "as"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::femto>() { return "fs"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::pico>() { return "ps"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::nano>() { return "ns"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::micro>() { return "µs"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::milli>() { return "ms"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::centi>() { return "cs"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::deci>() { return "ds"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::ratio<1>>() { return "s"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::deca>() { return "das"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::hecto>() { return "hs"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::kilo>() { return "ks"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::mega>() { return "Ms"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::giga>() { return "Gs"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::tera>() { return "Ts"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::peta>() { return "Ps"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::exa>() { return "Es"; }
|
|
template <> FMT_CONSTEXPR const char* get_units<std::ratio<60>>() {
|
|
return "m";
|
|
}
|
|
template <> FMT_CONSTEXPR const char* get_units<std::ratio<3600>>() {
|
|
return "h";
|
|
}
|
|
|
|
enum class numeric_system {
|
|
standard,
|
|
// Alternative numeric system, e.g. 十二 instead of 12 in ja_JP locale.
|
|
alternative
|
|
};
|
|
|
|
// Parses a put_time-like format string and invokes handler actions.
|
|
template <typename Char, typename Handler>
|
|
FMT_CONSTEXPR const Char* parse_chrono_format(const Char* begin,
|
|
const Char* end,
|
|
Handler&& handler) {
|
|
auto ptr = begin;
|
|
while (ptr != end) {
|
|
auto c = *ptr;
|
|
if (c == '}') break;
|
|
if (c != '%') {
|
|
++ptr;
|
|
continue;
|
|
}
|
|
if (begin != ptr) handler.on_text(begin, ptr);
|
|
++ptr; // consume '%'
|
|
if (ptr == end) FMT_THROW(format_error("invalid format"));
|
|
c = *ptr++;
|
|
switch (c) {
|
|
case '%':
|
|
handler.on_text(ptr - 1, ptr);
|
|
break;
|
|
case 'n': {
|
|
const Char newline[] = {'\n'};
|
|
handler.on_text(newline, newline + 1);
|
|
break;
|
|
}
|
|
case 't': {
|
|
const Char tab[] = {'\t'};
|
|
handler.on_text(tab, tab + 1);
|
|
break;
|
|
}
|
|
// Day of the week:
|
|
case 'a':
|
|
handler.on_abbr_weekday();
|
|
break;
|
|
case 'A':
|
|
handler.on_full_weekday();
|
|
break;
|
|
case 'w':
|
|
handler.on_dec0_weekday(numeric_system::standard);
|
|
break;
|
|
case 'u':
|
|
handler.on_dec1_weekday(numeric_system::standard);
|
|
break;
|
|
// Month:
|
|
case 'b':
|
|
handler.on_abbr_month();
|
|
break;
|
|
case 'B':
|
|
handler.on_full_month();
|
|
break;
|
|
// Hour, minute, second:
|
|
case 'H':
|
|
handler.on_24_hour(numeric_system::standard);
|
|
break;
|
|
case 'I':
|
|
handler.on_12_hour(numeric_system::standard);
|
|
break;
|
|
case 'M':
|
|
handler.on_minute(numeric_system::standard);
|
|
break;
|
|
case 'S':
|
|
handler.on_second(numeric_system::standard);
|
|
break;
|
|
// Other:
|
|
case 'c':
|
|
handler.on_datetime(numeric_system::standard);
|
|
break;
|
|
case 'x':
|
|
handler.on_loc_date(numeric_system::standard);
|
|
break;
|
|
case 'X':
|
|
handler.on_loc_time(numeric_system::standard);
|
|
break;
|
|
case 'D':
|
|
handler.on_us_date();
|
|
break;
|
|
case 'F':
|
|
handler.on_iso_date();
|
|
break;
|
|
case 'r':
|
|
handler.on_12_hour_time();
|
|
break;
|
|
case 'R':
|
|
handler.on_24_hour_time();
|
|
break;
|
|
case 'T':
|
|
handler.on_iso_time();
|
|
break;
|
|
case 'p':
|
|
handler.on_am_pm();
|
|
break;
|
|
case 'Q':
|
|
handler.on_duration_value();
|
|
break;
|
|
case 'q':
|
|
handler.on_duration_unit();
|
|
break;
|
|
case 'z':
|
|
handler.on_utc_offset();
|
|
break;
|
|
case 'Z':
|
|
handler.on_tz_name();
|
|
break;
|
|
// Alternative representation:
|
|
case 'E': {
|
|
if (ptr == end) FMT_THROW(format_error("invalid format"));
|
|
c = *ptr++;
|
|
switch (c) {
|
|
case 'c':
|
|
handler.on_datetime(numeric_system::alternative);
|
|
break;
|
|
case 'x':
|
|
handler.on_loc_date(numeric_system::alternative);
|
|
break;
|
|
case 'X':
|
|
handler.on_loc_time(numeric_system::alternative);
|
|
break;
|
|
default:
|
|
FMT_THROW(format_error("invalid format"));
|
|
}
|
|
break;
|
|
}
|
|
case 'O':
|
|
if (ptr == end) FMT_THROW(format_error("invalid format"));
|
|
c = *ptr++;
|
|
switch (c) {
|
|
case 'w':
|
|
handler.on_dec0_weekday(numeric_system::alternative);
|
|
break;
|
|
case 'u':
|
|
handler.on_dec1_weekday(numeric_system::alternative);
|
|
break;
|
|
case 'H':
|
|
handler.on_24_hour(numeric_system::alternative);
|
|
break;
|
|
case 'I':
|
|
handler.on_12_hour(numeric_system::alternative);
|
|
break;
|
|
case 'M':
|
|
handler.on_minute(numeric_system::alternative);
|
|
break;
|
|
case 'S':
|
|
handler.on_second(numeric_system::alternative);
|
|
break;
|
|
default:
|
|
FMT_THROW(format_error("invalid format"));
|
|
}
|
|
break;
|
|
default:
|
|
FMT_THROW(format_error("invalid format"));
|
|
}
|
|
begin = ptr;
|
|
}
|
|
if (begin != ptr) handler.on_text(begin, ptr);
|
|
return ptr;
|
|
}
|
|
|
|
struct chrono_format_checker {
|
|
FMT_NORETURN void report_no_date() { FMT_THROW(format_error("no date")); }
|
|
|
|
template <typename Char> void on_text(const Char*, const Char*) {}
|
|
FMT_NORETURN void on_abbr_weekday() { report_no_date(); }
|
|
FMT_NORETURN void on_full_weekday() { report_no_date(); }
|
|
FMT_NORETURN void on_dec0_weekday(numeric_system) { report_no_date(); }
|
|
FMT_NORETURN void on_dec1_weekday(numeric_system) { report_no_date(); }
|
|
FMT_NORETURN void on_abbr_month() { report_no_date(); }
|
|
FMT_NORETURN void on_full_month() { report_no_date(); }
|
|
void on_24_hour(numeric_system) {}
|
|
void on_12_hour(numeric_system) {}
|
|
void on_minute(numeric_system) {}
|
|
void on_second(numeric_system) {}
|
|
FMT_NORETURN void on_datetime(numeric_system) { report_no_date(); }
|
|
FMT_NORETURN void on_loc_date(numeric_system) { report_no_date(); }
|
|
FMT_NORETURN void on_loc_time(numeric_system) { report_no_date(); }
|
|
FMT_NORETURN void on_us_date() { report_no_date(); }
|
|
FMT_NORETURN void on_iso_date() { report_no_date(); }
|
|
void on_12_hour_time() {}
|
|
void on_24_hour_time() {}
|
|
void on_iso_time() {}
|
|
void on_am_pm() {}
|
|
void on_duration_value() {}
|
|
void on_duration_unit() {}
|
|
FMT_NORETURN void on_utc_offset() { report_no_date(); }
|
|
FMT_NORETURN void on_tz_name() { report_no_date(); }
|
|
};
|
|
|
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
|
inline bool isnan(T) {
|
|
return false;
|
|
}
|
|
template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
|
|
inline bool isnan(T value) {
|
|
return std::isnan(value);
|
|
}
|
|
|
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
|
inline bool isfinite(T) {
|
|
return true;
|
|
}
|
|
template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
|
|
inline bool isfinite(T value) {
|
|
return std::isfinite(value);
|
|
}
|
|
|
|
// Converts value to int and checks that it's in the range [0, upper).
|
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
|
inline int to_nonnegative_int(T value, int upper) {
|
|
FMT_ASSERT(value >= 0 && value <= upper, "invalid value");
|
|
(void)upper;
|
|
return static_cast<int>(value);
|
|
}
|
|
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
|
inline int to_nonnegative_int(T value, int upper) {
|
|
FMT_ASSERT(
|
|
std::isnan(value) || (value >= 0 && value <= static_cast<T>(upper)),
|
|
"invalid value");
|
|
(void)upper;
|
|
return static_cast<int>(value);
|
|
}
|
|
|
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
|
inline T mod(T x, int y) {
|
|
return x % static_cast<T>(y);
|
|
}
|
|
template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
|
|
inline T mod(T x, int y) {
|
|
return std::fmod(x, static_cast<T>(y));
|
|
}
|
|
|
|
// If T is an integral type, maps T to its unsigned counterpart, otherwise
|
|
// leaves it unchanged (unlike std::make_unsigned).
|
|
template <typename T, bool INTEGRAL = std::is_integral<T>::value>
|
|
struct make_unsigned_or_unchanged {
|
|
using type = T;
|
|
};
|
|
|
|
template <typename T> struct make_unsigned_or_unchanged<T, true> {
|
|
using type = typename std::make_unsigned<T>::type;
|
|
};
|
|
|
|
#if FMT_SAFE_DURATION_CAST
|
|
// throwing version of safe_duration_cast
|
|
template <typename To, typename FromRep, typename FromPeriod>
|
|
To fmt_safe_duration_cast(std::chrono::duration<FromRep, FromPeriod> from) {
|
|
int ec;
|
|
To to = safe_duration_cast::safe_duration_cast<To>(from, ec);
|
|
if (ec) FMT_THROW(format_error("cannot format duration"));
|
|
return to;
|
|
}
|
|
#endif
|
|
|
|
template <typename Rep, typename Period,
|
|
FMT_ENABLE_IF(std::is_integral<Rep>::value)>
|
|
inline std::chrono::duration<Rep, std::milli> get_milliseconds(
|
|
std::chrono::duration<Rep, Period> d) {
|
|
// this may overflow and/or the result may not fit in the
|
|
// target type.
|
|
#if FMT_SAFE_DURATION_CAST
|
|
using CommonSecondsType =
|
|
typename std::common_type<decltype(d), std::chrono::seconds>::type;
|
|
const auto d_as_common = fmt_safe_duration_cast<CommonSecondsType>(d);
|
|
const auto d_as_whole_seconds =
|
|
fmt_safe_duration_cast<std::chrono::seconds>(d_as_common);
|
|
// this conversion should be nonproblematic
|
|
const auto diff = d_as_common - d_as_whole_seconds;
|
|
const auto ms =
|
|
fmt_safe_duration_cast<std::chrono::duration<Rep, std::milli>>(diff);
|
|
return ms;
|
|
#else
|
|
auto s = std::chrono::duration_cast<std::chrono::seconds>(d);
|
|
return std::chrono::duration_cast<std::chrono::milliseconds>(d - s);
|
|
#endif
|
|
}
|
|
|
|
template <typename Rep, typename Period,
|
|
FMT_ENABLE_IF(std::is_floating_point<Rep>::value)>
|
|
inline std::chrono::duration<Rep, std::milli> get_milliseconds(
|
|
std::chrono::duration<Rep, Period> d) {
|
|
using common_type = typename std::common_type<Rep, std::intmax_t>::type;
|
|
auto ms = mod(d.count() * static_cast<common_type>(Period::num) /
|
|
static_cast<common_type>(Period::den) * 1000,
|
|
1000);
|
|
return std::chrono::duration<Rep, std::milli>(static_cast<Rep>(ms));
|
|
}
|
|
|
|
template <typename Char, typename Rep, typename OutputIt>
|
|
OutputIt format_duration_value(OutputIt out, Rep val, int precision) {
|
|
const Char pr_f[] = {'{', ':', '.', '{', '}', 'f', '}', 0};
|
|
if (precision >= 0) return format_to(out, pr_f, val, precision);
|
|
const Char fp_f[] = {'{', ':', 'g', '}', 0};
|
|
const Char format[] = {'{', '}', 0};
|
|
return format_to(out, std::is_floating_point<Rep>::value ? fp_f : format,
|
|
val);
|
|
}
|
|
template <typename Char, typename OutputIt>
|
|
OutputIt copy_unit(string_view unit, OutputIt out, Char) {
|
|
return std::copy(unit.begin(), unit.end(), out);
|
|
}
|
|
|
|
template <typename OutputIt>
|
|
OutputIt copy_unit(string_view unit, OutputIt out, wchar_t) {
|
|
// This works when wchar_t is UTF-32 because units only contain characters
|
|
// that have the same representation in UTF-16 and UTF-32.
|
|
utf8_to_utf16 u(unit);
|
|
return std::copy(u.c_str(), u.c_str() + u.size(), out);
|
|
}
|
|
|
|
template <typename Char, typename Period, typename OutputIt>
|
|
OutputIt format_duration_unit(OutputIt out) {
|
|
if (const char* unit = get_units<Period>())
|
|
return copy_unit(string_view(unit), out, Char());
|
|
const Char num_f[] = {'[', '{', '}', ']', 's', 0};
|
|
if (const_check(Period::den == 1)) return format_to(out, num_f, Period::num);
|
|
const Char num_def_f[] = {'[', '{', '}', '/', '{', '}', ']', 's', 0};
|
|
return format_to(out, num_def_f, Period::num, Period::den);
|
|
}
|
|
|
|
template <typename FormatContext, typename OutputIt, typename Rep,
|
|
typename Period>
|
|
struct chrono_formatter {
|
|
FormatContext& context;
|
|
OutputIt out;
|
|
int precision;
|
|
// rep is unsigned to avoid overflow.
|
|
using rep =
|
|
conditional_t<std::is_integral<Rep>::value && sizeof(Rep) < sizeof(int),
|
|
unsigned, typename make_unsigned_or_unchanged<Rep>::type>;
|
|
rep val;
|
|
using seconds = std::chrono::duration<rep>;
|
|
seconds s;
|
|
using milliseconds = std::chrono::duration<rep, std::milli>;
|
|
bool negative;
|
|
|
|
using char_type = typename FormatContext::char_type;
|
|
|
|
explicit chrono_formatter(FormatContext& ctx, OutputIt o,
|
|
std::chrono::duration<Rep, Period> d)
|
|
: context(ctx),
|
|
out(o),
|
|
val(static_cast<rep>(d.count())),
|
|
negative(false) {
|
|
if (d.count() < 0) {
|
|
val = 0 - val;
|
|
negative = true;
|
|
}
|
|
|
|
// this may overflow and/or the result may not fit in the
|
|
// target type.
|
|
#if FMT_SAFE_DURATION_CAST
|
|
// might need checked conversion (rep!=Rep)
|
|
auto tmpval = std::chrono::duration<rep, Period>(val);
|
|
s = fmt_safe_duration_cast<seconds>(tmpval);
|
|
#else
|
|
s = std::chrono::duration_cast<seconds>(
|
|
std::chrono::duration<rep, Period>(val));
|
|
#endif
|
|
}
|
|
|
|
// returns true if nan or inf, writes to out.
|
|
bool handle_nan_inf() {
|
|
if (isfinite(val)) {
|
|
return false;
|
|
}
|
|
if (isnan(val)) {
|
|
write_nan();
|
|
return true;
|
|
}
|
|
// must be +-inf
|
|
if (val > 0) {
|
|
write_pinf();
|
|
} else {
|
|
write_ninf();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Rep hour() const { return static_cast<Rep>(mod((s.count() / 3600), 24)); }
|
|
|
|
Rep hour12() const {
|
|
Rep hour = static_cast<Rep>(mod((s.count() / 3600), 12));
|
|
return hour <= 0 ? 12 : hour;
|
|
}
|
|
|
|
Rep minute() const { return static_cast<Rep>(mod((s.count() / 60), 60)); }
|
|
Rep second() const { return static_cast<Rep>(mod(s.count(), 60)); }
|
|
|
|
std::tm time() const {
|
|
auto time = std::tm();
|
|
time.tm_hour = to_nonnegative_int(hour(), 24);
|
|
time.tm_min = to_nonnegative_int(minute(), 60);
|
|
time.tm_sec = to_nonnegative_int(second(), 60);
|
|
return time;
|
|
}
|
|
|
|
void write_sign() {
|
|
if (negative) {
|
|
*out++ = '-';
|
|
negative = false;
|
|
}
|
|
}
|
|
|
|
void write(Rep value, int width) {
|
|
write_sign();
|
|
if (isnan(value)) return write_nan();
|
|
uint32_or_64_or_128_t<int> n =
|
|
to_unsigned(to_nonnegative_int(value, max_value<int>()));
|
|
int num_digits = detail::count_digits(n);
|
|
if (width > num_digits) out = std::fill_n(out, width - num_digits, '0');
|
|
out = format_decimal<char_type>(out, n, num_digits).end;
|
|
}
|
|
|
|
void write_nan() { std::copy_n("nan", 3, out); }
|
|
void write_pinf() { std::copy_n("inf", 3, out); }
|
|
void write_ninf() { std::copy_n("-inf", 4, out); }
|
|
|
|
void format_localized(const tm& time, char format, char modifier = 0) {
|
|
if (isnan(val)) return write_nan();
|
|
auto locale = context.locale().template get<std::locale>();
|
|
auto& facet = std::use_facet<std::time_put<char_type>>(locale);
|
|
std::basic_ostringstream<char_type> os;
|
|
os.imbue(locale);
|
|
facet.put(os, os, ' ', &time, format, modifier);
|
|
auto str = os.str();
|
|
std::copy(str.begin(), str.end(), out);
|
|
}
|
|
|
|
void on_text(const char_type* begin, const char_type* end) {
|
|
std::copy(begin, end, out);
|
|
}
|
|
|
|
// These are not implemented because durations don't have date information.
|
|
void on_abbr_weekday() {}
|
|
void on_full_weekday() {}
|
|
void on_dec0_weekday(numeric_system) {}
|
|
void on_dec1_weekday(numeric_system) {}
|
|
void on_abbr_month() {}
|
|
void on_full_month() {}
|
|
void on_datetime(numeric_system) {}
|
|
void on_loc_date(numeric_system) {}
|
|
void on_loc_time(numeric_system) {}
|
|
void on_us_date() {}
|
|
void on_iso_date() {}
|
|
void on_utc_offset() {}
|
|
void on_tz_name() {}
|
|
|
|
void on_24_hour(numeric_system ns) {
|
|
if (handle_nan_inf()) return;
|
|
|
|
if (ns == numeric_system::standard) return write(hour(), 2);
|
|
auto time = tm();
|
|
time.tm_hour = to_nonnegative_int(hour(), 24);
|
|
format_localized(time, 'H', 'O');
|
|
}
|
|
|
|
void on_12_hour(numeric_system ns) {
|
|
if (handle_nan_inf()) return;
|
|
|
|
if (ns == numeric_system::standard) return write(hour12(), 2);
|
|
auto time = tm();
|
|
time.tm_hour = to_nonnegative_int(hour12(), 12);
|
|
format_localized(time, 'I', 'O');
|
|
}
|
|
|
|
void on_minute(numeric_system ns) {
|
|
if (handle_nan_inf()) return;
|
|
|
|
if (ns == numeric_system::standard) return write(minute(), 2);
|
|
auto time = tm();
|
|
time.tm_min = to_nonnegative_int(minute(), 60);
|
|
format_localized(time, 'M', 'O');
|
|
}
|
|
|
|
void on_second(numeric_system ns) {
|
|
if (handle_nan_inf()) return;
|
|
|
|
if (ns == numeric_system::standard) {
|
|
write(second(), 2);
|
|
#if FMT_SAFE_DURATION_CAST
|
|
// convert rep->Rep
|
|
using duration_rep = std::chrono::duration<rep, Period>;
|
|
using duration_Rep = std::chrono::duration<Rep, Period>;
|
|
auto tmpval = fmt_safe_duration_cast<duration_Rep>(duration_rep{val});
|
|
#else
|
|
auto tmpval = std::chrono::duration<Rep, Period>(val);
|
|
#endif
|
|
auto ms = get_milliseconds(tmpval);
|
|
if (ms != std::chrono::milliseconds(0)) {
|
|
*out++ = '.';
|
|
write(ms.count(), 3);
|
|
}
|
|
return;
|
|
}
|
|
auto time = tm();
|
|
time.tm_sec = to_nonnegative_int(second(), 60);
|
|
format_localized(time, 'S', 'O');
|
|
}
|
|
|
|
void on_12_hour_time() {
|
|
if (handle_nan_inf()) return;
|
|
format_localized(time(), 'r');
|
|
}
|
|
|
|
void on_24_hour_time() {
|
|
if (handle_nan_inf()) {
|
|
*out++ = ':';
|
|
handle_nan_inf();
|
|
return;
|
|
}
|
|
|
|
write(hour(), 2);
|
|
*out++ = ':';
|
|
write(minute(), 2);
|
|
}
|
|
|
|
void on_iso_time() {
|
|
on_24_hour_time();
|
|
*out++ = ':';
|
|
if (handle_nan_inf()) return;
|
|
write(second(), 2);
|
|
}
|
|
|
|
void on_am_pm() {
|
|
if (handle_nan_inf()) return;
|
|
format_localized(time(), 'p');
|
|
}
|
|
|
|
void on_duration_value() {
|
|
if (handle_nan_inf()) return;
|
|
write_sign();
|
|
out = format_duration_value<char_type>(out, val, precision);
|
|
}
|
|
|
|
void on_duration_unit() {
|
|
out = format_duration_unit<char_type, Period>(out);
|
|
}
|
|
};
|
|
} // namespace detail
|
|
|
|
template <typename Rep, typename Period, typename Char>
|
|
struct formatter<std::chrono::duration<Rep, Period>, Char> {
|
|
private:
|
|
basic_format_specs<Char> specs;
|
|
int precision;
|
|
using arg_ref_type = detail::arg_ref<Char>;
|
|
arg_ref_type width_ref;
|
|
arg_ref_type precision_ref;
|
|
mutable basic_string_view<Char> format_str;
|
|
using duration = std::chrono::duration<Rep, Period>;
|
|
|
|
struct spec_handler {
|
|
formatter& f;
|
|
basic_format_parse_context<Char>& context;
|
|
basic_string_view<Char> format_str;
|
|
|
|
template <typename Id> FMT_CONSTEXPR arg_ref_type make_arg_ref(Id arg_id) {
|
|
context.check_arg_id(arg_id);
|
|
return arg_ref_type(arg_id);
|
|
}
|
|
|
|
FMT_CONSTEXPR arg_ref_type make_arg_ref(basic_string_view<Char> arg_id) {
|
|
context.check_arg_id(arg_id);
|
|
return arg_ref_type(arg_id);
|
|
}
|
|
|
|
FMT_CONSTEXPR arg_ref_type make_arg_ref(detail::auto_id) {
|
|
return arg_ref_type(context.next_arg_id());
|
|
}
|
|
|
|
void on_error(const char* msg) { FMT_THROW(format_error(msg)); }
|
|
void on_fill(basic_string_view<Char> fill) { f.specs.fill = fill; }
|
|
void on_align(align_t align) { f.specs.align = align; }
|
|
void on_width(int width) { f.specs.width = width; }
|
|
void on_precision(int _precision) { f.precision = _precision; }
|
|
void end_precision() {}
|
|
|
|
template <typename Id> void on_dynamic_width(Id arg_id) {
|
|
f.width_ref = make_arg_ref(arg_id);
|
|
}
|
|
|
|
template <typename Id> void on_dynamic_precision(Id arg_id) {
|
|
f.precision_ref = make_arg_ref(arg_id);
|
|
}
|
|
};
|
|
|
|
using iterator = typename basic_format_parse_context<Char>::iterator;
|
|
struct parse_range {
|
|
iterator begin;
|
|
iterator end;
|
|
};
|
|
|
|
FMT_CONSTEXPR parse_range do_parse(basic_format_parse_context<Char>& ctx) {
|
|
auto begin = ctx.begin(), end = ctx.end();
|
|
if (begin == end || *begin == '}') return {begin, begin};
|
|
spec_handler handler{*this, ctx, format_str};
|
|
begin = detail::parse_align(begin, end, handler);
|
|
if (begin == end) return {begin, begin};
|
|
begin = detail::parse_width(begin, end, handler);
|
|
if (begin == end) return {begin, begin};
|
|
if (*begin == '.') {
|
|
if (std::is_floating_point<Rep>::value)
|
|
begin = detail::parse_precision(begin, end, handler);
|
|
else
|
|
handler.on_error("precision not allowed for this argument type");
|
|
}
|
|
end = parse_chrono_format(begin, end, detail::chrono_format_checker());
|
|
return {begin, end};
|
|
}
|
|
|
|
public:
|
|
formatter() : precision(-1) {}
|
|
|
|
FMT_CONSTEXPR auto parse(basic_format_parse_context<Char>& ctx)
|
|
-> decltype(ctx.begin()) {
|
|
auto range = do_parse(ctx);
|
|
format_str = basic_string_view<Char>(
|
|
&*range.begin, detail::to_unsigned(range.end - range.begin));
|
|
return range.end;
|
|
}
|
|
|
|
template <typename FormatContext>
|
|
auto format(const duration& d, FormatContext& ctx) -> decltype(ctx.out()) {
|
|
auto begin = format_str.begin(), end = format_str.end();
|
|
// As a possible future optimization, we could avoid extra copying if width
|
|
// is not specified.
|
|
basic_memory_buffer<Char> buf;
|
|
auto out = std::back_inserter(buf);
|
|
detail::handle_dynamic_spec<detail::width_checker>(specs.width, width_ref,
|
|
ctx);
|
|
detail::handle_dynamic_spec<detail::precision_checker>(precision,
|
|
precision_ref, ctx);
|
|
if (begin == end || *begin == '}') {
|
|
out = detail::format_duration_value<Char>(out, d.count(), precision);
|
|
detail::format_duration_unit<Char, Period>(out);
|
|
} else {
|
|
detail::chrono_formatter<FormatContext, decltype(out), Rep, Period> f(
|
|
ctx, out, d);
|
|
f.precision = precision;
|
|
parse_chrono_format(begin, end, f);
|
|
}
|
|
return detail::write(
|
|
ctx.out(), basic_string_view<Char>(buf.data(), buf.size()), specs);
|
|
}
|
|
};
|
|
|
|
FMT_END_NAMESPACE
|
|
|
|
#endif // FMT_CHRONO_H_
|