simpleble 0.0.1

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 (341) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +88 -0
  3. data/LICENSE.txt +21 -0
  4. data/README.md +350 -0
  5. data/ext/simpleble/extconf.rb +57 -0
  6. data/ext/simpleble/simpleble_ruby.c +682 -0
  7. data/lib/simpleble/adapter.rb +10 -0
  8. data/lib/simpleble/characteristic.rb +36 -0
  9. data/lib/simpleble/descriptor.rb +8 -0
  10. data/lib/simpleble/exceptions.rb +25 -0
  11. data/lib/simpleble/peripheral.rb +64 -0
  12. data/lib/simpleble/service.rb +16 -0
  13. data/lib/simpleble/version.rb +3 -0
  14. data/lib/simpleble.rb +40 -0
  15. data/vendor/simpleble/build_simplecble/CMakeFiles/4.0.3/CompilerIdCXX/CMakeCXXCompilerId.cpp +920 -0
  16. data/vendor/simpleble/build_simplecble/CMakeFiles/4.0.3/CompilerIdCXX/apple-sdk.cpp +1 -0
  17. data/vendor/simpleble/build_simplecble/CMakeFiles/4.1.0/CompilerIdCXX/CMakeCXXCompilerId.cpp +949 -0
  18. data/vendor/simpleble/build_simplecble/CMakeFiles/4.1.0/CompilerIdCXX/apple-sdk.cpp +1 -0
  19. data/vendor/simpleble/build_simplecble/export/simplecble/export.h +43 -0
  20. data/vendor/simpleble/build_simplecble/simpleble/export/simpleble/export.h +43 -0
  21. data/vendor/simpleble/dependencies/external/kvn/kvn_bytearray.h +222 -0
  22. data/vendor/simpleble/dependencies/external/kvn/kvn_safe_callback.hpp +66 -0
  23. data/vendor/simpleble/dependencies/external/kvn/logfwd.hpp +35 -0
  24. data/vendor/simpleble/dependencies/internal/fmt/args.h +220 -0
  25. data/vendor/simpleble/dependencies/internal/fmt/base.h +2962 -0
  26. data/vendor/simpleble/dependencies/internal/fmt/chrono.h +2338 -0
  27. data/vendor/simpleble/dependencies/internal/fmt/color.h +610 -0
  28. data/vendor/simpleble/dependencies/internal/fmt/compile.h +539 -0
  29. data/vendor/simpleble/dependencies/internal/fmt/core.h +5 -0
  30. data/vendor/simpleble/dependencies/internal/fmt/format-inl.h +1949 -0
  31. data/vendor/simpleble/dependencies/internal/fmt/format.h +4244 -0
  32. data/vendor/simpleble/dependencies/internal/fmt/os.h +427 -0
  33. data/vendor/simpleble/dependencies/internal/fmt/ostream.h +166 -0
  34. data/vendor/simpleble/dependencies/internal/fmt/printf.h +633 -0
  35. data/vendor/simpleble/dependencies/internal/fmt/ranges.h +850 -0
  36. data/vendor/simpleble/dependencies/internal/fmt/std.h +726 -0
  37. data/vendor/simpleble/dependencies/internal/fmt/xchar.h +373 -0
  38. data/vendor/simpleble/dependencies/internal/simplejni/Common.hpp +579 -0
  39. data/vendor/simpleble/dependencies/internal/simplejni/References.hpp +151 -0
  40. data/vendor/simpleble/dependencies/internal/simplejni/Registry.hpp +183 -0
  41. data/vendor/simpleble/dependencies/internal/simplejni/VM.hpp +88 -0
  42. data/vendor/simpleble/examples/simpleble/src/connect.cpp +67 -0
  43. data/vendor/simpleble/examples/simpleble/src/connect_safe.cpp +91 -0
  44. data/vendor/simpleble/examples/simpleble/src/list_adapters.cpp +22 -0
  45. data/vendor/simpleble/examples/simpleble/src/list_adapters_safe.cpp +33 -0
  46. data/vendor/simpleble/examples/simpleble/src/list_paired.cpp +27 -0
  47. data/vendor/simpleble/examples/simpleble/src/multiconnect.cpp +60 -0
  48. data/vendor/simpleble/examples/simpleble/src/notify.cpp +85 -0
  49. data/vendor/simpleble/examples/simpleble/src/notify_multi.cpp +101 -0
  50. data/vendor/simpleble/examples/simpleble/src/power_cycle.cpp +43 -0
  51. data/vendor/simpleble/examples/simpleble/src/read.cpp +82 -0
  52. data/vendor/simpleble/examples/simpleble/src/scan.cpp +62 -0
  53. data/vendor/simpleble/examples/simpleble/src/utils.cpp +59 -0
  54. data/vendor/simpleble/examples/simpleble/src/utils.hpp +27 -0
  55. data/vendor/simpleble/examples/simpleble/src/write.cpp +81 -0
  56. data/vendor/simpleble/examples/simplebluez/ble_nus/ble_nus.cpp +138 -0
  57. data/vendor/simpleble/examples/simplebluez/connect/connect.cpp +119 -0
  58. data/vendor/simpleble/examples/simplebluez/list_adapters/list_adapters.cpp +44 -0
  59. data/vendor/simpleble/examples/simplebluez/list_paired/list_paired.cpp +48 -0
  60. data/vendor/simpleble/examples/simplebluez/notify/notify.cpp +145 -0
  61. data/vendor/simpleble/examples/simplebluez/pair/pair.cpp +181 -0
  62. data/vendor/simpleble/examples/simplebluez/read/read.cpp +141 -0
  63. data/vendor/simpleble/examples/simplebluez/scan/scan.cpp +85 -0
  64. data/vendor/simpleble/examples/simplecble/c/connect.c +163 -0
  65. data/vendor/simpleble/examples/simplecble/c/notify.c +217 -0
  66. data/vendor/simpleble/examples/simplecble/c/scan.c +164 -0
  67. data/vendor/simpleble/examples/simpledbus/notification.cpp +24 -0
  68. data/vendor/simpleble/hitl/src/test_sanity.cpp +48 -0
  69. data/vendor/simpleble/install_simplecble/include/simpleble/Adapter.h +102 -0
  70. data/vendor/simpleble/install_simplecble/include/simpleble/AdapterSafe.h +58 -0
  71. data/vendor/simpleble/install_simplecble/include/simpleble/Advanced.h +50 -0
  72. data/vendor/simpleble/install_simplecble/include/simpleble/Characteristic.h +39 -0
  73. data/vendor/simpleble/install_simplecble/include/simpleble/Config.h +55 -0
  74. data/vendor/simpleble/install_simplecble/include/simpleble/Descriptor.h +30 -0
  75. data/vendor/simpleble/install_simplecble/include/simpleble/Exceptions.h +72 -0
  76. data/vendor/simpleble/install_simplecble/include/simpleble/Logging.h +73 -0
  77. data/vendor/simpleble/install_simplecble/include/simpleble/Peripheral.h +82 -0
  78. data/vendor/simpleble/install_simplecble/include/simpleble/PeripheralSafe.h +64 -0
  79. data/vendor/simpleble/install_simplecble/include/simpleble/Service.h +34 -0
  80. data/vendor/simpleble/install_simplecble/include/simpleble/SimpleBLE.h +8 -0
  81. data/vendor/simpleble/install_simplecble/include/simpleble/Types.h +49 -0
  82. data/vendor/simpleble/install_simplecble/include/simpleble/Utils.h +13 -0
  83. data/vendor/simpleble/install_simplecble/include/simpleble/export.h +43 -0
  84. data/vendor/simpleble/install_simplecble/include/simpleble/kvn/kvn_bytearray.h +222 -0
  85. data/vendor/simpleble/install_simplecble/include/simplecble/adapter.h +188 -0
  86. data/vendor/simpleble/install_simplecble/include/simplecble/logging.h +37 -0
  87. data/vendor/simpleble/install_simplecble/include/simplecble/peripheral.h +304 -0
  88. data/vendor/simpleble/install_simplecble/include/simplecble/simpleble.h +21 -0
  89. data/vendor/simpleble/install_simplecble/include/simplecble/types.h +73 -0
  90. data/vendor/simpleble/install_simplecble/include/simplecble/utils.h +27 -0
  91. data/vendor/simpleble/simpleble/include/simpleble/Adapter.h +102 -0
  92. data/vendor/simpleble/simpleble/include/simpleble/AdapterSafe.h +58 -0
  93. data/vendor/simpleble/simpleble/include/simpleble/Characteristic.h +39 -0
  94. data/vendor/simpleble/simpleble/include/simpleble/Config.h +55 -0
  95. data/vendor/simpleble/simpleble/include/simpleble/Descriptor.h +30 -0
  96. data/vendor/simpleble/simpleble/include/simpleble/Exceptions.h +72 -0
  97. data/vendor/simpleble/simpleble/include/simpleble/Logging.h +73 -0
  98. data/vendor/simpleble/simpleble/include/simpleble/Peripheral.h +82 -0
  99. data/vendor/simpleble/simpleble/include/simpleble/PeripheralSafe.h +64 -0
  100. data/vendor/simpleble/simpleble/include/simpleble/Service.h +34 -0
  101. data/vendor/simpleble/simpleble/include/simpleble/SimpleBLE.h +8 -0
  102. data/vendor/simpleble/simpleble/include/simpleble/Types.h +49 -0
  103. data/vendor/simpleble/simpleble/include/simpleble/Utils.h +13 -0
  104. data/vendor/simpleble/simpleble/include/simpleble_c/adapter.h +188 -0
  105. data/vendor/simpleble/simpleble/include/simpleble_c/logging.h +37 -0
  106. data/vendor/simpleble/simpleble/include/simpleble_c/peripheral.h +304 -0
  107. data/vendor/simpleble/simpleble/include/simpleble_c/simpleble.h +21 -0
  108. data/vendor/simpleble/simpleble/include/simpleble_c/types.h +73 -0
  109. data/vendor/simpleble/simpleble/include/simpleble_c/utils.h +27 -0
  110. data/vendor/simpleble/simpleble/src/CommonUtils.h +63 -0
  111. data/vendor/simpleble/simpleble/src/Exceptions.cpp +31 -0
  112. data/vendor/simpleble/simpleble/src/Logging.cpp +136 -0
  113. data/vendor/simpleble/simpleble/src/LoggingInternal.h +85 -0
  114. data/vendor/simpleble/simpleble/src/Utils.cpp +24 -0
  115. data/vendor/simpleble/simpleble/src/backends/android/AdapterAndroid.cpp +101 -0
  116. data/vendor/simpleble/simpleble/src/backends/android/AdapterAndroid.h +70 -0
  117. data/vendor/simpleble/simpleble/src/backends/android/BackendAndroid.cpp +40 -0
  118. data/vendor/simpleble/simpleble/src/backends/android/BackendAndroid.h +20 -0
  119. data/vendor/simpleble/simpleble/src/backends/android/PeripheralAndroid.cpp +365 -0
  120. data/vendor/simpleble/simpleble/src/backends/android/PeripheralAndroid.h +90 -0
  121. data/vendor/simpleble/simpleble/src/backends/android/bridge/BluetoothGattCallback.cpp +432 -0
  122. data/vendor/simpleble/simpleble/src/backends/android/bridge/BluetoothGattCallback.h +102 -0
  123. data/vendor/simpleble/simpleble/src/backends/android/bridge/ScanCallback.cpp +142 -0
  124. data/vendor/simpleble/simpleble/src/backends/android/bridge/ScanCallback.h +55 -0
  125. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothAdapter.cpp +107 -0
  126. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothAdapter.h +55 -0
  127. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothDevice.cpp +68 -0
  128. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothDevice.h +54 -0
  129. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGatt.cpp +115 -0
  130. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGatt.h +75 -0
  131. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattCharacteristic.cpp +142 -0
  132. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattCharacteristic.h +66 -0
  133. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattDescriptor.cpp +67 -0
  134. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattDescriptor.h +46 -0
  135. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattService.cpp +106 -0
  136. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/BluetoothGattService.h +47 -0
  137. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/BluetoothScanner.cpp +47 -0
  138. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/BluetoothScanner.h +37 -0
  139. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/ScanRecord.cpp +69 -0
  140. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/ScanRecord.h +41 -0
  141. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/ScanResult.cpp +63 -0
  142. data/vendor/simpleble/simpleble/src/backends/android/types/android/bluetooth/le/ScanResult.h +42 -0
  143. data/vendor/simpleble/simpleble/src/backends/android/types/android/os/ParcelUUID.cpp +32 -0
  144. data/vendor/simpleble/simpleble/src/backends/android/types/android/os/ParcelUUID.h +30 -0
  145. data/vendor/simpleble/simpleble/src/backends/android/types/android/util/SparseArray.cpp +54 -0
  146. data/vendor/simpleble/simpleble/src/backends/android/types/android/util/SparseArray.h +37 -0
  147. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/Iterator.cpp +36 -0
  148. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/Iterator.h +28 -0
  149. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/List.cpp +29 -0
  150. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/List.h +27 -0
  151. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/Set.cpp +33 -0
  152. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/Set.h +28 -0
  153. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/UUID.cpp +26 -0
  154. data/vendor/simpleble/simpleble/src/backends/android/types/java/util/UUID.h +29 -0
  155. data/vendor/simpleble/simpleble/src/backends/common/AdapterBase.cpp +53 -0
  156. data/vendor/simpleble/simpleble/src/backends/common/AdapterBase.h +81 -0
  157. data/vendor/simpleble/simpleble/src/backends/common/AdapterBaseTypes.h +22 -0
  158. data/vendor/simpleble/simpleble/src/backends/common/BackendBase.h +20 -0
  159. data/vendor/simpleble/simpleble/src/backends/common/BackendUtils.h +33 -0
  160. data/vendor/simpleble/simpleble/src/backends/common/CharacteristicBase.cpp +28 -0
  161. data/vendor/simpleble/simpleble/src/backends/common/CharacteristicBase.h +38 -0
  162. data/vendor/simpleble/simpleble/src/backends/common/DescriptorBase.cpp +7 -0
  163. data/vendor/simpleble/simpleble/src/backends/common/DescriptorBase.h +19 -0
  164. data/vendor/simpleble/simpleble/src/backends/common/PeripheralBase.h +82 -0
  165. data/vendor/simpleble/simpleble/src/backends/common/ServiceBase.cpp +18 -0
  166. data/vendor/simpleble/simpleble/src/backends/common/ServiceBase.h +28 -0
  167. data/vendor/simpleble/simpleble/src/backends/linux/AdapterLinux.cpp +102 -0
  168. data/vendor/simpleble/simpleble/src/backends/linux/AdapterLinux.h +55 -0
  169. data/vendor/simpleble/simpleble/src/backends/linux/BackendBluez.cpp +87 -0
  170. data/vendor/simpleble/simpleble/src/backends/linux/PeripheralLinux.cpp +376 -0
  171. data/vendor/simpleble/simpleble/src/backends/linux/PeripheralLinux.h +90 -0
  172. data/vendor/simpleble/simpleble/src/backends/macos/AdapterBaseMacOS.h +29 -0
  173. data/vendor/simpleble/simpleble/src/backends/macos/AdapterMac.h +78 -0
  174. data/vendor/simpleble/simpleble/src/backends/macos/PeripheralBaseMacOS.h +49 -0
  175. data/vendor/simpleble/simpleble/src/backends/macos/PeripheralMac.h +81 -0
  176. data/vendor/simpleble/simpleble/src/backends/macos/Utils.h +9 -0
  177. data/vendor/simpleble/simpleble/src/backends/plain/AdapterPlain.cpp +65 -0
  178. data/vendor/simpleble/simpleble/src/backends/plain/AdapterPlain.h +49 -0
  179. data/vendor/simpleble/simpleble/src/backends/plain/BackendPlain.cpp +30 -0
  180. data/vendor/simpleble/simpleble/src/backends/plain/PeripheralPlain.cpp +159 -0
  181. data/vendor/simpleble/simpleble/src/backends/plain/PeripheralPlain.h +72 -0
  182. data/vendor/simpleble/simpleble/src/backends/windows/AdapterWindows.cpp +330 -0
  183. data/vendor/simpleble/simpleble/src/backends/windows/AdapterWindows.h +89 -0
  184. data/vendor/simpleble/simpleble/src/backends/windows/BackendWinRT.cpp +67 -0
  185. data/vendor/simpleble/simpleble/src/backends/windows/BackendWinRT.h +18 -0
  186. data/vendor/simpleble/simpleble/src/backends/windows/MtaManager.cpp +49 -0
  187. data/vendor/simpleble/simpleble/src/backends/windows/MtaManager.h +90 -0
  188. data/vendor/simpleble/simpleble/src/backends/windows/PeripheralWindows.cpp +487 -0
  189. data/vendor/simpleble/simpleble/src/backends/windows/PeripheralWindows.h +129 -0
  190. data/vendor/simpleble/simpleble/src/backends/windows/Utils.cpp +146 -0
  191. data/vendor/simpleble/simpleble/src/backends/windows/Utils.h +47 -0
  192. data/vendor/simpleble/simpleble/src/builders/BuildVec.h +32 -0
  193. data/vendor/simpleble/simpleble/src/builders/BuilderBase.h +87 -0
  194. data/vendor/simpleble/simpleble/src/external/TaskRunner.hpp +99 -0
  195. data/vendor/simpleble/simpleble/src/external/ThreadRunner.h +52 -0
  196. data/vendor/simpleble/simpleble/src/external/kvn_safe_callback.hpp +66 -0
  197. data/vendor/simpleble/simpleble/src/external/kvn_safe_map.hpp +94 -0
  198. data/vendor/simpleble/simpleble/src/external/kvn_threadrunner.hpp +70 -0
  199. data/vendor/simpleble/simpleble/src/external/logfwd.hpp +35 -0
  200. data/vendor/simpleble/simpleble/src/frontends/base/Adapter.cpp +111 -0
  201. data/vendor/simpleble/simpleble/src/frontends/base/Backend.cpp +83 -0
  202. data/vendor/simpleble/simpleble/src/frontends/base/Backend.h +76 -0
  203. data/vendor/simpleble/simpleble/src/frontends/base/Characteristic.cpp +56 -0
  204. data/vendor/simpleble/simpleble/src/frontends/base/Descriptor.cpp +21 -0
  205. data/vendor/simpleble/simpleble/src/frontends/base/Peripheral.cpp +113 -0
  206. data/vendor/simpleble/simpleble/src/frontends/base/Service.cpp +26 -0
  207. data/vendor/simpleble/simpleble/src/frontends/safe/AdapterSafe.cpp +158 -0
  208. data/vendor/simpleble/simpleble/src/frontends/safe/PeripheralSafe.cpp +219 -0
  209. data/vendor/simpleble/simpleble/src_c/adapter.cpp +204 -0
  210. data/vendor/simpleble/simpleble/src_c/logging.cpp +19 -0
  211. data/vendor/simpleble/simpleble/src_c/peripheral.cpp +444 -0
  212. data/vendor/simpleble/simpleble/src_c/simpleble.cpp +5 -0
  213. data/vendor/simpleble/simpleble/src_c/utils.cpp +15 -0
  214. data/vendor/simpleble/simpleble/test/src/main.cpp +8 -0
  215. data/vendor/simpleble/simpleble/test/src/test_bytearray.cpp +246 -0
  216. data/vendor/simpleble/simpleble/test/src/test_utils.cpp +24 -0
  217. data/vendor/simpleble/simplebluez/include/simplebluez/Adapter.h +46 -0
  218. data/vendor/simpleble/simplebluez/include/simplebluez/Agent.h +52 -0
  219. data/vendor/simpleble/simplebluez/include/simplebluez/Bluez.h +37 -0
  220. data/vendor/simpleble/simplebluez/include/simplebluez/BluezOrg.h +22 -0
  221. data/vendor/simpleble/simplebluez/include/simplebluez/BluezOrgBluez.h +26 -0
  222. data/vendor/simpleble/simplebluez/include/simplebluez/BluezRoot.h +32 -0
  223. data/vendor/simpleble/simplebluez/include/simplebluez/Characteristic.h +46 -0
  224. data/vendor/simpleble/simplebluez/include/simplebluez/Descriptor.h +33 -0
  225. data/vendor/simpleble/simplebluez/include/simplebluez/Device.h +64 -0
  226. data/vendor/simpleble/simplebluez/include/simplebluez/Exceptions.h +43 -0
  227. data/vendor/simpleble/simplebluez/include/simplebluez/Service.h +27 -0
  228. data/vendor/simpleble/simplebluez/include/simplebluez/Types.h +9 -0
  229. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/Adapter1.h +49 -0
  230. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/Agent1.h +100 -0
  231. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/AgentManager1.h +24 -0
  232. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/Battery1.h +32 -0
  233. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/Device1.h +62 -0
  234. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/GattCharacteristic1.h +48 -0
  235. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/GattDescriptor1.h +41 -0
  236. data/vendor/simpleble/simplebluez/include/simplebluez/interfaces/GattService1.h +28 -0
  237. data/vendor/simpleble/simplebluez/src/Adapter.cpp +78 -0
  238. data/vendor/simpleble/simplebluez/src/Agent.cpp +72 -0
  239. data/vendor/simpleble/simplebluez/src/Bluez.cpp +45 -0
  240. data/vendor/simpleble/simplebluez/src/BluezOrg.cpp +20 -0
  241. data/vendor/simpleble/simplebluez/src/BluezOrgBluez.cpp +23 -0
  242. data/vendor/simpleble/simplebluez/src/BluezRoot.cpp +44 -0
  243. data/vendor/simpleble/simplebluez/src/Characteristic.cpp +64 -0
  244. data/vendor/simpleble/simplebluez/src/Descriptor.cpp +27 -0
  245. data/vendor/simpleble/simplebluez/src/Device.cpp +104 -0
  246. data/vendor/simpleble/simplebluez/src/Exceptions.cpp +29 -0
  247. data/vendor/simpleble/simplebluez/src/Logging.cpp +52 -0
  248. data/vendor/simpleble/simplebluez/src/Logging.h +57 -0
  249. data/vendor/simpleble/simplebluez/src/Service.cpp +33 -0
  250. data/vendor/simpleble/simplebluez/src/interfaces/Adapter1.cpp +123 -0
  251. data/vendor/simpleble/simplebluez/src/interfaces/Agent1.cpp +143 -0
  252. data/vendor/simpleble/simplebluez/src/interfaces/AgentManager1.cpp +34 -0
  253. data/vendor/simpleble/simplebluez/src/interfaces/Battery1.cpp +30 -0
  254. data/vendor/simpleble/simplebluez/src/interfaces/Device1.cpp +170 -0
  255. data/vendor/simpleble/simplebluez/src/interfaces/GattCharacteristic1.cpp +118 -0
  256. data/vendor/simpleble/simplebluez/src/interfaces/GattDescriptor1.cpp +78 -0
  257. data/vendor/simpleble/simplebluez/src/interfaces/GattService1.cpp +28 -0
  258. data/vendor/simpleble/simplebluez/test/src/helpers/PythonRunner.cpp +53 -0
  259. data/vendor/simpleble/simplebluez/test/src/helpers/PythonRunner.h +24 -0
  260. data/vendor/simpleble/simplebluez/test/src/main.cpp +16 -0
  261. data/vendor/simpleble/simplecble/include/simplecble/adapter.h +188 -0
  262. data/vendor/simpleble/simplecble/include/simplecble/logging.h +37 -0
  263. data/vendor/simpleble/simplecble/include/simplecble/peripheral.h +304 -0
  264. data/vendor/simpleble/simplecble/include/simplecble/simpleble.h +21 -0
  265. data/vendor/simpleble/simplecble/include/simplecble/types.h +73 -0
  266. data/vendor/simpleble/simplecble/include/simplecble/utils.h +27 -0
  267. data/vendor/simpleble/simplecble/src/adapter.cpp +204 -0
  268. data/vendor/simpleble/simplecble/src/logging.cpp +19 -0
  269. data/vendor/simpleble/simplecble/src/peripheral.cpp +444 -0
  270. data/vendor/simpleble/simplecble/src/simpleble.cpp +5 -0
  271. data/vendor/simpleble/simplecble/src/utils.cpp +15 -0
  272. data/vendor/simpleble/simpledbus/include/simpledbus/advanced/Interface.h +67 -0
  273. data/vendor/simpleble/simpledbus/include/simpledbus/advanced/InterfaceRegistry.h +64 -0
  274. data/vendor/simpleble/simpledbus/include/simpledbus/advanced/Proxy.h +117 -0
  275. data/vendor/simpleble/simpledbus/include/simpledbus/base/Connection.h +50 -0
  276. data/vendor/simpleble/simpledbus/include/simpledbus/base/Exceptions.h +56 -0
  277. data/vendor/simpleble/simpledbus/include/simpledbus/base/Holder.h +147 -0
  278. data/vendor/simpleble/simpledbus/include/simpledbus/base/Logging.h +57 -0
  279. data/vendor/simpleble/simpledbus/include/simpledbus/base/Message.h +89 -0
  280. data/vendor/simpleble/simpledbus/include/simpledbus/base/Path.h +24 -0
  281. data/vendor/simpleble/simpledbus/include/simpledbus/interfaces/ObjectManager.h +26 -0
  282. data/vendor/simpleble/simpledbus/src/advanced/Interface.cpp +155 -0
  283. data/vendor/simpleble/simpledbus/src/advanced/Proxy.cpp +375 -0
  284. data/vendor/simpleble/simpledbus/src/base/Connection.cpp +222 -0
  285. data/vendor/simpleble/simpledbus/src/base/Exceptions.cpp +39 -0
  286. data/vendor/simpleble/simpledbus/src/base/Holder.cpp +739 -0
  287. data/vendor/simpleble/simpledbus/src/base/Logging.cpp +52 -0
  288. data/vendor/simpleble/simpledbus/src/base/Message.cpp +622 -0
  289. data/vendor/simpleble/simpledbus/src/base/Path.cpp +129 -0
  290. data/vendor/simpleble/simpledbus/src/interfaces/ObjectManager.cpp +58 -0
  291. data/vendor/simpleble/simpledbus/test/src/helpers/PythonRunner.cpp +53 -0
  292. data/vendor/simpleble/simpledbus/test/src/helpers/PythonRunner.h +24 -0
  293. data/vendor/simpleble/simpledbus/test/src/main.cpp +16 -0
  294. data/vendor/simpleble/simpledbus/test/src/test_holder.cpp +184 -0
  295. data/vendor/simpleble/simpledbus/test/src/test_message.cpp +397 -0
  296. data/vendor/simpleble/simpledbus/test/src/test_path.cpp +76 -0
  297. data/vendor/simpleble/simpledbus/test/src/test_proxy_children.cpp +109 -0
  298. data/vendor/simpleble/simpledbus/test/src/test_proxy_interfaces.cpp +102 -0
  299. data/vendor/simpleble/simpledbus/test/src/test_proxy_lifetime.cpp +18 -0
  300. data/vendor/simpleble/simpledroidble/simpledroidble/src/main/cpp/ThreadRunner.h +89 -0
  301. data/vendor/simpleble/simpledroidble/simpledroidble/src/main/cpp/android_utils.cpp +55 -0
  302. data/vendor/simpleble/simpledroidble/simpledroidble/src/main/cpp/android_utils.h +18 -0
  303. data/vendor/simpleble/simpledroidble/simpledroidble/src/main/cpp/simpleble_android.cpp +813 -0
  304. data/vendor/simpleble/simplejavable/cpp/src/core/AdapterWrapper.cpp +37 -0
  305. data/vendor/simpleble/simplejavable/cpp/src/core/AdapterWrapper.h +23 -0
  306. data/vendor/simpleble/simplejavable/cpp/src/core/Cache.cpp +58 -0
  307. data/vendor/simpleble/simplejavable/cpp/src/core/Cache.h +34 -0
  308. data/vendor/simpleble/simplejavable/cpp/src/core/PeripheralWrapper.cpp +17 -0
  309. data/vendor/simpleble/simplejavable/cpp/src/core/PeripheralWrapper.h +19 -0
  310. data/vendor/simpleble/simplejavable/cpp/src/java/lang/ArrayList.cpp +257 -0
  311. data/vendor/simpleble/simplejavable/cpp/src/java/lang/ArrayList.h +78 -0
  312. data/vendor/simpleble/simplejavable/cpp/src/java/lang/HashMap.cpp +96 -0
  313. data/vendor/simpleble/simplejavable/cpp/src/java/lang/HashMap.h +54 -0
  314. data/vendor/simpleble/simplejavable/cpp/src/java/lang/Integer.cpp +93 -0
  315. data/vendor/simpleble/simplejavable/cpp/src/java/lang/Integer.h +49 -0
  316. data/vendor/simpleble/simplejavable/cpp/src/java/lang/Iterator.cpp +85 -0
  317. data/vendor/simpleble/simplejavable/cpp/src/java/lang/Iterator.h +48 -0
  318. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/AdapterCallback.cpp +91 -0
  319. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/AdapterCallback.h +90 -0
  320. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Characteristic.cpp +35 -0
  321. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Characteristic.h +60 -0
  322. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/DataCallback.cpp +33 -0
  323. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/DataCallback.h +45 -0
  324. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Descriptor.cpp +30 -0
  325. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Descriptor.h +49 -0
  326. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/PeripheralCallback.cpp +41 -0
  327. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/PeripheralCallback.h +51 -0
  328. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Service.cpp +33 -0
  329. data/vendor/simpleble/simplejavable/cpp/src/org/simplejavable/Service.h +54 -0
  330. data/vendor/simpleble/simplejavable/cpp/src/simplejavable.cpp +361 -0
  331. data/vendor/simpleble/simplepyble/src/main.cpp +55 -0
  332. data/vendor/simpleble/simplepyble/src/wrap_adapter.cpp +106 -0
  333. data/vendor/simpleble/simplepyble/src/wrap_characteristic.cpp +56 -0
  334. data/vendor/simpleble/simplepyble/src/wrap_config.cpp +136 -0
  335. data/vendor/simpleble/simplepyble/src/wrap_descriptor.cpp +21 -0
  336. data/vendor/simpleble/simplepyble/src/wrap_peripheral.cpp +217 -0
  337. data/vendor/simpleble/simplepyble/src/wrap_service.cpp +32 -0
  338. data/vendor/simpleble/simplepyble/src/wrap_types.cpp +21 -0
  339. data/vendor/simpleble/simplersble/src/bindings/Bindings.cpp +338 -0
  340. data/vendor/simpleble/simplersble/src/bindings/Bindings.hpp +178 -0
  341. metadata +430 -0
