181ad6265SDimitry Andric // -*- C++ -*-
281ad6265SDimitry Andric //===----------------------------------------------------------------------===//
381ad6265SDimitry Andric //
481ad6265SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
581ad6265SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
681ad6265SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
781ad6265SDimitry Andric //
881ad6265SDimitry Andric //===----------------------------------------------------------------------===//
981ad6265SDimitry Andric 
1081ad6265SDimitry Andric #ifndef _LIBCPP___CHRONO_HH_MM_SS_H
1181ad6265SDimitry Andric #define _LIBCPP___CHRONO_HH_MM_SS_H
1281ad6265SDimitry Andric 
1381ad6265SDimitry Andric #include <__chrono/duration.h>
1481ad6265SDimitry Andric #include <__chrono/time_point.h>
1581ad6265SDimitry Andric #include <__config>
1606c3fb27SDimitry Andric #include <__type_traits/common_type.h>
1781ad6265SDimitry Andric #include <ratio>
1881ad6265SDimitry Andric 
1981ad6265SDimitry Andric #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
2081ad6265SDimitry Andric #  pragma GCC system_header
2181ad6265SDimitry Andric #endif
2281ad6265SDimitry Andric 
2306c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 20
2481ad6265SDimitry Andric 
2581ad6265SDimitry Andric _LIBCPP_BEGIN_NAMESPACE_STD
2681ad6265SDimitry Andric 
27*cb14a3feSDimitry Andric namespace chrono {
2881ad6265SDimitry Andric 
2981ad6265SDimitry Andric template <class _Duration>
30*cb14a3feSDimitry Andric class hh_mm_ss {
3181ad6265SDimitry Andric private:
3281ad6265SDimitry Andric   static_assert(__is_duration<_Duration>::value, "template parameter of hh_mm_ss must be a std::chrono::duration");
3381ad6265SDimitry Andric   using __CommonType = common_type_t<_Duration, chrono::seconds>;
3481ad6265SDimitry Andric 
__pow10(unsigned __exp)35*cb14a3feSDimitry Andric   _LIBCPP_HIDE_FROM_ABI static constexpr uint64_t __pow10(unsigned __exp) {
3681ad6265SDimitry Andric     uint64_t __ret = 1;
3781ad6265SDimitry Andric     for (unsigned __i = 0; __i < __exp; ++__i)
3881ad6265SDimitry Andric       __ret *= 10U;
3981ad6265SDimitry Andric     return __ret;
4081ad6265SDimitry Andric   }
4181ad6265SDimitry Andric 
42*cb14a3feSDimitry Andric   _LIBCPP_HIDE_FROM_ABI static constexpr unsigned __width(uint64_t __n, uint64_t __d = 10, unsigned __w = 0) {
4381ad6265SDimitry Andric     if (__n >= 2 && __d != 0 && __w < 19)
4481ad6265SDimitry Andric       return 1 + __width(__n, __d % __n * 10, __w + 1);
4581ad6265SDimitry Andric     return 0;
4681ad6265SDimitry Andric   }
4781ad6265SDimitry Andric 
4881ad6265SDimitry Andric public:
49*cb14a3feSDimitry Andric   _LIBCPP_HIDE_FROM_ABI static unsigned constexpr fractional_width =
50*cb14a3feSDimitry Andric       __width(__CommonType::period::den) < 19 ? __width(__CommonType::period::den) : 6u;
5181ad6265SDimitry Andric   using precision = duration<typename __CommonType::rep, ratio<1, __pow10(fractional_width)>>;
5281ad6265SDimitry Andric 
hh_mm_ss()5381ad6265SDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr hh_mm_ss() noexcept : hh_mm_ss{_Duration::zero()} {}
5481ad6265SDimitry Andric 
hh_mm_ss(_Duration __d)55*cb14a3feSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr explicit hh_mm_ss(_Duration __d) noexcept
56*cb14a3feSDimitry Andric       : __is_neg_(__d < _Duration(0)),
57bdd1243dSDimitry Andric         __h_(chrono::duration_cast<chrono::hours>(chrono::abs(__d))),
58bdd1243dSDimitry Andric         __m_(chrono::duration_cast<chrono::minutes>(chrono::abs(__d) - hours())),
59bdd1243dSDimitry Andric         __s_(chrono::duration_cast<chrono::seconds>(chrono::abs(__d) - hours() - minutes())),
60*cb14a3feSDimitry Andric         __f_(chrono::duration_cast<precision>(chrono::abs(__d) - hours() - minutes() - seconds())) {}
6181ad6265SDimitry Andric 
is_negative()62bdd1243dSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr bool is_negative() const noexcept { return __is_neg_; }
hours()63bdd1243dSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr chrono::hours hours() const noexcept { return __h_; }
minutes()64bdd1243dSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr chrono::minutes minutes() const noexcept { return __m_; }
seconds()65bdd1243dSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr chrono::seconds seconds() const noexcept { return __s_; }
subseconds()66bdd1243dSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr precision subseconds() const noexcept { return __f_; }
6781ad6265SDimitry Andric 
to_duration()68*cb14a3feSDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr precision to_duration() const noexcept {
69bdd1243dSDimitry Andric     auto __dur = __h_ + __m_ + __s_ + __f_;
70bdd1243dSDimitry Andric     return __is_neg_ ? -__dur : __dur;
7181ad6265SDimitry Andric   }
7281ad6265SDimitry Andric 
precision()7381ad6265SDimitry Andric   _LIBCPP_HIDE_FROM_ABI constexpr explicit operator precision() const noexcept { return to_duration(); }
7481ad6265SDimitry Andric 
7581ad6265SDimitry Andric private:
76bdd1243dSDimitry Andric   bool __is_neg_;
77bdd1243dSDimitry Andric   chrono::hours __h_;
78bdd1243dSDimitry Andric   chrono::minutes __m_;
79bdd1243dSDimitry Andric   chrono::seconds __s_;
80bdd1243dSDimitry Andric   precision __f_;
8181ad6265SDimitry Andric };
8206c3fb27SDimitry Andric _LIBCPP_CTAD_SUPPORTED_FOR_TYPE(hh_mm_ss);
8381ad6265SDimitry Andric 
is_am(const hours & __h)84*cb14a3feSDimitry Andric _LIBCPP_HIDE_FROM_ABI inline constexpr bool is_am(const hours& __h) noexcept {
85*cb14a3feSDimitry Andric   return __h >= hours(0) && __h < hours(12);
86*cb14a3feSDimitry Andric }
is_pm(const hours & __h)87*cb14a3feSDimitry Andric _LIBCPP_HIDE_FROM_ABI inline constexpr bool is_pm(const hours& __h) noexcept {
88*cb14a3feSDimitry Andric   return __h >= hours(12) && __h < hours(24);
8981ad6265SDimitry Andric }
9081ad6265SDimitry Andric 
make12(const hours & __h)91*cb14a3feSDimitry Andric _LIBCPP_HIDE_FROM_ABI inline constexpr hours make12(const hours& __h) noexcept {
92*cb14a3feSDimitry Andric   if (__h == hours(0))
93*cb14a3feSDimitry Andric     return hours(12);
94*cb14a3feSDimitry Andric   else if (__h <= hours(12))
95*cb14a3feSDimitry Andric     return __h;
96*cb14a3feSDimitry Andric   else
97*cb14a3feSDimitry Andric     return __h - hours(12);
98*cb14a3feSDimitry Andric }
99*cb14a3feSDimitry Andric 
make24(const hours & __h,bool __is_pm)100*cb14a3feSDimitry Andric _LIBCPP_HIDE_FROM_ABI inline constexpr hours make24(const hours& __h, bool __is_pm) noexcept {
10181ad6265SDimitry Andric   if (__is_pm)
10281ad6265SDimitry Andric     return __h == hours(12) ? __h : __h + hours(12);
10381ad6265SDimitry Andric   else
10481ad6265SDimitry Andric     return __h == hours(12) ? hours(0) : __h;
10581ad6265SDimitry Andric }
10681ad6265SDimitry Andric } // namespace chrono
10781ad6265SDimitry Andric 
10881ad6265SDimitry Andric _LIBCPP_END_NAMESPACE_STD
10981ad6265SDimitry Andric 
11006c3fb27SDimitry Andric #endif // _LIBCPP_STD_VER >= 20
11181ad6265SDimitry Andric 
11281ad6265SDimitry Andric #endif // _LIBCPP___CHRONO_HH_MM_SS_H
113