mqt-core 3.3.2__cp314-cp314t-macosx_11_0_x86_64.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (546) hide show
  1. mqt/core/__init__.py +77 -0
  2. mqt/core/__main__.py +55 -0
  3. mqt/core/_commands.py +52 -0
  4. mqt/core/_compat/__init__.py +11 -0
  5. mqt/core/_compat/typing.py +29 -0
  6. mqt/core/_version.py +34 -0
  7. mqt/core/_version.pyi +12 -0
  8. mqt/core/dd.cpython-314t-darwin.so +0 -0
  9. mqt/core/dd.pyi +1016 -0
  10. mqt/core/dd_evaluation.py +368 -0
  11. mqt/core/fomac.cpython-314t-darwin.so +0 -0
  12. mqt/core/fomac.pyi +125 -0
  13. mqt/core/include/mqt-core/algorithms/BernsteinVazirani.hpp +39 -0
  14. mqt/core/include/mqt-core/algorithms/GHZState.hpp +18 -0
  15. mqt/core/include/mqt-core/algorithms/Grover.hpp +33 -0
  16. mqt/core/include/mqt-core/algorithms/QFT.hpp +21 -0
  17. mqt/core/include/mqt-core/algorithms/QPE.hpp +30 -0
  18. mqt/core/include/mqt-core/algorithms/RandomCliffordCircuit.hpp +22 -0
  19. mqt/core/include/mqt-core/algorithms/StatePreparation.hpp +43 -0
  20. mqt/core/include/mqt-core/algorithms/WState.hpp +18 -0
  21. mqt/core/include/mqt-core/algorithms/mqt_core_algorithms_export.h +43 -0
  22. mqt/core/include/mqt-core/boost/config/abi/borland_prefix.hpp +27 -0
  23. mqt/core/include/mqt-core/boost/config/abi/borland_suffix.hpp +12 -0
  24. mqt/core/include/mqt-core/boost/config/abi/msvc_prefix.hpp +22 -0
  25. mqt/core/include/mqt-core/boost/config/abi/msvc_suffix.hpp +8 -0
  26. mqt/core/include/mqt-core/boost/config/abi_prefix.hpp +25 -0
  27. mqt/core/include/mqt-core/boost/config/abi_suffix.hpp +25 -0
  28. mqt/core/include/mqt-core/boost/config/assert_cxx03.hpp +211 -0
  29. mqt/core/include/mqt-core/boost/config/assert_cxx11.hpp +212 -0
  30. mqt/core/include/mqt-core/boost/config/assert_cxx14.hpp +47 -0
  31. mqt/core/include/mqt-core/boost/config/assert_cxx17.hpp +65 -0
  32. mqt/core/include/mqt-core/boost/config/assert_cxx20.hpp +59 -0
  33. mqt/core/include/mqt-core/boost/config/assert_cxx23.hpp +41 -0
  34. mqt/core/include/mqt-core/boost/config/assert_cxx98.hpp +23 -0
  35. mqt/core/include/mqt-core/boost/config/auto_link.hpp +525 -0
  36. mqt/core/include/mqt-core/boost/config/compiler/borland.hpp +342 -0
  37. mqt/core/include/mqt-core/boost/config/compiler/clang.hpp +370 -0
  38. mqt/core/include/mqt-core/boost/config/compiler/clang_version.hpp +89 -0
  39. mqt/core/include/mqt-core/boost/config/compiler/codegear.hpp +389 -0
  40. mqt/core/include/mqt-core/boost/config/compiler/comeau.hpp +59 -0
  41. mqt/core/include/mqt-core/boost/config/compiler/common_edg.hpp +185 -0
  42. mqt/core/include/mqt-core/boost/config/compiler/compaq_cxx.hpp +19 -0
  43. mqt/core/include/mqt-core/boost/config/compiler/cray.hpp +446 -0
  44. mqt/core/include/mqt-core/boost/config/compiler/diab.hpp +26 -0
  45. mqt/core/include/mqt-core/boost/config/compiler/digitalmars.hpp +146 -0
  46. mqt/core/include/mqt-core/boost/config/compiler/gcc.hpp +386 -0
  47. mqt/core/include/mqt-core/boost/config/compiler/gcc_xml.hpp +115 -0
  48. mqt/core/include/mqt-core/boost/config/compiler/greenhills.hpp +28 -0
  49. mqt/core/include/mqt-core/boost/config/compiler/hp_acc.hpp +153 -0
  50. mqt/core/include/mqt-core/boost/config/compiler/intel.hpp +577 -0
  51. mqt/core/include/mqt-core/boost/config/compiler/kai.hpp +33 -0
  52. mqt/core/include/mqt-core/boost/config/compiler/metrowerks.hpp +201 -0
  53. mqt/core/include/mqt-core/boost/config/compiler/mpw.hpp +143 -0
  54. mqt/core/include/mqt-core/boost/config/compiler/nvcc.hpp +64 -0
  55. mqt/core/include/mqt-core/boost/config/compiler/pathscale.hpp +141 -0
  56. mqt/core/include/mqt-core/boost/config/compiler/pgi.hpp +23 -0
  57. mqt/core/include/mqt-core/boost/config/compiler/sgi_mipspro.hpp +29 -0
  58. mqt/core/include/mqt-core/boost/config/compiler/sunpro_cc.hpp +225 -0
  59. mqt/core/include/mqt-core/boost/config/compiler/vacpp.hpp +189 -0
  60. mqt/core/include/mqt-core/boost/config/compiler/visualc.hpp +398 -0
  61. mqt/core/include/mqt-core/boost/config/compiler/xlcpp.hpp +303 -0
  62. mqt/core/include/mqt-core/boost/config/compiler/xlcpp_zos.hpp +174 -0
  63. mqt/core/include/mqt-core/boost/config/detail/cxx_composite.hpp +218 -0
  64. mqt/core/include/mqt-core/boost/config/detail/posix_features.hpp +95 -0
  65. mqt/core/include/mqt-core/boost/config/detail/select_compiler_config.hpp +157 -0
  66. mqt/core/include/mqt-core/boost/config/detail/select_platform_config.hpp +147 -0
  67. mqt/core/include/mqt-core/boost/config/detail/select_stdlib_config.hpp +121 -0
  68. mqt/core/include/mqt-core/boost/config/detail/suffix.hpp +1334 -0
  69. mqt/core/include/mqt-core/boost/config/header_deprecated.hpp +26 -0
  70. mqt/core/include/mqt-core/boost/config/helper_macros.hpp +37 -0
  71. mqt/core/include/mqt-core/boost/config/no_tr1/cmath.hpp +28 -0
  72. mqt/core/include/mqt-core/boost/config/no_tr1/complex.hpp +28 -0
  73. mqt/core/include/mqt-core/boost/config/no_tr1/functional.hpp +28 -0
  74. mqt/core/include/mqt-core/boost/config/no_tr1/memory.hpp +28 -0
  75. mqt/core/include/mqt-core/boost/config/no_tr1/utility.hpp +28 -0
  76. mqt/core/include/mqt-core/boost/config/platform/aix.hpp +33 -0
  77. mqt/core/include/mqt-core/boost/config/platform/amigaos.hpp +15 -0
  78. mqt/core/include/mqt-core/boost/config/platform/beos.hpp +26 -0
  79. mqt/core/include/mqt-core/boost/config/platform/bsd.hpp +83 -0
  80. mqt/core/include/mqt-core/boost/config/platform/cloudabi.hpp +18 -0
  81. mqt/core/include/mqt-core/boost/config/platform/cray.hpp +18 -0
  82. mqt/core/include/mqt-core/boost/config/platform/cygwin.hpp +71 -0
  83. mqt/core/include/mqt-core/boost/config/platform/haiku.hpp +31 -0
  84. mqt/core/include/mqt-core/boost/config/platform/hpux.hpp +87 -0
  85. mqt/core/include/mqt-core/boost/config/platform/irix.hpp +31 -0
  86. mqt/core/include/mqt-core/boost/config/platform/linux.hpp +106 -0
  87. mqt/core/include/mqt-core/boost/config/platform/macos.hpp +87 -0
  88. mqt/core/include/mqt-core/boost/config/platform/qnxnto.hpp +31 -0
  89. mqt/core/include/mqt-core/boost/config/platform/solaris.hpp +31 -0
  90. mqt/core/include/mqt-core/boost/config/platform/symbian.hpp +97 -0
  91. mqt/core/include/mqt-core/boost/config/platform/vms.hpp +25 -0
  92. mqt/core/include/mqt-core/boost/config/platform/vxworks.hpp +422 -0
  93. mqt/core/include/mqt-core/boost/config/platform/wasm.hpp +23 -0
  94. mqt/core/include/mqt-core/boost/config/platform/win32.hpp +90 -0
  95. mqt/core/include/mqt-core/boost/config/platform/zos.hpp +32 -0
  96. mqt/core/include/mqt-core/boost/config/pragma_message.hpp +31 -0
  97. mqt/core/include/mqt-core/boost/config/requires_threads.hpp +92 -0
  98. mqt/core/include/mqt-core/boost/config/stdlib/dinkumware.hpp +324 -0
  99. mqt/core/include/mqt-core/boost/config/stdlib/libcomo.hpp +93 -0
  100. mqt/core/include/mqt-core/boost/config/stdlib/libcpp.hpp +180 -0
  101. mqt/core/include/mqt-core/boost/config/stdlib/libstdcpp3.hpp +482 -0
  102. mqt/core/include/mqt-core/boost/config/stdlib/modena.hpp +79 -0
  103. mqt/core/include/mqt-core/boost/config/stdlib/msl.hpp +98 -0
  104. mqt/core/include/mqt-core/boost/config/stdlib/roguewave.hpp +208 -0
  105. mqt/core/include/mqt-core/boost/config/stdlib/sgi.hpp +168 -0
  106. mqt/core/include/mqt-core/boost/config/stdlib/stlport.hpp +258 -0
  107. mqt/core/include/mqt-core/boost/config/stdlib/vacpp.hpp +74 -0
  108. mqt/core/include/mqt-core/boost/config/stdlib/xlcpp_zos.hpp +61 -0
  109. mqt/core/include/mqt-core/boost/config/user.hpp +133 -0
  110. mqt/core/include/mqt-core/boost/config/warning_disable.hpp +47 -0
  111. mqt/core/include/mqt-core/boost/config/workaround.hpp +305 -0
  112. mqt/core/include/mqt-core/boost/config.hpp +67 -0
  113. mqt/core/include/mqt-core/boost/cstdint.hpp +556 -0
  114. mqt/core/include/mqt-core/boost/cxx11_char_types.hpp +70 -0
  115. mqt/core/include/mqt-core/boost/detail/workaround.hpp +10 -0
  116. mqt/core/include/mqt-core/boost/limits.hpp +146 -0
  117. mqt/core/include/mqt-core/boost/multiprecision/complex128.hpp +24 -0
  118. mqt/core/include/mqt-core/boost/multiprecision/complex_adaptor.hpp +1046 -0
  119. mqt/core/include/mqt-core/boost/multiprecision/concepts/mp_number_archetypes.hpp +257 -0
  120. mqt/core/include/mqt-core/boost/multiprecision/cpp_bin_float/io.hpp +698 -0
  121. mqt/core/include/mqt-core/boost/multiprecision/cpp_bin_float/transcendental.hpp +157 -0
  122. mqt/core/include/mqt-core/boost/multiprecision/cpp_bin_float.hpp +2297 -0
  123. mqt/core/include/mqt-core/boost/multiprecision/cpp_complex.hpp +12 -0
  124. mqt/core/include/mqt-core/boost/multiprecision/cpp_dec_float.hpp +3690 -0
  125. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/add.hpp +368 -0
  126. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/add_unsigned.hpp +387 -0
  127. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/bitwise.hpp +889 -0
  128. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/checked.hpp +178 -0
  129. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/comparison.hpp +374 -0
  130. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/cpp_int_config.hpp +161 -0
  131. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/divide.hpp +703 -0
  132. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/import_export.hpp +248 -0
  133. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/intel_intrinsics.hpp +138 -0
  134. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/limits.hpp +282 -0
  135. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/literals.hpp +295 -0
  136. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/misc.hpp +1457 -0
  137. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/multiply.hpp +848 -0
  138. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/serialize.hpp +211 -0
  139. mqt/core/include/mqt-core/boost/multiprecision/cpp_int/value_pack.hpp +42 -0
  140. mqt/core/include/mqt-core/boost/multiprecision/cpp_int.hpp +2360 -0
  141. mqt/core/include/mqt-core/boost/multiprecision/debug_adaptor.hpp +760 -0
  142. mqt/core/include/mqt-core/boost/multiprecision/detail/assert.hpp +29 -0
  143. mqt/core/include/mqt-core/boost/multiprecision/detail/atomic.hpp +62 -0
  144. mqt/core/include/mqt-core/boost/multiprecision/detail/bitscan.hpp +317 -0
  145. mqt/core/include/mqt-core/boost/multiprecision/detail/check_cpp11_config.hpp +64 -0
  146. mqt/core/include/mqt-core/boost/multiprecision/detail/constexpr.hpp +88 -0
  147. mqt/core/include/mqt-core/boost/multiprecision/detail/default_ops.hpp +4052 -0
  148. mqt/core/include/mqt-core/boost/multiprecision/detail/digits.hpp +49 -0
  149. mqt/core/include/mqt-core/boost/multiprecision/detail/dynamic_array.hpp +44 -0
  150. mqt/core/include/mqt-core/boost/multiprecision/detail/empty_value.hpp +87 -0
  151. mqt/core/include/mqt-core/boost/multiprecision/detail/endian.hpp +35 -0
  152. mqt/core/include/mqt-core/boost/multiprecision/detail/et_ops.hpp +1831 -0
  153. mqt/core/include/mqt-core/boost/multiprecision/detail/float128_functions.hpp +95 -0
  154. mqt/core/include/mqt-core/boost/multiprecision/detail/float_string_cvt.hpp +333 -0
  155. mqt/core/include/mqt-core/boost/multiprecision/detail/fpclassify.hpp +101 -0
  156. mqt/core/include/mqt-core/boost/multiprecision/detail/functions/constants.hpp +288 -0
  157. mqt/core/include/mqt-core/boost/multiprecision/detail/functions/pow.hpp +905 -0
  158. mqt/core/include/mqt-core/boost/multiprecision/detail/functions/trig.hpp +1058 -0
  159. mqt/core/include/mqt-core/boost/multiprecision/detail/functions/trunc.hpp +82 -0
  160. mqt/core/include/mqt-core/boost/multiprecision/detail/generic_interconvert.hpp +687 -0
  161. mqt/core/include/mqt-core/boost/multiprecision/detail/hash.hpp +56 -0
  162. mqt/core/include/mqt-core/boost/multiprecision/detail/integer_ops.hpp +474 -0
  163. mqt/core/include/mqt-core/boost/multiprecision/detail/itos.hpp +39 -0
  164. mqt/core/include/mqt-core/boost/multiprecision/detail/min_max.hpp +106 -0
  165. mqt/core/include/mqt-core/boost/multiprecision/detail/no_et_ops.hpp +661 -0
  166. mqt/core/include/mqt-core/boost/multiprecision/detail/no_exceptions_support.hpp +55 -0
  167. mqt/core/include/mqt-core/boost/multiprecision/detail/number_base.hpp +1656 -0
  168. mqt/core/include/mqt-core/boost/multiprecision/detail/number_compare.hpp +848 -0
  169. mqt/core/include/mqt-core/boost/multiprecision/detail/precision.hpp +313 -0
  170. mqt/core/include/mqt-core/boost/multiprecision/detail/rebind.hpp +19 -0
  171. mqt/core/include/mqt-core/boost/multiprecision/detail/standalone_config.hpp +148 -0
  172. mqt/core/include/mqt-core/boost/multiprecision/detail/static_array.hpp +42 -0
  173. mqt/core/include/mqt-core/boost/multiprecision/detail/string_helpers.hpp +48 -0
  174. mqt/core/include/mqt-core/boost/multiprecision/detail/tables.hpp +80 -0
  175. mqt/core/include/mqt-core/boost/multiprecision/detail/ublas_interop.hpp +75 -0
  176. mqt/core/include/mqt-core/boost/multiprecision/detail/uniform_int_distribution.hpp +212 -0
  177. mqt/core/include/mqt-core/boost/multiprecision/detail/utype_helper.hpp +374 -0
  178. mqt/core/include/mqt-core/boost/multiprecision/eigen.hpp +248 -0
  179. mqt/core/include/mqt-core/boost/multiprecision/float128.hpp +920 -0
  180. mqt/core/include/mqt-core/boost/multiprecision/fwd.hpp +268 -0
  181. mqt/core/include/mqt-core/boost/multiprecision/gmp.hpp +4060 -0
  182. mqt/core/include/mqt-core/boost/multiprecision/integer.hpp +363 -0
  183. mqt/core/include/mqt-core/boost/multiprecision/logged_adaptor.hpp +834 -0
  184. mqt/core/include/mqt-core/boost/multiprecision/miller_rabin.hpp +221 -0
  185. mqt/core/include/mqt-core/boost/multiprecision/mpc.hpp +1721 -0
  186. mqt/core/include/mqt-core/boost/multiprecision/mpfi.hpp +2559 -0
  187. mqt/core/include/mqt-core/boost/multiprecision/mpfr.hpp +3644 -0
  188. mqt/core/include/mqt-core/boost/multiprecision/number.hpp +2500 -0
  189. mqt/core/include/mqt-core/boost/multiprecision/random.hpp +23 -0
  190. mqt/core/include/mqt-core/boost/multiprecision/rational_adaptor.hpp +1289 -0
  191. mqt/core/include/mqt-core/boost/multiprecision/tommath.hpp +1034 -0
  192. mqt/core/include/mqt-core/boost/multiprecision/traits/explicit_conversion.hpp +67 -0
  193. mqt/core/include/mqt-core/boost/multiprecision/traits/extract_exponent_type.hpp +28 -0
  194. mqt/core/include/mqt-core/boost/multiprecision/traits/is_backend.hpp +91 -0
  195. mqt/core/include/mqt-core/boost/multiprecision/traits/is_byte_container.hpp +51 -0
  196. mqt/core/include/mqt-core/boost/multiprecision/traits/is_complex.hpp +22 -0
  197. mqt/core/include/mqt-core/boost/multiprecision/traits/is_convertible_arithmetic.hpp +51 -0
  198. mqt/core/include/mqt-core/boost/multiprecision/traits/is_restricted_conversion.hpp +47 -0
  199. mqt/core/include/mqt-core/boost/multiprecision/traits/is_variable_precision.hpp +25 -0
  200. mqt/core/include/mqt-core/boost/multiprecision/traits/max_digits10.hpp +79 -0
  201. mqt/core/include/mqt-core/boost/multiprecision/traits/std_integer_traits.hpp +90 -0
  202. mqt/core/include/mqt-core/boost/multiprecision/traits/transcendental_reduction_type.hpp +21 -0
  203. mqt/core/include/mqt-core/boost/version.hpp +32 -0
  204. mqt/core/include/mqt-core/circuit_optimizer/CircuitOptimizer.hpp +119 -0
  205. mqt/core/include/mqt-core/circuit_optimizer/mqt_core_circuit_optimizer_export.h +43 -0
  206. mqt/core/include/mqt-core/datastructures/DirectedAcyclicGraph.hpp +117 -0
  207. mqt/core/include/mqt-core/datastructures/DirectedGraph.hpp +158 -0
  208. mqt/core/include/mqt-core/datastructures/DisjointSet.hpp +50 -0
  209. mqt/core/include/mqt-core/datastructures/Layer.hpp +172 -0
  210. mqt/core/include/mqt-core/datastructures/SymmetricMatrix.hpp +57 -0
  211. mqt/core/include/mqt-core/datastructures/UndirectedGraph.hpp +227 -0
  212. mqt/core/include/mqt-core/datastructures/mqt_core_ds_export.h +43 -0
  213. mqt/core/include/mqt-core/dd/Approximation.hpp +45 -0
  214. mqt/core/include/mqt-core/dd/CachedEdge.hpp +174 -0
  215. mqt/core/include/mqt-core/dd/Complex.hpp +165 -0
  216. mqt/core/include/mqt-core/dd/ComplexNumbers.hpp +150 -0
  217. mqt/core/include/mqt-core/dd/ComplexValue.hpp +184 -0
  218. mqt/core/include/mqt-core/dd/ComputeTable.hpp +183 -0
  219. mqt/core/include/mqt-core/dd/DDDefinitions.hpp +139 -0
  220. mqt/core/include/mqt-core/dd/DDpackageConfig.hpp +104 -0
  221. mqt/core/include/mqt-core/dd/DensityNoiseTable.hpp +114 -0
  222. mqt/core/include/mqt-core/dd/Edge.hpp +416 -0
  223. mqt/core/include/mqt-core/dd/Export.hpp +438 -0
  224. mqt/core/include/mqt-core/dd/FunctionalityConstruction.hpp +75 -0
  225. mqt/core/include/mqt-core/dd/GateMatrixDefinitions.hpp +43 -0
  226. mqt/core/include/mqt-core/dd/LinkedListBase.hpp +45 -0
  227. mqt/core/include/mqt-core/dd/MemoryManager.hpp +193 -0
  228. mqt/core/include/mqt-core/dd/Node.hpp +223 -0
  229. mqt/core/include/mqt-core/dd/NoiseFunctionality.hpp +144 -0
  230. mqt/core/include/mqt-core/dd/Operations.hpp +306 -0
  231. mqt/core/include/mqt-core/dd/Package.hpp +2036 -0
  232. mqt/core/include/mqt-core/dd/Package_fwd.hpp +22 -0
  233. mqt/core/include/mqt-core/dd/RealNumber.hpp +255 -0
  234. mqt/core/include/mqt-core/dd/RealNumberUniqueTable.hpp +217 -0
  235. mqt/core/include/mqt-core/dd/Simulation.hpp +98 -0
  236. mqt/core/include/mqt-core/dd/StateGeneration.hpp +143 -0
  237. mqt/core/include/mqt-core/dd/StochasticNoiseOperationTable.hpp +88 -0
  238. mqt/core/include/mqt-core/dd/UnaryComputeTable.hpp +121 -0
  239. mqt/core/include/mqt-core/dd/UniqueTable.hpp +243 -0
  240. mqt/core/include/mqt-core/dd/mqt_core_dd_export.h +43 -0
  241. mqt/core/include/mqt-core/dd/statistics/MemoryManagerStatistics.hpp +84 -0
  242. mqt/core/include/mqt-core/dd/statistics/PackageStatistics.hpp +55 -0
  243. mqt/core/include/mqt-core/dd/statistics/Statistics.hpp +48 -0
  244. mqt/core/include/mqt-core/dd/statistics/TableStatistics.hpp +79 -0
  245. mqt/core/include/mqt-core/dd/statistics/UniqueTableStatistics.hpp +31 -0
  246. mqt/core/include/mqt-core/fomac/FoMaC.hpp +568 -0
  247. mqt/core/include/mqt-core/ir/Definitions.hpp +108 -0
  248. mqt/core/include/mqt-core/ir/Permutation.hpp +213 -0
  249. mqt/core/include/mqt-core/ir/QuantumComputation.hpp +596 -0
  250. mqt/core/include/mqt-core/ir/Register.hpp +125 -0
  251. mqt/core/include/mqt-core/ir/mqt_core_ir_export.h +43 -0
  252. mqt/core/include/mqt-core/ir/operations/AodOperation.hpp +92 -0
  253. mqt/core/include/mqt-core/ir/operations/CompoundOperation.hpp +212 -0
  254. mqt/core/include/mqt-core/ir/operations/Control.hpp +142 -0
  255. mqt/core/include/mqt-core/ir/operations/Expression.hpp +847 -0
  256. mqt/core/include/mqt-core/ir/operations/IfElseOperation.hpp +169 -0
  257. mqt/core/include/mqt-core/ir/operations/NonUnitaryOperation.hpp +118 -0
  258. mqt/core/include/mqt-core/ir/operations/OpType.hpp +120 -0
  259. mqt/core/include/mqt-core/ir/operations/OpType.inc +76 -0
  260. mqt/core/include/mqt-core/ir/operations/Operation.hpp +247 -0
  261. mqt/core/include/mqt-core/ir/operations/StandardOperation.hpp +140 -0
  262. mqt/core/include/mqt-core/ir/operations/SymbolicOperation.hpp +144 -0
  263. mqt/core/include/mqt-core/mqt_na_qdmi/device.h +602 -0
  264. mqt/core/include/mqt-core/mqt_na_qdmi/types.h +78 -0
  265. mqt/core/include/mqt-core/na/NAComputation.hpp +185 -0
  266. mqt/core/include/mqt-core/na/device/Device.hpp +410 -0
  267. mqt/core/include/mqt-core/na/device/DeviceMemberInitializers.hpp +724 -0
  268. mqt/core/include/mqt-core/na/device/Generator.hpp +447 -0
  269. mqt/core/include/mqt-core/na/entities/Atom.hpp +62 -0
  270. mqt/core/include/mqt-core/na/entities/Location.hpp +154 -0
  271. mqt/core/include/mqt-core/na/entities/Zone.hpp +95 -0
  272. mqt/core/include/mqt-core/na/fomac/Device.hpp +169 -0
  273. mqt/core/include/mqt-core/na/mqt_core_na_export.h +43 -0
  274. mqt/core/include/mqt-core/na/operations/GlobalCZOp.hpp +38 -0
  275. mqt/core/include/mqt-core/na/operations/GlobalOp.hpp +58 -0
  276. mqt/core/include/mqt-core/na/operations/GlobalRYOp.hpp +42 -0
  277. mqt/core/include/mqt-core/na/operations/LoadOp.hpp +89 -0
  278. mqt/core/include/mqt-core/na/operations/LocalOp.hpp +56 -0
  279. mqt/core/include/mqt-core/na/operations/LocalRZOp.hpp +42 -0
  280. mqt/core/include/mqt-core/na/operations/LocalUOp.hpp +49 -0
  281. mqt/core/include/mqt-core/na/operations/MoveOp.hpp +66 -0
  282. mqt/core/include/mqt-core/na/operations/Op.hpp +62 -0
  283. mqt/core/include/mqt-core/na/operations/ShuttlingOp.hpp +51 -0
  284. mqt/core/include/mqt-core/na/operations/StoreOp.hpp +87 -0
  285. mqt/core/include/mqt-core/qasm3/Exception.hpp +85 -0
  286. mqt/core/include/mqt-core/qasm3/Gate.hpp +65 -0
  287. mqt/core/include/mqt-core/qasm3/Importer.hpp +192 -0
  288. mqt/core/include/mqt-core/qasm3/InstVisitor.hpp +145 -0
  289. mqt/core/include/mqt-core/qasm3/NestedEnvironment.hpp +41 -0
  290. mqt/core/include/mqt-core/qasm3/Parser.hpp +170 -0
  291. mqt/core/include/mqt-core/qasm3/Scanner.hpp +73 -0
  292. mqt/core/include/mqt-core/qasm3/Statement.hpp +486 -0
  293. mqt/core/include/mqt-core/qasm3/Statement_fwd.hpp +39 -0
  294. mqt/core/include/mqt-core/qasm3/StdGates.hpp +232 -0
  295. mqt/core/include/mqt-core/qasm3/Token.hpp +198 -0
  296. mqt/core/include/mqt-core/qasm3/Types.hpp +238 -0
  297. mqt/core/include/mqt-core/qasm3/Types_fwd.hpp +22 -0
  298. mqt/core/include/mqt-core/qasm3/mqt_core_qasm_export.h +43 -0
  299. mqt/core/include/mqt-core/qasm3/passes/CompilerPass.hpp +22 -0
  300. mqt/core/include/mqt-core/qasm3/passes/ConstEvalPass.hpp +102 -0
  301. mqt/core/include/mqt-core/qasm3/passes/TypeCheckPass.hpp +124 -0
  302. mqt/core/include/mqt-core/qdmi/Driver.hpp +431 -0
  303. mqt/core/include/mqt-core/zx/FunctionalityConstruction.hpp +125 -0
  304. mqt/core/include/mqt-core/zx/Rational.hpp +318 -0
  305. mqt/core/include/mqt-core/zx/Rules.hpp +132 -0
  306. mqt/core/include/mqt-core/zx/Simplify.hpp +182 -0
  307. mqt/core/include/mqt-core/zx/Utils.hpp +212 -0
  308. mqt/core/include/mqt-core/zx/ZXDefinitions.hpp +93 -0
  309. mqt/core/include/mqt-core/zx/ZXDiagram.hpp +480 -0
  310. mqt/core/include/mqt-core/zx/mqt_core_zx_export.h +43 -0
  311. mqt/core/include/nlohmann/adl_serializer.hpp +55 -0
  312. mqt/core/include/nlohmann/byte_container_with_subtype.hpp +103 -0
  313. mqt/core/include/nlohmann/detail/abi_macros.hpp +111 -0
  314. mqt/core/include/nlohmann/detail/conversions/from_json.hpp +577 -0
  315. mqt/core/include/nlohmann/detail/conversions/to_chars.hpp +1118 -0
  316. mqt/core/include/nlohmann/detail/conversions/to_json.hpp +479 -0
  317. mqt/core/include/nlohmann/detail/exceptions.hpp +291 -0
  318. mqt/core/include/nlohmann/detail/hash.hpp +129 -0
  319. mqt/core/include/nlohmann/detail/input/binary_reader.hpp +3068 -0
  320. mqt/core/include/nlohmann/detail/input/input_adapters.hpp +549 -0
  321. mqt/core/include/nlohmann/detail/input/json_sax.hpp +986 -0
  322. mqt/core/include/nlohmann/detail/input/lexer.hpp +1643 -0
  323. mqt/core/include/nlohmann/detail/input/parser.hpp +519 -0
  324. mqt/core/include/nlohmann/detail/input/position_t.hpp +37 -0
  325. mqt/core/include/nlohmann/detail/iterators/internal_iterator.hpp +35 -0
  326. mqt/core/include/nlohmann/detail/iterators/iter_impl.hpp +760 -0
  327. mqt/core/include/nlohmann/detail/iterators/iteration_proxy.hpp +235 -0
  328. mqt/core/include/nlohmann/detail/iterators/iterator_traits.hpp +61 -0
  329. mqt/core/include/nlohmann/detail/iterators/json_reverse_iterator.hpp +130 -0
  330. mqt/core/include/nlohmann/detail/iterators/primitive_iterator.hpp +132 -0
  331. mqt/core/include/nlohmann/detail/json_custom_base_class.hpp +39 -0
  332. mqt/core/include/nlohmann/detail/json_pointer.hpp +988 -0
  333. mqt/core/include/nlohmann/detail/json_ref.hpp +78 -0
  334. mqt/core/include/nlohmann/detail/macro_scope.hpp +595 -0
  335. mqt/core/include/nlohmann/detail/macro_unscope.hpp +46 -0
  336. mqt/core/include/nlohmann/detail/meta/call_std/begin.hpp +17 -0
  337. mqt/core/include/nlohmann/detail/meta/call_std/end.hpp +17 -0
  338. mqt/core/include/nlohmann/detail/meta/cpp_future.hpp +171 -0
  339. mqt/core/include/nlohmann/detail/meta/detected.hpp +70 -0
  340. mqt/core/include/nlohmann/detail/meta/identity_tag.hpp +21 -0
  341. mqt/core/include/nlohmann/detail/meta/is_sax.hpp +159 -0
  342. mqt/core/include/nlohmann/detail/meta/std_fs.hpp +29 -0
  343. mqt/core/include/nlohmann/detail/meta/type_traits.hpp +795 -0
  344. mqt/core/include/nlohmann/detail/meta/void_t.hpp +24 -0
  345. mqt/core/include/nlohmann/detail/output/binary_writer.hpp +1850 -0
  346. mqt/core/include/nlohmann/detail/output/output_adapters.hpp +147 -0
  347. mqt/core/include/nlohmann/detail/output/serializer.hpp +988 -0
  348. mqt/core/include/nlohmann/detail/string_concat.hpp +146 -0
  349. mqt/core/include/nlohmann/detail/string_escape.hpp +72 -0
  350. mqt/core/include/nlohmann/detail/string_utils.hpp +37 -0
  351. mqt/core/include/nlohmann/detail/value_t.hpp +118 -0
  352. mqt/core/include/nlohmann/json.hpp +5306 -0
  353. mqt/core/include/nlohmann/json_fwd.hpp +75 -0
  354. mqt/core/include/nlohmann/ordered_map.hpp +359 -0
  355. mqt/core/include/nlohmann/thirdparty/hedley/hedley.hpp +2045 -0
  356. mqt/core/include/nlohmann/thirdparty/hedley/hedley_undef.hpp +158 -0
  357. mqt/core/include/qdmi/qdmi/client.h +990 -0
  358. mqt/core/include/qdmi/qdmi/constants.h +1139 -0
  359. mqt/core/include/qdmi/qdmi/device.h +602 -0
  360. mqt/core/include/qdmi/qdmi/types.h +78 -0
  361. mqt/core/include/spdlog/async.h +99 -0
  362. mqt/core/include/spdlog/async_logger-inl.h +84 -0
  363. mqt/core/include/spdlog/async_logger.h +74 -0
  364. mqt/core/include/spdlog/cfg/argv.h +40 -0
  365. mqt/core/include/spdlog/cfg/env.h +36 -0
  366. mqt/core/include/spdlog/cfg/helpers-inl.h +107 -0
  367. mqt/core/include/spdlog/cfg/helpers.h +29 -0
  368. mqt/core/include/spdlog/common-inl.h +68 -0
  369. mqt/core/include/spdlog/common.h +406 -0
  370. mqt/core/include/spdlog/details/backtracer-inl.h +63 -0
  371. mqt/core/include/spdlog/details/backtracer.h +45 -0
  372. mqt/core/include/spdlog/details/circular_q.h +115 -0
  373. mqt/core/include/spdlog/details/console_globals.h +28 -0
  374. mqt/core/include/spdlog/details/file_helper-inl.h +153 -0
  375. mqt/core/include/spdlog/details/file_helper.h +61 -0
  376. mqt/core/include/spdlog/details/fmt_helper.h +141 -0
  377. mqt/core/include/spdlog/details/log_msg-inl.h +44 -0
  378. mqt/core/include/spdlog/details/log_msg.h +40 -0
  379. mqt/core/include/spdlog/details/log_msg_buffer-inl.h +54 -0
  380. mqt/core/include/spdlog/details/log_msg_buffer.h +32 -0
  381. mqt/core/include/spdlog/details/mpmc_blocking_q.h +177 -0
  382. mqt/core/include/spdlog/details/null_mutex.h +35 -0
  383. mqt/core/include/spdlog/details/os-inl.h +606 -0
  384. mqt/core/include/spdlog/details/os.h +127 -0
  385. mqt/core/include/spdlog/details/periodic_worker-inl.h +26 -0
  386. mqt/core/include/spdlog/details/periodic_worker.h +58 -0
  387. mqt/core/include/spdlog/details/registry-inl.h +270 -0
  388. mqt/core/include/spdlog/details/registry.h +131 -0
  389. mqt/core/include/spdlog/details/synchronous_factory.h +22 -0
  390. mqt/core/include/spdlog/details/tcp_client-windows.h +135 -0
  391. mqt/core/include/spdlog/details/tcp_client.h +127 -0
  392. mqt/core/include/spdlog/details/thread_pool-inl.h +126 -0
  393. mqt/core/include/spdlog/details/thread_pool.h +117 -0
  394. mqt/core/include/spdlog/details/udp_client-windows.h +98 -0
  395. mqt/core/include/spdlog/details/udp_client.h +81 -0
  396. mqt/core/include/spdlog/details/windows_include.h +11 -0
  397. mqt/core/include/spdlog/fmt/bin_to_hex.h +224 -0
  398. mqt/core/include/spdlog/fmt/bundled/args.h +220 -0
  399. mqt/core/include/spdlog/fmt/bundled/base.h +2989 -0
  400. mqt/core/include/spdlog/fmt/bundled/chrono.h +2330 -0
  401. mqt/core/include/spdlog/fmt/bundled/color.h +637 -0
  402. mqt/core/include/spdlog/fmt/bundled/compile.h +539 -0
  403. mqt/core/include/spdlog/fmt/bundled/core.h +5 -0
  404. mqt/core/include/spdlog/fmt/bundled/fmt.license.rst +27 -0
  405. mqt/core/include/spdlog/fmt/bundled/format-inl.h +1948 -0
  406. mqt/core/include/spdlog/fmt/bundled/format.h +4244 -0
  407. mqt/core/include/spdlog/fmt/bundled/os.h +427 -0
  408. mqt/core/include/spdlog/fmt/bundled/ostream.h +167 -0
  409. mqt/core/include/spdlog/fmt/bundled/printf.h +633 -0
  410. mqt/core/include/spdlog/fmt/bundled/ranges.h +850 -0
  411. mqt/core/include/spdlog/fmt/bundled/std.h +728 -0
  412. mqt/core/include/spdlog/fmt/bundled/xchar.h +369 -0
  413. mqt/core/include/spdlog/fmt/chrono.h +23 -0
  414. mqt/core/include/spdlog/fmt/compile.h +23 -0
  415. mqt/core/include/spdlog/fmt/fmt.h +30 -0
  416. mqt/core/include/spdlog/fmt/ostr.h +23 -0
  417. mqt/core/include/spdlog/fmt/ranges.h +23 -0
  418. mqt/core/include/spdlog/fmt/std.h +24 -0
  419. mqt/core/include/spdlog/fmt/xchar.h +23 -0
  420. mqt/core/include/spdlog/formatter.h +17 -0
  421. mqt/core/include/spdlog/fwd.h +18 -0
  422. mqt/core/include/spdlog/logger-inl.h +198 -0
  423. mqt/core/include/spdlog/logger.h +379 -0
  424. mqt/core/include/spdlog/mdc.h +52 -0
  425. mqt/core/include/spdlog/pattern_formatter-inl.h +1340 -0
  426. mqt/core/include/spdlog/pattern_formatter.h +118 -0
  427. mqt/core/include/spdlog/sinks/android_sink.h +137 -0
  428. mqt/core/include/spdlog/sinks/ansicolor_sink-inl.h +142 -0
  429. mqt/core/include/spdlog/sinks/ansicolor_sink.h +116 -0
  430. mqt/core/include/spdlog/sinks/base_sink-inl.h +59 -0
  431. mqt/core/include/spdlog/sinks/base_sink.h +51 -0
  432. mqt/core/include/spdlog/sinks/basic_file_sink-inl.h +48 -0
  433. mqt/core/include/spdlog/sinks/basic_file_sink.h +66 -0
  434. mqt/core/include/spdlog/sinks/callback_sink.h +56 -0
  435. mqt/core/include/spdlog/sinks/daily_file_sink.h +254 -0
  436. mqt/core/include/spdlog/sinks/dist_sink.h +81 -0
  437. mqt/core/include/spdlog/sinks/dup_filter_sink.h +91 -0
  438. mqt/core/include/spdlog/sinks/hourly_file_sink.h +193 -0
  439. mqt/core/include/spdlog/sinks/kafka_sink.h +119 -0
  440. mqt/core/include/spdlog/sinks/mongo_sink.h +108 -0
  441. mqt/core/include/spdlog/sinks/msvc_sink.h +68 -0
  442. mqt/core/include/spdlog/sinks/null_sink.h +41 -0
  443. mqt/core/include/spdlog/sinks/ostream_sink.h +43 -0
  444. mqt/core/include/spdlog/sinks/qt_sinks.h +304 -0
  445. mqt/core/include/spdlog/sinks/ringbuffer_sink.h +67 -0
  446. mqt/core/include/spdlog/sinks/rotating_file_sink-inl.h +179 -0
  447. mqt/core/include/spdlog/sinks/rotating_file_sink.h +93 -0
  448. mqt/core/include/spdlog/sinks/sink-inl.h +22 -0
  449. mqt/core/include/spdlog/sinks/sink.h +34 -0
  450. mqt/core/include/spdlog/sinks/stdout_color_sinks-inl.h +38 -0
  451. mqt/core/include/spdlog/sinks/stdout_color_sinks.h +49 -0
  452. mqt/core/include/spdlog/sinks/stdout_sinks-inl.h +127 -0
  453. mqt/core/include/spdlog/sinks/stdout_sinks.h +84 -0
  454. mqt/core/include/spdlog/sinks/syslog_sink.h +104 -0
  455. mqt/core/include/spdlog/sinks/systemd_sink.h +121 -0
  456. mqt/core/include/spdlog/sinks/tcp_sink.h +75 -0
  457. mqt/core/include/spdlog/sinks/udp_sink.h +69 -0
  458. mqt/core/include/spdlog/sinks/win_eventlog_sink.h +260 -0
  459. mqt/core/include/spdlog/sinks/wincolor_sink-inl.h +172 -0
  460. mqt/core/include/spdlog/sinks/wincolor_sink.h +82 -0
  461. mqt/core/include/spdlog/spdlog-inl.h +96 -0
  462. mqt/core/include/spdlog/spdlog.h +357 -0
  463. mqt/core/include/spdlog/stopwatch.h +66 -0
  464. mqt/core/include/spdlog/tweakme.h +148 -0
  465. mqt/core/include/spdlog/version.h +11 -0
  466. mqt/core/ir/__init__.pyi +2078 -0
  467. mqt/core/ir/operations.pyi +1011 -0
  468. mqt/core/ir/registers.pyi +91 -0
  469. mqt/core/ir/symbolic.pyi +177 -0
  470. mqt/core/ir.cpython-314t-darwin.so +0 -0
  471. mqt/core/lib/libmqt-core-algorithms.3.3.2.dylib +0 -0
  472. mqt/core/lib/libmqt-core-algorithms.3.3.dylib +0 -0
  473. mqt/core/lib/libmqt-core-algorithms.dylib +0 -0
  474. mqt/core/lib/libmqt-core-circuit-optimizer.3.3.2.dylib +0 -0
  475. mqt/core/lib/libmqt-core-circuit-optimizer.3.3.dylib +0 -0
  476. mqt/core/lib/libmqt-core-circuit-optimizer.dylib +0 -0
  477. mqt/core/lib/libmqt-core-dd.3.3.2.dylib +0 -0
  478. mqt/core/lib/libmqt-core-dd.3.3.dylib +0 -0
  479. mqt/core/lib/libmqt-core-dd.dylib +0 -0
  480. mqt/core/lib/libmqt-core-ds.3.3.2.dylib +0 -0
  481. mqt/core/lib/libmqt-core-ds.3.3.dylib +0 -0
  482. mqt/core/lib/libmqt-core-ds.dylib +0 -0
  483. mqt/core/lib/libmqt-core-fomac.3.3.2.dylib +0 -0
  484. mqt/core/lib/libmqt-core-fomac.3.3.dylib +0 -0
  485. mqt/core/lib/libmqt-core-fomac.dylib +0 -0
  486. mqt/core/lib/libmqt-core-ir.3.3.2.dylib +0 -0
  487. mqt/core/lib/libmqt-core-ir.3.3.dylib +0 -0
  488. mqt/core/lib/libmqt-core-ir.dylib +0 -0
  489. mqt/core/lib/libmqt-core-na-fomac.3.3.2.dylib +0 -0
  490. mqt/core/lib/libmqt-core-na-fomac.3.3.dylib +0 -0
  491. mqt/core/lib/libmqt-core-na-fomac.dylib +0 -0
  492. mqt/core/lib/libmqt-core-na.3.3.2.dylib +0 -0
  493. mqt/core/lib/libmqt-core-na.3.3.dylib +0 -0
  494. mqt/core/lib/libmqt-core-na.dylib +0 -0
  495. mqt/core/lib/libmqt-core-qasm.3.3.2.dylib +0 -0
  496. mqt/core/lib/libmqt-core-qasm.3.3.dylib +0 -0
  497. mqt/core/lib/libmqt-core-qasm.dylib +0 -0
  498. mqt/core/lib/libmqt-core-qdmi-driver.3.3.2.dylib +0 -0
  499. mqt/core/lib/libmqt-core-qdmi-driver.3.3.dylib +0 -0
  500. mqt/core/lib/libmqt-core-qdmi-driver.dylib +0 -0
  501. mqt/core/lib/libmqt-core-qdmi-na-device-gen.a +0 -0
  502. mqt/core/lib/libmqt-core-qdmi-na-device.3.3.2.dylib +0 -0
  503. mqt/core/lib/libmqt-core-qdmi-na-device.3.3.dylib +0 -0
  504. mqt/core/lib/libmqt-core-qdmi-na-device.dylib +0 -0
  505. mqt/core/lib/libmqt-core-zx.3.3.2.dylib +0 -0
  506. mqt/core/lib/libmqt-core-zx.3.3.dylib +0 -0
  507. mqt/core/lib/libmqt-core-zx.dylib +0 -0
  508. mqt/core/lib/libspdlog.a +0 -0
  509. mqt/core/lib/pkgconfig/spdlog.pc +13 -0
  510. mqt/core/na/__init__.py +12 -0
  511. mqt/core/na/fomac.cpython-314t-darwin.so +0 -0
  512. mqt/core/na/fomac.pyi +117 -0
  513. mqt/core/plugins/__init__.py +9 -0
  514. mqt/core/plugins/qiskit/__init__.py +19 -0
  515. mqt/core/plugins/qiskit/mqt_to_qiskit.py +420 -0
  516. mqt/core/plugins/qiskit/qiskit_to_mqt.py +562 -0
  517. mqt/core/py.typed +2 -0
  518. mqt/core/share/cmake/mqt-core/AddMQTPythonBinding.cmake +55 -0
  519. mqt/core/share/cmake/mqt-core/Cache.cmake +33 -0
  520. mqt/core/share/cmake/mqt-core/FindGMP.cmake +103 -0
  521. mqt/core/share/cmake/mqt-core/PackageAddTest.cmake +46 -0
  522. mqt/core/share/cmake/mqt-core/PreventInSourceBuilds.cmake +25 -0
  523. mqt/core/share/cmake/mqt-core/StandardProjectSettings.cmake +87 -0
  524. mqt/core/share/cmake/mqt-core/mqt-core-config-version.cmake +85 -0
  525. mqt/core/share/cmake/mqt-core/mqt-core-config.cmake +52 -0
  526. mqt/core/share/cmake/mqt-core/mqt-core-targets-release.cmake +141 -0
  527. mqt/core/share/cmake/mqt-core/mqt-core-targets.cmake +446 -0
  528. mqt/core/share/cmake/nlohmann_json/nlohmann_jsonConfig.cmake +15 -0
  529. mqt/core/share/cmake/nlohmann_json/nlohmann_jsonConfigVersion.cmake +20 -0
  530. mqt/core/share/cmake/nlohmann_json/nlohmann_jsonTargets.cmake +109 -0
  531. mqt/core/share/cmake/qdmi/Cache.cmake +44 -0
  532. mqt/core/share/cmake/qdmi/PrefixHandling.cmake +78 -0
  533. mqt/core/share/cmake/qdmi/prefix_defs.txt +26 -0
  534. mqt/core/share/cmake/qdmi/qdmi-config-version.cmake +85 -0
  535. mqt/core/share/cmake/qdmi/qdmi-config.cmake +42 -0
  536. mqt/core/share/cmake/qdmi/qdmi-targets.cmake +129 -0
  537. mqt/core/share/cmake/spdlog/spdlogConfig.cmake +44 -0
  538. mqt/core/share/cmake/spdlog/spdlogConfigTargets-release.cmake +19 -0
  539. mqt/core/share/cmake/spdlog/spdlogConfigTargets.cmake +118 -0
  540. mqt/core/share/cmake/spdlog/spdlogConfigVersion.cmake +65 -0
  541. mqt/core/share/pkgconfig/nlohmann_json.pc +7 -0
  542. mqt_core-3.3.2.dist-info/METADATA +210 -0
  543. mqt_core-3.3.2.dist-info/RECORD +546 -0
  544. mqt_core-3.3.2.dist-info/WHEEL +6 -0
  545. mqt_core-3.3.2.dist-info/entry_points.txt +4 -0
  546. mqt_core-3.3.2.dist-info/licenses/LICENSE.md +22 -0