@@ -0,0 +1,2338 @@
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
+ namespace fmt_detail {
26
+ struct time_zone {
27
+ template <typename Duration, typename T>
28
+ auto to_sys(T)
29
+ -> std::chrono::time_point<std::chrono::system_clock, Duration> {
30
+ return {};
31
+ }
32
+ };
33
+ template <typename... T> inline auto current_zone(T...) -> time_zone* {
34
+ return nullptr;
35
+ }
36
+
37
+ template <typename... T> inline void _tzset(T...) {}
38
+ } // namespace fmt_detail
39
+
40
+ FMT_BEGIN_NAMESPACE
41
+
42
+ // Enable safe chrono durations, unless explicitly disabled.
43
+ #ifndef FMT_SAFE_DURATION_CAST
44
+ # define FMT_SAFE_DURATION_CAST 1
45
+ #endif
46
+ #if FMT_SAFE_DURATION_CAST
47
+
48
+ // For conversion between std::chrono::durations without undefined
49
+ // behaviour or erroneous results.
50
+ // This is a stripped down version of duration_cast, for inclusion in fmt.
51
+ // See https://github.com/pauldreik/safe_duration_cast
52
+ //
53
+ // Copyright Paul Dreik 2019
54
+ namespace safe_duration_cast {
55
+
56
+ template <typename To, typename From,
57
+ FMT_ENABLE_IF(!std::is_same<From, To>::value &&
58
+ std::numeric_limits<From>::is_signed ==
59
+ std::numeric_limits<To>::is_signed)>
60
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
61
+ -> To {
62
+ ec = 0;
63
+ using F = std::numeric_limits<From>;
64
+ using T = std::numeric_limits<To>;
65
+ static_assert(F::is_integer, "From must be integral");
66
+ static_assert(T::is_integer, "To must be integral");
67
+
68
+ // A and B are both signed, or both unsigned.
69
+ if (detail::const_check(F::digits <= T::digits)) {
70
+ // From fits in To without any problem.
71
+ } else {
72
+ // From does not always fit in To, resort to a dynamic check.
73
+ if (from < (T::min)() || from > (T::max)()) {
74
+ // outside range.
75
+ ec = 1;
76
+ return {};
77
+ }
78
+ }
79
+ return static_cast<To>(from);
80
+ }
81
+
82
+ /// Converts From to To, without loss. If the dynamic value of from
83
+ /// can't be converted to To without loss, ec is set.
84
+ template <typename To, typename From,
85
+ FMT_ENABLE_IF(!std::is_same<From, To>::value &&
86
+ std::numeric_limits<From>::is_signed !=
87
+ std::numeric_limits<To>::is_signed)>
88
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
89
+ -> To {
90
+ ec = 0;
91
+ using F = std::numeric_limits<From>;
92
+ using T = std::numeric_limits<To>;
93
+ static_assert(F::is_integer, "From must be integral");
94
+ static_assert(T::is_integer, "To must be integral");
95
+
96
+ if (detail::const_check(F::is_signed && !T::is_signed)) {
97
+ // From may be negative, not allowed!
98
+ if (fmt::detail::is_negative(from)) {
99
+ ec = 1;
100
+ return {};
101
+ }
102
+ // From is positive. Can it always fit in To?
103
+ if (detail::const_check(F::digits > T::digits) &&
104
+ from > static_cast<From>(detail::max_value<To>())) {
105
+ ec = 1;
106
+ return {};
107
+ }
108
+ }
109
+
110
+ if (detail::const_check(!F::is_signed && T::is_signed &&
111
+ F::digits >= T::digits) &&
112
+ from > static_cast<From>(detail::max_value<To>())) {
113
+ ec = 1;
114
+ return {};
115
+ }
116
+ return static_cast<To>(from); // Lossless conversion.
117
+ }
118
+
119
+ template <typename To, typename From,
120
+ FMT_ENABLE_IF(std::is_same<From, To>::value)>
121
+ FMT_CONSTEXPR auto lossless_integral_conversion(const From from, int& ec)
122
+ -> To {
123
+ ec = 0;
124
+ return from;
125
+ } // function
126
+
127
+ // clang-format off
128
+ /**
129
+ * converts From to To if possible, otherwise ec is set.
130
+ *
131
+ * input | output
132
+ * ---------------------------------|---------------
133
+ * NaN | NaN
134
+ * Inf | Inf
135
+ * normal, fits in output | converted (possibly lossy)
136
+ * normal, does not fit in output | ec is set
137
+ * subnormal | best effort
138
+ * -Inf | -Inf
139
+ */
140
+ // clang-format on
141
+ template <typename To, typename From,
142
+ FMT_ENABLE_IF(!std::is_same<From, To>::value)>
143
+ FMT_CONSTEXPR auto safe_float_conversion(const From from, int& ec) -> To {
144
+ ec = 0;
145
+ using T = std::numeric_limits<To>;
146
+ static_assert(std::is_floating_point<From>::value, "From must be floating");
147
+ static_assert(std::is_floating_point<To>::value, "To must be floating");
148
+
149
+ // catch the only happy case
150
+ if (std::isfinite(from)) {
151
+ if (from >= T::lowest() && from <= (T::max)()) {
152
+ return static_cast<To>(from);
153
+ }
154
+ // not within range.
155
+ ec = 1;
156
+ return {};
157
+ }
158
+
159
+ // nan and inf will be preserved
160
+ return static_cast<To>(from);
161
+ } // function
162
+
163
+ template <typename To, typename From,
164
+ FMT_ENABLE_IF(std::is_same<From, To>::value)>
165
+ FMT_CONSTEXPR auto safe_float_conversion(const From from, int& ec) -> To {
166
+ ec = 0;
167
+ static_assert(std::is_floating_point<From>::value, "From must be floating");
168
+ return from;
169
+ }
170
+
171
+ /// Safe duration_cast between floating point durations
172
+ template <typename To, typename FromRep, typename FromPeriod,
173
+ FMT_ENABLE_IF(std::is_floating_point<FromRep>::value),
174
+ FMT_ENABLE_IF(std::is_floating_point<typename To::rep>::value)>
175
+ auto safe_duration_cast(std::chrono::duration<FromRep, FromPeriod> from,
176
+ int& ec) -> To {
177
+ using From = std::chrono::duration<FromRep, FromPeriod>;
178
+ ec = 0;
179
+ if (std::isnan(from.count())) {
180
+ // nan in, gives nan out. easy.
181
+ return To{std::numeric_limits<typename To::rep>::quiet_NaN()};
182
+ }
183
+ // maybe we should also check if from is denormal, and decide what to do about
184
+ // it.
185
+
186
+ // +-inf should be preserved.
187
+ if (std::isinf(from.count())) {
188
+ return To{from.count()};
189
+ }
190
+
191
+ // the basic idea is that we need to convert from count() in the from type
192
+ // to count() in the To type, by multiplying it with this:
193
+ struct Factor
194
+ : std::ratio_divide<typename From::period, typename To::period> {};
195
+
196
+ static_assert(Factor::num > 0, "num must be positive");
197
+ static_assert(Factor::den > 0, "den must be positive");
198
+
199
+ // the conversion is like this: multiply from.count() with Factor::num
200
+ // /Factor::den and convert it to To::rep, all this without
201
+ // overflow/underflow. let's start by finding a suitable type that can hold
202
+ // both To, From and Factor::num
203
+ using IntermediateRep =
204
+ typename std::common_type<typename From::rep, typename To::rep,
205
+ decltype(Factor::num)>::type;
206
+
207
+ // force conversion of From::rep -> IntermediateRep to be safe,
208
+ // even if it will never happen be narrowing in this context.
209
+ IntermediateRep count =
210
+ safe_float_conversion<IntermediateRep>(from.count(), ec);
211
+ if (ec) {
212
+ return {};
213
+ }
214
+
215
+ // multiply with Factor::num without overflow or underflow
216
+ if (detail::const_check(Factor::num != 1)) {
217
+ constexpr auto max1 = detail::max_value<IntermediateRep>() /
218
+ static_cast<IntermediateRep>(Factor::num);
219
+ if (count > max1) {
220
+ ec = 1;
221
+ return {};
222
+ }
223
+ constexpr auto min1 = std::numeric_limits<IntermediateRep>::lowest() /
224
+ static_cast<IntermediateRep>(Factor::num);
225
+ if (count < min1) {
226
+ ec = 1;
227
+ return {};
228
+ }
229
+ count *= static_cast<IntermediateRep>(Factor::num);
230
+ }
231
+
232
+ // this can't go wrong, right? den>0 is checked earlier.
233
+ if (detail::const_check(Factor::den != 1)) {
234
+ using common_t = typename std::common_type<IntermediateRep, intmax_t>::type;
235
+ count /= static_cast<common_t>(Factor::den);
236
+ }
237
+
238
+ // convert to the to type, safely
239
+ using ToRep = typename To::rep;
240
+
241
+ const ToRep tocount = safe_float_conversion<ToRep>(count, ec);
242
+ if (ec) {
243
+ return {};
244
+ }
245
+ return To{tocount};
246
+ }
247
+ } // namespace safe_duration_cast
248
+ #endif
249
+
250
+ namespace detail {
251
+
252
+ // Check if std::chrono::utc_time is available.
253
+ #ifdef FMT_USE_UTC_TIME
254
+ // Use the provided definition.
255
+ #elif defined(__cpp_lib_chrono)
256
+ # define FMT_USE_UTC_TIME (__cpp_lib_chrono >= 201907L)
257
+ #else
258
+ # define FMT_USE_UTC_TIME 0
259
+ #endif
260
+ #if FMT_USE_UTC_TIME
261
+ using utc_clock = std::chrono::utc_clock;
262
+ #else
263
+ struct utc_clock {
264
+ template <typename T> void to_sys(T);
265
+ };
266
+ #endif
267
+
268
+ // Check if std::chrono::local_time is available.
269
+ #ifdef FMT_USE_LOCAL_TIME
270
+ // Use the provided definition.
271
+ #elif defined(__cpp_lib_chrono)
272
+ # define FMT_USE_LOCAL_TIME (__cpp_lib_chrono >= 201907L)
273
+ #else
274
+ # define FMT_USE_LOCAL_TIME 0
275
+ #endif
276
+ #if FMT_USE_LOCAL_TIME
277
+ using local_t = std::chrono::local_t;
278
+ #else
279
+ struct local_t {};
280
+ #endif
281
+
282
+ } // namespace detail
283
+
284
+ template <typename Duration>
285
+ using sys_time = std::chrono::time_point<std::chrono::system_clock, Duration>;
286
+
287
+ template <typename Duration>
288
+ using utc_time = std::chrono::time_point<detail::utc_clock, Duration>;
289
+
290
+ template <class Duration>
291
+ using local_time = std::chrono::time_point<detail::local_t, Duration>;
292
+
293
+ namespace detail {
294
+
295
+ // Prevents expansion of a preceding token as a function-style macro.
296
+ // Usage: f FMT_NOMACRO()
297
+ #define FMT_NOMACRO
298
+
299
+ template <typename T = void> struct null {};
300
+ inline auto localtime_r FMT_NOMACRO(...) -> null<> { return null<>(); }
301
+ inline auto localtime_s(...) -> null<> { return null<>(); }
302
+ inline auto gmtime_r(...) -> null<> { return null<>(); }
303
+ inline auto gmtime_s(...) -> null<> { return null<>(); }
304
+
305
+ // It is defined here and not in ostream.h because the latter has expensive
306
+ // includes.
307
+ template <typename StreamBuf> class formatbuf : public StreamBuf {
308
+ private:
309
+ using char_type = typename StreamBuf::char_type;
310
+ using streamsize = decltype(std::declval<StreamBuf>().sputn(nullptr, 0));
311
+ using int_type = typename StreamBuf::int_type;
312
+ using traits_type = typename StreamBuf::traits_type;
313
+
314
+ buffer<char_type>& buffer_;
315
+
316
+ public:
317
+ explicit formatbuf(buffer<char_type>& buf) : buffer_(buf) {}
318
+
319
+ protected:
320
+ // The put area is always empty. This makes the implementation simpler and has
321
+ // the advantage that the streambuf and the buffer are always in sync and
322
+ // sputc never writes into uninitialized memory. A disadvantage is that each
323
+ // call to sputc always results in a (virtual) call to overflow. There is no
324
+ // disadvantage here for sputn since this always results in a call to xsputn.
325
+
326
+ auto overflow(int_type ch) -> int_type override {
327
+ if (!traits_type::eq_int_type(ch, traits_type::eof()))
328
+ buffer_.push_back(static_cast<char_type>(ch));
329
+ return ch;
330
+ }
331
+
332
+ auto xsputn(const char_type* s, streamsize count) -> streamsize override {
333
+ buffer_.append(s, s + count);
334
+ return count;
335
+ }
336
+ };
337
+
338
+ inline auto get_classic_locale() -> const std::locale& {
339
+ static const auto& locale = std::locale::classic();
340
+ return locale;
341
+ }
342
+
343
+ template <typename CodeUnit> struct codecvt_result {
344
+ static constexpr const size_t max_size = 32;
345
+ CodeUnit buf[max_size];
346
+ CodeUnit* end;
347
+ };
348
+
349
+ template <typename CodeUnit>
350
+ void write_codecvt(codecvt_result<CodeUnit>& out, string_view in,
351
+ const std::locale& loc) {
352
+ FMT_PRAGMA_CLANG(diagnostic push)
353
+ FMT_PRAGMA_CLANG(diagnostic ignored "-Wdeprecated")
354
+ auto& f = std::use_facet<std::codecvt<CodeUnit, char, std::mbstate_t>>(loc);
355
+ FMT_PRAGMA_CLANG(diagnostic pop)
356
+ auto mb = std::mbstate_t();
357
+ const char* from_next = nullptr;
358
+ auto result = f.in(mb, in.begin(), in.end(), from_next, std::begin(out.buf),
359
+ std::end(out.buf), out.end);
360
+ if (result != std::codecvt_base::ok)
361
+ FMT_THROW(format_error("failed to format time"));
362
+ }
363
+
364
+ template <typename OutputIt>
365
+ auto write_encoded_tm_str(OutputIt out, string_view in, const std::locale& loc)
366
+ -> OutputIt {
367
+ if (const_check(detail::use_utf8) && loc != get_classic_locale()) {
368
+ // char16_t and char32_t codecvts are broken in MSVC (linkage errors) and
369
+ // gcc-4.
370
+ #if FMT_MSC_VERSION != 0 || \
371
+ (defined(__GLIBCXX__) && \
372
+ (!defined(_GLIBCXX_USE_DUAL_ABI) || _GLIBCXX_USE_DUAL_ABI == 0))
373
+ // The _GLIBCXX_USE_DUAL_ABI macro is always defined in libstdc++ from gcc-5
374
+ // and newer.
375
+ using code_unit = wchar_t;
376
+ #else
377
+ using code_unit = char32_t;
378
+ #endif
379
+
380
+ using unit_t = codecvt_result<code_unit>;
381
+ unit_t unit;
382
+ write_codecvt(unit, in, loc);
383
+ // In UTF-8 is used one to four one-byte code units.
384
+ auto u =
385
+ to_utf8<code_unit, basic_memory_buffer<char, unit_t::max_size * 4>>();
386
+ if (!u.convert({unit.buf, to_unsigned(unit.end - unit.buf)}))
387
+ FMT_THROW(format_error("failed to format time"));
388
+ return copy<char>(u.c_str(), u.c_str() + u.size(), out);
389
+ }
390
+ return copy<char>(in.data(), in.data() + in.size(), out);
391
+ }
392
+
393
+ template <typename Char, typename OutputIt,
394
+ FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
395
+ auto write_tm_str(OutputIt out, string_view sv, const std::locale& loc)
396
+ -> OutputIt {
397
+ codecvt_result<Char> unit;
398
+ write_codecvt(unit, sv, loc);
399
+ return copy<Char>(unit.buf, unit.end, out);
400
+ }
401
+
402
+ template <typename Char, typename OutputIt,
403
+ FMT_ENABLE_IF(std::is_same<Char, char>::value)>
404
+ auto write_tm_str(OutputIt out, string_view sv, const std::locale& loc)
405
+ -> OutputIt {
406
+ return write_encoded_tm_str(out, sv, loc);
407
+ }
408
+
409
+ template <typename Char>
410
+ inline void do_write(buffer<Char>& buf, const std::tm& time,
411
+ const std::locale& loc, char format, char modifier) {
412
+ auto&& format_buf = formatbuf<std::basic_streambuf<Char>>(buf);
413
+ auto&& os = std::basic_ostream<Char>(&format_buf);
414
+ os.imbue(loc);
415
+ const auto& facet = std::use_facet<std::time_put<Char>>(loc);
416
+ auto end = facet.put(os, os, Char(' '), &time, format, modifier);
417
+ if (end.failed()) FMT_THROW(format_error("failed to format time"));
418
+ }
419
+
420
+ template <typename Char, typename OutputIt,
421
+ FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
422
+ auto write(OutputIt out, const std::tm& time, const std::locale& loc,
423
+ char format, char modifier = 0) -> OutputIt {
424
+ auto&& buf = get_buffer<Char>(out);
425
+ do_write<Char>(buf, time, loc, format, modifier);
426
+ return get_iterator(buf, out);
427
+ }
428
+
429
+ template <typename Char, typename OutputIt,
430
+ FMT_ENABLE_IF(std::is_same<Char, char>::value)>
431
+ auto write(OutputIt out, const std::tm& time, const std::locale& loc,
432
+ char format, char modifier = 0) -> OutputIt {
433
+ auto&& buf = basic_memory_buffer<Char>();
434
+ do_write<char>(buf, time, loc, format, modifier);
435
+ return write_encoded_tm_str(out, string_view(buf.data(), buf.size()), loc);
436
+ }
437
+
438
+ template <typename Rep1, typename Rep2>
439
+ struct is_same_arithmetic_type
440
+ : public std::integral_constant<bool,
441
+ (std::is_integral<Rep1>::value &&
442
+ std::is_integral<Rep2>::value) ||
443
+ (std::is_floating_point<Rep1>::value &&
444
+ std::is_floating_point<Rep2>::value)> {
445
+ };
446
+
447
+ FMT_NORETURN inline void throw_duration_error() {
448
+ FMT_THROW(format_error("cannot format duration"));
449
+ }
450
+
451
+ // Cast one integral duration to another with an overflow check.
452
+ template <typename To, typename FromRep, typename FromPeriod,
453
+ FMT_ENABLE_IF(std::is_integral<FromRep>::value&&
454
+ std::is_integral<typename To::rep>::value)>
455
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
456
+ #if !FMT_SAFE_DURATION_CAST
457
+ return std::chrono::duration_cast<To>(from);
458
+ #else
459
+ // The conversion factor: to.count() == factor * from.count().
460
+ using factor = std::ratio_divide<FromPeriod, typename To::period>;
461
+
462
+ using common_rep = typename std::common_type<FromRep, typename To::rep,
463
+ decltype(factor::num)>::type;
464
+
465
+ int ec = 0;
466
+ auto count = safe_duration_cast::lossless_integral_conversion<common_rep>(
467
+ from.count(), ec);
468
+ if (ec) throw_duration_error();
469
+
470
+ // Multiply from.count() by factor and check for overflow.
471
+ if (const_check(factor::num != 1)) {
472
+ if (count > max_value<common_rep>() / factor::num) throw_duration_error();
473
+ const auto min = (std::numeric_limits<common_rep>::min)() / factor::num;
474
+ if (const_check(!std::is_unsigned<common_rep>::value) && count < min)
475
+ throw_duration_error();
476
+ count *= factor::num;
477
+ }
478
+ if (const_check(factor::den != 1)) count /= factor::den;
479
+ auto to =
480
+ To(safe_duration_cast::lossless_integral_conversion<typename To::rep>(
481
+ count, ec));
482
+ if (ec) throw_duration_error();
483
+ return to;
484
+ #endif
485
+ }
486
+
487
+ template <typename To, typename FromRep, typename FromPeriod,
488
+ FMT_ENABLE_IF(std::is_floating_point<FromRep>::value&&
489
+ std::is_floating_point<typename To::rep>::value)>
490
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
491
+ #if FMT_SAFE_DURATION_CAST
492
+ // Throwing version of safe_duration_cast is only available for
493
+ // integer to integer or float to float casts.
494
+ int ec;
495
+ To to = safe_duration_cast::safe_duration_cast<To>(from, ec);
496
+ if (ec) throw_duration_error();
497
+ return to;
498
+ #else
499
+ // Standard duration cast, may overflow.
500
+ return std::chrono::duration_cast<To>(from);
501
+ #endif
502
+ }
503
+
504
+ template <
505
+ typename To, typename FromRep, typename FromPeriod,
506
+ FMT_ENABLE_IF(!is_same_arithmetic_type<FromRep, typename To::rep>::value)>
507
+ auto duration_cast(std::chrono::duration<FromRep, FromPeriod> from) -> To {
508
+ // Mixed integer <-> float cast is not supported by safe_duration_cast.
509
+ return std::chrono::duration_cast<To>(from);
510
+ }
511
+
512
+ template <typename Duration>
513
+ auto to_time_t(sys_time<Duration> time_point) -> std::time_t {
514
+ // Cannot use std::chrono::system_clock::to_time_t since this would first
515
+ // require a cast to std::chrono::system_clock::time_point, which could
516
+ // overflow.
517
+ return detail::duration_cast<std::chrono::duration<std::time_t>>(
518
+ time_point.time_since_epoch())
519
+ .count();
520
+ }
521
+
522
+ // Workaround a bug in libstdc++ which sets __cpp_lib_chrono to 201907 without
523
+ // providing current_zone(): https://github.com/fmtlib/fmt/issues/4160.
524
+ template <typename T> FMT_CONSTEXPR auto has_current_zone() -> bool {
525
+ using namespace std::chrono;
526
+ using namespace fmt_detail;
527
+ return !std::is_same<decltype(current_zone()), fmt_detail::time_zone*>::value;
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
+ 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_ENABLE_IF(detail::has_current_zone<Duration>())>
577
+ inline auto localtime(std::chrono::local_time<Duration> time) -> std::tm {
578
+ using namespace std::chrono;
579
+ using namespace fmt_detail;
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
+ struct tm_format_checker : null_chrono_spec_handler<tm_format_checker> {
915
+ FMT_NORETURN inline void unsupported() {
916
+ FMT_THROW(format_error("no format"));
917
+ }
918
+
919
+ template <typename Char>
920
+ FMT_CONSTEXPR void on_text(const Char*, const Char*) {}
921
+ FMT_CONSTEXPR void on_year(numeric_system, pad_type) {}
922
+ FMT_CONSTEXPR void on_short_year(numeric_system) {}
923
+ FMT_CONSTEXPR void on_offset_year() {}
924
+ FMT_CONSTEXPR void on_century(numeric_system) {}
925
+ FMT_CONSTEXPR void on_iso_week_based_year() {}
926
+ FMT_CONSTEXPR void on_iso_week_based_short_year() {}
927
+ FMT_CONSTEXPR void on_abbr_weekday() {}
928
+ FMT_CONSTEXPR void on_full_weekday() {}
929
+ FMT_CONSTEXPR void on_dec0_weekday(numeric_system) {}
930
+ FMT_CONSTEXPR void on_dec1_weekday(numeric_system) {}
931
+ FMT_CONSTEXPR void on_abbr_month() {}
932
+ FMT_CONSTEXPR void on_full_month() {}
933
+ FMT_CONSTEXPR void on_dec_month(numeric_system, pad_type) {}
934
+ FMT_CONSTEXPR void on_dec0_week_of_year(numeric_system, pad_type) {}
935
+ FMT_CONSTEXPR void on_dec1_week_of_year(numeric_system, pad_type) {}
936
+ FMT_CONSTEXPR void on_iso_week_of_year(numeric_system, pad_type) {}
937
+ FMT_CONSTEXPR void on_day_of_year(pad_type) {}
938
+ FMT_CONSTEXPR void on_day_of_month(numeric_system, pad_type) {}
939
+ FMT_CONSTEXPR void on_24_hour(numeric_system, pad_type) {}
940
+ FMT_CONSTEXPR void on_12_hour(numeric_system, pad_type) {}
941
+ FMT_CONSTEXPR void on_minute(numeric_system, pad_type) {}
942
+ FMT_CONSTEXPR void on_second(numeric_system, pad_type) {}
943
+ FMT_CONSTEXPR void on_datetime(numeric_system) {}
944
+ FMT_CONSTEXPR void on_loc_date(numeric_system) {}
945
+ FMT_CONSTEXPR void on_loc_time(numeric_system) {}
946
+ FMT_CONSTEXPR void on_us_date() {}
947
+ FMT_CONSTEXPR void on_iso_date() {}
948
+ FMT_CONSTEXPR void on_12_hour_time() {}
949
+ FMT_CONSTEXPR void on_24_hour_time() {}
950
+ FMT_CONSTEXPR void on_iso_time() {}
951
+ FMT_CONSTEXPR void on_am_pm() {}
952
+ FMT_CONSTEXPR void on_utc_offset(numeric_system) {}
953
+ FMT_CONSTEXPR void on_tz_name() {}
954
+ };
955
+
956
+ inline auto tm_wday_full_name(int wday) -> const char* {
957
+ static constexpr const char* full_name_list[] = {
958
+ "Sunday", "Monday", "Tuesday", "Wednesday",
959
+ "Thursday", "Friday", "Saturday"};
960
+ return wday >= 0 && wday <= 6 ? full_name_list[wday] : "?";
961
+ }
962
+ inline auto tm_wday_short_name(int wday) -> const char* {
963
+ static constexpr const char* short_name_list[] = {"Sun", "Mon", "Tue", "Wed",
964
+ "Thu", "Fri", "Sat"};
965
+ return wday >= 0 && wday <= 6 ? short_name_list[wday] : "???";
966
+ }
967
+
968
+ inline auto tm_mon_full_name(int mon) -> const char* {
969
+ static constexpr const char* full_name_list[] = {
970
+ "January", "February", "March", "April", "May", "June",
971
+ "July", "August", "September", "October", "November", "December"};
972
+ return mon >= 0 && mon <= 11 ? full_name_list[mon] : "?";
973
+ }
974
+ inline auto tm_mon_short_name(int mon) -> const char* {
975
+ static constexpr const char* short_name_list[] = {
976
+ "Jan", "Feb", "Mar", "Apr", "May", "Jun",
977
+ "Jul", "Aug", "Sep", "Oct", "Nov", "Dec",
978
+ };
979
+ return mon >= 0 && mon <= 11 ? short_name_list[mon] : "???";
980
+ }
981
+
982
+ template <typename T, typename = void>
983
+ struct has_member_data_tm_gmtoff : std::false_type {};
984
+ template <typename T>
985
+ struct has_member_data_tm_gmtoff<T, void_t<decltype(T::tm_gmtoff)>>
986
+ : std::true_type {};
987
+
988
+ template <typename T, typename = void>
989
+ struct has_member_data_tm_zone : std::false_type {};
990
+ template <typename T>
991
+ struct has_member_data_tm_zone<T, void_t<decltype(T::tm_zone)>>
992
+ : std::true_type {};
993
+
994
+ inline void tzset_once() {
995
+ static bool init = []() {
996
+ using namespace fmt_detail;
997
+ _tzset();
998
+ return false;
999
+ }();
1000
+ ignore_unused(init);
1001
+ }
1002
+
1003
+ // Converts value to Int and checks that it's in the range [0, upper).
1004
+ template <typename T, typename Int, FMT_ENABLE_IF(std::is_integral<T>::value)>
1005
+ inline auto to_nonnegative_int(T value, Int upper) -> Int {
1006
+ if (!std::is_unsigned<Int>::value &&
1007
+ (value < 0 || to_unsigned(value) > to_unsigned(upper))) {
1008
+ FMT_THROW(fmt::format_error("chrono value is out of range"));
1009
+ }
1010
+ return static_cast<Int>(value);
1011
+ }
1012
+ template <typename T, typename Int, FMT_ENABLE_IF(!std::is_integral<T>::value)>
1013
+ inline auto to_nonnegative_int(T value, Int upper) -> Int {
1014
+ auto int_value = static_cast<Int>(value);
1015
+ if (int_value < 0 || value > static_cast<T>(upper))
1016
+ FMT_THROW(format_error("invalid value"));
1017
+ return int_value;
1018
+ }
1019
+
1020
+ constexpr auto pow10(std::uint32_t n) -> long long {
1021
+ return n == 0 ? 1 : 10 * pow10(n - 1);
1022
+ }
1023
+
1024
+ // Counts the number of fractional digits in the range [0, 18] according to the
1025
+ // C++20 spec. If more than 18 fractional digits are required then returns 6 for
1026
+ // microseconds precision.
1027
+ template <long long Num, long long Den, int N = 0,
1028
+ bool Enabled = (N < 19) && (Num <= max_value<long long>() / 10)>
1029
+ struct count_fractional_digits {
1030
+ static constexpr int value =
1031
+ Num % Den == 0 ? N : count_fractional_digits<Num * 10, Den, N + 1>::value;
1032
+ };
1033
+
1034
+ // Base case that doesn't instantiate any more templates
1035
+ // in order to avoid overflow.
1036
+ template <long long Num, long long Den, int N>
1037
+ struct count_fractional_digits<Num, Den, N, false> {
1038
+ static constexpr int value = (Num % Den == 0) ? N : 6;
1039
+ };
1040
+
1041
+ // Format subseconds which are given as an integer type with an appropriate
1042
+ // number of digits.
1043
+ template <typename Char, typename OutputIt, typename Duration>
1044
+ void write_fractional_seconds(OutputIt& out, Duration d, int precision = -1) {
1045
+ constexpr auto num_fractional_digits =
1046
+ count_fractional_digits<Duration::period::num,
1047
+ Duration::period::den>::value;
1048
+
1049
+ using subsecond_precision = std::chrono::duration<
1050
+ typename std::common_type<typename Duration::rep,
1051
+ std::chrono::seconds::rep>::type,
1052
+ std::ratio<1, pow10(num_fractional_digits)>>;
1053
+
1054
+ const auto fractional = d - detail::duration_cast<std::chrono::seconds>(d);
1055
+ const auto subseconds =
1056
+ std::chrono::treat_as_floating_point<
1057
+ typename subsecond_precision::rep>::value
1058
+ ? fractional.count()
1059
+ : detail::duration_cast<subsecond_precision>(fractional).count();
1060
+ auto n = static_cast<uint32_or_64_or_128_t<long long>>(subseconds);
1061
+ const int num_digits = count_digits(n);
1062
+
1063
+ int leading_zeroes = (std::max)(0, num_fractional_digits - num_digits);
1064
+ if (precision < 0) {
1065
+ FMT_ASSERT(!std::is_floating_point<typename Duration::rep>::value, "");
1066
+ if (std::ratio_less<typename subsecond_precision::period,
1067
+ std::chrono::seconds::period>::value) {
1068
+ *out++ = '.';
1069
+ out = detail::fill_n(out, leading_zeroes, '0');
1070
+ out = format_decimal<Char>(out, n, num_digits);
1071
+ }
1072
+ } else if (precision > 0) {
1073
+ *out++ = '.';
1074
+ leading_zeroes = min_of(leading_zeroes, precision);
1075
+ int remaining = precision - leading_zeroes;
1076
+ out = detail::fill_n(out, leading_zeroes, '0');
1077
+ if (remaining < num_digits) {
1078
+ int num_truncated_digits = num_digits - remaining;
1079
+ n /= to_unsigned(pow10(to_unsigned(num_truncated_digits)));
1080
+ if (n != 0) out = format_decimal<Char>(out, n, remaining);
1081
+ return;
1082
+ }
1083
+ if (n != 0) {
1084
+ out = format_decimal<Char>(out, n, num_digits);
1085
+ remaining -= num_digits;
1086
+ }
1087
+ out = detail::fill_n(out, remaining, '0');
1088
+ }
1089
+ }
1090
+
1091
+ // Format subseconds which are given as a floating point type with an
1092
+ // appropriate number of digits. We cannot pass the Duration here, as we
1093
+ // explicitly need to pass the Rep value in the chrono_formatter.
1094
+ template <typename Duration>
1095
+ void write_floating_seconds(memory_buffer& buf, Duration duration,
1096
+ int num_fractional_digits = -1) {
1097
+ using rep = typename Duration::rep;
1098
+ FMT_ASSERT(std::is_floating_point<rep>::value, "");
1099
+
1100
+ auto val = duration.count();
1101
+
1102
+ if (num_fractional_digits < 0) {
1103
+ // For `std::round` with fallback to `round`:
1104
+ // On some toolchains `std::round` is not available (e.g. GCC 6).
1105
+ using namespace std;
1106
+ num_fractional_digits =
1107
+ count_fractional_digits<Duration::period::num,
1108
+ Duration::period::den>::value;
1109
+ if (num_fractional_digits < 6 && static_cast<rep>(round(val)) != val)
1110
+ num_fractional_digits = 6;
1111
+ }
1112
+
1113
+ fmt::format_to(std::back_inserter(buf), FMT_STRING("{:.{}f}"),
1114
+ std::fmod(val * static_cast<rep>(Duration::period::num) /
1115
+ static_cast<rep>(Duration::period::den),
1116
+ static_cast<rep>(60)),
1117
+ num_fractional_digits);
1118
+ }
1119
+
1120
+ template <typename OutputIt, typename Char,
1121
+ typename Duration = std::chrono::seconds>
1122
+ class tm_writer {
1123
+ private:
1124
+ static constexpr int days_per_week = 7;
1125
+
1126
+ const std::locale& loc_;
1127
+ const bool is_classic_;
1128
+ OutputIt out_;
1129
+ const Duration* subsecs_;
1130
+ const std::tm& tm_;
1131
+
1132
+ auto tm_sec() const noexcept -> int {
1133
+ FMT_ASSERT(tm_.tm_sec >= 0 && tm_.tm_sec <= 61, "");
1134
+ return tm_.tm_sec;
1135
+ }
1136
+ auto tm_min() const noexcept -> int {
1137
+ FMT_ASSERT(tm_.tm_min >= 0 && tm_.tm_min <= 59, "");
1138
+ return tm_.tm_min;
1139
+ }
1140
+ auto tm_hour() const noexcept -> int {
1141
+ FMT_ASSERT(tm_.tm_hour >= 0 && tm_.tm_hour <= 23, "");
1142
+ return tm_.tm_hour;
1143
+ }
1144
+ auto tm_mday() const noexcept -> int {
1145
+ FMT_ASSERT(tm_.tm_mday >= 1 && tm_.tm_mday <= 31, "");
1146
+ return tm_.tm_mday;
1147
+ }
1148
+ auto tm_mon() const noexcept -> int {
1149
+ FMT_ASSERT(tm_.tm_mon >= 0 && tm_.tm_mon <= 11, "");
1150
+ return tm_.tm_mon;
1151
+ }
1152
+ auto tm_year() const noexcept -> long long { return 1900ll + tm_.tm_year; }
1153
+ auto tm_wday() const noexcept -> int {
1154
+ FMT_ASSERT(tm_.tm_wday >= 0 && tm_.tm_wday <= 6, "");
1155
+ return tm_.tm_wday;
1156
+ }
1157
+ auto tm_yday() const noexcept -> int {
1158
+ FMT_ASSERT(tm_.tm_yday >= 0 && tm_.tm_yday <= 365, "");
1159
+ return tm_.tm_yday;
1160
+ }
1161
+
1162
+ auto tm_hour12() const noexcept -> int {
1163
+ const auto h = tm_hour();
1164
+ const auto z = h < 12 ? h : h - 12;
1165
+ return z == 0 ? 12 : z;
1166
+ }
1167
+
1168
+ // POSIX and the C Standard are unclear or inconsistent about what %C and %y
1169
+ // do if the year is negative or exceeds 9999. Use the convention that %C
1170
+ // concatenated with %y yields the same output as %Y, and that %Y contains at
1171
+ // least 4 characters, with more only if necessary.
1172
+ auto split_year_lower(long long year) const noexcept -> int {
1173
+ auto l = year % 100;
1174
+ if (l < 0) l = -l; // l in [0, 99]
1175
+ return static_cast<int>(l);
1176
+ }
1177
+
1178
+ // Algorithm: https://en.wikipedia.org/wiki/ISO_week_date.
1179
+ auto iso_year_weeks(long long curr_year) const noexcept -> int {
1180
+ const auto prev_year = curr_year - 1;
1181
+ const auto curr_p =
1182
+ (curr_year + curr_year / 4 - curr_year / 100 + curr_year / 400) %
1183
+ days_per_week;
1184
+ const auto prev_p =
1185
+ (prev_year + prev_year / 4 - prev_year / 100 + prev_year / 400) %
1186
+ days_per_week;
1187
+ return 52 + ((curr_p == 4 || prev_p == 3) ? 1 : 0);
1188
+ }
1189
+ auto iso_week_num(int tm_yday, int tm_wday) const noexcept -> int {
1190
+ return (tm_yday + 11 - (tm_wday == 0 ? days_per_week : tm_wday)) /
1191
+ days_per_week;
1192
+ }
1193
+ auto tm_iso_week_year() const noexcept -> long long {
1194
+ const auto year = tm_year();
1195
+ const auto w = iso_week_num(tm_yday(), tm_wday());
1196
+ if (w < 1) return year - 1;
1197
+ if (w > iso_year_weeks(year)) return year + 1;
1198
+ return year;
1199
+ }
1200
+ auto tm_iso_week_of_year() const noexcept -> int {
1201
+ const auto year = tm_year();
1202
+ const auto w = iso_week_num(tm_yday(), tm_wday());
1203
+ if (w < 1) return iso_year_weeks(year - 1);
1204
+ if (w > iso_year_weeks(year)) return 1;
1205
+ return w;
1206
+ }
1207
+
1208
+ void write1(int value) {
1209
+ *out_++ = static_cast<char>('0' + to_unsigned(value) % 10);
1210
+ }
1211
+ void write2(int value) {
1212
+ const char* d = digits2(to_unsigned(value) % 100);
1213
+ *out_++ = *d++;
1214
+ *out_++ = *d;
1215
+ }
1216
+ void write2(int value, pad_type pad) {
1217
+ unsigned int v = to_unsigned(value) % 100;
1218
+ if (v >= 10) {
1219
+ const char* d = digits2(v);
1220
+ *out_++ = *d++;
1221
+ *out_++ = *d;
1222
+ } else {
1223
+ out_ = detail::write_padding(out_, pad);
1224
+ *out_++ = static_cast<char>('0' + v);
1225
+ }
1226
+ }
1227
+
1228
+ void write_year_extended(long long year, pad_type pad) {
1229
+ // At least 4 characters.
1230
+ int width = 4;
1231
+ bool negative = year < 0;
1232
+ if (negative) {
1233
+ year = 0 - year;
1234
+ --width;
1235
+ }
1236
+ uint32_or_64_or_128_t<long long> n = to_unsigned(year);
1237
+ const int num_digits = count_digits(n);
1238
+ if (negative && pad == pad_type::zero) *out_++ = '-';
1239
+ if (width > num_digits) {
1240
+ out_ = detail::write_padding(out_, pad, width - num_digits);
1241
+ }
1242
+ if (negative && pad != pad_type::zero) *out_++ = '-';
1243
+ out_ = format_decimal<Char>(out_, n, num_digits);
1244
+ }
1245
+ void write_year(long long year, pad_type pad) {
1246
+ write_year_extended(year, pad);
1247
+ }
1248
+
1249
+ void write_utc_offset(long long offset, numeric_system ns) {
1250
+ if (offset < 0) {
1251
+ *out_++ = '-';
1252
+ offset = -offset;
1253
+ } else {
1254
+ *out_++ = '+';
1255
+ }
1256
+ offset /= 60;
1257
+ write2(static_cast<int>(offset / 60));
1258
+ if (ns != numeric_system::standard) *out_++ = ':';
1259
+ write2(static_cast<int>(offset % 60));
1260
+ }
1261
+
1262
+ template <typename T, FMT_ENABLE_IF(has_member_data_tm_gmtoff<T>::value)>
1263
+ void format_utc_offset_impl(const T& tm, numeric_system ns) {
1264
+ write_utc_offset(tm.tm_gmtoff, ns);
1265
+ }
1266
+ template <typename T, FMT_ENABLE_IF(!has_member_data_tm_gmtoff<T>::value)>
1267
+ void format_utc_offset_impl(const T& tm, numeric_system ns) {
1268
+ #if defined(_WIN32) && defined(_UCRT)
1269
+ tzset_once();
1270
+ long offset = 0;
1271
+ _get_timezone(&offset);
1272
+ if (tm.tm_isdst) {
1273
+ long dstbias = 0;
1274
+ _get_dstbias(&dstbias);
1275
+ offset += dstbias;
1276
+ }
1277
+ write_utc_offset(-offset, ns);
1278
+ #else
1279
+ if (ns == numeric_system::standard) return format_localized('z');
1280
+
1281
+ // Extract timezone offset from timezone conversion functions.
1282
+ std::tm gtm = tm;
1283
+ std::time_t gt = std::mktime(&gtm);
1284
+ std::tm ltm = gmtime(gt);
1285
+ std::time_t lt = std::mktime(&ltm);
1286
+ long long offset = gt - lt;
1287
+ write_utc_offset(offset, ns);
1288
+ #endif
1289
+ }
1290
+
1291
+ template <typename T, FMT_ENABLE_IF(has_member_data_tm_zone<T>::value)>
1292
+ void format_tz_name_impl(const T& tm) {
1293
+ if (is_classic_)
1294
+ out_ = write_tm_str<Char>(out_, tm.tm_zone, loc_);
1295
+ else
1296
+ format_localized('Z');
1297
+ }
1298
+ template <typename T, FMT_ENABLE_IF(!has_member_data_tm_zone<T>::value)>
1299
+ void format_tz_name_impl(const T&) {
1300
+ format_localized('Z');
1301
+ }
1302
+
1303
+ void format_localized(char format, char modifier = 0) {
1304
+ out_ = write<Char>(out_, tm_, loc_, format, modifier);
1305
+ }
1306
+
1307
+ public:
1308
+ tm_writer(const std::locale& loc, OutputIt out, const std::tm& tm,
1309
+ const Duration* subsecs = nullptr)
1310
+ : loc_(loc),
1311
+ is_classic_(loc_ == get_classic_locale()),
1312
+ out_(out),
1313
+ subsecs_(subsecs),
1314
+ tm_(tm) {}
1315
+
1316
+ auto out() const -> OutputIt { return out_; }
1317
+
1318
+ FMT_CONSTEXPR void on_text(const Char* begin, const Char* end) {
1319
+ out_ = copy<Char>(begin, end, out_);
1320
+ }
1321
+
1322
+ void on_abbr_weekday() {
1323
+ if (is_classic_)
1324
+ out_ = write(out_, tm_wday_short_name(tm_wday()));
1325
+ else
1326
+ format_localized('a');
1327
+ }
1328
+ void on_full_weekday() {
1329
+ if (is_classic_)
1330
+ out_ = write(out_, tm_wday_full_name(tm_wday()));
1331
+ else
1332
+ format_localized('A');
1333
+ }
1334
+ void on_dec0_weekday(numeric_system ns) {
1335
+ if (is_classic_ || ns == numeric_system::standard) return write1(tm_wday());
1336
+ format_localized('w', 'O');
1337
+ }
1338
+ void on_dec1_weekday(numeric_system ns) {
1339
+ if (is_classic_ || ns == numeric_system::standard) {
1340
+ auto wday = tm_wday();
1341
+ write1(wday == 0 ? days_per_week : wday);
1342
+ } else {
1343
+ format_localized('u', 'O');
1344
+ }
1345
+ }
1346
+
1347
+ void on_abbr_month() {
1348
+ if (is_classic_)
1349
+ out_ = write(out_, tm_mon_short_name(tm_mon()));
1350
+ else
1351
+ format_localized('b');
1352
+ }
1353
+ void on_full_month() {
1354
+ if (is_classic_)
1355
+ out_ = write(out_, tm_mon_full_name(tm_mon()));
1356
+ else
1357
+ format_localized('B');
1358
+ }
1359
+
1360
+ void on_datetime(numeric_system ns) {
1361
+ if (is_classic_) {
1362
+ on_abbr_weekday();
1363
+ *out_++ = ' ';
1364
+ on_abbr_month();
1365
+ *out_++ = ' ';
1366
+ on_day_of_month(numeric_system::standard, pad_type::space);
1367
+ *out_++ = ' ';
1368
+ on_iso_time();
1369
+ *out_++ = ' ';
1370
+ on_year(numeric_system::standard, pad_type::space);
1371
+ } else {
1372
+ format_localized('c', ns == numeric_system::standard ? '\0' : 'E');
1373
+ }
1374
+ }
1375
+ void on_loc_date(numeric_system ns) {
1376
+ if (is_classic_)
1377
+ on_us_date();
1378
+ else
1379
+ format_localized('x', ns == numeric_system::standard ? '\0' : 'E');
1380
+ }
1381
+ void on_loc_time(numeric_system ns) {
1382
+ if (is_classic_)
1383
+ on_iso_time();
1384
+ else
1385
+ format_localized('X', ns == numeric_system::standard ? '\0' : 'E');
1386
+ }
1387
+ void on_us_date() {
1388
+ char buf[8];
1389
+ write_digit2_separated(buf, to_unsigned(tm_mon() + 1),
1390
+ to_unsigned(tm_mday()),
1391
+ to_unsigned(split_year_lower(tm_year())), '/');
1392
+ out_ = copy<Char>(std::begin(buf), std::end(buf), out_);
1393
+ }
1394
+ void on_iso_date() {
1395
+ auto year = tm_year();
1396
+ char buf[10];
1397
+ size_t offset = 0;
1398
+ if (year >= 0 && year < 10000) {
1399
+ write2digits(buf, static_cast<size_t>(year / 100));
1400
+ } else {
1401
+ offset = 4;
1402
+ write_year_extended(year, pad_type::zero);
1403
+ year = 0;
1404
+ }
1405
+ write_digit2_separated(buf + 2, static_cast<unsigned>(year % 100),
1406
+ to_unsigned(tm_mon() + 1), to_unsigned(tm_mday()),
1407
+ '-');
1408
+ out_ = copy<Char>(std::begin(buf) + offset, std::end(buf), out_);
1409
+ }
1410
+
1411
+ void on_utc_offset(numeric_system ns) { format_utc_offset_impl(tm_, ns); }
1412
+ void on_tz_name() { format_tz_name_impl(tm_); }
1413
+
1414
+ void on_year(numeric_system ns, pad_type pad) {
1415
+ if (is_classic_ || ns == numeric_system::standard)
1416
+ return write_year(tm_year(), pad);
1417
+ format_localized('Y', 'E');
1418
+ }
1419
+ void on_short_year(numeric_system ns) {
1420
+ if (is_classic_ || ns == numeric_system::standard)
1421
+ return write2(split_year_lower(tm_year()));
1422
+ format_localized('y', 'O');
1423
+ }
1424
+ void on_offset_year() {
1425
+ if (is_classic_) return write2(split_year_lower(tm_year()));
1426
+ format_localized('y', 'E');
1427
+ }
1428
+
1429
+ void on_century(numeric_system ns) {
1430
+ if (is_classic_ || ns == numeric_system::standard) {
1431
+ auto year = tm_year();
1432
+ auto upper = year / 100;
1433
+ if (year >= -99 && year < 0) {
1434
+ // Zero upper on negative year.
1435
+ *out_++ = '-';
1436
+ *out_++ = '0';
1437
+ } else if (upper >= 0 && upper < 100) {
1438
+ write2(static_cast<int>(upper));
1439
+ } else {
1440
+ out_ = write<Char>(out_, upper);
1441
+ }
1442
+ } else {
1443
+ format_localized('C', 'E');
1444
+ }
1445
+ }
1446
+
1447
+ void on_dec_month(numeric_system ns, pad_type pad) {
1448
+ if (is_classic_ || ns == numeric_system::standard)
1449
+ return write2(tm_mon() + 1, pad);
1450
+ format_localized('m', 'O');
1451
+ }
1452
+
1453
+ void on_dec0_week_of_year(numeric_system ns, pad_type pad) {
1454
+ if (is_classic_ || ns == numeric_system::standard)
1455
+ return write2((tm_yday() + days_per_week - tm_wday()) / days_per_week,
1456
+ pad);
1457
+ format_localized('U', 'O');
1458
+ }
1459
+ void on_dec1_week_of_year(numeric_system ns, pad_type pad) {
1460
+ if (is_classic_ || ns == numeric_system::standard) {
1461
+ auto wday = tm_wday();
1462
+ write2((tm_yday() + days_per_week -
1463
+ (wday == 0 ? (days_per_week - 1) : (wday - 1))) /
1464
+ days_per_week,
1465
+ pad);
1466
+ } else {
1467
+ format_localized('W', 'O');
1468
+ }
1469
+ }
1470
+ void on_iso_week_of_year(numeric_system ns, pad_type pad) {
1471
+ if (is_classic_ || ns == numeric_system::standard)
1472
+ return write2(tm_iso_week_of_year(), pad);
1473
+ format_localized('V', 'O');
1474
+ }
1475
+
1476
+ void on_iso_week_based_year() {
1477
+ write_year(tm_iso_week_year(), pad_type::zero);
1478
+ }
1479
+ void on_iso_week_based_short_year() {
1480
+ write2(split_year_lower(tm_iso_week_year()));
1481
+ }
1482
+
1483
+ void on_day_of_year(pad_type pad) {
1484
+ auto yday = tm_yday() + 1;
1485
+ auto digit1 = yday / 100;
1486
+ if (digit1 != 0) {
1487
+ write1(digit1);
1488
+ } else {
1489
+ out_ = detail::write_padding(out_, pad);
1490
+ }
1491
+ write2(yday % 100, pad);
1492
+ }
1493
+
1494
+ void on_day_of_month(numeric_system ns, pad_type pad) {
1495
+ if (is_classic_ || ns == numeric_system::standard)
1496
+ return write2(tm_mday(), pad);
1497
+ format_localized('d', 'O');
1498
+ }
1499
+
1500
+ void on_24_hour(numeric_system ns, pad_type pad) {
1501
+ if (is_classic_ || ns == numeric_system::standard)
1502
+ return write2(tm_hour(), pad);
1503
+ format_localized('H', 'O');
1504
+ }
1505
+ void on_12_hour(numeric_system ns, pad_type pad) {
1506
+ if (is_classic_ || ns == numeric_system::standard)
1507
+ return write2(tm_hour12(), pad);
1508
+ format_localized('I', 'O');
1509
+ }
1510
+ void on_minute(numeric_system ns, pad_type pad) {
1511
+ if (is_classic_ || ns == numeric_system::standard)
1512
+ return write2(tm_min(), pad);
1513
+ format_localized('M', 'O');
1514
+ }
1515
+
1516
+ void on_second(numeric_system ns, pad_type pad) {
1517
+ if (is_classic_ || ns == numeric_system::standard) {
1518
+ write2(tm_sec(), pad);
1519
+ if (subsecs_) {
1520
+ if (std::is_floating_point<typename Duration::rep>::value) {
1521
+ auto buf = memory_buffer();
1522
+ write_floating_seconds(buf, *subsecs_);
1523
+ if (buf.size() > 1) {
1524
+ // Remove the leading "0", write something like ".123".
1525
+ out_ = copy<Char>(buf.begin() + 1, buf.end(), out_);
1526
+ }
1527
+ } else {
1528
+ write_fractional_seconds<Char>(out_, *subsecs_);
1529
+ }
1530
+ }
1531
+ } else {
1532
+ // Currently no formatting of subseconds when a locale is set.
1533
+ format_localized('S', 'O');
1534
+ }
1535
+ }
1536
+
1537
+ void on_12_hour_time() {
1538
+ if (is_classic_) {
1539
+ char buf[8];
1540
+ write_digit2_separated(buf, to_unsigned(tm_hour12()),
1541
+ to_unsigned(tm_min()), to_unsigned(tm_sec()), ':');
1542
+ out_ = copy<Char>(std::begin(buf), std::end(buf), out_);
1543
+ *out_++ = ' ';
1544
+ on_am_pm();
1545
+ } else {
1546
+ format_localized('r');
1547
+ }
1548
+ }
1549
+ void on_24_hour_time() {
1550
+ write2(tm_hour());
1551
+ *out_++ = ':';
1552
+ write2(tm_min());
1553
+ }
1554
+ void on_iso_time() {
1555
+ on_24_hour_time();
1556
+ *out_++ = ':';
1557
+ on_second(numeric_system::standard, pad_type::zero);
1558
+ }
1559
+
1560
+ void on_am_pm() {
1561
+ if (is_classic_) {
1562
+ *out_++ = tm_hour() < 12 ? 'A' : 'P';
1563
+ *out_++ = 'M';
1564
+ } else {
1565
+ format_localized('p');
1566
+ }
1567
+ }
1568
+
1569
+ // These apply to chrono durations but not tm.
1570
+ void on_duration_value() {}
1571
+ void on_duration_unit() {}
1572
+ };
1573
+
1574
+ struct chrono_format_checker : null_chrono_spec_handler<chrono_format_checker> {
1575
+ bool has_precision_integral = false;
1576
+
1577
+ FMT_NORETURN inline void unsupported() { FMT_THROW(format_error("no date")); }
1578
+
1579
+ template <typename Char>
1580
+ FMT_CONSTEXPR void on_text(const Char*, const Char*) {}
1581
+ FMT_CONSTEXPR void on_day_of_year(pad_type) {}
1582
+ FMT_CONSTEXPR void on_24_hour(numeric_system, pad_type) {}
1583
+ FMT_CONSTEXPR void on_12_hour(numeric_system, pad_type) {}
1584
+ FMT_CONSTEXPR void on_minute(numeric_system, pad_type) {}
1585
+ FMT_CONSTEXPR void on_second(numeric_system, pad_type) {}
1586
+ FMT_CONSTEXPR void on_12_hour_time() {}
1587
+ FMT_CONSTEXPR void on_24_hour_time() {}
1588
+ FMT_CONSTEXPR void on_iso_time() {}
1589
+ FMT_CONSTEXPR void on_am_pm() {}
1590
+ FMT_CONSTEXPR void on_duration_value() const {
1591
+ if (has_precision_integral)
1592
+ FMT_THROW(format_error("precision not allowed for this argument type"));
1593
+ }
1594
+ FMT_CONSTEXPR void on_duration_unit() {}
1595
+ };
1596
+
1597
+ template <typename T,
1598
+ FMT_ENABLE_IF(std::is_integral<T>::value&& has_isfinite<T>::value)>
1599
+ inline auto isfinite(T) -> bool {
1600
+ return true;
1601
+ }
1602
+
1603
+ template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
1604
+ inline auto mod(T x, int y) -> T {
1605
+ return x % static_cast<T>(y);
1606
+ }
1607
+ template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
1608
+ inline auto mod(T x, int y) -> T {
1609
+ return std::fmod(x, static_cast<T>(y));
1610
+ }
1611
+
1612
+ // If T is an integral type, maps T to its unsigned counterpart, otherwise
1613
+ // leaves it unchanged (unlike std::make_unsigned).
1614
+ template <typename T, bool INTEGRAL = std::is_integral<T>::value>
1615
+ struct make_unsigned_or_unchanged {
1616
+ using type = T;
1617
+ };
1618
+
1619
+ template <typename T> struct make_unsigned_or_unchanged<T, true> {
1620
+ using type = typename std::make_unsigned<T>::type;
1621
+ };
1622
+
1623
+ template <typename Rep, typename Period,
1624
+ FMT_ENABLE_IF(std::is_integral<Rep>::value)>
1625
+ inline auto get_milliseconds(std::chrono::duration<Rep, Period> d)
1626
+ -> std::chrono::duration<Rep, std::milli> {
1627
+ // this may overflow and/or the result may not fit in the
1628
+ // target type.
1629
+ #if FMT_SAFE_DURATION_CAST
1630
+ using CommonSecondsType =
1631
+ typename std::common_type<decltype(d), std::chrono::seconds>::type;
1632
+ const auto d_as_common = detail::duration_cast<CommonSecondsType>(d);
1633
+ const auto d_as_whole_seconds =
1634
+ detail::duration_cast<std::chrono::seconds>(d_as_common);
1635
+ // this conversion should be nonproblematic
1636
+ const auto diff = d_as_common - d_as_whole_seconds;
1637
+ const auto ms =
1638
+ detail::duration_cast<std::chrono::duration<Rep, std::milli>>(diff);
1639
+ return ms;
1640
+ #else
1641
+ auto s = detail::duration_cast<std::chrono::seconds>(d);
1642
+ return detail::duration_cast<std::chrono::milliseconds>(d - s);
1643
+ #endif
1644
+ }
1645
+
1646
+ template <typename Char, typename Rep, typename OutputIt,
1647
+ FMT_ENABLE_IF(std::is_integral<Rep>::value)>
1648
+ auto format_duration_value(OutputIt out, Rep val, int) -> OutputIt {
1649
+ return write<Char>(out, val);
1650
+ }
1651
+
1652
+ template <typename Char, typename Rep, typename OutputIt,
1653
+ FMT_ENABLE_IF(std::is_floating_point<Rep>::value)>
1654
+ auto format_duration_value(OutputIt out, Rep val, int precision) -> OutputIt {
1655
+ auto specs = format_specs();
1656
+ specs.precision = precision;
1657
+ specs.set_type(precision >= 0 ? presentation_type::fixed
1658
+ : presentation_type::general);
1659
+ return write<Char>(out, val, specs);
1660
+ }
1661
+
1662
+ template <typename Char, typename OutputIt>
1663
+ auto copy_unit(string_view unit, OutputIt out, Char) -> OutputIt {
1664
+ return copy<Char>(unit.begin(), unit.end(), out);
1665
+ }
1666
+
1667
+ template <typename OutputIt>
1668
+ auto copy_unit(string_view unit, OutputIt out, wchar_t) -> OutputIt {
1669
+ // This works when wchar_t is UTF-32 because units only contain characters
1670
+ // that have the same representation in UTF-16 and UTF-32.
1671
+ utf8_to_utf16 u(unit);
1672
+ return copy<wchar_t>(u.c_str(), u.c_str() + u.size(), out);
1673
+ }
1674
+
1675
+ template <typename Char, typename Period, typename OutputIt>
1676
+ auto format_duration_unit(OutputIt out) -> OutputIt {
1677
+ if (const char* unit = get_units<Period>())
1678
+ return copy_unit(string_view(unit), out, Char());
1679
+ *out++ = '[';
1680
+ out = write<Char>(out, Period::num);
1681
+ if (const_check(Period::den != 1)) {
1682
+ *out++ = '/';
1683
+ out = write<Char>(out, Period::den);
1684
+ }
1685
+ *out++ = ']';
1686
+ *out++ = 's';
1687
+ return out;
1688
+ }
1689
+
1690
+ class get_locale {
1691
+ private:
1692
+ union {
1693
+ std::locale locale_;
1694
+ };
1695
+ bool has_locale_ = false;
1696
+
1697
+ public:
1698
+ inline get_locale(bool localized, locale_ref loc) : has_locale_(localized) {
1699
+ if (localized)
1700
+ ::new (&locale_) std::locale(loc.template get<std::locale>());
1701
+ }
1702
+ inline ~get_locale() {
1703
+ if (has_locale_) locale_.~locale();
1704
+ }
1705
+ inline operator const std::locale&() const {
1706
+ return has_locale_ ? locale_ : get_classic_locale();
1707
+ }
1708
+ };
1709
+
1710
+ template <typename FormatContext, typename OutputIt, typename Rep,
1711
+ typename Period>
1712
+ struct chrono_formatter {
1713
+ FormatContext& context;
1714
+ OutputIt out;
1715
+ int precision;
1716
+ bool localized = false;
1717
+ // rep is unsigned to avoid overflow.
1718
+ using rep =
1719
+ conditional_t<std::is_integral<Rep>::value && sizeof(Rep) < sizeof(int),
1720
+ unsigned, typename make_unsigned_or_unchanged<Rep>::type>;
1721
+ rep val;
1722
+ using seconds = std::chrono::duration<rep>;
1723
+ seconds s;
1724
+ using milliseconds = std::chrono::duration<rep, std::milli>;
1725
+ bool negative;
1726
+
1727
+ using char_type = typename FormatContext::char_type;
1728
+ using tm_writer_type = tm_writer<OutputIt, char_type>;
1729
+
1730
+ chrono_formatter(FormatContext& ctx, OutputIt o,
1731
+ std::chrono::duration<Rep, Period> d)
1732
+ : context(ctx),
1733
+ out(o),
1734
+ val(static_cast<rep>(d.count())),
1735
+ negative(false) {
1736
+ if (d.count() < 0) {
1737
+ val = 0 - val;
1738
+ negative = true;
1739
+ }
1740
+
1741
+ // this may overflow and/or the result may not fit in the
1742
+ // target type.
1743
+ // might need checked conversion (rep!=Rep)
1744
+ s = detail::duration_cast<seconds>(std::chrono::duration<rep, Period>(val));
1745
+ }
1746
+
1747
+ // returns true if nan or inf, writes to out.
1748
+ auto handle_nan_inf() -> bool {
1749
+ if (isfinite(val)) {
1750
+ return false;
1751
+ }
1752
+ if (isnan(val)) {
1753
+ write_nan();
1754
+ return true;
1755
+ }
1756
+ // must be +-inf
1757
+ if (val > 0) {
1758
+ write_pinf();
1759
+ } else {
1760
+ write_ninf();
1761
+ }
1762
+ return true;
1763
+ }
1764
+
1765
+ auto days() const -> Rep { return static_cast<Rep>(s.count() / 86400); }
1766
+ auto hour() const -> Rep {
1767
+ return static_cast<Rep>(mod((s.count() / 3600), 24));
1768
+ }
1769
+
1770
+ auto hour12() const -> Rep {
1771
+ Rep hour = static_cast<Rep>(mod((s.count() / 3600), 12));
1772
+ return hour <= 0 ? 12 : hour;
1773
+ }
1774
+
1775
+ auto minute() const -> Rep {
1776
+ return static_cast<Rep>(mod((s.count() / 60), 60));
1777
+ }
1778
+ auto second() const -> Rep { return static_cast<Rep>(mod(s.count(), 60)); }
1779
+
1780
+ auto time() const -> std::tm {
1781
+ auto time = std::tm();
1782
+ time.tm_hour = to_nonnegative_int(hour(), 24);
1783
+ time.tm_min = to_nonnegative_int(minute(), 60);
1784
+ time.tm_sec = to_nonnegative_int(second(), 60);
1785
+ return time;
1786
+ }
1787
+
1788
+ void write_sign() {
1789
+ if (negative) {
1790
+ *out++ = '-';
1791
+ negative = false;
1792
+ }
1793
+ }
1794
+
1795
+ void write(Rep value, int width, pad_type pad = pad_type::zero) {
1796
+ write_sign();
1797
+ if (isnan(value)) return write_nan();
1798
+ uint32_or_64_or_128_t<int> n =
1799
+ to_unsigned(to_nonnegative_int(value, max_value<int>()));
1800
+ int num_digits = detail::count_digits(n);
1801
+ if (width > num_digits) {
1802
+ out = detail::write_padding(out, pad, width - num_digits);
1803
+ }
1804
+ out = format_decimal<char_type>(out, n, num_digits);
1805
+ }
1806
+
1807
+ void write_nan() { std::copy_n("nan", 3, out); }
1808
+ void write_pinf() { std::copy_n("inf", 3, out); }
1809
+ void write_ninf() { std::copy_n("-inf", 4, out); }
1810
+
1811
+ template <typename Callback, typename... Args>
1812
+ void format_tm(const tm& time, Callback cb, Args... args) {
1813
+ if (isnan(val)) return write_nan();
1814
+ get_locale loc(localized, context.locale());
1815
+ auto w = tm_writer_type(loc, out, time);
1816
+ (w.*cb)(args...);
1817
+ out = w.out();
1818
+ }
1819
+
1820
+ void on_text(const char_type* begin, const char_type* end) {
1821
+ copy<char_type>(begin, end, out);
1822
+ }
1823
+
1824
+ // These are not implemented because durations don't have date information.
1825
+ void on_abbr_weekday() {}
1826
+ void on_full_weekday() {}
1827
+ void on_dec0_weekday(numeric_system) {}
1828
+ void on_dec1_weekday(numeric_system) {}
1829
+ void on_abbr_month() {}
1830
+ void on_full_month() {}
1831
+ void on_datetime(numeric_system) {}
1832
+ void on_loc_date(numeric_system) {}
1833
+ void on_loc_time(numeric_system) {}
1834
+ void on_us_date() {}
1835
+ void on_iso_date() {}
1836
+ void on_utc_offset(numeric_system) {}
1837
+ void on_tz_name() {}
1838
+ void on_year(numeric_system, pad_type) {}
1839
+ void on_short_year(numeric_system) {}
1840
+ void on_offset_year() {}
1841
+ void on_century(numeric_system) {}
1842
+ void on_iso_week_based_year() {}
1843
+ void on_iso_week_based_short_year() {}
1844
+ void on_dec_month(numeric_system, pad_type) {}
1845
+ void on_dec0_week_of_year(numeric_system, pad_type) {}
1846
+ void on_dec1_week_of_year(numeric_system, pad_type) {}
1847
+ void on_iso_week_of_year(numeric_system, pad_type) {}
1848
+ void on_day_of_month(numeric_system, pad_type) {}
1849
+
1850
+ void on_day_of_year(pad_type) {
1851
+ if (handle_nan_inf()) return;
1852
+ write(days(), 0);
1853
+ }
1854
+
1855
+ void on_24_hour(numeric_system ns, pad_type pad) {
1856
+ if (handle_nan_inf()) return;
1857
+
1858
+ if (ns == numeric_system::standard) return write(hour(), 2, pad);
1859
+ auto time = tm();
1860
+ time.tm_hour = to_nonnegative_int(hour(), 24);
1861
+ format_tm(time, &tm_writer_type::on_24_hour, ns, pad);
1862
+ }
1863
+
1864
+ void on_12_hour(numeric_system ns, pad_type pad) {
1865
+ if (handle_nan_inf()) return;
1866
+
1867
+ if (ns == numeric_system::standard) return write(hour12(), 2, pad);
1868
+ auto time = tm();
1869
+ time.tm_hour = to_nonnegative_int(hour12(), 12);
1870
+ format_tm(time, &tm_writer_type::on_12_hour, ns, pad);
1871
+ }
1872
+
1873
+ void on_minute(numeric_system ns, pad_type pad) {
1874
+ if (handle_nan_inf()) return;
1875
+
1876
+ if (ns == numeric_system::standard) return write(minute(), 2, pad);
1877
+ auto time = tm();
1878
+ time.tm_min = to_nonnegative_int(minute(), 60);
1879
+ format_tm(time, &tm_writer_type::on_minute, ns, pad);
1880
+ }
1881
+
1882
+ void on_second(numeric_system ns, pad_type pad) {
1883
+ if (handle_nan_inf()) return;
1884
+
1885
+ if (ns == numeric_system::standard) {
1886
+ if (std::is_floating_point<rep>::value) {
1887
+ auto buf = memory_buffer();
1888
+ write_floating_seconds(buf, std::chrono::duration<rep, Period>(val),
1889
+ precision);
1890
+ if (negative) *out++ = '-';
1891
+ if (buf.size() < 2 || buf[1] == '.') {
1892
+ out = detail::write_padding(out, pad);
1893
+ }
1894
+ out = copy<char_type>(buf.begin(), buf.end(), out);
1895
+ } else {
1896
+ write(second(), 2, pad);
1897
+ write_fractional_seconds<char_type>(
1898
+ out, std::chrono::duration<rep, Period>(val), precision);
1899
+ }
1900
+ return;
1901
+ }
1902
+ auto time = tm();
1903
+ time.tm_sec = to_nonnegative_int(second(), 60);
1904
+ format_tm(time, &tm_writer_type::on_second, ns, pad);
1905
+ }
1906
+
1907
+ void on_12_hour_time() {
1908
+ if (handle_nan_inf()) return;
1909
+ format_tm(time(), &tm_writer_type::on_12_hour_time);
1910
+ }
1911
+
1912
+ void on_24_hour_time() {
1913
+ if (handle_nan_inf()) {
1914
+ *out++ = ':';
1915
+ handle_nan_inf();
1916
+ return;
1917
+ }
1918
+
1919
+ write(hour(), 2);
1920
+ *out++ = ':';
1921
+ write(minute(), 2);
1922
+ }
1923
+
1924
+ void on_iso_time() {
1925
+ on_24_hour_time();
1926
+ *out++ = ':';
1927
+ if (handle_nan_inf()) return;
1928
+ on_second(numeric_system::standard, pad_type::zero);
1929
+ }
1930
+
1931
+ void on_am_pm() {
1932
+ if (handle_nan_inf()) return;
1933
+ format_tm(time(), &tm_writer_type::on_am_pm);
1934
+ }
1935
+
1936
+ void on_duration_value() {
1937
+ if (handle_nan_inf()) return;
1938
+ write_sign();
1939
+ out = format_duration_value<char_type>(out, val, precision);
1940
+ }
1941
+
1942
+ void on_duration_unit() {
1943
+ out = format_duration_unit<char_type, Period>(out);
1944
+ }
1945
+ };
1946
+
1947
+ } // namespace detail
1948
+
1949
+ #if defined(__cpp_lib_chrono) && __cpp_lib_chrono >= 201907
1950
+ using weekday = std::chrono::weekday;
1951
+ using day = std::chrono::day;
1952
+ using month = std::chrono::month;
1953
+ using year = std::chrono::year;
1954
+ using year_month_day = std::chrono::year_month_day;
1955
+ #else
1956
+ // A fallback version of weekday.
1957
+ class weekday {
1958
+ private:
1959
+ unsigned char value_;
1960
+
1961
+ public:
1962
+ weekday() = default;
1963
+ constexpr explicit weekday(unsigned wd) noexcept
1964
+ : value_(static_cast<unsigned char>(wd != 7 ? wd : 0)) {}
1965
+ constexpr auto c_encoding() const noexcept -> unsigned { return value_; }
1966
+ };
1967
+
1968
+ class day {
1969
+ private:
1970
+ unsigned char value_;
1971
+
1972
+ public:
1973
+ day() = default;
1974
+ constexpr explicit day(unsigned d) noexcept
1975
+ : value_(static_cast<unsigned char>(d)) {}
1976
+ constexpr explicit operator unsigned() const noexcept { return value_; }
1977
+ };
1978
+
1979
+ class month {
1980
+ private:
1981
+ unsigned char value_;
1982
+
1983
+ public:
1984
+ month() = default;
1985
+ constexpr explicit month(unsigned m) noexcept
1986
+ : value_(static_cast<unsigned char>(m)) {}
1987
+ constexpr explicit operator unsigned() const noexcept { return value_; }
1988
+ };
1989
+
1990
+ class year {
1991
+ private:
1992
+ int value_;
1993
+
1994
+ public:
1995
+ year() = default;
1996
+ constexpr explicit year(int y) noexcept : value_(y) {}
1997
+ constexpr explicit operator int() const noexcept { return value_; }
1998
+ };
1999
+
2000
+ class year_month_day {
2001
+ private:
2002
+ fmt::year year_;
2003
+ fmt::month month_;
2004
+ fmt::day day_;
2005
+
2006
+ public:
2007
+ year_month_day() = default;
2008
+ constexpr year_month_day(const year& y, const month& m, const day& d) noexcept
2009
+ : year_(y), month_(m), day_(d) {}
2010
+ constexpr auto year() const noexcept -> fmt::year { return year_; }
2011
+ constexpr auto month() const noexcept -> fmt::month { return month_; }
2012
+ constexpr auto day() const noexcept -> fmt::day { return day_; }
2013
+ };
2014
+ #endif
2015
+
2016
+ template <typename Char>
2017
+ struct formatter<weekday, Char> : private formatter<std::tm, Char> {
2018
+ private:
2019
+ bool localized_ = false;
2020
+ bool use_tm_formatter_ = false;
2021
+
2022
+ public:
2023
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2024
+ auto it = ctx.begin(), end = ctx.end();
2025
+ if (it != end && *it == 'L') {
2026
+ ++it;
2027
+ localized_ = true;
2028
+ return it;
2029
+ }
2030
+ use_tm_formatter_ = it != end && *it != '}';
2031
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2032
+ }
2033
+
2034
+ template <typename FormatContext>
2035
+ auto format(weekday wd, FormatContext& ctx) const -> decltype(ctx.out()) {
2036
+ auto time = std::tm();
2037
+ time.tm_wday = static_cast<int>(wd.c_encoding());
2038
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2039
+ detail::get_locale loc(localized_, ctx.locale());
2040
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2041
+ w.on_abbr_weekday();
2042
+ return w.out();
2043
+ }
2044
+ };
2045
+
2046
+ template <typename Char>
2047
+ struct formatter<day, Char> : private formatter<std::tm, Char> {
2048
+ private:
2049
+ bool use_tm_formatter_ = false;
2050
+
2051
+ public:
2052
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2053
+ auto it = ctx.begin(), end = ctx.end();
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(day d, FormatContext& ctx) const -> decltype(ctx.out()) {
2060
+ auto time = std::tm();
2061
+ time.tm_mday = static_cast<int>(static_cast<unsigned>(d));
2062
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2063
+ detail::get_locale loc(false, ctx.locale());
2064
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2065
+ w.on_day_of_month(detail::numeric_system::standard, detail::pad_type::zero);
2066
+ return w.out();
2067
+ }
2068
+ };
2069
+
2070
+ template <typename Char>
2071
+ struct formatter<month, Char> : private formatter<std::tm, Char> {
2072
+ private:
2073
+ bool localized_ = false;
2074
+ bool use_tm_formatter_ = false;
2075
+
2076
+ public:
2077
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2078
+ auto it = ctx.begin(), end = ctx.end();
2079
+ if (it != end && *it == 'L') {
2080
+ ++it;
2081
+ localized_ = true;
2082
+ return it;
2083
+ }
2084
+ use_tm_formatter_ = it != end && *it != '}';
2085
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2086
+ }
2087
+
2088
+ template <typename FormatContext>
2089
+ auto format(month m, FormatContext& ctx) const -> decltype(ctx.out()) {
2090
+ auto time = std::tm();
2091
+ time.tm_mon = static_cast<int>(static_cast<unsigned>(m)) - 1;
2092
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2093
+ detail::get_locale loc(localized_, ctx.locale());
2094
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2095
+ w.on_abbr_month();
2096
+ return w.out();
2097
+ }
2098
+ };
2099
+
2100
+ template <typename Char>
2101
+ struct formatter<year, Char> : private formatter<std::tm, Char> {
2102
+ private:
2103
+ bool use_tm_formatter_ = false;
2104
+
2105
+ public:
2106
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2107
+ auto it = ctx.begin(), end = ctx.end();
2108
+ use_tm_formatter_ = it != end && *it != '}';
2109
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2110
+ }
2111
+
2112
+ template <typename FormatContext>
2113
+ auto format(year y, FormatContext& ctx) const -> decltype(ctx.out()) {
2114
+ auto time = std::tm();
2115
+ time.tm_year = static_cast<int>(y) - 1900;
2116
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2117
+ detail::get_locale loc(false, ctx.locale());
2118
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2119
+ w.on_year(detail::numeric_system::standard, detail::pad_type::zero);
2120
+ return w.out();
2121
+ }
2122
+ };
2123
+
2124
+ template <typename Char>
2125
+ struct formatter<year_month_day, Char> : private formatter<std::tm, Char> {
2126
+ private:
2127
+ bool use_tm_formatter_ = false;
2128
+
2129
+ public:
2130
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2131
+ auto it = ctx.begin(), end = ctx.end();
2132
+ use_tm_formatter_ = it != end && *it != '}';
2133
+ return use_tm_formatter_ ? formatter<std::tm, Char>::parse(ctx) : it;
2134
+ }
2135
+
2136
+ template <typename FormatContext>
2137
+ auto format(year_month_day val, FormatContext& ctx) const
2138
+ -> decltype(ctx.out()) {
2139
+ auto time = std::tm();
2140
+ time.tm_year = static_cast<int>(val.year()) - 1900;
2141
+ time.tm_mon = static_cast<int>(static_cast<unsigned>(val.month())) - 1;
2142
+ time.tm_mday = static_cast<int>(static_cast<unsigned>(val.day()));
2143
+ if (use_tm_formatter_) return formatter<std::tm, Char>::format(time, ctx);
2144
+ detail::get_locale loc(true, ctx.locale());
2145
+ auto w = detail::tm_writer<decltype(ctx.out()), Char>(loc, ctx.out(), time);
2146
+ w.on_iso_date();
2147
+ return w.out();
2148
+ }
2149
+ };
2150
+
2151
+ template <typename Rep, typename Period, typename Char>
2152
+ struct formatter<std::chrono::duration<Rep, Period>, Char> {
2153
+ private:
2154
+ format_specs specs_;
2155
+ detail::arg_ref<Char> width_ref_;
2156
+ detail::arg_ref<Char> precision_ref_;
2157
+ bool localized_ = false;
2158
+ basic_string_view<Char> fmt_;
2159
+
2160
+ public:
2161
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2162
+ auto it = ctx.begin(), end = ctx.end();
2163
+ if (it == end || *it == '}') return it;
2164
+
2165
+ it = detail::parse_align(it, end, specs_);
2166
+ if (it == end) return it;
2167
+
2168
+ Char c = *it;
2169
+ if ((c >= '0' && c <= '9') || c == '{') {
2170
+ it = detail::parse_width(it, end, specs_, width_ref_, ctx);
2171
+ if (it == end) return it;
2172
+ }
2173
+
2174
+ auto checker = detail::chrono_format_checker();
2175
+ if (*it == '.') {
2176
+ checker.has_precision_integral = !std::is_floating_point<Rep>::value;
2177
+ it = detail::parse_precision(it, end, specs_, precision_ref_, ctx);
2178
+ }
2179
+ if (it != end && *it == 'L') {
2180
+ localized_ = true;
2181
+ ++it;
2182
+ }
2183
+ end = detail::parse_chrono_format(it, end, checker);
2184
+ fmt_ = {it, detail::to_unsigned(end - it)};
2185
+ return end;
2186
+ }
2187
+
2188
+ template <typename FormatContext>
2189
+ auto format(std::chrono::duration<Rep, Period> d, FormatContext& ctx) const
2190
+ -> decltype(ctx.out()) {
2191
+ auto specs = specs_;
2192
+ auto precision = specs.precision;
2193
+ specs.precision = -1;
2194
+ auto begin = fmt_.begin(), end = fmt_.end();
2195
+ // As a possible future optimization, we could avoid extra copying if width
2196
+ // is not specified.
2197
+ auto buf = basic_memory_buffer<Char>();
2198
+ auto out = basic_appender<Char>(buf);
2199
+ detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
2200
+ ctx);
2201
+ detail::handle_dynamic_spec(specs.dynamic_precision(), precision,
2202
+ precision_ref_, ctx);
2203
+ if (begin == end || *begin == '}') {
2204
+ out = detail::format_duration_value<Char>(out, d.count(), precision);
2205
+ detail::format_duration_unit<Char, Period>(out);
2206
+ } else {
2207
+ using chrono_formatter =
2208
+ detail::chrono_formatter<FormatContext, decltype(out), Rep, Period>;
2209
+ auto f = chrono_formatter(ctx, out, d);
2210
+ f.precision = precision;
2211
+ f.localized = localized_;
2212
+ detail::parse_chrono_format(begin, end, f);
2213
+ }
2214
+ return detail::write(
2215
+ ctx.out(), basic_string_view<Char>(buf.data(), buf.size()), specs);
2216
+ }
2217
+ };
2218
+
2219
+ template <typename Char> struct formatter<std::tm, Char> {
2220
+ private:
2221
+ format_specs specs_;
2222
+ detail::arg_ref<Char> width_ref_;
2223
+
2224
+ protected:
2225
+ basic_string_view<Char> fmt_;
2226
+
2227
+ template <typename Duration, typename FormatContext>
2228
+ auto do_format(const std::tm& tm, FormatContext& ctx,
2229
+ const Duration* subsecs) const -> decltype(ctx.out()) {
2230
+ auto specs = specs_;
2231
+ auto buf = basic_memory_buffer<Char>();
2232
+ auto out = basic_appender<Char>(buf);
2233
+ detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
2234
+ ctx);
2235
+
2236
+ auto loc_ref = ctx.locale();
2237
+ detail::get_locale loc(static_cast<bool>(loc_ref), loc_ref);
2238
+ auto w =
2239
+ detail::tm_writer<decltype(out), Char, Duration>(loc, out, tm, subsecs);
2240
+ detail::parse_chrono_format(fmt_.begin(), fmt_.end(), w);
2241
+ return detail::write(
2242
+ ctx.out(), basic_string_view<Char>(buf.data(), buf.size()), specs);
2243
+ }
2244
+
2245
+ public:
2246
+ FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
2247
+ auto it = ctx.begin(), end = ctx.end();
2248
+ if (it == end || *it == '}') return it;
2249
+
2250
+ it = detail::parse_align(it, end, specs_);
2251
+ if (it == end) return it;
2252
+
2253
+ Char c = *it;
2254
+ if ((c >= '0' && c <= '9') || c == '{') {
2255
+ it = detail::parse_width(it, end, specs_, width_ref_, ctx);
2256
+ if (it == end) return it;
2257
+ }
2258
+
2259
+ end = detail::parse_chrono_format(it, end, detail::tm_format_checker());
2260
+ // Replace the default format string only if the new spec is not empty.
2261
+ if (end != it) fmt_ = {it, detail::to_unsigned(end - it)};
2262
+ return end;
2263
+ }
2264
+
2265
+ template <typename FormatContext>
2266
+ auto format(const std::tm& tm, FormatContext& ctx) const
2267
+ -> decltype(ctx.out()) {
2268
+ return do_format<std::chrono::seconds>(tm, ctx, nullptr);
2269
+ }
2270
+ };
2271
+
2272
+ template <typename Char, typename Duration>
2273
+ struct formatter<sys_time<Duration>, Char> : formatter<std::tm, Char> {
2274
+ FMT_CONSTEXPR formatter() {
2275
+ this->fmt_ = detail::string_literal<Char, '%', 'F', ' ', '%', 'T'>();
2276
+ }
2277
+
2278
+ template <typename FormatContext>
2279
+ auto format(sys_time<Duration> val, FormatContext& ctx) const
2280
+ -> decltype(ctx.out()) {
2281
+ std::tm tm = gmtime(val);
2282
+ using period = typename Duration::period;
2283
+ if (detail::const_check(
2284
+ period::num == 1 && period::den == 1 &&
2285
+ !std::is_floating_point<typename Duration::rep>::value)) {
2286
+ return formatter<std::tm, Char>::format(tm, ctx);
2287
+ }
2288
+ Duration epoch = val.time_since_epoch();
2289
+ Duration subsecs = detail::duration_cast<Duration>(
2290
+ epoch - detail::duration_cast<std::chrono::seconds>(epoch));
2291
+ if (subsecs.count() < 0) {
2292
+ auto second = detail::duration_cast<Duration>(std::chrono::seconds(1));
2293
+ if (tm.tm_sec != 0)
2294
+ --tm.tm_sec;
2295
+ else
2296
+ tm = gmtime(val - second);
2297
+ subsecs += detail::duration_cast<Duration>(std::chrono::seconds(1));
2298
+ }
2299
+ return formatter<std::tm, Char>::do_format(tm, ctx, &subsecs);
2300
+ }
2301
+ };
2302
+
2303
+ template <typename Duration, typename Char>
2304
+ struct formatter<utc_time<Duration>, Char>
2305
+ : formatter<sys_time<Duration>, Char> {
2306
+ template <typename FormatContext>
2307
+ auto format(utc_time<Duration> val, FormatContext& ctx) const
2308
+ -> decltype(ctx.out()) {
2309
+ return formatter<sys_time<Duration>, Char>::format(
2310
+ detail::utc_clock::to_sys(val), ctx);
2311
+ }
2312
+ };
2313
+
2314
+ template <typename Duration, typename Char>
2315
+ struct formatter<local_time<Duration>, Char> : formatter<std::tm, Char> {
2316
+ FMT_CONSTEXPR formatter() {
2317
+ this->fmt_ = detail::string_literal<Char, '%', 'F', ' ', '%', 'T'>();
2318
+ }
2319
+
2320
+ template <typename FormatContext>
2321
+ auto format(local_time<Duration> val, FormatContext& ctx) const
2322
+ -> decltype(ctx.out()) {
2323
+ using period = typename Duration::period;
2324
+ if (period::num == 1 && period::den == 1 &&
2325
+ !std::is_floating_point<typename Duration::rep>::value) {
2326
+ return formatter<std::tm, Char>::format(localtime(val), ctx);
2327
+ }
2328
+ auto epoch = val.time_since_epoch();
2329
+ auto subsecs = detail::duration_cast<Duration>(
2330
+ epoch - detail::duration_cast<std::chrono::seconds>(epoch));
2331
+ return formatter<std::tm, Char>::do_format(localtime(val), ctx, &subsecs);
2332
+ }
2333
+ };
2334
+
2335
+ FMT_END_EXPORT
2336
+ FMT_END_NAMESPACE
2337
+
2338
+ #endif // FMT_CHRONO_H_