@@ -0,0 +1,2330 @@
1
+ // Formatting library for C++ - chrono support
2
+ //
3
+ // Copyright (c) 2012 - present, Victor Zverovich
4
+ // All rights reserved.
5
+ //
6
+ // For the license information refer to format.h.
7
+
8
+ #ifndef FMT_CHRONO_H_
9
+ #define FMT_CHRONO_H_
10
+
11
+ #ifndef FMT_MODULE
12
+ # include <algorithm>
13
+ # include <chrono>
14
+ # include <cmath> // std::isfinite
15
+ # include <cstring> // std::memcpy
16
+ # include <ctime>
17
+ # include <iterator>
18
+ # include <locale>
19
+ # include <ostream>
20
+ # include <type_traits>
21
+ #endif
22
+
23
+ #include "format.h"
24
+
25
+ FMT_BEGIN_NAMESPACE
26
+
27
+ // Enable safe chrono durations, unless explicitly disabled.
28
+ #ifndef FMT_SAFE_DURATION_CAST
29
+ # define FMT_SAFE_DURATION_CAST 1
30
+ #endif
31
+ #if FMT_SAFE_DURATION_CAST
32
+
33
+ // For conversion between std::chrono::durations without undefined
34
+ // behaviour or erroneous results.
35
+ // This is a stripped down version of duration_cast, for inclusion in fmt.
36
+ // See https://github.com/pauldreik/safe_duration_cast
37
+ //
38
+ // Copyright Paul Dreik 2019
39
+ namespace safe_duration_cast {
40
+
41
+ template <typename To, typename From,
42
+ FMT_ENABLE_IF(!std::is_same<From, To>::value &&
43
+ std::numeric_limits<From>::is_signed ==
44
+ std::numeric_limits<To>::is_signed)>
45
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
46
+ -> To {
47
+ ec = 0;
48
+ using F = std::numeric_limits<From>;
49
+ using T = std::numeric_limits<To>;
50
+ static_assert(F::is_integer, "From must be integral");
51
+ static_assert(T::is_integer, "To must be integral");
52
+
53
+ // A and B are both signed, or both unsigned.
54
+ if (detail::const_check(F::digits <= T::digits)) {
55
+ // From fits in To without any problem.
56
+ } else {
57
+ // From does not always fit in To, resort to a dynamic check.
58
+ if (from < (T::min)() || from > (T::max)()) {
59
+ // outside range.
60
+ ec = 1;
61
+ return {};
62
+ }
63
+ }
64
+ return static_cast<To>(from);
65
+ }
66
+
67
+ /// Converts From to To, without loss. If the dynamic value of from
68
+ /// can't be converted to To without loss, ec is set.
69
+ template <typename To, typename From,
70
+ FMT_ENABLE_IF(!std::is_same<From, To>::value &&
71
+ std::numeric_limits<From>::is_signed !=
72
+ std::numeric_limits<To>::is_signed)>
73
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
74
+ -> To {
75
+ ec = 0;
76
+ using F = std::numeric_limits<From>;
77
+ using T = std::numeric_limits<To>;
78
+ static_assert(F::is_integer, "From must be integral");
79
+ static_assert(T::is_integer, "To must be integral");
80
+
81
+ if (detail::const_check(F::is_signed && !T::is_signed)) {
82
+ // From may be negative, not allowed!
83
+ if (fmt::detail::is_negative(from)) {
84
+ ec = 1;
85
+ return {};
86
+ }
87
+ // From is positive. Can it always fit in To?
88
+ if (detail::const_check(F::digits > T::digits) &&
89
+ from > static_cast<From>(detail::max_value<To>())) {
90
+ ec = 1;
91
+ return {};
92
+ }
93
+ }
94
+
95
+ if (detail::const_check(!F::is_signed && T::is_signed &&
96
+ F::digits >= T::digits) &&
97
+ from > static_cast<From>(detail::max_value<To>())) {
98
+ ec = 1;
99
+ return {};
100
+ }
101
+ return static_cast<To>(from); // Lossless conversion.
102
+ }
103
+
104
+ template <typename To, typename From,
105
+ FMT_ENABLE_IF(std::is_same<From, To>::value)>
106
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
107
+ -> To {
108
+ ec = 0;
109
+ return from;
110
+ } // function
111
+
112
+ // clang-format off
113
+ /**
114
+ * converts From to To if possible, otherwise ec is set.
115
+ *
116
+ * input | output
117
+ * ---------------------------------|---------------
118
+ * NaN | NaN
119
+ * Inf | Inf
120
+ * normal, fits in output | converted (possibly lossy)
121
+ * normal, does not fit in output | ec is set
122
+ * subnormal | best effort
123
+ * -Inf | -Inf
124
+ */
125
+ // clang-format on
126
+ template <typename To, typename From,
127
+ FMT_ENABLE_IF(!std::is_same<From, To>::value)>
128
+ FMT_CONSTEXPR auto safe_float_conversion(const From from, int& ec) -> To {
129
+ ec = 0;
130
+ using T = std::numeric_limits<To>;
131
+ static_assert(std::is_floating_point<From>::value, "From must be floating");
132
+ static_assert(std::is_floating_point<To>::value, "To must be floating");
133
+
134
+ // catch the only happy case
135
+ if (std::isfinite(from)) {
136
+ if (from >= T::lowest() && from <= (T::max)()) {
137
+ return static_cast<To>(from);
138
+ }
139
+ // not within range.
140
+ ec = 1;
141
+ return {};
142
+ }
143
+
144
+ // nan and inf will be preserved
145
+ return static_cast<To>(from);
146
+ } // function
147
+
148
+ template <typename To, typename From,
149
+ FMT_ENABLE_IF(std::is_same<From, To>::value)>
150
+ FMT_CONSTEXPR auto safe_float_conversion(const From from, int& ec) -> To {
151
+ ec = 0;
152
+ static_assert(std::is_floating_point<From>::value, "From must be floating");
153
+ return from;
154
+ }
155
+
156
+ /// Safe duration_cast between floating point durations
157
+ template <typename To, typename FromRep, typename FromPeriod,
158
+ FMT_ENABLE_IF(std::is_floating_point<FromRep>::value),
159
+ FMT_ENABLE_IF(std::is_floating_point<typename To::rep>::value)>
160
+ auto safe_duration_cast(std::chrono::duration<FromRep, FromPeriod> from,
161
+ int& ec) -> To {
162
+ using From = std::chrono::duration<FromRep, FromPeriod>;
163
+ ec = 0;
164
+ if (std::isnan(from.count())) {
165
+ // nan in, gives nan out. easy.
166
+ return To{std::numeric_limits<typename To::rep>::quiet_NaN()};
167
+ }
168
+ // maybe we should also check if from is denormal, and decide what to do about
169
+ // it.
170
+
171
+ // +-inf should be preserved.
172
+ if (std::isinf(from.count())) {
173
+ return To{from.count()};
174
+ }
175
+
176
+ // the basic idea is that we need to convert from count() in the from type
177
+ // to count() in the To type, by multiplying it with this:
178
+ struct Factor
179
+ : std::ratio_divide<typename From::period, typename To::period> {};
180
+
181
+ static_assert(Factor::num > 0, "num must be positive");
182
+ static_assert(Factor::den > 0, "den must be positive");
183
+
184
+ // the conversion is like this: multiply from.count() with Factor::num
185
+ // /Factor::den and convert it to To::rep, all this without
186
+ // overflow/underflow. let's start by finding a suitable type that can hold
187
+ // both To, From and Factor::num
188
+ using IntermediateRep =
189
+ typename std::common_type<typename From::rep, typename To::rep,
190
+ decltype(Factor::num)>::type;
191
+
192
+ // force conversion of From::rep -> IntermediateRep to be safe,
193
+ // even if it will never happen be narrowing in this context.
194
+ IntermediateRep count =
195
+ safe_float_conversion<IntermediateRep>(from.count(), ec);
196
+ if (ec) {
197
+ return {};
198
+ }
199
+
200
+ // multiply with Factor::num without overflow or underflow
201
+ if (detail::const_check(Factor::num != 1)) {
202
+ constexpr auto max1 = detail::max_value<IntermediateRep>() /
203
+ static_cast<IntermediateRep>(Factor::num);
204
+ if (count > max1) {
205
+ ec = 1;
206
+ return {};
207
+ }
208
+ constexpr auto min1 = std::numeric_limits<IntermediateRep>::lowest() /
209
+ static_cast<IntermediateRep>(Factor::num);
210
+ if (count < min1) {
211
+ ec = 1;
212
+ return {};
213
+ }
214
+ count *= static_cast<IntermediateRep>(Factor::num);
215
+ }
216
+
217
+ // this can't go wrong, right? den>0 is checked earlier.
218
+ if (detail::const_check(Factor::den != 1)) {
219
+ using common_t = typename std::common_type<IntermediateRep, intmax_t>::type;
220
+ count /= static_cast<common_t>(Factor::den);
221
+ }
222
+
223
+ // convert to the to type, safely
224
+ using ToRep = typename To::rep;
225
+
226
+ const ToRep tocount = safe_float_conversion<ToRep>(count, ec);
227
+ if (ec) {
228
+ return {};
229
+ }
230
+ return To{tocount};
231
+ }
232
+ } // namespace safe_duration_cast
233
+ #endif
234
+
235
+ namespace detail {
236
+
237
+ // Check if std::chrono::utc_time is available.
238
+ #ifdef FMT_USE_UTC_TIME
239
+ // Use the provided definition.
240
+ #elif defined(__cpp_lib_chrono)
241
+ # define FMT_USE_UTC_TIME (__cpp_lib_chrono >= 201907L)
242
+ #else
243
+ # define FMT_USE_UTC_TIME 0
244
+ #endif
245
+ #if FMT_USE_UTC_TIME
246
+ using utc_clock = std::chrono::utc_clock;
247
+ #else
248
+ struct utc_clock {
249
+ template <typename T> void to_sys(T);
250
+ };
251
+ #endif
252
+
253
+ // Check if std::chrono::local_time is available.
254
+ #ifdef FMT_USE_LOCAL_TIME
255
+ // Use the provided definition.
256
+ #elif defined(__cpp_lib_chrono)
257
+ # define FMT_USE_LOCAL_TIME (__cpp_lib_chrono >= 201907L)
258
+ #else
259
+ # define FMT_USE_LOCAL_TIME 0
260
+ #endif
261
+ #if FMT_USE_LOCAL_TIME
262
+ using local_t = std::chrono::local_t;
263
+ #else
264
+ struct local_t {};
265
+ #endif
266
+
267
+ } // namespace detail
268
+
269
+ template <typename Duration>
270
+ using sys_time = std::chrono::time_point<std::chrono::system_clock, Duration>;
271
+
272
+ template <typename Duration>
273
+ using utc_time = std::chrono::time_point<detail::utc_clock, Duration>;
274
+
275
+ template <class Duration>
276
+ using local_time = std::chrono::time_point<detail::local_t, Duration>;
277
+
278
+ namespace detail {
279
+
280
+ // Prevents expansion of a preceding token as a function-style macro.
281
+ // Usage: f FMT_NOMACRO()
282
+ #define FMT_NOMACRO
283
+
284
+ template <typename T = void> struct null {};
285
+ inline auto localtime_r FMT_NOMACRO(...) -> null<> { return null<>(); }
286
+ inline auto localtime_s(...) -> null<> { return null<>(); }
287
+ inline auto gmtime_r(...) -> null<> { return null<>(); }
288
+ inline auto gmtime_s(...) -> null<> { return null<>(); }
289
+
290
+ // It is defined here and not in ostream.h because the latter has expensive
291
+ // includes.
292
+ template <typename StreamBuf> class formatbuf : public StreamBuf {
293
+ private:
294
+ using char_type = typename StreamBuf::char_type;
295
+ using streamsize = decltype(std::declval<StreamBuf>().sputn(nullptr, 0));
296
+ using int_type = typename StreamBuf::int_type;
297
+ using traits_type = typename StreamBuf::traits_type;
298
+
299
+ buffer<char_type>& buffer_;
300
+
301
+ public:
302
+ explicit formatbuf(buffer<char_type>& buf) : buffer_(buf) {}
303
+
304
+ protected:
305
+ // The put area is always empty. This makes the implementation simpler and has
306
+ // the advantage that the streambuf and the buffer are always in sync and
307
+ // sputc never writes into uninitialized memory. A disadvantage is that each
308
+ // call to sputc always results in a (virtual) call to overflow. There is no
309
+ // disadvantage here for sputn since this always results in a call to xsputn.
310
+
311
+ auto overflow(int_type ch) -> int_type override {
312
+ if (!traits_type::eq_int_type(ch, traits_type::eof()))
313
+ buffer_.push_back(static_cast<char_type>(ch));
314
+ return ch;
315
+ }
316
+
317
+ auto xsputn(const char_type* s, streamsize count) -> streamsize override {
318
+ buffer_.append(s, s + count);
319
+ return count;
320
+ }
321
+ };
322
+
323
+ inline auto get_classic_locale() -> const std::locale& {
324
+ static const auto& locale = std::locale::classic();
325
+ return locale;
326
+ }
327
+
328
+ template <typename CodeUnit> struct codecvt_result {
329
+ static constexpr const size_t max_size = 32;
330
+ CodeUnit buf[max_size];
331
+ CodeUnit* end;
332
+ };
333
+
334
+ template <typename CodeUnit>
335
+ void write_codecvt(codecvt_result<CodeUnit>& out, string_view in,
336
+ const std::locale& loc) {
337
+ FMT_PRAGMA_CLANG(diagnostic push)
338
+ FMT_PRAGMA_CLANG(diagnostic ignored "-Wdeprecated")
339
+ auto& f = std::use_facet<std::codecvt<CodeUnit, char, std::mbstate_t>>(loc);
340
+ FMT_PRAGMA_CLANG(diagnostic pop)
341
+ auto mb = std::mbstate_t();
342
+ const char* from_next = nullptr;
343
+ auto result = f.in(mb, in.begin(), in.end(), from_next, std::begin(out.buf),
344
+ std::end(out.buf), out.end);
345
+ if (result != std::codecvt_base::ok)
346
+ FMT_THROW(format_error("failed to format time"));
347
+ }
348
+
349
+ template <typename OutputIt>
350
+ auto write_encoded_tm_str(OutputIt out, string_view in, const std::locale& loc)
351
+ -> OutputIt {
352
+ if (const_check(detail::use_utf8) && loc != get_classic_locale()) {
353
+ // char16_t and char32_t codecvts are broken in MSVC (linkage errors) and
354
+ // gcc-4.
355
+ #if FMT_MSC_VERSION != 0 || \
356
+ (defined(__GLIBCXX__) && \
357
+ (!defined(_GLIBCXX_USE_DUAL_ABI) || _GLIBCXX_USE_DUAL_ABI == 0))
358
+ // The _GLIBCXX_USE_DUAL_ABI macro is always defined in libstdc++ from gcc-5
359
+ // and newer.
360
+ using code_unit = wchar_t;
361
+ #else
362
+ using code_unit = char32_t;
363
+ #endif
364
+
365
+ using unit_t = codecvt_result<code_unit>;
366
+ unit_t unit;
367
+ write_codecvt(unit, in, loc);
368
+ // In UTF-8 is used one to four one-byte code units.
369
+ auto u =
370
+ to_utf8<code_unit, basic_memory_buffer<char, unit_t::max_size * 4>>();
371
+ if (!u.convert({unit.buf, to_unsigned(unit.end - unit.buf)}))
372
+ FMT_THROW(format_error("failed to format time"));
373
+ return copy<char>(u.c_str(), u.c_str() + u.size(), out);
374
+ }
375
+ return copy<char>(in.data(), in.data() + in.size(), out);
376
+ }
377
+
378
+ template <typename Char, typename OutputIt,
379
+ FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
380
+ auto write_tm_str(OutputIt out, string_view sv, const std::locale& loc)
381
+ -> OutputIt {
382
+ codecvt_result<Char> unit;
383
+ write_codecvt(unit, sv, loc);
384
+ return copy<Char>(unit.buf, unit.end, out);
385
+ }
386
+
387
+ template <typename Char, typename OutputIt,
388
+ FMT_ENABLE_IF(std::is_same<Char, char>::value)>
389
+ auto write_tm_str(OutputIt out, string_view sv, const std::locale& loc)
390
+ -> OutputIt {
391
+ return write_encoded_tm_str(out, sv, loc);
392
+ }
393
+
394
+ template <typename Char>
395
+ inline void do_write(buffer<Char>& buf, const std::tm& time,
396
+ const std::locale& loc, char format, char modifier) {
397
+ auto&& format_buf = formatbuf<std::basic_streambuf<Char>>(buf);
398
+ auto&& os = std::basic_ostream<Char>(&format_buf);
399
+ os.imbue(loc);
400
+ const auto& facet = std::use_facet<std::time_put<Char>>(loc);
401
+ auto end = facet.put(os, os, Char(' '), &time, format, modifier);
402
+ if (end.failed()) FMT_THROW(format_error("failed to format time"));
403
+ }
404
+
405
+ template <typename Char, typename OutputIt,
406
+ FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
407
+ auto write(OutputIt out, const std::tm& time, const std::locale& loc,
408
+ char format, char modifier = 0) -> OutputIt {
409
+ auto&& buf = get_buffer<Char>(out);
410
+ do_write<Char>(buf, time, loc, format, modifier);
411
+ return get_iterator(buf, out);
412
+ }
413
+
414
+ template <typename Char, typename OutputIt,
415
+ FMT_ENABLE_IF(std::is_same<Char, char>::value)>
416
+ auto write(OutputIt out, const std::tm& time, const std::locale& loc,
417
+ char format, char modifier = 0) -> OutputIt {
418
+ auto&& buf = basic_memory_buffer<Char>();
419
+ do_write<char>(buf, time, loc, format, modifier);
420
+ return write_encoded_tm_str(out, string_view(buf.data(), buf.size()), loc);
421
+ }
422
+
423
+ template <typename T, typename U>
424
+ using is_similar_arithmetic_type =
425
+ bool_constant<(std::is_integral<T>::value && std::is_integral<U>::value) ||
426
+ (std::is_floating_point<T>::value &&
427
+ std::is_floating_point<U>::value)>;
428
+
429
+ FMT_NORETURN inline void throw_duration_error() {
430
+ FMT_THROW(format_error("cannot format duration"));
431
+ }
432
+
433
+ // Cast one integral duration to another with an overflow check.
434
+ template <typename To, typename FromRep, typename FromPeriod,
435
+ FMT_ENABLE_IF(std::is_integral<FromRep>::value&&
436
+ std::is_integral<typename To::rep>::value)>
437
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
438
+ #if !FMT_SAFE_DURATION_CAST
439
+ return std::chrono::duration_cast<To>(from);
440
+ #else
441
+ // The conversion factor: to.count() == factor * from.count().
442
+ using factor = std::ratio_divide<FromPeriod, typename To::period>;
443
+
444
+ using common_rep = typename std::common_type<FromRep, typename To::rep,
445
+ decltype(factor::num)>::type;
446
+
447
+ int ec = 0;
448
+ auto count = safe_duration_cast::lossless_integral_conversion<common_rep>(
449
+ from.count(), ec);
450
+ if (ec) throw_duration_error();
451
+
452
+ // Multiply from.count() by factor and check for overflow.
453
+ if (const_check(factor::num != 1)) {
454
+ if (count > max_value<common_rep>() / factor::num) throw_duration_error();
455
+ const auto min = (std::numeric_limits<common_rep>::min)() / factor::num;
456
+ if (const_check(!std::is_unsigned<common_rep>::value) && count < min)
457
+ throw_duration_error();
458
+ count *= factor::num;
459
+ }
460
+ if (const_check(factor::den != 1)) count /= factor::den;
461
+ auto to =
462
+ To(safe_duration_cast::lossless_integral_conversion<typename To::rep>(
463
+ count, ec));
464
+ if (ec) throw_duration_error();
465
+ return to;
466
+ #endif
467
+ }
468
+
469
+ template <typename To, typename FromRep, typename FromPeriod,
470
+ FMT_ENABLE_IF(std::is_floating_point<FromRep>::value&&
471
+ std::is_floating_point<typename To::rep>::value)>
472
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
473
+ #if FMT_SAFE_DURATION_CAST
474
+ // Throwing version of safe_duration_cast is only available for
475
+ // integer to integer or float to float casts.
476
+ int ec;
477
+ To to = safe_duration_cast::safe_duration_cast<To>(from, ec);
478
+ if (ec) throw_duration_error();
479
+ return to;
480
+ #else
481
+ // Standard duration cast, may overflow.
482
+ return std::chrono::duration_cast<To>(from);
483
+ #endif
484
+ }
485
+
486
+ template <typename To, typename FromRep, typename FromPeriod,
487
+ FMT_ENABLE_IF(
488
+ !is_similar_arithmetic_type<FromRep, typename To::rep>::value)>
489
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
490
+ // Mixed integer <-> float cast is not supported by safe_duration_cast.
491
+ return std::chrono::duration_cast<To>(from);
492
+ }
493
+
494
+ template <typename Duration>
495
+ auto to_time_t(sys_time<Duration> time_point) -> std::time_t {
496
+ // Cannot use std::chrono::system_clock::to_time_t since this would first
497
+ // require a cast to std::chrono::system_clock::time_point, which could
498
+ // overflow.
499
+ return detail::duration_cast<std::chrono::duration<std::time_t>>(
500
+ time_point.time_since_epoch())
501
+ .count();
502
+ }
503
+
504
+ namespace tz {
505
+
506
+ // DEPRECATED!
507
+ struct time_zone {
508
+ template <typename Duration, typename LocalTime>
509
+ auto to_sys(LocalTime) -> sys_time<Duration> {
510
+ return {};
511
+ }
512
+ };
513
+ template <typename... T> auto current_zone(T...) -> time_zone* {
514
+ return nullptr;
515
+ }
516
+
517
+ template <typename... T> void _tzset(T...) {}
518
+ } // namespace tz
519
+
520
+ // DEPRECATED!
521
+ inline void tzset_once() {
522
+ static bool init = []() {
523
+ using namespace tz;
524
+ _tzset();
525
+ return false;
526
+ }();
527
+ ignore_unused(init);
528
+ }
529
+ } // namespace detail
530
+
531
+ FMT_BEGIN_EXPORT
532
+
533
+ /**
534
+ * Converts given time since epoch as `std::time_t` value into calendar time,
535
+ * expressed in local time. Unlike `std::localtime`, this function is
536
+ * thread-safe on most platforms.
537
+ */
538
+ FMT_DEPRECATED inline auto localtime(std::time_t time) -> std::tm {
539
+ struct dispatcher {
540
+ std::time_t time_;
541
+ std::tm tm_;
542
+
543
+ inline dispatcher(std::time_t t) : time_(t) {}
544
+
545
+ inline auto run() -> bool {
546
+ using namespace fmt::detail;
547
+ return handle(localtime_r(&time_, &tm_));
548
+ }
549
+
550
+ inline auto handle(std::tm* tm) -> bool { return tm != nullptr; }
551
+
552
+ inline auto handle(detail::null<>) -> bool {
553
+ using namespace fmt::detail;
554
+ return fallback(localtime_s(&tm_, &time_));
555
+ }
556
+
557
+ inline auto fallback(int res) -> bool { return res == 0; }
558
+
559
+ #if !FMT_MSC_VERSION
560
+ inline auto fallback(detail::null<>) -> bool {
561
+ using namespace fmt::detail;
562
+ std::tm* tm = std::localtime(&time_);
563
+ if (tm) tm_ = *tm;
564
+ return tm != nullptr;
565
+ }
566
+ #endif
567
+ };
568
+ dispatcher lt(time);
569
+ // Too big time values may be unsupported.
570
+ if (!lt.run()) FMT_THROW(format_error("time_t value out of range"));
571
+ return lt.tm_;
572
+ }
573
+
574
+ #if FMT_USE_LOCAL_TIME
575
+ template <typename Duration>
576
+ FMT_DEPRECATED auto localtime(std::chrono::local_time<Duration> time)
577
+ -> std::tm {
578
+ using namespace std::chrono;
579
+ using namespace detail::tz;
580
+ return localtime(detail::to_time_t(current_zone()->to_sys<Duration>(time)));
581
+ }
582
+ #endif
583
+
584
+ /**
585
+ * Converts given time since epoch as `std::time_t` value into calendar time,
586
+ * expressed in Coordinated Universal Time (UTC). Unlike `std::gmtime`, this
587
+ * function is thread-safe on most platforms.
588
+ */
589
+ inline auto gmtime(std::time_t time) -> std::tm {
590
+ struct dispatcher {
591
+ std::time_t time_;
592
+ std::tm tm_;
593
+
594
+ inline dispatcher(std::time_t t) : time_(t) {}
595
+
596
+ inline auto run() -> bool {
597
+ using namespace fmt::detail;
598
+ return handle(gmtime_r(&time_, &tm_));
599
+ }
600
+
601
+ inline auto handle(std::tm* tm) -> bool { return tm != nullptr; }
602
+
603
+ inline auto handle(detail::null<>) -> bool {
604
+ using namespace fmt::detail;
605
+ return fallback(gmtime_s(&tm_, &time_));
606
+ }
607
+
608
+ inline auto fallback(int res) -> bool { return res == 0; }
609
+
610
+ #if !FMT_MSC_VERSION
611
+ inline auto fallback(detail::null<>) -> bool {
612
+ std::tm* tm = std::gmtime(&time_);
613
+ if (tm) tm_ = *tm;
614
+ return tm != nullptr;
615
+ }
616
+ #endif
617
+ };
618
+ auto gt = dispatcher(time);
619
+ // Too big time values may be unsupported.
620
+ if (!gt.run()) FMT_THROW(format_error("time_t value out of range"));
621
+ return gt.tm_;
622
+ }
623
+
624
+ template <typename Duration>
625
+ inline auto gmtime(sys_time<Duration> time_point) -> std::tm {
626
+ return gmtime(detail::to_time_t(time_point));
627
+ }
628
+
629
+ namespace detail {
630
+
631
+ // Writes two-digit numbers a, b and c separated by sep to buf.
632
+ // The method by Pavel Novikov based on
633
+ // https://johnnylee-sde.github.io/Fast-unsigned-integer-to-time-string/.
634
+ inline void write_digit2_separated(char* buf, unsigned a, unsigned b,
635
+ unsigned c, char sep) {
636
+ unsigned long long digits =
637
+ a | (b << 24) | (static_cast<unsigned long long>(c) << 48);
638
+ // Convert each value to BCD.
639
+ // We have x = a * 10 + b and we want to convert it to BCD y = a * 16 + b.
640
+ // The difference is
641
+ // y - x = a * 6
642
+ // a can be found from x:
643
+ // a = floor(x / 10)
644
+ // then
645
+ // y = x + a * 6 = x + floor(x / 10) * 6
646
+ // floor(x / 10) is (x * 205) >> 11 (needs 16 bits).
647
+ digits += (((digits * 205) >> 11) & 0x000f00000f00000f) * 6;
648
+ // Put low nibbles to high bytes and high nibbles to low bytes.
649
+ digits = ((digits & 0x00f00000f00000f0) >> 4) |
650
+ ((digits & 0x000f00000f00000f) << 8);
651
+ auto usep = static_cast<unsigned long long>(sep);
652
+ // Add ASCII '0' to each digit byte and insert separators.
653
+ digits |= 0x3030003030003030 | (usep << 16) | (usep << 40);
654
+
655
+ constexpr const size_t len = 8;
656
+ if (const_check(is_big_endian())) {
657
+ char tmp[len];
658
+ std::memcpy(tmp, &digits, len);
659
+ std::reverse_copy(tmp, tmp + len, buf);
660
+ } else {
661
+ std::memcpy(buf, &digits, len);
662
+ }
663
+ }
664
+
665
+ template <typename Period>
666
+ FMT_CONSTEXPR inline auto get_units() -> const char* {
667
+ if (std::is_same<Period, std::atto>::value) return "as";
668
+ if (std::is_same<Period, std::femto>::value) return "fs";
669
+ if (std::is_same<Period, std::pico>::value) return "ps";
670
+ if (std::is_same<Period, std::nano>::value) return "ns";
671
+ if (std::is_same<Period, std::micro>::value)
672
+ return detail::use_utf8 ? "µs" : "us";
673
+ if (std::is_same<Period, std::milli>::value) return "ms";
674
+ if (std::is_same<Period, std::centi>::value) return "cs";
675
+ if (std::is_same<Period, std::deci>::value) return "ds";
676
+ if (std::is_same<Period, std::ratio<1>>::value) return "s";
677
+ if (std::is_same<Period, std::deca>::value) return "das";
678
+ if (std::is_same<Period, std::hecto>::value) return "hs";
679
+ if (std::is_same<Period, std::kilo>::value) return "ks";
680
+ if (std::is_same<Period, std::mega>::value) return "Ms";
681
+ if (std::is_same<Period, std::giga>::value) return "Gs";
682
+ if (std::is_same<Period, std::tera>::value) return "Ts";
683
+ if (std::is_same<Period, std::peta>::value) return "Ps";
684
+ if (std::is_same<Period, std::exa>::value) return "Es";
685
+ if (std::is_same<Period, std::ratio<60>>::value) return "min";
686
+ if (std::is_same<Period, std::ratio<3600>>::value) return "h";
687
+ if (std::is_same<Period, std::ratio<86400>>::value) return "d";
688
+ return nullptr;
689
+ }
690
+
691
+ enum class numeric_system {
692
+ standard,
693
+ // Alternative numeric system, e.g. 十二 instead of 12 in ja_JP locale.
694
+ alternative
695
+ };
696
+
697
+ // Glibc extensions for formatting numeric values.
698
+ enum class pad_type {
699
+ // Pad a numeric result string with zeros (the default).
700
+ zero,
701
+ // Do not pad a numeric result string.
702
+ none,
703
+ // Pad a numeric result string with spaces.
704
+ space,
705
+ };
706
+
707
+ template <typename OutputIt>
708
+ auto write_padding(OutputIt out, pad_type pad, int width) -> OutputIt {
709
+ if (pad == pad_type::none) return out;
710
+ return detail::fill_n(out, width, pad == pad_type::space ? ' ' : '0');
711
+ }
712
+
713
+ template <typename OutputIt>
714
+ auto write_padding(OutputIt out, pad_type pad) -> OutputIt {
715
+ if (pad != pad_type::none) *out++ = pad == pad_type::space ? ' ' : '0';
716
+ return out;
717
+ }
718
+
719
+ // Parses a put_time-like format string and invokes handler actions.
720
+ template <typename Char, typename Handler>
721
+ FMT_CONSTEXPR auto parse_chrono_format(const Char* begin, const Char* end,
722
+ Handler&& handler) -> const Char* {
723
+ if (begin == end || *begin == '}') return begin;
724
+ if (*begin != '%') FMT_THROW(format_error("invalid format"));
725
+ auto ptr = begin;
726
+ while (ptr != end) {
727
+ pad_type pad = pad_type::zero;
728
+ auto c = *ptr;
729
+ if (c == '}') break;
730
+ if (c != '%') {
731
+ ++ptr;
732
+ continue;
733
+ }
734
+ if (begin != ptr) handler.on_text(begin, ptr);
735
+ ++ptr; // consume '%'
736
+ if (ptr == end) FMT_THROW(format_error("invalid format"));
737
+ c = *ptr;
738
+ switch (c) {
739
+ case '_':
740
+ pad = pad_type::space;
741
+ ++ptr;
742
+ break;
743
+ case '-':
744
+ pad = pad_type::none;
745
+ ++ptr;
746
+ break;
747
+ }
748
+ if (ptr == end) FMT_THROW(format_error("invalid format"));
749
+ c = *ptr++;
750
+ switch (c) {
751
+ case '%': handler.on_text(ptr - 1, ptr); break;
752
+ case 'n': {
753
+ const Char newline[] = {'\n'};
754
+ handler.on_text(newline, newline + 1);
755
+ break;
756
+ }
757
+ case 't': {
758
+ const Char tab[] = {'\t'};
759
+ handler.on_text(tab, tab + 1);
760
+ break;
761
+ }
762
+ // Year:
763
+ case 'Y': handler.on_year(numeric_system::standard, pad); break;
764
+ case 'y': handler.on_short_year(numeric_system::standard); break;
765
+ case 'C': handler.on_century(numeric_system::standard); break;
766
+ case 'G': handler.on_iso_week_based_year(); break;
767
+ case 'g': handler.on_iso_week_based_short_year(); break;
768
+ // Day of the week:
769
+ case 'a': handler.on_abbr_weekday(); break;
770
+ case 'A': handler.on_full_weekday(); break;
771
+ case 'w': handler.on_dec0_weekday(numeric_system::standard); break;
772
+ case 'u': handler.on_dec1_weekday(numeric_system::standard); break;
773
+ // Month:
774
+ case 'b':
775
+ case 'h': handler.on_abbr_month(); break;
776
+ case 'B': handler.on_full_month(); break;
777
+ case 'm': handler.on_dec_month(numeric_system::standard, pad); break;
778
+ // Day of the year/month:
779
+ case 'U':
780
+ handler.on_dec0_week_of_year(numeric_system::standard, pad);
781
+ break;
782
+ case 'W':
783
+ handler.on_dec1_week_of_year(numeric_system::standard, pad);
784
+ break;
785
+ case 'V': handler.on_iso_week_of_year(numeric_system::standard, pad); break;
786
+ case 'j': handler.on_day_of_year(pad); break;
787
+ case 'd': handler.on_day_of_month(numeric_system::standard, pad); break;
788
+ case 'e':
789
+ handler.on_day_of_month(numeric_system::standard, pad_type::space);
790
+ break;
791
+ // Hour, minute, second:
792
+ case 'H': handler.on_24_hour(numeric_system::standard, pad); break;
793
+ case 'I': handler.on_12_hour(numeric_system::standard, pad); break;
794
+ case 'M': handler.on_minute(numeric_system::standard, pad); break;
795
+ case 'S': handler.on_second(numeric_system::standard, pad); break;
796
+ // Other:
797
+ case 'c': handler.on_datetime(numeric_system::standard); break;
798
+ case 'x': handler.on_loc_date(numeric_system::standard); break;
799
+ case 'X': handler.on_loc_time(numeric_system::standard); break;
800
+ case 'D': handler.on_us_date(); break;
801
+ case 'F': handler.on_iso_date(); break;
802
+ case 'r': handler.on_12_hour_time(); break;
803
+ case 'R': handler.on_24_hour_time(); break;
804
+ case 'T': handler.on_iso_time(); break;
805
+ case 'p': handler.on_am_pm(); break;
806
+ case 'Q': handler.on_duration_value(); break;
807
+ case 'q': handler.on_duration_unit(); break;
808
+ case 'z': handler.on_utc_offset(numeric_system::standard); break;
809
+ case 'Z': handler.on_tz_name(); break;
810
+ // Alternative representation:
811
+ case 'E': {
812
+ if (ptr == end) FMT_THROW(format_error("invalid format"));
813
+ c = *ptr++;
814
+ switch (c) {
815
+ case 'Y': handler.on_year(numeric_system::alternative, pad); break;
816
+ case 'y': handler.on_offset_year(); break;
817
+ case 'C': handler.on_century(numeric_system::alternative); break;
818
+ case 'c': handler.on_datetime(numeric_system::alternative); break;
819
+ case 'x': handler.on_loc_date(numeric_system::alternative); break;
820
+ case 'X': handler.on_loc_time(numeric_system::alternative); break;
821
+ case 'z': handler.on_utc_offset(numeric_system::alternative); break;
822
+ default: FMT_THROW(format_error("invalid format"));
823
+ }
824
+ break;
825
+ }
826
+ case 'O':
827
+ if (ptr == end) FMT_THROW(format_error("invalid format"));
828
+ c = *ptr++;
829
+ switch (c) {
830
+ case 'y': handler.on_short_year(numeric_system::alternative); break;
831
+ case 'm': handler.on_dec_month(numeric_system::alternative, pad); break;
832
+ case 'U':
833
+ handler.on_dec0_week_of_year(numeric_system::alternative, pad);
834
+ break;
835
+ case 'W':
836
+ handler.on_dec1_week_of_year(numeric_system::alternative, pad);
837
+ break;
838
+ case 'V':
839
+ handler.on_iso_week_of_year(numeric_system::alternative, pad);
840
+ break;
841
+ case 'd':
842
+ handler.on_day_of_month(numeric_system::alternative, pad);
843
+ break;
844
+ case 'e':
845
+ handler.on_day_of_month(numeric_system::alternative, pad_type::space);
846
+ break;
847
+ case 'w': handler.on_dec0_weekday(numeric_system::alternative); break;
848
+ case 'u': handler.on_dec1_weekday(numeric_system::alternative); break;
849
+ case 'H': handler.on_24_hour(numeric_system::alternative, pad); break;
850
+ case 'I': handler.on_12_hour(numeric_system::alternative, pad); break;
851
+ case 'M': handler.on_minute(numeric_system::alternative, pad); break;
852
+ case 'S': handler.on_second(numeric_system::alternative, pad); break;
853
+ case 'z': handler.on_utc_offset(numeric_system::alternative); break;
854
+ default: FMT_THROW(format_error("invalid format"));
855
+ }
856
+ break;
857
+ default: FMT_THROW(format_error("invalid format"));
858
+ }
859
+ begin = ptr;
860
+ }
861
+ if (begin != ptr) handler.on_text(begin, ptr);
862
+ return ptr;
863
+ }
864
+
865
+ template <typename Derived> struct null_chrono_spec_handler {
866
+ FMT_CONSTEXPR void unsupported() {
867
+ static_cast<Derived*>(this)->unsupported();
868
+ }
869
+ FMT_CONSTEXPR void on_year(numeric_system, pad_type) { unsupported(); }
870
+ FMT_CONSTEXPR void on_short_year(numeric_system) { unsupported(); }
871
+ FMT_CONSTEXPR void on_offset_year() { unsupported(); }
872
+ FMT_CONSTEXPR void on_century(numeric_system) { unsupported(); }
873
+ FMT_CONSTEXPR void on_iso_week_based_year() { unsupported(); }
874
+ FMT_CONSTEXPR void on_iso_week_based_short_year() { unsupported(); }
875
+ FMT_CONSTEXPR void on_abbr_weekday() { unsupported(); }
876
+ FMT_CONSTEXPR void on_full_weekday() { unsupported(); }
877
+ FMT_CONSTEXPR void on_dec0_weekday(numeric_system) { unsupported(); }
878
+ FMT_CONSTEXPR void on_dec1_weekday(numeric_system) { unsupported(); }
879
+ FMT_CONSTEXPR void on_abbr_month() { unsupported(); }
880
+ FMT_CONSTEXPR void on_full_month() { unsupported(); }
881
+ FMT_CONSTEXPR void on_dec_month(numeric_system, pad_type) { unsupported(); }
882
+ FMT_CONSTEXPR void on_dec0_week_of_year(numeric_system, pad_type) {
883
+ unsupported();
884
+ }
885
+ FMT_CONSTEXPR void on_dec1_week_of_year(numeric_system, pad_type) {
886
+ unsupported();
887
+ }
888
+ FMT_CONSTEXPR void on_iso_week_of_year(numeric_system, pad_type) {
889
+ unsupported();
890
+ }
891
+ FMT_CONSTEXPR void on_day_of_year(pad_type) { unsupported(); }
892
+ FMT_CONSTEXPR void on_day_of_month(numeric_system, pad_type) {
893
+ unsupported();
894
+ }
895
+ FMT_CONSTEXPR void on_24_hour(numeric_system) { unsupported(); }
896
+ FMT_CONSTEXPR void on_12_hour(numeric_system) { unsupported(); }
897
+ FMT_CONSTEXPR void on_minute(numeric_system) { unsupported(); }
898
+ FMT_CONSTEXPR void on_second(numeric_system) { unsupported(); }
899
+ FMT_CONSTEXPR void on_datetime(numeric_system) { unsupported(); }
900
+ FMT_CONSTEXPR void on_loc_date(numeric_system) { unsupported(); }
901
+ FMT_CONSTEXPR void on_loc_time(numeric_system) { unsupported(); }
902
+ FMT_CONSTEXPR void on_us_date() { unsupported(); }
903
+ FMT_CONSTEXPR void on_iso_date() { unsupported(); }
904
+ FMT_CONSTEXPR void on_12_hour_time() { unsupported(); }
905
+ FMT_CONSTEXPR void on_24_hour_time() { unsupported(); }
906
+ FMT_CONSTEXPR void on_iso_time() { unsupported(); }
907
+ FMT_CONSTEXPR void on_am_pm() { unsupported(); }
908
+ FMT_CONSTEXPR void on_duration_value() { unsupported(); }
909
+ FMT_CONSTEXPR void on_duration_unit() { unsupported(); }
910
+ FMT_CONSTEXPR void on_utc_offset(numeric_system) { unsupported(); }
911
+ FMT_CONSTEXPR void on_tz_name() { unsupported(); }
912
+ };
913
+
914
+ class tm_format_checker : public null_chrono_spec_handler<tm_format_checker> {
915
+ private:
916
+ bool has_timezone_ = false;
917
+
918
+ public:
919
+ constexpr explicit tm_format_checker(bool has_timezone)
920
+ : has_timezone_(has_timezone) {}
921
+
922
+ FMT_NORETURN inline void unsupported() {
923
+ FMT_THROW(format_error("no format"));
924
+ }
925
+
926
+ template <typename Char>
927
+ FMT_CONSTEXPR void on_text(const Char*, const Char*) {}
928
+ FMT_CONSTEXPR void on_year(numeric_system, pad_type) {}
929
+ FMT_CONSTEXPR void on_short_year(numeric_system) {}
930
+ FMT_CONSTEXPR void on_offset_year() {}
931
+ FMT_CONSTEXPR void on_century(numeric_system) {}
932
+ FMT_CONSTEXPR void on_iso_week_based_year() {}
933
+ FMT_CONSTEXPR void on_iso_week_based_short_year() {}
934
+ FMT_CONSTEXPR void on_abbr_weekday() {}
935
+ FMT_CONSTEXPR void on_full_weekday() {}
936
+ FMT_CONSTEXPR void on_dec0_weekday(numeric_system) {}
937
+ FMT_CONSTEXPR void on_dec1_weekday(numeric_system) {}
938
+ FMT_CONSTEXPR void on_abbr_month() {}
939
+ FMT_CONSTEXPR void on_full_month() {}
940
+ FMT_CONSTEXPR void on_dec_month(numeric_system, pad_type) {}
941
+ FMT_CONSTEXPR void on_dec0_week_of_year(numeric_system, pad_type) {}
942
+ FMT_CONSTEXPR void on_dec1_week_of_year(numeric_system, pad_type) {}
943
+ FMT_CONSTEXPR void on_iso_week_of_year(numeric_system, pad_type) {}
944
+ FMT_CONSTEXPR void on_day_of_year(pad_type) {}
945
+ FMT_CONSTEXPR void on_day_of_month(numeric_system, pad_type) {}
946
+ FMT_CONSTEXPR void on_24_hour(numeric_system, pad_type) {}
947
+ FMT_CONSTEXPR void on_12_hour(numeric_system, pad_type) {}
948
+ FMT_CONSTEXPR void on_minute(numeric_system, pad_type) {}
949
+ FMT_CONSTEXPR void on_second(numeric_system, pad_type) {}
950
+ FMT_CONSTEXPR void on_datetime(numeric_system) {}
951
+ FMT_CONSTEXPR void on_loc_date(numeric_system) {}
952
+ FMT_CONSTEXPR void on_loc_time(numeric_system) {}
953
+ FMT_CONSTEXPR void on_us_date() {}
954
+ FMT_CONSTEXPR void on_iso_date() {}
955
+ FMT_CONSTEXPR void on_12_hour_time() {}
956
+ FMT_CONSTEXPR void on_24_hour_time() {}
957
+ FMT_CONSTEXPR void on_iso_time() {}
958
+ FMT_CONSTEXPR void on_am_pm() {}
959
+ FMT_CONSTEXPR void on_utc_offset(numeric_system) {
960
+ if (!has_timezone_) FMT_THROW(format_error("no timezone"));
961
+ }
962
+ FMT_CONSTEXPR void on_tz_name() {
963
+ if (!has_timezone_) FMT_THROW(format_error("no timezone"));
964
+ }
965
+ };
966
+
967
+ inline auto tm_wday_full_name(int wday) -> const char* {
968
+ static constexpr const char* full_name_list[] = {
969
+ "Sunday", "Monday", "Tuesday", "Wednesday",
970
+ "Thursday", "Friday", "Saturday"};
971
+ return wday >= 0 && wday <= 6 ? full_name_list[wday] : "?";
972
+ }
973
+ inline auto tm_wday_short_name(int wday) -> const char* {
974
+ static constexpr const char* short_name_list[] = {"Sun", "Mon", "Tue", "Wed",
975
+ "Thu", "Fri", "Sat"};
976
+ return wday >= 0 && wday <= 6 ? short_name_list[wday] : "???";
977
+ }
978
+
979
+ inline auto tm_mon_full_name(int mon) -> const char* {
980
+ static constexpr const char* full_name_list[] = {
981
+ "January", "February", "March", "April", "May", "June",
982
+ "July", "August", "September", "October", "November", "December"};
983
+ return mon >= 0 && mon <= 11 ? full_name_list[mon] : "?";
984
+ }
985
+ inline auto tm_mon_short_name(int mon) -> const char* {
986
+ static constexpr const char* short_name_list[] = {
987
+ "Jan", "Feb", "Mar", "Apr", "May", "Jun",
988
+ "Jul", "Aug", "Sep", "Oct", "Nov", "Dec",
989
+ };
990
+ return mon >= 0 && mon <= 11 ? short_name_list[mon] : "???";
991
+ }
992
+
993
+ template <typename T, typename = void>
994
+ struct has_tm_gmtoff : std::false_type {};
995
+ template <typename T>
996
+ struct has_tm_gmtoff<T, void_t<decltype(T::tm_gmtoff)>> : std::true_type {};
997
+
998
+ template <typename T, typename = void> struct has_tm_zone : std::false_type {};
999
+ template <typename T>
1000
+ struct has_tm_zone<T, void_t<decltype(T::tm_zone)>> : std::true_type {};
1001
+
1002
+ template <typename T, FMT_ENABLE_IF(has_tm_zone<T>::value)>
1003
+ bool set_tm_zone(T& time, char* tz) {
1004
+ time.tm_zone = tz;
1005
+ return true;
1006
+ }
1007
+ template <typename T, FMT_ENABLE_IF(!has_tm_zone<T>::value)>
1008
+ bool set_tm_zone(T&, char*) {
1009
+ return false;
1010
+ }
1011
+
1012
+ inline char* utc() {
1013
+ static char tz[] = "UTC";
1014
+ return tz;
1015
+ }
1016
+
1017
+ // Converts value to Int and checks that it's in the range [0, upper).
1018
+ template <typename T, typename Int, FMT_ENABLE_IF(std::is_integral<T>::value)>
1019
+ inline auto to_nonnegative_int(T value, Int upper) -> Int {
1020
+ if (!std::is_unsigned<Int>::value &&
1021
+ (value < 0 || to_unsigned(value) > to_unsigned(upper))) {
1022
+ FMT_THROW(format_error("chrono value is out of range"));
1023
+ }
1024
+ return static_cast<Int>(value);
1025
+ }
1026
+ template <typename T, typename Int, FMT_ENABLE_IF(!std::is_integral<T>::value)>
1027
+ inline auto to_nonnegative_int(T value, Int upper) -> Int {
1028
+ auto int_value = static_cast<Int>(value);
1029
+ if (int_value < 0 || value > static_cast<T>(upper))
1030
+ FMT_THROW(format_error("invalid value"));
1031
+ return int_value;
1032
+ }
1033
+
1034
+ constexpr auto pow10(std::uint32_t n) -> long long {
1035
+ return n == 0 ? 1 : 10 * pow10(n - 1);
1036
+ }
1037
+
1038
+ // Counts the number of fractional digits in the range [0, 18] according to the
1039
+ // C++20 spec. If more than 18 fractional digits are required then returns 6 for
1040
+ // microseconds precision.
1041
+ template <long long Num, long long Den, int N = 0,
1042
+ bool Enabled = (N < 19) && (Num <= max_value<long long>() / 10)>
1043
+ struct count_fractional_digits {
1044
+ static constexpr int value =
1045
+ Num % Den == 0 ? N : count_fractional_digits<Num * 10, Den, N + 1>::value;
1046
+ };
1047
+
1048
+ // Base case that doesn't instantiate any more templates
1049
+ // in order to avoid overflow.
1050
+ template <long long Num, long long Den, int N>
1051
+ struct count_fractional_digits<Num, Den, N, false> {
1052
+ static constexpr int value = (Num % Den == 0) ? N : 6;
1053
+ };
1054
+
1055
+ // Format subseconds which are given as an integer type with an appropriate
1056
+ // number of digits.
1057
+ template <typename Char, typename OutputIt, typename Duration>
1058
+ void write_fractional_seconds(OutputIt& out, Duration d, int precision = -1) {
1059
+ constexpr auto num_fractional_digits =
1060
+ count_fractional_digits<Duration::period::num,
1061
+ Duration::period::den>::value;
1062
+
1063
+ using subsecond_precision = std::chrono::duration<
1064
+ typename std::common_type<typename Duration::rep,
1065
+ std::chrono::seconds::rep>::type,
1066
+ std::ratio<1, pow10(num_fractional_digits)>>;
1067
+
1068
+ const auto fractional = d - detail::duration_cast<std::chrono::seconds>(d);
1069
+ const auto subseconds =
1070
+ std::chrono::treat_as_floating_point<
1071
+ typename subsecond_precision::rep>::value
1072
+ ? fractional.count()
1073
+ : detail::duration_cast<subsecond_precision>(fractional).count();
1074
+ auto n = static_cast<uint32_or_64_or_128_t<long long>>(subseconds);
1075
+ const int num_digits = count_digits(n);
1076
+
1077
+ int leading_zeroes = (std::max)(0, num_fractional_digits - num_digits);
1078
+ if (precision < 0) {
1079
+ FMT_ASSERT(!std::is_floating_point<typename Duration::rep>::value, "");
1080
+ if (std::ratio_less<typename subsecond_precision::period,
1081
+ std::chrono::seconds::period>::value) {
1082
+ *out++ = '.';
1083
+ out = detail::fill_n(out, leading_zeroes, '0');
1084
+ out = format_decimal<Char>(out, n, num_digits);
1085
+ }
1086
+ } else if (precision > 0) {
1087
+ *out++ = '.';
1088
+ leading_zeroes = min_of(leading_zeroes, precision);
1089
+ int remaining = precision - leading_zeroes;
1090
+ out = detail::fill_n(out, leading_zeroes, '0');
1091
+ if (remaining < num_digits) {
1092
+ int num_truncated_digits = num_digits - remaining;
1093
+ n /= to_unsigned(pow10(to_unsigned(num_truncated_digits)));
1094
+ if (n != 0) out = format_decimal<Char>(out, n, remaining);
1095
+ return;
1096
+ }
1097
+ if (n != 0) {
1098
+ out = format_decimal<Char>(out, n, num_digits);
1099
+ remaining -= num_digits;
1100
+ }
1101
+ out = detail::fill_n(out, remaining, '0');
1102
+ }
1103
+ }
1104
+
1105
+ // Format subseconds which are given as a floating point type with an
1106
+ // appropriate number of digits. We cannot pass the Duration here, as we
1107
+ // explicitly need to pass the Rep value in the duration_formatter.
1108
+ template <typename Duration>
1109
+ void write_floating_seconds(memory_buffer& buf, Duration duration,
1110
+ int num_fractional_digits = -1) {
1111
+ using rep = typename Duration::rep;
1112
+ FMT_ASSERT(std::is_floating_point<rep>::value, "");
1113
+
1114
+ auto val = duration.count();
1115
+
1116
+ if (num_fractional_digits < 0) {
1117
+ // For `std::round` with fallback to `round`:
1118
+ // On some toolchains `std::round` is not available (e.g. GCC 6).
1119
+ using namespace std;
1120
+ num_fractional_digits =
1121
+ count_fractional_digits<Duration::period::num,
1122
+ Duration::period::den>::value;
1123
+ if (num_fractional_digits < 6 && static_cast<rep>(round(val)) != val)
1124
+ num_fractional_digits = 6;
1125
+ }
1126
+
1127
+ fmt::format_to(std::back_inserter(buf), FMT_STRING("{:.{}f}"),
1128
+ std::fmod(val * static_cast<rep>(Duration::period::num) /
1129
+ static_cast<rep>(Duration::period::den),
1130
+ static_cast<rep>(60)),
1131
+ num_fractional_digits);
1132
+ }
1133
+
1134
+ template <typename OutputIt, typename Char,
1135
+ typename Duration = std::chrono::seconds>
1136
+ class tm_writer {
1137
+ private:
1138
+ static constexpr int days_per_week = 7;
1139
+
1140
+ const std::locale& loc_;
1141
+ bool is_classic_;
1142
+ OutputIt out_;
1143
+ const Duration* subsecs_;
1144
+ const std::tm& tm_;
1145
+
1146
+ auto tm_sec() const noexcept -> int {
1147
+ FMT_ASSERT(tm_.tm_sec >= 0 && tm_.tm_sec <= 61, "");
1148
+ return tm_.tm_sec;
1149
+ }
1150
+ auto tm_min() const noexcept -> int {
1151
+ FMT_ASSERT(tm_.tm_min >= 0 && tm_.tm_min <= 59, "");
1152
+ return tm_.tm_min;
1153
+ }
1154
+ auto tm_hour() const noexcept -> int {
1155
+ FMT_ASSERT(tm_.tm_hour >= 0 && tm_.tm_hour <= 23, "");
1156
+ return tm_.tm_hour;
1157
+ }
1158
+ auto tm_mday() const noexcept -> int {
1159
+ FMT_ASSERT(tm_.tm_mday >= 1 && tm_.tm_mday <= 31, "");
1160
+ return tm_.tm_mday;
1161
+ }
1162
+ auto tm_mon() const noexcept -> int {
1163
+ FMT_ASSERT(tm_.tm_mon >= 0 && tm_.tm_mon <= 11, "");
1164
+ return tm_.tm_mon;
1165
+ }
1166
+ auto tm_year() const noexcept -> long long { return 1900ll + tm_.tm_year; }
1167
+ auto tm_wday() const noexcept -> int {
1168
+ FMT_ASSERT(tm_.tm_wday >= 0 && tm_.tm_wday <= 6, "");
1169
+ return tm_.tm_wday;
1170
+ }
1171
+ auto tm_yday() const noexcept -> int {
1172
+ FMT_ASSERT(tm_.tm_yday >= 0 && tm_.tm_yday <= 365, "");
1173
+ return tm_.tm_yday;
1174
+ }
1175
+
1176
+ auto tm_hour12() const noexcept -> int {
1177
+ auto h = tm_hour();
1178
+ auto z = h < 12 ? h : h - 12;
1179
+ return z == 0 ? 12 : z;
1180
+ }
1181
+
1182
+ // POSIX and the C Standard are unclear or inconsistent about what %C and %y
1183
+ // do if the year is negative or exceeds 9999. Use the convention that %C
1184
+ // concatenated with %y yields the same output as %Y, and that %Y contains at
1185
+ // least 4 characters, with more only if necessary.
1186
+ auto split_year_lower(long long year) const noexcept -> int {
1187
+ auto l = year % 100;
1188
+ if (l < 0) l = -l; // l in [0, 99]
1189
+ return static_cast<int>(l);
1190
+ }
1191
+
1192
+ // Algorithm: https://en.wikipedia.org/wiki/ISO_week_date.
1193
+ auto iso_year_weeks(long long curr_year) const noexcept -> int {
1194
+ auto prev_year = curr_year - 1;
1195
+ auto curr_p =
1196
+ (curr_year + curr_year / 4 - curr_year / 100 + curr_year / 400) %
1197
+ days_per_week;
1198
+ auto prev_p =
1199
+ (prev_year + prev_year / 4 - prev_year / 100 + prev_year / 400) %
1200
+ days_per_week;
1201
+ return 52 + ((curr_p == 4 || prev_p == 3) ? 1 : 0);
1202
+ }
1203
+ auto iso_week_num(int tm_yday, int tm_wday) const noexcept -> int {
1204
+ return (tm_yday + 11 - (tm_wday == 0 ? days_per_week : tm_wday)) /
1205
+ days_per_week;
1206
+ }
1207
+ auto tm_iso_week_year() const noexcept -> long long {
1208
+ auto year = tm_year();
1209
+ auto w = iso_week_num(tm_yday(), tm_wday());
1210
+ if (w < 1) return year - 1;
1211
+ if (w > iso_year_weeks(year)) return year + 1;
1212
+ return year;
1213
+ }
1214
+ auto tm_iso_week_of_year() const noexcept -> int {
1215
+ auto year = tm_year();
1216
+ auto w = iso_week_num(tm_yday(), tm_wday());
1217
+ if (w < 1) return iso_year_weeks(year - 1);
1218
+ if (w > iso_year_weeks(year)) return 1;
1219
+ return w;
1220
+ }
1221
+
1222
+ void write1(int value) {
1223
+ *out_++ = static_cast<char>('0' + to_unsigned(value) % 10);
1224
+ }
1225
+ void write2(int value) {
1226
+ const char* d = digits2(to_unsigned(value) % 100);
1227
+ *out_++ = *d++;
1228
+ *out_++ = *d;
1229
+ }
1230
+ void write2(int value, pad_type pad) {
1231
+ unsigned int v = to_unsigned(value) % 100;
1232
+ if (v >= 10) {
1233
+ const char* d = digits2(v);
1234
+ *out_++ = *d++;
1235
+ *out_++ = *d;
1236
+ } else {
1237
+ out_ = detail::write_padding(out_, pad);
1238
+ *out_++ = static_cast<char>('0' + v);
1239
+ }
1240
+ }
1241
+
1242
+ void write_year_extended(long long year, pad_type pad) {
1243
+ // At least 4 characters.
1244
+ int width = 4;
1245
+ bool negative = year < 0;
1246
+ if (negative) {
1247
+ year = 0 - year;
1248
+ --width;
1249
+ }
1250
+ uint32_or_64_or_128_t<long long> n = to_unsigned(year);
1251
+ const int num_digits = count_digits(n);
1252
+ if (negative && pad == pad_type::zero) *out_++ = '-';
1253
+ if (width > num_digits)
1254
+ out_ = detail::write_padding(out_, pad, width - num_digits);
1255
+ if (negative && pad != pad_type::zero) *out_++ = '-';
1256
+ out_ = format_decimal<Char>(out_, n, num_digits);
1257
+ }
1258
+ void write_year(long long year, pad_type pad) {
1259
+ write_year_extended(year, pad);
1260
+ }
1261
+
1262
+ void write_utc_offset(long long offset, numeric_system ns) {
1263
+ if (offset < 0) {
1264
+ *out_++ = '-';
1265
+ offset = -offset;
1266
+ } else {
1267
+ *out_++ = '+';
1268
+ }
1269
+ offset /= 60;
1270
+ write2(static_cast<int>(offset / 60));
1271
+ if (ns != numeric_system::standard) *out_++ = ':';
1272
+ write2(static_cast<int>(offset % 60));
1273
+ }
1274
+
1275
+ template <typename T, FMT_ENABLE_IF(has_tm_gmtoff<T>::value)>
1276
+ void format_utc_offset(const T& tm, numeric_system ns) {
1277
+ write_utc_offset(tm.tm_gmtoff, ns);
1278
+ }
1279
+ template <typename T, FMT_ENABLE_IF(!has_tm_gmtoff<T>::value)>
1280
+ void format_utc_offset(const T&, numeric_system ns) {
1281
+ write_utc_offset(0, ns);
1282
+ }
1283
+
1284
+ template <typename T, FMT_ENABLE_IF(has_tm_zone<T>::value)>
1285
+ void format_tz_name(const T& tm) {
1286
+ out_ = write_tm_str<Char>(out_, tm.tm_zone, loc_);
1287
+ }
1288
+ template <typename T, FMT_ENABLE_IF(!has_tm_zone<T>::value)>
1289
+ void format_tz_name(const T&) {
1290
+ out_ = std::copy_n(utc(), 3, out_);
1291
+ }
1292
+
1293
+ void format_localized(char format, char modifier = 0) {
1294
+ out_ = write<Char>(out_, tm_, loc_, format, modifier);
1295
+ }
1296
+
1297
+ public:
1298
+ tm_writer(const std::locale& loc, OutputIt out, const std::tm& tm,
1299
+ const Duration* subsecs = nullptr)
1300
+ : loc_(loc),
1301
+ is_classic_(loc_ == get_classic_locale()),
1302
+ out_(out),
1303
+ subsecs_(subsecs),
1304
+ tm_(tm) {}
1305
+
1306
+ auto out() const -> OutputIt { return out_; }
1307
+
1308
+ FMT_CONSTEXPR void on_text(const Char* begin, const Char* end) {
1309
+ out_ = copy<Char>(begin, end, out_);
1310
+ }
1311
+
1312
+ void on_abbr_weekday() {
1313
+ if (is_classic_)
1314
+ out_ = write(out_, tm_wday_short_name(tm_wday()));
1315
+ else
1316
+ format_localized('a');
1317
+ }
1318
+ void on_full_weekday() {
1319
+ if (is_classic_)
1320
+ out_ = write(out_, tm_wday_full_name(tm_wday()));
1321
+ else
1322
+ format_localized('A');
1323
+ }
1324
+ void on_dec0_weekday(numeric_system ns) {
1325
+ if (is_classic_ || ns == numeric_system::standard) return write1(tm_wday());
1326
+ format_localized('w', 'O');
1327
+ }
1328
+ void on_dec1_weekday(numeric_system ns) {
1329
+ if (is_classic_ || ns == numeric_system::standard) {
1330
+ auto wday = tm_wday();
1331
+ write1(wday == 0 ? days_per_week : wday);
1332
+ } else {
1333
+ format_localized('u', 'O');
1334
+ }
1335
+ }
1336
+
1337
+ void on_abbr_month() {
1338
+ if (is_classic_)
1339
+ out_ = write(out_, tm_mon_short_name(tm_mon()));
1340
+ else
1341
+ format_localized('b');
1342
+ }
1343
+ void on_full_month() {
1344
+ if (is_classic_)
1345
+ out_ = write(out_, tm_mon_full_name(tm_mon()));
1346
+ else
1347
+ format_localized('B');
1348
+ }
1349
+
1350
+ void on_datetime(numeric_system ns) {
1351
+ if (is_classic_) {
1352
+ on_abbr_weekday();
1353
+ *out_++ = ' ';
1354
+ on_abbr_month();
1355
+ *out_++ = ' ';
1356
+ on_day_of_month(numeric_system::standard, pad_type::space);
1357
+ *out_++ = ' ';
1358
+ on_iso_time();
1359
+ *out_++ = ' ';
1360
+ on_year(numeric_system::standard, pad_type::space);
1361
+ } else {
1362
+ format_localized('c', ns == numeric_system::standard ? '\0' : 'E');
1363
+ }
1364
+ }
1365
+ void on_loc_date(numeric_system ns) {
1366
+ if (is_classic_)
1367
+ on_us_date();
1368
+ else
1369
+ format_localized('x', ns == numeric_system::standard ? '\0' : 'E');
1370
+ }
1371
+ void on_loc_time(numeric_system ns) {
1372
+ if (is_classic_)
1373
+ on_iso_time();
1374
+ else
1375
+ format_localized('X', ns == numeric_system::standard ? '\0' : 'E');
1376
+ }
1377
+ void on_us_date() {
1378
+ char buf[8];
1379
+ write_digit2_separated(buf, to_unsigned(tm_mon() + 1),
1380
+ to_unsigned(tm_mday()),
1381
+ to_unsigned(split_year_lower(tm_year())), '/');
1382
+ out_ = copy<Char>(std::begin(buf), std::end(buf), out_);
1383
+ }
1384
+ void on_iso_date() {
1385
+ auto year = tm_year();
1386
+ char buf[10];
1387
+ size_t offset = 0;
1388
+ if (year >= 0 && year < 10000) {
1389
+ write2digits(buf, static_cast<size_t>(year / 100));
1390
+ } else {
1391
+ offset = 4;
1392
+ write_year_extended(year, pad_type::zero);
1393
+ year = 0;
1394
+ }
1395
+ write_digit2_separated(buf + 2, static_cast<unsigned>(year % 100),
1396
+ to_unsigned(tm_mon() + 1), to_unsigned(tm_mday()),
1397
+ '-');
1398
+ out_ = copy<Char>(std::begin(buf) + offset, std::end(buf), out_);
1399
+ }
1400
+
1401
+ void on_utc_offset(numeric_system ns) { format_utc_offset(tm_, ns); }
1402
+ void on_tz_name() { format_tz_name(tm_); }
1403
+
1404
+ void on_year(numeric_system ns, pad_type pad) {
1405
+ if (is_classic_ || ns == numeric_system::standard)
1406
+ return write_year(tm_year(), pad);
1407
+ format_localized('Y', 'E');
1408
+ }
1409
+ void on_short_year(numeric_system ns) {
1410
+ if (is_classic_ || ns == numeric_system::standard)
1411
+ return write2(split_year_lower(tm_year()));
1412
+ format_localized('y', 'O');
1413
+ }
1414
+ void on_offset_year() {
1415
+ if (is_classic_) return write2(split_year_lower(tm_year()));
1416
+ format_localized('y', 'E');
1417
+ }
1418
+
1419
+ void on_century(numeric_system ns) {
1420
+ if (is_classic_ || ns == numeric_system::standard) {
1421
+ auto year = tm_year();
1422
+ auto upper = year / 100;
1423
+ if (year >= -99 && year < 0) {
1424
+ // Zero upper on negative year.
1425
+ *out_++ = '-';
1426
+ *out_++ = '0';
1427
+ } else if (upper >= 0 && upper < 100) {
1428
+ write2(static_cast<int>(upper));
1429
+ } else {
1430
+ out_ = write<Char>(out_, upper);
1431
+ }
1432
+ } else {
1433
+ format_localized('C', 'E');
1434
+ }
1435
+ }
1436
+
1437
+ void on_dec_month(numeric_system ns, pad_type pad) {
1438
+ if (is_classic_ || ns == numeric_system::standard)
1439
+ return write2(tm_mon() + 1, pad);
1440
+ format_localized('m', 'O');
1441
+ }
1442
+
1443
+ void on_dec0_week_of_year(numeric_system ns, pad_type pad) {
1444
+ if (is_classic_ || ns == numeric_system::standard)
1445
+ return write2((tm_yday() + days_per_week - tm_wday()) / days_per_week,
1446
+ pad);
1447
+ format_localized('U', 'O');
1448
+ }
1449
+ void on_dec1_week_of_year(numeric_system ns, pad_type pad) {
1450
+ if (is_classic_ || ns == numeric_system::standard) {
1451
+ auto wday = tm_wday();
1452
+ write2((tm_yday() + days_per_week -
1453
+ (wday == 0 ? (days_per_week - 1) : (wday - 1))) /
1454
+ days_per_week,
1455
+ pad);
1456
+ } else {
1457
+ format_localized('W', 'O');
1458
+ }
1459
+ }
1460
+ void on_iso_week_of_year(numeric_system ns, pad_type pad) {
1461
+ if (is_classic_ || ns == numeric_system::standard)
1462
+ return write2(tm_iso_week_of_year(), pad);
1463
+ format_localized('V', 'O');
1464
+ }
1465
+
1466
+ void on_iso_week_based_year() {
1467
+ write_year(tm_iso_week_year(), pad_type::zero);
1468
+ }
1469
+ void on_iso_week_based_short_year() {
1470
+ write2(split_year_lower(tm_iso_week_year()));
1471
+ }
1472
+
1473
+ void on_day_of_year(pad_type pad) {
1474
+ auto yday = tm_yday() + 1;
1475
+ auto digit1 = yday / 100;
1476
+ if (digit1 != 0)
1477
+ write1(digit1);
1478
+ else
1479
+ out_ = detail::write_padding(out_, pad);
1480
+ write2(yday % 100, pad);
1481
+ }
1482
+
1483
+ void on_day_of_month(numeric_system ns, pad_type pad) {
1484
+ if (is_classic_ || ns == numeric_system::standard)
1485
+ return write2(tm_mday(), pad);
1486
+ format_localized('d', 'O');
1487
+ }
1488
+
1489
+ void on_24_hour(numeric_system ns, pad_type pad) {
1490
+ if (is_classic_ || ns == numeric_system::standard)
1491
+ return write2(tm_hour(), pad);
1492
+ format_localized('H', 'O');
1493
+ }
1494
+ void on_12_hour(numeric_system ns, pad_type pad) {
1495
+ if (is_classic_ || ns == numeric_system::standard)
1496
+ return write2(tm_hour12(), pad);
1497
+ format_localized('I', 'O');
1498
+ }
1499
+ void on_minute(numeric_system ns, pad_type pad) {
1500
+ if (is_classic_ || ns == numeric_system::standard)
1501
+ return write2(tm_min(), pad);
1502
+ format_localized('M', 'O');
1503
+ }
1504
+
1505
+ void on_second(numeric_system ns, pad_type pad) {
1506
+ if (is_classic_ || ns == numeric_system::standard) {
1507
+ write2(tm_sec(), pad);
1508
+ if (subsecs_) {
1509
+ if (std::is_floating_point<typename Duration::rep>::value) {
1510
+ auto buf = memory_buffer();
1511
+ write_floating_seconds(buf, *subsecs_);
1512
+ if (buf.size() > 1) {
1513
+ // Remove the leading "0", write something like ".123".
1514
+ out_ = copy<Char>(buf.begin() + 1, buf.end(), out_);
1515
+ }
1516
+ } else {
1517
+ write_fractional_seconds<Char>(out_, *subsecs_);
1518
+ }
1519
+ }
1520
+ } else {
1521
+ // Currently no formatting of subseconds when a locale is set.
1522
+ format_localized('S', 'O');
1523
+ }
1524
+ }
1525
+
1526
+ void on_12_hour_time() {
1527
+ if (is_classic_) {
1528
+ char buf[8];
1529
+ write_digit2_separated(buf, to_unsigned(tm_hour12()),
1530
+ to_unsigned(tm_min()), to_unsigned(tm_sec()), ':');
1531
+ out_ = copy<Char>(std::begin(buf), std::end(buf), out_);
1532
+ *out_++ = ' ';
1533
+ on_am_pm();
1534
+ } else {
1535
+ format_localized('r');
1536
+ }
1537
+ }
1538
+ void on_24_hour_time() {
1539
+ write2(tm_hour());
1540
+ *out_++ = ':';
1541
+ write2(tm_min());
1542
+ }
1543
+ void on_iso_time() {
1544
+ on_24_hour_time();
1545
+ *out_++ = ':';
1546
+ on_second(numeric_system::standard, pad_type::zero);
1547
+ }
1548
+
1549
+ void on_am_pm() {
1550
+ if (is_classic_) {
1551
+ *out_++ = tm_hour() < 12 ? 'A' : 'P';
1552
+ *out_++ = 'M';
1553
+ } else {
1554
+ format_localized('p');
1555
+ }
1556
+ }
1557
+
1558
+ // These apply to chrono durations but not tm.
1559
+ void on_duration_value() {}
1560
+ void on_duration_unit() {}
1561
+ };
1562
+
1563
+ struct chrono_format_checker : null_chrono_spec_handler<chrono_format_checker> {
1564
+ bool has_precision_integral = false;
1565
+
1566
+ FMT_NORETURN inline void unsupported() { FMT_THROW(format_error("no date")); }
1567
+
1568
+ template <typename Char>
1569
+ FMT_CONSTEXPR void on_text(const Char*, const Char*) {}
1570
+ FMT_CONSTEXPR void on_day_of_year(pad_type) {}
1571
+ FMT_CONSTEXPR void on_24_hour(numeric_system, pad_type) {}
1572
+ FMT_CONSTEXPR void on_12_hour(numeric_system, pad_type) {}
1573
+ FMT_CONSTEXPR void on_minute(numeric_system, pad_type) {}
1574
+ FMT_CONSTEXPR void on_second(numeric_system, pad_type) {}
1575
+ FMT_CONSTEXPR void on_12_hour_time() {}
1576
+ FMT_CONSTEXPR void on_24_hour_time() {}
1577
+ FMT_CONSTEXPR void on_iso_time() {}
1578
+ FMT_CONSTEXPR void on_am_pm() {}
1579
+ FMT_CONSTEXPR void on_duration_value() const {
1580
+ if (has_precision_integral)
1581
+ FMT_THROW(format_error("precision not allowed for this argument type"));
1582
+ }
1583
+ FMT_CONSTEXPR void on_duration_unit() {}
1584
+ };
1585
+
1586
+ template <typename T,
1587
+ FMT_ENABLE_IF(std::is_integral<T>::value&& has_isfinite<T>::value)>
1588
+ inline auto isfinite(T) -> bool {
1589
+ return true;
1590
+ }
1591
+
1592
+ template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
1593
+ inline auto mod(T x, int y) -> T {
1594
+ return x % static_cast<T>(y);
1595
+ }
1596
+ template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
1597
+ inline auto mod(T x, int y) -> T {
1598
+ return std::fmod(x, static_cast<T>(y));
1599
+ }
1600
+
1601
+ // If T is an integral type, maps T to its unsigned counterpart, otherwise
1602
+ // leaves it unchanged (unlike std::make_unsigned).
1603
+ template <typename T, bool INTEGRAL = std::is_integral<T>::value>
1604
+ struct make_unsigned_or_unchanged {
1605
+ using type = T;
1606
+ };
1607
+
1608
+ template <typename T> struct make_unsigned_or_unchanged<T, true> {
1609
+ using type = typename std::make_unsigned<T>::type;
1610
+ };
1611
+
1612
+ template <typename Rep, typename Period,
1613
+ FMT_ENABLE_IF(std::is_integral<Rep>::value)>
1614
+ inline auto get_milliseconds(std::chrono::duration<Rep, Period> d)
1615
+ -> std::chrono::duration<Rep, std::milli> {
1616
+ // This may overflow and/or the result may not fit in the target type.
1617
+ #if FMT_SAFE_DURATION_CAST
1618
+ using common_seconds_type =
1619
+ typename std::common_type<decltype(d), std::chrono::seconds>::type;
1620
+ auto d_as_common = detail::duration_cast<common_seconds_type>(d);
1621
+ auto d_as_whole_seconds =
1622
+ detail::duration_cast<std::chrono::seconds>(d_as_common);
1623
+ // This conversion should be nonproblematic.
1624
+ auto diff = d_as_common - d_as_whole_seconds;
1625
+ auto ms = detail::duration_cast<std::chrono::duration<Rep, std::milli>>(diff);
1626
+ return ms;
1627
+ #else
1628
+ auto s = detail::duration_cast<std::chrono::seconds>(d);
1629
+ return detail::duration_cast<std::chrono::milliseconds>(d - s);
1630
+ #endif
1631
+ }
1632
+
1633
+ template <typename Char, typename Rep, typename OutputIt,
1634
+ FMT_ENABLE_IF(std::is_integral<Rep>::value)>
1635
+ auto format_duration_value(OutputIt out, Rep val, int) -> OutputIt {
1636
+ return write<Char>(out, val);
1637
+ }
1638
+
1639
+ template <typename Char, typename Rep, typename OutputIt,
1640
+ FMT_ENABLE_IF(std::is_floating_point<Rep>::value)>
1641
+ auto format_duration_value(OutputIt out, Rep val, int precision) -> OutputIt {
1642
+ auto specs = format_specs();
1643
+ specs.precision = precision;
1644
+ specs.set_type(precision >= 0 ? presentation_type::fixed
1645
+ : presentation_type::general);
1646
+ return write<Char>(out, val, specs);
1647
+ }
1648
+
1649
+ template <typename Char, typename OutputIt>
1650
+ auto copy_unit(string_view unit, OutputIt out, Char) -> OutputIt {
1651
+ return copy<Char>(unit.begin(), unit.end(), out);
1652
+ }
1653
+
1654
+ template <typename OutputIt>
1655
+ auto copy_unit(string_view unit, OutputIt out, wchar_t) -> OutputIt {
1656
+ // This works when wchar_t is UTF-32 because units only contain characters
1657
+ // that have the same representation in UTF-16 and UTF-32.
1658
+ utf8_to_utf16 u(unit);
1659
+ return copy<wchar_t>(u.c_str(), u.c_str() + u.size(), out);
1660
+ }
1661
+
1662
+ template <typename Char, typename Period, typename OutputIt>
1663
+ auto format_duration_unit(OutputIt out) -> OutputIt {
1664
+ if (const char* unit = get_units<Period>())
1665
+ return copy_unit(string_view(unit), out, Char());
1666
+ *out++ = '[';
1667
+ out = write<Char>(out, Period::num);
1668
+ if (const_check(Period::den != 1)) {
1669
+ *out++ = '/';
1670
+ out = write<Char>(out, Period::den);
1671
+ }
1672
+ *out++ = ']';
1673
+ *out++ = 's';
1674
+ return out;
1675
+ }
1676
+
1677
+ class get_locale {
1678
+ private:
1679
+ union {
1680
+ std::locale locale_;
1681
+ };
1682
+ bool has_locale_ = false;
1683
+
1684
+ public:
1685
+ inline get_locale(bool localized, locale_ref loc) : has_locale_(localized) {
1686
+ if (localized)
1687
+ ::new (&locale_) std::locale(loc.template get<std::locale>());
1688
+ }
1689
+ inline ~get_locale() {
1690
+ if (has_locale_) locale_.~locale();
1691
+ }
1692
+ inline operator const std::locale&() const {
1693
+ return has_locale_ ? locale_ : get_classic_locale();
1694
+ }
1695
+ };
1696
+
1697
+ template <typename Char, typename Rep, typename Period>
1698
+ struct duration_formatter {
1699
+ using iterator = basic_appender<Char>;
1700
+ iterator out;
1701
+ // rep is unsigned to avoid overflow.
1702
+ using rep =
1703
+ conditional_t<std::is_integral<Rep>::value && sizeof(Rep) < sizeof(int),
1704
+ unsigned, typename make_unsigned_or_unchanged<Rep>::type>;
1705
+ rep val;
1706
+ int precision;
1707
+ locale_ref locale;
1708
+ bool localized = false;
1709
+ using seconds = std::chrono::duration<rep>;
1710
+ seconds s;
1711
+ using milliseconds = std::chrono::duration<rep, std::milli>;
1712
+ bool negative;
1713
+
1714
+ using tm_writer_type = tm_writer<iterator, Char>;
1715
+
1716
+ duration_formatter(iterator o, std::chrono::duration<Rep, Period> d,
1717
+ locale_ref loc)
1718
+ : out(o), val(static_cast<rep>(d.count())), locale(loc), negative(false) {
1719
+ if (d.count() < 0) {
1720
+ val = 0 - val;
1721
+ negative = true;
1722
+ }
1723
+
1724
+ // this may overflow and/or the result may not fit in the
1725
+ // target type.
1726
+ // might need checked conversion (rep!=Rep)
1727
+ s = detail::duration_cast<seconds>(std::chrono::duration<rep, Period>(val));
1728
+ }
1729
+
1730
+ // returns true if nan or inf, writes to out.
1731
+ auto handle_nan_inf() -> bool {
1732
+ if (isfinite(val)) return false;
1733
+ if (isnan(val)) {
1734
+ write_nan();
1735
+ return true;
1736
+ }
1737
+ // must be +-inf
1738
+ if (val > 0)
1739
+ std::copy_n("inf", 3, out);
1740
+ else
1741
+ std::copy_n("-inf", 4, out);
1742
+ return true;
1743
+ }
1744
+
1745
+ auto days() const -> Rep { return static_cast<Rep>(s.count() / 86400); }
1746
+ auto hour() const -> Rep {
1747
+ return static_cast<Rep>(mod((s.count() / 3600), 24));
1748
+ }
1749
+
1750
+ auto hour12() const -> Rep {
1751
+ Rep hour = static_cast<Rep>(mod((s.count() / 3600), 12));
1752
+ return hour <= 0 ? 12 : hour;
1753
+ }
1754
+
1755
+ auto minute() const -> Rep {
1756
+ return static_cast<Rep>(mod((s.count() / 60), 60));
1757
+ }
1758
+ auto second() const -> Rep { return static_cast<Rep>(mod(s.count(), 60)); }
1759
+
1760
+ auto time() const -> std::tm {
1761
+ auto time = std::tm();
1762
+ time.tm_hour = to_nonnegative_int(hour(), 24);
1763
+ time.tm_min = to_nonnegative_int(minute(), 60);
1764
+ time.tm_sec = to_nonnegative_int(second(), 60);
1765
+ return time;
1766
+ }
1767
+
1768
+ void write_sign() {
1769
+ if (!negative) return;
1770
+ *out++ = '-';
1771
+ negative = false;
1772
+ }
1773
+
1774
+ void write(Rep value, int width, pad_type pad = pad_type::zero) {
1775
+ write_sign();
1776
+ if (isnan(value)) return write_nan();
1777
+ uint32_or_64_or_128_t<int> n =
1778
+ to_unsigned(to_nonnegative_int(value, max_value<int>()));
1779
+ int num_digits = detail::count_digits(n);
1780
+ if (width > num_digits) {
1781
+ out = detail::write_padding(out, pad, width - num_digits);
1782
+ }
1783
+ out = format_decimal<Char>(out, n, num_digits);
1784
+ }
1785
+
1786
+ void write_nan() { std::copy_n("nan", 3, out); }
1787
+
1788
+ template <typename Callback, typename... Args>
1789
+ void format_tm(const tm& time, Callback cb, Args... args) {
1790
+ if (isnan(val)) return write_nan();
1791
+ get_locale loc(localized, locale);
1792
+ auto w = tm_writer_type(loc, out, time);
1793
+ (w.*cb)(args...);
1794
+ out = w.out();
1795
+ }
1796
+
1797
+ void on_text(const Char* begin, const Char* end) {
1798
+ copy<Char>(begin, end, out);
1799
+ }
1800
+
1801
+ // These are not implemented because durations don't have date information.
1802
+ void on_abbr_weekday() {}
1803
+ void on_full_weekday() {}
1804
+ void on_dec0_weekday(numeric_system) {}
1805
+ void on_dec1_weekday(numeric_system) {}
1806
+ void on_abbr_month() {}
1807
+ void on_full_month() {}
1808
+ void on_datetime(numeric_system) {}
1809
+ void on_loc_date(numeric_system) {}
1810
+ void on_loc_time(numeric_system) {}
1811
+ void on_us_date() {}
1812
+ void on_iso_date() {}
1813
+ void on_utc_offset(numeric_system) {}
1814
+ void on_tz_name() {}
1815
+ void on_year(numeric_system, pad_type) {}
1816
+ void on_short_year(numeric_system) {}
1817
+ void on_offset_year() {}
1818
+ void on_century(numeric_system) {}
1819
+ void on_iso_week_based_year() {}
1820
+ void on_iso_week_based_short_year() {}
1821
+ void on_dec_month(numeric_system, pad_type) {}
1822
+ void on_dec0_week_of_year(numeric_system, pad_type) {}
1823
+ void on_dec1_week_of_year(numeric_system, pad_type) {}
1824
+ void on_iso_week_of_year(numeric_system, pad_type) {}
1825
+ void on_day_of_month(numeric_system, pad_type) {}
1826
+
1827
+ void on_day_of_year(pad_type) {
1828
+ if (handle_nan_inf()) return;
1829
+ write(days(), 0);
1830
+ }
1831
+
1832
+ void on_24_hour(numeric_system ns, pad_type pad) {
1833
+ if (handle_nan_inf()) return;
1834
+
1835
+ if (ns == numeric_system::standard) return write(hour(), 2, pad);
1836
+ auto time = tm();
1837
+ time.tm_hour = to_nonnegative_int(hour(), 24);
1838
+ format_tm(time, &tm_writer_type::on_24_hour, ns, pad);
1839
+ }
1840
+
1841
+ void on_12_hour(numeric_system ns, pad_type pad) {
1842
+ if (handle_nan_inf()) return;
1843
+
1844
+ if (ns == numeric_system::standard) return write(hour12(), 2, pad);
1845
+ auto time = tm();
1846
+ time.tm_hour = to_nonnegative_int(hour12(), 12);
1847
+ format_tm(time, &tm_writer_type::on_12_hour, ns, pad);
1848
+ }
1849
+
1850
+ void on_minute(numeric_system ns, pad_type pad) {
1851
+ if (handle_nan_inf()) return;
1852
+
1853
+ if (ns == numeric_system::standard) return write(minute(), 2, pad);
1854
+ auto time = tm();
1855
+ time.tm_min = to_nonnegative_int(minute(), 60);
1856
+ format_tm(time, &tm_writer_type::on_minute, ns, pad);
1857
+ }
1858
+
1859
+ void on_second(numeric_system ns, pad_type pad) {
1860
+ if (handle_nan_inf()) return;
1861
+
1862
+ if (ns == numeric_system::standard) {
1863
+ if (std::is_floating_point<rep>::value) {
1864
+ auto buf = memory_buffer();
1865
+ write_floating_seconds(buf, std::chrono::duration<rep, Period>(val),
1866
+ precision);
1867
+ if (negative) *out++ = '-';
1868
+ if (buf.size() < 2 || buf[1] == '.')
1869
+ out = detail::write_padding(out, pad);
1870
+ out = copy<Char>(buf.begin(), buf.end(), out);
1871
+ } else {
1872
+ write(second(), 2, pad);
1873
+ write_fractional_seconds<Char>(
1874
+ out, std::chrono::duration<rep, Period>(val), precision);
1875
+ }
1876
+ return;
1877
+ }
1878
+ auto time = tm();
1879
+ time.tm_sec = to_nonnegative_int(second(), 60);
1880
+ format_tm(time, &tm_writer_type::on_second, ns, pad);
1881
+ }
1882
+
1883
+ void on_12_hour_time() {
1884
+ if (handle_nan_inf()) return;
1885
+ format_tm(time(), &tm_writer_type::on_12_hour_time);
1886
+ }
1887
+
1888
+ void on_24_hour_time() {
1889
+ if (handle_nan_inf()) {
1890
+ *out++ = ':';
1891
+ handle_nan_inf();
1892
+ return;
1893
+ }
1894
+
1895
+ write(hour(), 2);
1896
+ *out++ = ':';
1897
+ write(minute(), 2);
1898
+ }
1899
+
1900
+ void on_iso_time() {
1901
+ on_24_hour_time();
1902
+ *out++ = ':';
1903
+ if (handle_nan_inf()) return;
1904
+ on_second(numeric_system::standard, pad_type::zero);
1905
+ }
1906
+
1907
+ void on_am_pm() {
1908
+ if (handle_nan_inf()) return;
1909
+ format_tm(time(), &tm_writer_type::on_am_pm);
1910
+ }
1911
+
1912
+ void on_duration_value() {
1913
+ if (handle_nan_inf()) return;
1914
+ write_sign();
1915
+ out = format_duration_value<Char>(out, val, precision);
1916
+ }
1917
+
1918
+ void on_duration_unit() { out = format_duration_unit<Char, Period>(out); }
1919
+ };
1920
+
1921
+ } // namespace detail
1922
+
1923
+ #if defined(__cpp_lib_chrono) && __cpp_lib_chrono >= 201907
1924
+ using weekday = std::chrono::weekday;
1925
+ using day = std::chrono::day;
1926
+ using month = std::chrono::month;
1927
+ using year = std::chrono::year;
1928
+ using year_month_day = std::chrono::year_month_day;
1929
+ #else
1930
+ // A fallback version of weekday.
1931
+ class weekday {
1932
+ private:
1933
+ unsigned char value_;
1934
+
1935
+ public:
1936
+ weekday() = default;
1937
+ constexpr explicit weekday(unsigned wd) noexcept
1938
+ : value_(static_cast<unsigned char>(wd != 7 ? wd : 0)) {}
1939
+ constexpr auto c_encoding() const noexcept -> unsigned { return value_; }
1940
+ };
1941
+
1942
+ class day {
1943
+ private:
1944
+ unsigned char value_;
1945
+
1946
+ public:
1947
+ day() = default;
1948
+ constexpr explicit day(unsigned d) noexcept
1949
+ : value_(static_cast<unsigned char>(d)) {}
1950
+ constexpr explicit operator unsigned() const noexcept { return value_; }
1951
+ };
1952
+
1953
+ class month {
1954
+ private:
1955
+ unsigned char value_;
1956
+
1957
+ public:
1958
+ month() = default;
1959
+ constexpr explicit month(unsigned m) noexcept
1960
+ : value_(static_cast<unsigned char>(m)) {}
1961
+ constexpr explicit operator unsigned() const noexcept { return value_; }
1962
+ };
1963
+
1964
+ class year {
1965
+ private:
1966
+ int value_;
1967
+
1968
+ public:
1969
+ year() = default;
1970
+ constexpr explicit year(int y) noexcept : value_(y) {}
1971
+ constexpr explicit operator int() const noexcept { return value_; }
1972
+ };
1973
+
1974
+ class year_month_day {
1975
+ private:
1976
+ fmt::year year_;
1977
+ fmt::month month_;
1978
+ fmt::day day_;
1979
+
1980
+ public:
1981
+ year_month_day() = default;
1982
+ constexpr year_month_day(const year& y, const month& m, const day& d) noexcept
1983
+ : year_(y), month_(m), day_(d) {}
1984
+ constexpr auto year() const noexcept -> fmt::year { return year_; }
1985
+ constexpr auto month() const noexcept -> fmt::month { return month_; }
1986
+ constexpr auto day() const noexcept -> fmt::day { return day_; }
1987
+ };
1988
+ #endif // __cpp_lib_chrono >= 201907
1989
+
1990
+ template <typename Char>
1991
+ struct formatter<weekday, Char> : private formatter<std::tm, Char> {
1992
+ private:
1993
+ bool use_tm_formatter_ = false;
1994
+
1995
+ public:
1996
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
1997
+ auto it = ctx.begin(), end = ctx.end();
1998
+ if (it != end && *it == 'L') {
1999
+ ++it;
2000
+ this->set_localized();
2001
+ }
2002
+ use_tm_formatter_ = it != end && *it != '}';
2003
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2004
+ }
2005
+
2006
+ template <typename FormatContext>
2007
+ auto format(weekday wd, FormatContext& ctx) const -> decltype(ctx.out()) {
2008
+ auto time = std::tm();
2009
+ time.tm_wday = static_cast<int>(wd.c_encoding());
2010
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2011
+ detail::get_locale loc(this->localized(), ctx.locale());
2012
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2013
+ w.on_abbr_weekday();
2014
+ return w.out();
2015
+ }
2016
+ };
2017
+
2018
+ template <typename Char>
2019
+ struct formatter<day, Char> : private formatter<std::tm, Char> {
2020
+ private:
2021
+ bool use_tm_formatter_ = false;
2022
+
2023
+ public:
2024
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2025
+ auto it = ctx.begin(), end = ctx.end();
2026
+ use_tm_formatter_ = it != end && *it != '}';
2027
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2028
+ }
2029
+
2030
+ template <typename FormatContext>
2031
+ auto format(day d, FormatContext& ctx) const -> decltype(ctx.out()) {
2032
+ auto time = std::tm();
2033
+ time.tm_mday = static_cast<int>(static_cast<unsigned>(d));
2034
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2035
+ detail::get_locale loc(false, ctx.locale());
2036
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2037
+ w.on_day_of_month(detail::numeric_system::standard, detail::pad_type::zero);
2038
+ return w.out();
2039
+ }
2040
+ };
2041
+
2042
+ template <typename Char>
2043
+ struct formatter<month, Char> : private formatter<std::tm, Char> {
2044
+ private:
2045
+ bool use_tm_formatter_ = false;
2046
+
2047
+ public:
2048
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2049
+ auto it = ctx.begin(), end = ctx.end();
2050
+ if (it != end && *it == 'L') {
2051
+ ++it;
2052
+ this->set_localized();
2053
+ }
2054
+ use_tm_formatter_ = it != end && *it != '}';
2055
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2056
+ }
2057
+
2058
+ template <typename FormatContext>
2059
+ auto format(month m, FormatContext& ctx) const -> decltype(ctx.out()) {
2060
+ auto time = std::tm();
2061
+ time.tm_mon = static_cast<int>(static_cast<unsigned>(m)) - 1;
2062
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2063
+ detail::get_locale loc(this->localized(), ctx.locale());
2064
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2065
+ w.on_abbr_month();
2066
+ return w.out();
2067
+ }
2068
+ };
2069
+
2070
+ template <typename Char>
2071
+ struct formatter<year, Char> : private formatter<std::tm, Char> {
2072
+ private:
2073
+ bool use_tm_formatter_ = false;
2074
+
2075
+ public:
2076
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2077
+ auto it = ctx.begin(), end = ctx.end();
2078
+ use_tm_formatter_ = it != end && *it != '}';
2079
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2080
+ }
2081
+
2082
+ template <typename FormatContext>
2083
+ auto format(year y, FormatContext& ctx) const -> decltype(ctx.out()) {
2084
+ auto time = std::tm();
2085
+ time.tm_year = static_cast<int>(y) - 1900;
2086
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2087
+ detail::get_locale loc(false, ctx.locale());
2088
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2089
+ w.on_year(detail::numeric_system::standard, detail::pad_type::zero);
2090
+ return w.out();
2091
+ }
2092
+ };
2093
+
2094
+ template <typename Char>
2095
+ struct formatter<year_month_day, Char> : private formatter<std::tm, Char> {
2096
+ private:
2097
+ bool use_tm_formatter_ = false;
2098
+
2099
+ public:
2100
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2101
+ auto it = ctx.begin(), end = ctx.end();
2102
+ use_tm_formatter_ = it != end && *it != '}';
2103
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2104
+ }
2105
+
2106
+ template <typename FormatContext>
2107
+ auto format(year_month_day val, FormatContext& ctx) const
2108
+ -> decltype(ctx.out()) {
2109
+ auto time = std::tm();
2110
+ time.tm_year = static_cast<int>(val.year()) - 1900;
2111
+ time.tm_mon = static_cast<int>(static_cast<unsigned>(val.month())) - 1;
2112
+ time.tm_mday = static_cast<int>(static_cast<unsigned>(val.day()));
2113
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2114
+ detail::get_locale loc(true, ctx.locale());
2115
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2116
+ w.on_iso_date();
2117
+ return w.out();
2118
+ }
2119
+ };
2120
+
2121
+ template <typename Rep, typename Period, typename Char>
2122
+ struct formatter<std::chrono::duration<Rep, Period>, Char> {
2123
+ private:
2124
+ format_specs specs_;
2125
+ detail::arg_ref<Char> width_ref_;
2126
+ detail::arg_ref<Char> precision_ref_;
2127
+ basic_string_view<Char> fmt_;
2128
+
2129
+ public:
2130
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2131
+ auto it = ctx.begin(), end = ctx.end();
2132
+ if (it == end || *it == '}') return it;
2133
+
2134
+ it = detail::parse_align(it, end, specs_);
2135
+ if (it == end) return it;
2136
+
2137
+ Char c = *it;
2138
+ if ((c >= '0' && c <= '9') || c == '{') {
2139
+ it = detail::parse_width(it, end, specs_, width_ref_, ctx);
2140
+ if (it == end) return it;
2141
+ }
2142
+
2143
+ auto checker = detail::chrono_format_checker();
2144
+ if (*it == '.') {
2145
+ checker.has_precision_integral = !std::is_floating_point<Rep>::value;
2146
+ it = detail::parse_precision(it, end, specs_, precision_ref_, ctx);
2147
+ }
2148
+ if (it != end && *it == 'L') {
2149
+ specs_.set_localized();
2150
+ ++it;
2151
+ }
2152
+ end = detail::parse_chrono_format(it, end, checker);
2153
+ fmt_ = {it, detail::to_unsigned(end - it)};
2154
+ return end;
2155
+ }
2156
+
2157
+ template <typename FormatContext>
2158
+ auto format(std::chrono::duration<Rep, Period> d, FormatContext& ctx) const
2159
+ -> decltype(ctx.out()) {
2160
+ auto specs = specs_;
2161
+ auto precision = specs.precision;
2162
+ specs.precision = -1;
2163
+ auto begin = fmt_.begin(), end = fmt_.end();
2164
+ // As a possible future optimization, we could avoid extra copying if width
2165
+ // is not specified.
2166
+ auto buf = basic_memory_buffer<Char>();
2167
+ auto out = basic_appender<Char>(buf);
2168
+ detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
2169
+ ctx);
2170
+ detail::handle_dynamic_spec(specs.dynamic_precision(), precision,
2171
+ precision_ref_, ctx);
2172
+ if (begin == end || *begin == '}') {
2173
+ out = detail::format_duration_value<Char>(out, d.count(), precision);
2174
+ detail::format_duration_unit<Char, Period>(out);
2175
+ } else {
2176
+ auto f =
2177
+ detail::duration_formatter<Char, Rep, Period>(out, d, ctx.locale());
2178
+ f.precision = precision;
2179
+ f.localized = specs_.localized();
2180
+ detail::parse_chrono_format(begin, end, f);
2181
+ }
2182
+ return detail::write(
2183
+ ctx.out(), basic_string_view<Char>(buf.data(), buf.size()), specs);
2184
+ }
2185
+ };
2186
+
2187
+ template <typename Char> struct formatter<std::tm, Char> {
2188
+ private:
2189
+ format_specs specs_;
2190
+ detail::arg_ref<Char> width_ref_;
2191
+ basic_string_view<Char> fmt_ =
2192
+ detail::string_literal<Char, '%', 'F', ' ', '%', 'T'>();
2193
+
2194
+ protected:
2195
+ auto localized() const -> bool { return specs_.localized(); }
2196
+ FMT_CONSTEXPR void set_localized() { specs_.set_localized(); }
2197
+
2198
+ FMT_CONSTEXPR auto do_parse(parse_context<Char>& ctx, bool has_timezone)
2199
+ -> const Char* {
2200
+ auto it = ctx.begin(), end = ctx.end();
2201
+ if (it == end || *it == '}') return it;
2202
+
2203
+ it = detail::parse_align(it, end, specs_);
2204
+ if (it == end) return it;
2205
+
2206
+ Char c = *it;
2207
+ if ((c >= '0' && c <= '9') || c == '{') {
2208
+ it = detail::parse_width(it, end, specs_, width_ref_, ctx);
2209
+ if (it == end) return it;
2210
+ }
2211
+
2212
+ if (*it == 'L') {
2213
+ specs_.set_localized();
2214
+ ++it;
2215
+ }
2216
+
2217
+ end = detail::parse_chrono_format(it, end,
2218
+ detail::tm_format_checker(has_timezone));
2219
+ // Replace the default format string only if the new spec is not empty.
2220
+ if (end != it) fmt_ = {it, detail::to_unsigned(end - it)};
2221
+ return end;
2222
+ }
2223
+
2224
+ template <typename Duration, typename FormatContext>
2225
+ auto do_format(const std::tm& tm, FormatContext& ctx,
2226
+ const Duration* subsecs) const -> decltype(ctx.out()) {
2227
+ auto specs = specs_;
2228
+ auto buf = basic_memory_buffer<Char>();
2229
+ auto out = basic_appender<Char>(buf);
2230
+ detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
2231
+ ctx);
2232
+
2233
+ auto loc_ref = specs.localized() ? ctx.locale() : detail::locale_ref();
2234
+ detail::get_locale loc(static_cast<bool>(loc_ref), loc_ref);
2235
+ auto w = detail::tm_writer<basic_appender<Char>, Char, Duration>(
2236
+ loc, out, tm, subsecs);
2237
+ detail::parse_chrono_format(fmt_.begin(), fmt_.end(), w);
2238
+ return detail::write(
2239
+ ctx.out(), basic_string_view<Char>(buf.data(), buf.size()), specs);
2240
+ }
2241
+
2242
+ public:
2243
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2244
+ return do_parse(ctx, detail::has_tm_gmtoff<std::tm>::value);
2245
+ }
2246
+
2247
+ template <typename FormatContext>
2248
+ auto format(const std::tm& tm, FormatContext& ctx) const
2249
+ -> decltype(ctx.out()) {
2250
+ return do_format<std::chrono::seconds>(tm, ctx, nullptr);
2251
+ }
2252
+ };
2253
+
2254
+ // DEPRECATED! Reversed order of template parameters.
2255
+ template <typename Char, typename Duration>
2256
+ struct formatter<sys_time<Duration>, Char> : private formatter<std::tm, Char> {
2257
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2258
+ return this->do_parse(ctx, true);
2259
+ }
2260
+
2261
+ template <typename FormatContext>
2262
+ auto format(sys_time<Duration> val, FormatContext& ctx) const
2263
+ -> decltype(ctx.out()) {
2264
+ std::tm tm = gmtime(val);
2265
+ using period = typename Duration::period;
2266
+ if (detail::const_check(
2267
+ period::num == 1 && period::den == 1 &&
2268
+ !std::is_floating_point<typename Duration::rep>::value)) {
2269
+ detail::set_tm_zone(tm, detail::utc());
2270
+ return formatter<std::tm, Char>::format(tm, ctx);
2271
+ }
2272
+ Duration epoch = val.time_since_epoch();
2273
+ Duration subsecs = detail::duration_cast<Duration>(
2274
+ epoch - detail::duration_cast<std::chrono::seconds>(epoch));
2275
+ if (subsecs.count() < 0) {
2276
+ auto second = detail::duration_cast<Duration>(std::chrono::seconds(1));
2277
+ if (tm.tm_sec != 0) {
2278
+ --tm.tm_sec;
2279
+ } else {
2280
+ tm = gmtime(val - second);
2281
+ detail::set_tm_zone(tm, detail::utc());
2282
+ }
2283
+ subsecs += second;
2284
+ }
2285
+ return formatter<std::tm, Char>::do_format(tm, ctx, &subsecs);
2286
+ }
2287
+ };
2288
+
2289
+ template <typename Duration, typename Char>
2290
+ struct formatter<utc_time<Duration>, Char>
2291
+ : formatter<sys_time<Duration>, Char> {
2292
+ template <typename FormatContext>
2293
+ auto format(utc_time<Duration> val, FormatContext& ctx) const
2294
+ -> decltype(ctx.out()) {
2295
+ return formatter<sys_time<Duration>, Char>::format(
2296
+ detail::utc_clock::to_sys(val), ctx);
2297
+ }
2298
+ };
2299
+
2300
+ template <typename Duration, typename Char>
2301
+ struct formatter<local_time<Duration>, Char>
2302
+ : private formatter<std::tm, Char> {
2303
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2304
+ return this->do_parse(ctx, false);
2305
+ }
2306
+
2307
+ template <typename FormatContext>
2308
+ auto format(local_time<Duration> val, FormatContext& ctx) const
2309
+ -> decltype(ctx.out()) {
2310
+ auto time_since_epoch = val.time_since_epoch();
2311
+ auto seconds_since_epoch =
2312
+ detail::duration_cast<std::chrono::seconds>(time_since_epoch);
2313
+ // Use gmtime to prevent time zone conversion since local_time has an
2314
+ // unspecified time zone.
2315
+ std::tm t = gmtime(seconds_since_epoch.count());
2316
+ using period = typename Duration::period;
2317
+ if (period::num == 1 && period::den == 1 &&
2318
+ !std::is_floating_point<typename Duration::rep>::value) {
2319
+ return formatter<std::tm, Char>::format(t, ctx);
2320
+ }
2321
+ auto subsecs =
2322
+ detail::duration_cast<Duration>(time_since_epoch - seconds_since_epoch);
2323
+ return formatter<std::tm, Char>::do_format(t, ctx, &subsecs);
2324
+ }
2325
+ };
2326
+
2327
+ FMT_END_EXPORT
2328
+ FMT_END_NAMESPACE
2329
+
2330
+ #endif // FMT_CHRONO_H_