xref: /freebsd/contrib/llvm-project/libcxx/include/__chrono/duration.h (revision 5f757f3ff9144b609b3c433dfd370cc6bdc191ad)
104eeddc0SDimitry Andric // -*- C++ -*-
204eeddc0SDimitry Andric //===----------------------------------------------------------------------===//
304eeddc0SDimitry Andric //
404eeddc0SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
504eeddc0SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
604eeddc0SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
704eeddc0SDimitry Andric //
804eeddc0SDimitry Andric //===----------------------------------------------------------------------===//
904eeddc0SDimitry Andric 
1004eeddc0SDimitry Andric #ifndef _LIBCPP___CHRONO_DURATION_H
1104eeddc0SDimitry Andric #define _LIBCPP___CHRONO_DURATION_H
1204eeddc0SDimitry Andric 
1306c3fb27SDimitry Andric #include <__compare/ordering.h>
1406c3fb27SDimitry Andric #include <__compare/three_way_comparable.h>
1504eeddc0SDimitry Andric #include <__config>
16bdd1243dSDimitry Andric #include <__type_traits/common_type.h>
17bdd1243dSDimitry Andric #include <__type_traits/enable_if.h>
18bdd1243dSDimitry Andric #include <__type_traits/is_convertible.h>
19bdd1243dSDimitry Andric #include <__type_traits/is_floating_point.h>
2004eeddc0SDimitry Andric #include <limits>
2104eeddc0SDimitry Andric #include <ratio>
2204eeddc0SDimitry Andric 
2304eeddc0SDimitry Andric #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
2404eeddc0SDimitry Andric #  pragma GCC system_header
2504eeddc0SDimitry Andric #endif
2604eeddc0SDimitry Andric 
2704eeddc0SDimitry Andric _LIBCPP_PUSH_MACROS
2804eeddc0SDimitry Andric #include <__undef_macros>
2904eeddc0SDimitry Andric 
3004eeddc0SDimitry Andric _LIBCPP_BEGIN_NAMESPACE_STD
3104eeddc0SDimitry Andric 
3204eeddc0SDimitry Andric namespace chrono
3304eeddc0SDimitry Andric {
3404eeddc0SDimitry Andric 
3504eeddc0SDimitry Andric template <class _Rep, class _Period = ratio<1> > class _LIBCPP_TEMPLATE_VIS duration;
3604eeddc0SDimitry Andric 
3704eeddc0SDimitry Andric template <class _Tp>
3804eeddc0SDimitry Andric struct __is_duration : false_type {};
3904eeddc0SDimitry Andric 
4004eeddc0SDimitry Andric template <class _Rep, class _Period>
4104eeddc0SDimitry Andric struct __is_duration<duration<_Rep, _Period> > : true_type  {};
4204eeddc0SDimitry Andric 
4304eeddc0SDimitry Andric template <class _Rep, class _Period>
4404eeddc0SDimitry Andric struct __is_duration<const duration<_Rep, _Period> > : true_type  {};
4504eeddc0SDimitry Andric 
4604eeddc0SDimitry Andric template <class _Rep, class _Period>
4704eeddc0SDimitry Andric struct __is_duration<volatile duration<_Rep, _Period> > : true_type  {};
4804eeddc0SDimitry Andric 
4904eeddc0SDimitry Andric template <class _Rep, class _Period>
5004eeddc0SDimitry Andric struct __is_duration<const volatile duration<_Rep, _Period> > : true_type  {};
5104eeddc0SDimitry Andric 
5204eeddc0SDimitry Andric } // namespace chrono
5304eeddc0SDimitry Andric 
5404eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
5504eeddc0SDimitry Andric struct _LIBCPP_TEMPLATE_VIS common_type<chrono::duration<_Rep1, _Period1>,
5604eeddc0SDimitry Andric                                          chrono::duration<_Rep2, _Period2> >
5704eeddc0SDimitry Andric {
5804eeddc0SDimitry Andric     typedef chrono::duration<typename common_type<_Rep1, _Rep2>::type,
5904eeddc0SDimitry Andric                              typename __ratio_gcd<_Period1, _Period2>::type> type;
6004eeddc0SDimitry Andric };
6104eeddc0SDimitry Andric 
6204eeddc0SDimitry Andric namespace chrono {
6304eeddc0SDimitry Andric 
6404eeddc0SDimitry Andric // duration_cast
6504eeddc0SDimitry Andric 
6604eeddc0SDimitry Andric template <class _FromDuration, class _ToDuration,
6704eeddc0SDimitry Andric           class _Period = typename ratio_divide<typename _FromDuration::period, typename _ToDuration::period>::type,
6804eeddc0SDimitry Andric           bool = _Period::num == 1,
6904eeddc0SDimitry Andric           bool = _Period::den == 1>
7004eeddc0SDimitry Andric struct __duration_cast;
7104eeddc0SDimitry Andric 
7204eeddc0SDimitry Andric template <class _FromDuration, class _ToDuration, class _Period>
7304eeddc0SDimitry Andric struct __duration_cast<_FromDuration, _ToDuration, _Period, true, true>
7404eeddc0SDimitry Andric {
75*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
7604eeddc0SDimitry Andric     _ToDuration operator()(const _FromDuration& __fd) const
7704eeddc0SDimitry Andric     {
7804eeddc0SDimitry Andric         return _ToDuration(static_cast<typename _ToDuration::rep>(__fd.count()));
7904eeddc0SDimitry Andric     }
8004eeddc0SDimitry Andric };
8104eeddc0SDimitry Andric 
8204eeddc0SDimitry Andric template <class _FromDuration, class _ToDuration, class _Period>
8304eeddc0SDimitry Andric struct __duration_cast<_FromDuration, _ToDuration, _Period, true, false>
8404eeddc0SDimitry Andric {
85*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
8604eeddc0SDimitry Andric     _ToDuration operator()(const _FromDuration& __fd) const
8704eeddc0SDimitry Andric     {
8804eeddc0SDimitry Andric         typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;
8904eeddc0SDimitry Andric         return _ToDuration(static_cast<typename _ToDuration::rep>(
9004eeddc0SDimitry Andric                            static_cast<_Ct>(__fd.count()) / static_cast<_Ct>(_Period::den)));
9104eeddc0SDimitry Andric     }
9204eeddc0SDimitry Andric };
9304eeddc0SDimitry Andric 
9404eeddc0SDimitry Andric template <class _FromDuration, class _ToDuration, class _Period>
9504eeddc0SDimitry Andric struct __duration_cast<_FromDuration, _ToDuration, _Period, false, true>
9604eeddc0SDimitry Andric {
97*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
9804eeddc0SDimitry Andric     _ToDuration operator()(const _FromDuration& __fd) const
9904eeddc0SDimitry Andric     {
10004eeddc0SDimitry Andric         typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;
10104eeddc0SDimitry Andric         return _ToDuration(static_cast<typename _ToDuration::rep>(
10204eeddc0SDimitry Andric                            static_cast<_Ct>(__fd.count()) * static_cast<_Ct>(_Period::num)));
10304eeddc0SDimitry Andric     }
10404eeddc0SDimitry Andric };
10504eeddc0SDimitry Andric 
10604eeddc0SDimitry Andric template <class _FromDuration, class _ToDuration, class _Period>
10704eeddc0SDimitry Andric struct __duration_cast<_FromDuration, _ToDuration, _Period, false, false>
10804eeddc0SDimitry Andric {
109*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
11004eeddc0SDimitry Andric     _ToDuration operator()(const _FromDuration& __fd) const
11104eeddc0SDimitry Andric     {
11204eeddc0SDimitry Andric         typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;
11304eeddc0SDimitry Andric         return _ToDuration(static_cast<typename _ToDuration::rep>(
11404eeddc0SDimitry Andric                            static_cast<_Ct>(__fd.count()) * static_cast<_Ct>(_Period::num)
11504eeddc0SDimitry Andric                                                           / static_cast<_Ct>(_Period::den)));
11604eeddc0SDimitry Andric     }
11704eeddc0SDimitry Andric };
11804eeddc0SDimitry Andric 
119*5f757f3fSDimitry Andric template <class _ToDuration, class _Rep, class _Period, __enable_if_t<__is_duration<_ToDuration>::value, int> = 0>
120*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
12104eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
12204eeddc0SDimitry Andric _ToDuration
12304eeddc0SDimitry Andric duration_cast(const duration<_Rep, _Period>& __fd)
12404eeddc0SDimitry Andric {
12504eeddc0SDimitry Andric     return __duration_cast<duration<_Rep, _Period>, _ToDuration>()(__fd);
12604eeddc0SDimitry Andric }
12704eeddc0SDimitry Andric 
12804eeddc0SDimitry Andric template <class _Rep>
12904eeddc0SDimitry Andric struct _LIBCPP_TEMPLATE_VIS treat_as_floating_point : is_floating_point<_Rep> {};
13004eeddc0SDimitry Andric 
13106c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 17
13204eeddc0SDimitry Andric template <class _Rep>
13304eeddc0SDimitry Andric inline constexpr bool treat_as_floating_point_v = treat_as_floating_point<_Rep>::value;
13404eeddc0SDimitry Andric #endif
13504eeddc0SDimitry Andric 
13604eeddc0SDimitry Andric template <class _Rep>
13704eeddc0SDimitry Andric struct _LIBCPP_TEMPLATE_VIS duration_values
13804eeddc0SDimitry Andric {
13904eeddc0SDimitry Andric public:
140*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR _Rep zero() _NOEXCEPT {return _Rep(0);}
141*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR _Rep max()  _NOEXCEPT {return numeric_limits<_Rep>::max();}
142*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR _Rep min()  _NOEXCEPT {return numeric_limits<_Rep>::lowest();}
14304eeddc0SDimitry Andric };
14404eeddc0SDimitry Andric 
14506c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 17
146*5f757f3fSDimitry Andric template <class _ToDuration, class _Rep, class _Period, enable_if_t<__is_duration<_ToDuration>::value, int> = 0>
147*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
14804eeddc0SDimitry Andric _ToDuration
14904eeddc0SDimitry Andric floor(const duration<_Rep, _Period>& __d)
15004eeddc0SDimitry Andric {
151bdd1243dSDimitry Andric     _ToDuration __t = chrono::duration_cast<_ToDuration>(__d);
15204eeddc0SDimitry Andric     if (__t > __d)
15304eeddc0SDimitry Andric         __t = __t - _ToDuration{1};
15404eeddc0SDimitry Andric     return __t;
15504eeddc0SDimitry Andric }
15604eeddc0SDimitry Andric 
157*5f757f3fSDimitry Andric template <class _ToDuration, class _Rep, class _Period, enable_if_t<__is_duration<_ToDuration>::value, int> = 0>
158*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
15904eeddc0SDimitry Andric _ToDuration
16004eeddc0SDimitry Andric ceil(const duration<_Rep, _Period>& __d)
16104eeddc0SDimitry Andric {
162bdd1243dSDimitry Andric     _ToDuration __t = chrono::duration_cast<_ToDuration>(__d);
16304eeddc0SDimitry Andric     if (__t < __d)
16404eeddc0SDimitry Andric         __t = __t + _ToDuration{1};
16504eeddc0SDimitry Andric     return __t;
16604eeddc0SDimitry Andric }
16704eeddc0SDimitry Andric 
168*5f757f3fSDimitry Andric template <class _ToDuration, class _Rep, class _Period, enable_if_t<__is_duration<_ToDuration>::value, int> = 0>
169*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
17004eeddc0SDimitry Andric _ToDuration
17104eeddc0SDimitry Andric round(const duration<_Rep, _Period>& __d)
17204eeddc0SDimitry Andric {
173bdd1243dSDimitry Andric     _ToDuration __lower = chrono::floor<_ToDuration>(__d);
17404eeddc0SDimitry Andric     _ToDuration __upper = __lower + _ToDuration{1};
17506c3fb27SDimitry Andric     auto __lower_diff   = __d - __lower;
17606c3fb27SDimitry Andric     auto __upper_diff   = __upper - __d;
17706c3fb27SDimitry Andric     if (__lower_diff < __upper_diff)
17804eeddc0SDimitry Andric         return __lower;
17906c3fb27SDimitry Andric     if (__lower_diff > __upper_diff)
18004eeddc0SDimitry Andric         return __upper;
18104eeddc0SDimitry Andric     return __lower.count() & 1 ? __upper : __lower;
18204eeddc0SDimitry Andric }
18304eeddc0SDimitry Andric #endif
18404eeddc0SDimitry Andric 
18504eeddc0SDimitry Andric // duration
18604eeddc0SDimitry Andric 
18704eeddc0SDimitry Andric template <class _Rep, class _Period>
18804eeddc0SDimitry Andric class _LIBCPP_TEMPLATE_VIS duration
18904eeddc0SDimitry Andric {
19004eeddc0SDimitry Andric     static_assert(!__is_duration<_Rep>::value, "A duration representation can not be a duration");
19104eeddc0SDimitry Andric     static_assert(__is_ratio<_Period>::value, "Second template parameter of duration must be a std::ratio");
19204eeddc0SDimitry Andric     static_assert(_Period::num > 0, "duration period must be positive");
19304eeddc0SDimitry Andric 
19404eeddc0SDimitry Andric     template <class _R1, class _R2>
19504eeddc0SDimitry Andric     struct __no_overflow
19604eeddc0SDimitry Andric     {
19704eeddc0SDimitry Andric     private:
19804eeddc0SDimitry Andric         static const intmax_t __gcd_n1_n2 = __static_gcd<_R1::num, _R2::num>::value;
19904eeddc0SDimitry Andric         static const intmax_t __gcd_d1_d2 = __static_gcd<_R1::den, _R2::den>::value;
20004eeddc0SDimitry Andric         static const intmax_t __n1 = _R1::num / __gcd_n1_n2;
20104eeddc0SDimitry Andric         static const intmax_t __d1 = _R1::den / __gcd_d1_d2;
20204eeddc0SDimitry Andric         static const intmax_t __n2 = _R2::num / __gcd_n1_n2;
20304eeddc0SDimitry Andric         static const intmax_t __d2 = _R2::den / __gcd_d1_d2;
20404eeddc0SDimitry Andric         static const intmax_t max = -((intmax_t(1) << (sizeof(intmax_t) * CHAR_BIT - 1)) + 1);
20504eeddc0SDimitry Andric 
20604eeddc0SDimitry Andric         template <intmax_t _Xp, intmax_t _Yp, bool __overflow>
20704eeddc0SDimitry Andric         struct __mul    // __overflow == false
20804eeddc0SDimitry Andric         {
20904eeddc0SDimitry Andric             static const intmax_t value = _Xp * _Yp;
21004eeddc0SDimitry Andric         };
21104eeddc0SDimitry Andric 
21204eeddc0SDimitry Andric         template <intmax_t _Xp, intmax_t _Yp>
21304eeddc0SDimitry Andric         struct __mul<_Xp, _Yp, true>
21404eeddc0SDimitry Andric         {
21504eeddc0SDimitry Andric             static const intmax_t value = 1;
21604eeddc0SDimitry Andric         };
21704eeddc0SDimitry Andric 
21804eeddc0SDimitry Andric     public:
21904eeddc0SDimitry Andric         static const bool value = (__n1 <= max / __d2) && (__n2 <= max / __d1);
22004eeddc0SDimitry Andric         typedef ratio<__mul<__n1, __d2, !value>::value,
22104eeddc0SDimitry Andric                       __mul<__n2, __d1, !value>::value> type;
22204eeddc0SDimitry Andric     };
22304eeddc0SDimitry Andric 
22404eeddc0SDimitry Andric public:
22504eeddc0SDimitry Andric     typedef _Rep rep;
22604eeddc0SDimitry Andric     typedef typename _Period::type period;
22704eeddc0SDimitry Andric private:
22804eeddc0SDimitry Andric     rep __rep_;
22904eeddc0SDimitry Andric public:
23004eeddc0SDimitry Andric 
23104eeddc0SDimitry Andric #ifndef _LIBCPP_CXX03_LANG
23206c3fb27SDimitry Andric         constexpr duration() = default;
23304eeddc0SDimitry Andric #else
23406c3fb27SDimitry Andric         _LIBCPP_HIDE_FROM_ABI duration() {}
23504eeddc0SDimitry Andric #endif
23604eeddc0SDimitry Andric 
237*5f757f3fSDimitry Andric     template <class _Rep2, __enable_if_t<is_convertible<const _Rep2&, rep>::value &&
23804eeddc0SDimitry Andric                                          (treat_as_floating_point<rep>::value ||
239*5f757f3fSDimitry Andric                                           !treat_as_floating_point<_Rep2>::value), int> = 0>
240*5f757f3fSDimitry Andric         _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
241*5f757f3fSDimitry Andric         explicit duration(const _Rep2& __r)
24204eeddc0SDimitry Andric                 : __rep_(__r) {}
24304eeddc0SDimitry Andric 
24404eeddc0SDimitry Andric     // conversions
245*5f757f3fSDimitry Andric     template <class _Rep2, class _Period2, __enable_if_t<__no_overflow<_Period2, period>::value && (
24604eeddc0SDimitry Andric                                                             treat_as_floating_point<rep>::value ||
24704eeddc0SDimitry Andric                                                             (__no_overflow<_Period2, period>::type::den == 1 &&
248*5f757f3fSDimitry Andric                                                              !treat_as_floating_point<_Rep2>::value)), int> = 0>
249*5f757f3fSDimitry Andric         _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
250*5f757f3fSDimitry Andric         duration(const duration<_Rep2, _Period2>& __d)
25104eeddc0SDimitry Andric                 : __rep_(chrono::duration_cast<duration>(__d).count()) {}
25204eeddc0SDimitry Andric 
25304eeddc0SDimitry Andric     // observer
25404eeddc0SDimitry Andric 
255*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR rep count() const {return __rep_;}
25604eeddc0SDimitry Andric 
25704eeddc0SDimitry Andric     // arithmetic
25804eeddc0SDimitry Andric 
259*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR typename common_type<duration>::type operator+() const {return typename common_type<duration>::type(*this);}
260*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR typename common_type<duration>::type operator-() const {return typename common_type<duration>::type(-__rep_);}
261*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator++()      {++__rep_; return *this;}
262*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration  operator++(int)   {return duration(__rep_++);}
263*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator--()      {--__rep_; return *this;}
264*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration  operator--(int)   {return duration(__rep_--);}
26504eeddc0SDimitry Andric 
266*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator+=(const duration& __d) {__rep_ += __d.count(); return *this;}
267*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator-=(const duration& __d) {__rep_ -= __d.count(); return *this;}
26804eeddc0SDimitry Andric 
269*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator*=(const rep& __rhs) {__rep_ *= __rhs; return *this;}
270*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator/=(const rep& __rhs) {__rep_ /= __rhs; return *this;}
271*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator%=(const rep& __rhs) {__rep_ %= __rhs; return *this;}
272*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX17 duration& operator%=(const duration& __rhs) {__rep_ %= __rhs.count(); return *this;}
27304eeddc0SDimitry Andric 
27404eeddc0SDimitry Andric     // special values
27504eeddc0SDimitry Andric 
276*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR duration zero() _NOEXCEPT {return duration(duration_values<rep>::zero());}
277*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR duration min()  _NOEXCEPT {return duration(duration_values<rep>::min());}
278*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI static _LIBCPP_CONSTEXPR duration max()  _NOEXCEPT {return duration(duration_values<rep>::max());}
27904eeddc0SDimitry Andric };
28004eeddc0SDimitry Andric 
28104eeddc0SDimitry Andric typedef duration<long long,         nano> nanoseconds;
28204eeddc0SDimitry Andric typedef duration<long long,        micro> microseconds;
28304eeddc0SDimitry Andric typedef duration<long long,        milli> milliseconds;
28404eeddc0SDimitry Andric typedef duration<long long              > seconds;
28504eeddc0SDimitry Andric typedef duration<     long, ratio<  60> > minutes;
28604eeddc0SDimitry Andric typedef duration<     long, ratio<3600> > hours;
28706c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 20
28804eeddc0SDimitry Andric typedef duration<     int, ratio_multiply<ratio<24>, hours::period>>         days;
28904eeddc0SDimitry Andric typedef duration<     int, ratio_multiply<ratio<7>,   days::period>>         weeks;
29004eeddc0SDimitry Andric typedef duration<     int, ratio_multiply<ratio<146097, 400>, days::period>> years;
29104eeddc0SDimitry Andric typedef duration<     int, ratio_divide<years::period, ratio<12>>>           months;
29204eeddc0SDimitry Andric #endif
29304eeddc0SDimitry Andric // Duration ==
29404eeddc0SDimitry Andric 
29504eeddc0SDimitry Andric template <class _LhsDuration, class _RhsDuration>
29604eeddc0SDimitry Andric struct __duration_eq
29704eeddc0SDimitry Andric {
298*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
29904eeddc0SDimitry Andric     bool operator()(const _LhsDuration& __lhs, const _RhsDuration& __rhs) const
30004eeddc0SDimitry Andric         {
30104eeddc0SDimitry Andric             typedef typename common_type<_LhsDuration, _RhsDuration>::type _Ct;
30204eeddc0SDimitry Andric             return _Ct(__lhs).count() == _Ct(__rhs).count();
30304eeddc0SDimitry Andric         }
30404eeddc0SDimitry Andric };
30504eeddc0SDimitry Andric 
30604eeddc0SDimitry Andric template <class _LhsDuration>
30704eeddc0SDimitry Andric struct __duration_eq<_LhsDuration, _LhsDuration>
30804eeddc0SDimitry Andric {
309*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
31004eeddc0SDimitry Andric     bool operator()(const _LhsDuration& __lhs, const _LhsDuration& __rhs) const
31104eeddc0SDimitry Andric         {return __lhs.count() == __rhs.count();}
31204eeddc0SDimitry Andric };
31304eeddc0SDimitry Andric 
31404eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
315*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
31604eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
31704eeddc0SDimitry Andric bool
31804eeddc0SDimitry Andric operator==(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
31904eeddc0SDimitry Andric {
32004eeddc0SDimitry Andric     return __duration_eq<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >()(__lhs, __rhs);
32104eeddc0SDimitry Andric }
32204eeddc0SDimitry Andric 
32306c3fb27SDimitry Andric #if _LIBCPP_STD_VER <= 17
32406c3fb27SDimitry Andric 
32504eeddc0SDimitry Andric // Duration !=
32604eeddc0SDimitry Andric 
32704eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
328*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
32904eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
33004eeddc0SDimitry Andric bool
33104eeddc0SDimitry Andric operator!=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
33204eeddc0SDimitry Andric {
33304eeddc0SDimitry Andric     return !(__lhs == __rhs);
33404eeddc0SDimitry Andric }
33504eeddc0SDimitry Andric 
33606c3fb27SDimitry Andric #endif // _LIBCPP_STD_VER <= 17
33706c3fb27SDimitry Andric 
33804eeddc0SDimitry Andric // Duration <
33904eeddc0SDimitry Andric 
34004eeddc0SDimitry Andric template <class _LhsDuration, class _RhsDuration>
34104eeddc0SDimitry Andric struct __duration_lt
34204eeddc0SDimitry Andric {
343*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
34404eeddc0SDimitry Andric     bool operator()(const _LhsDuration& __lhs, const _RhsDuration& __rhs) const
34504eeddc0SDimitry Andric         {
34604eeddc0SDimitry Andric             typedef typename common_type<_LhsDuration, _RhsDuration>::type _Ct;
34704eeddc0SDimitry Andric             return _Ct(__lhs).count() < _Ct(__rhs).count();
34804eeddc0SDimitry Andric         }
34904eeddc0SDimitry Andric };
35004eeddc0SDimitry Andric 
35104eeddc0SDimitry Andric template <class _LhsDuration>
35204eeddc0SDimitry Andric struct __duration_lt<_LhsDuration, _LhsDuration>
35304eeddc0SDimitry Andric {
354*5f757f3fSDimitry Andric     _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR
35504eeddc0SDimitry Andric     bool operator()(const _LhsDuration& __lhs, const _LhsDuration& __rhs) const
35604eeddc0SDimitry Andric         {return __lhs.count() < __rhs.count();}
35704eeddc0SDimitry Andric };
35804eeddc0SDimitry Andric 
35904eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
360*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
36104eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
36204eeddc0SDimitry Andric bool
36304eeddc0SDimitry Andric operator< (const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
36404eeddc0SDimitry Andric {
36504eeddc0SDimitry Andric     return __duration_lt<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >()(__lhs, __rhs);
36604eeddc0SDimitry Andric }
36704eeddc0SDimitry Andric 
36804eeddc0SDimitry Andric // Duration >
36904eeddc0SDimitry Andric 
37004eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
371*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
37204eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
37304eeddc0SDimitry Andric bool
37404eeddc0SDimitry Andric operator> (const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
37504eeddc0SDimitry Andric {
37604eeddc0SDimitry Andric     return __rhs < __lhs;
37704eeddc0SDimitry Andric }
37804eeddc0SDimitry Andric 
37904eeddc0SDimitry Andric // Duration <=
38004eeddc0SDimitry Andric 
38104eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
382*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
38304eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
38404eeddc0SDimitry Andric bool
38504eeddc0SDimitry Andric operator<=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
38604eeddc0SDimitry Andric {
38704eeddc0SDimitry Andric     return !(__rhs < __lhs);
38804eeddc0SDimitry Andric }
38904eeddc0SDimitry Andric 
39004eeddc0SDimitry Andric // Duration >=
39104eeddc0SDimitry Andric 
39204eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
393*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
39404eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
39504eeddc0SDimitry Andric bool
39604eeddc0SDimitry Andric operator>=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
39704eeddc0SDimitry Andric {
39804eeddc0SDimitry Andric     return !(__lhs < __rhs);
39904eeddc0SDimitry Andric }
40004eeddc0SDimitry Andric 
40106c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 20
40206c3fb27SDimitry Andric 
40306c3fb27SDimitry Andric template<class _Rep1, class _Period1, class _Rep2, class _Period2>
40406c3fb27SDimitry Andric   requires three_way_comparable<common_type_t<_Rep1, _Rep2>>
40506c3fb27SDimitry Andric _LIBCPP_HIDE_FROM_ABI
40606c3fb27SDimitry Andric constexpr auto operator<=>(const duration<_Rep1, _Period1>& __lhs,
40706c3fb27SDimitry Andric                            const duration<_Rep2, _Period2>& __rhs)
40806c3fb27SDimitry Andric {
40906c3fb27SDimitry Andric     using _Ct = common_type_t<duration<_Rep1, _Period1>, duration<_Rep2, _Period2>>;
41006c3fb27SDimitry Andric     return _Ct(__lhs).count() <=> _Ct(__rhs).count();
41106c3fb27SDimitry Andric }
41206c3fb27SDimitry Andric 
41306c3fb27SDimitry Andric #endif // _LIBCPP_STD_VER >= 20
41406c3fb27SDimitry Andric 
41504eeddc0SDimitry Andric // Duration +
41604eeddc0SDimitry Andric 
41704eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
418*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
41904eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
42004eeddc0SDimitry Andric typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type
42104eeddc0SDimitry Andric operator+(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
42204eeddc0SDimitry Andric {
42304eeddc0SDimitry Andric     typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;
42404eeddc0SDimitry Andric     return _Cd(_Cd(__lhs).count() + _Cd(__rhs).count());
42504eeddc0SDimitry Andric }
42604eeddc0SDimitry Andric 
42704eeddc0SDimitry Andric // Duration -
42804eeddc0SDimitry Andric 
42904eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
430*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
43104eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
43204eeddc0SDimitry Andric typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type
43304eeddc0SDimitry Andric operator-(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
43404eeddc0SDimitry Andric {
43504eeddc0SDimitry Andric     typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;
43604eeddc0SDimitry Andric     return _Cd(_Cd(__lhs).count() - _Cd(__rhs).count());
43704eeddc0SDimitry Andric }
43804eeddc0SDimitry Andric 
43904eeddc0SDimitry Andric // Duration *
44004eeddc0SDimitry Andric 
441*5f757f3fSDimitry Andric template <class _Rep1, class _Period, class _Rep2,
442*5f757f3fSDimitry Andric           __enable_if_t<is_convertible<_Rep2, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>
443*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
44404eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
44504eeddc0SDimitry Andric duration<typename common_type<_Rep1, _Rep2>::type, _Period>
44604eeddc0SDimitry Andric operator*(const duration<_Rep1, _Period>& __d, const _Rep2& __s)
44704eeddc0SDimitry Andric {
44804eeddc0SDimitry Andric     typedef typename common_type<_Rep1, _Rep2>::type _Cr;
44904eeddc0SDimitry Andric     typedef duration<_Cr, _Period> _Cd;
45004eeddc0SDimitry Andric     return _Cd(_Cd(__d).count() * static_cast<_Cr>(__s));
45104eeddc0SDimitry Andric }
45204eeddc0SDimitry Andric 
453*5f757f3fSDimitry Andric template <class _Rep1, class _Period, class _Rep2,
454*5f757f3fSDimitry Andric           __enable_if_t<is_convertible<_Rep1, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>
455*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
45604eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
45704eeddc0SDimitry Andric duration<typename common_type<_Rep1, _Rep2>::type, _Period>
45804eeddc0SDimitry Andric operator*(const _Rep1& __s, const duration<_Rep2, _Period>& __d)
45904eeddc0SDimitry Andric {
46004eeddc0SDimitry Andric     return __d * __s;
46104eeddc0SDimitry Andric }
46204eeddc0SDimitry Andric 
46304eeddc0SDimitry Andric // Duration /
46404eeddc0SDimitry Andric 
465*5f757f3fSDimitry Andric template <class _Rep1, class _Period, class _Rep2,
466*5f757f3fSDimitry Andric           __enable_if_t<!__is_duration<_Rep2>::value && is_convertible<_Rep2, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>
467*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
46804eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
46904eeddc0SDimitry Andric duration<typename common_type<_Rep1, _Rep2>::type, _Period>
47004eeddc0SDimitry Andric operator/(const duration<_Rep1, _Period>& __d, const _Rep2& __s)
47104eeddc0SDimitry Andric {
47204eeddc0SDimitry Andric     typedef typename common_type<_Rep1, _Rep2>::type _Cr;
47304eeddc0SDimitry Andric     typedef duration<_Cr, _Period> _Cd;
47404eeddc0SDimitry Andric     return _Cd(_Cd(__d).count() / static_cast<_Cr>(__s));
47504eeddc0SDimitry Andric }
47604eeddc0SDimitry Andric 
47704eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
478*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
47904eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
48004eeddc0SDimitry Andric typename common_type<_Rep1, _Rep2>::type
48104eeddc0SDimitry Andric operator/(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
48204eeddc0SDimitry Andric {
48304eeddc0SDimitry Andric     typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Ct;
48404eeddc0SDimitry Andric     return _Ct(__lhs).count() / _Ct(__rhs).count();
48504eeddc0SDimitry Andric }
48604eeddc0SDimitry Andric 
48704eeddc0SDimitry Andric // Duration %
48804eeddc0SDimitry Andric 
489*5f757f3fSDimitry Andric template <class _Rep1, class _Period, class _Rep2,
490*5f757f3fSDimitry Andric           __enable_if_t<!__is_duration<_Rep2>::value && is_convertible<_Rep2, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>
491*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
49204eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
49304eeddc0SDimitry Andric duration<typename common_type<_Rep1, _Rep2>::type, _Period>
49404eeddc0SDimitry Andric operator%(const duration<_Rep1, _Period>& __d, const _Rep2& __s)
49504eeddc0SDimitry Andric {
49604eeddc0SDimitry Andric     typedef typename common_type<_Rep1, _Rep2>::type _Cr;
49704eeddc0SDimitry Andric     typedef duration<_Cr, _Period> _Cd;
49804eeddc0SDimitry Andric     return _Cd(_Cd(__d).count() % static_cast<_Cr>(__s));
49904eeddc0SDimitry Andric }
50004eeddc0SDimitry Andric 
50104eeddc0SDimitry Andric template <class _Rep1, class _Period1, class _Rep2, class _Period2>
502*5f757f3fSDimitry Andric inline _LIBCPP_HIDE_FROM_ABI
50304eeddc0SDimitry Andric _LIBCPP_CONSTEXPR
50404eeddc0SDimitry Andric typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type
50504eeddc0SDimitry Andric operator%(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs)
50604eeddc0SDimitry Andric {
50704eeddc0SDimitry Andric     typedef typename common_type<_Rep1, _Rep2>::type _Cr;
50804eeddc0SDimitry Andric     typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;
50904eeddc0SDimitry Andric     return _Cd(static_cast<_Cr>(_Cd(__lhs).count()) % static_cast<_Cr>(_Cd(__rhs).count()));
51004eeddc0SDimitry Andric }
51104eeddc0SDimitry Andric 
51204eeddc0SDimitry Andric } // namespace chrono
51304eeddc0SDimitry Andric 
51406c3fb27SDimitry Andric #if _LIBCPP_STD_VER >= 14
51504eeddc0SDimitry Andric // Suffixes for duration literals [time.duration.literals]
51604eeddc0SDimitry Andric inline namespace literals
51704eeddc0SDimitry Andric {
51804eeddc0SDimitry Andric   inline namespace chrono_literals
51904eeddc0SDimitry Andric   {
52004eeddc0SDimitry Andric 
521bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::hours operator""h(unsigned long long __h)
52204eeddc0SDimitry Andric     {
52304eeddc0SDimitry Andric         return chrono::hours(static_cast<chrono::hours::rep>(__h));
52404eeddc0SDimitry Andric     }
52504eeddc0SDimitry Andric 
526bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double, ratio<3600,1>> operator""h(long double __h)
52704eeddc0SDimitry Andric     {
52804eeddc0SDimitry Andric         return chrono::duration<long double, ratio<3600,1>>(__h);
52904eeddc0SDimitry Andric     }
53004eeddc0SDimitry Andric 
53104eeddc0SDimitry Andric 
532bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::minutes operator""min(unsigned long long __m)
53304eeddc0SDimitry Andric     {
53404eeddc0SDimitry Andric         return chrono::minutes(static_cast<chrono::minutes::rep>(__m));
53504eeddc0SDimitry Andric     }
53604eeddc0SDimitry Andric 
537bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double, ratio<60,1>> operator""min(long double __m)
53804eeddc0SDimitry Andric     {
53904eeddc0SDimitry Andric         return chrono::duration<long double, ratio<60,1>> (__m);
54004eeddc0SDimitry Andric     }
54104eeddc0SDimitry Andric 
54204eeddc0SDimitry Andric 
543bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::seconds operator""s(unsigned long long __s)
54404eeddc0SDimitry Andric     {
54504eeddc0SDimitry Andric         return chrono::seconds(static_cast<chrono::seconds::rep>(__s));
54604eeddc0SDimitry Andric     }
54704eeddc0SDimitry Andric 
548bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double> operator""s(long double __s)
54904eeddc0SDimitry Andric     {
55004eeddc0SDimitry Andric         return chrono::duration<long double> (__s);
55104eeddc0SDimitry Andric     }
55204eeddc0SDimitry Andric 
55304eeddc0SDimitry Andric 
554bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::milliseconds operator""ms(unsigned long long __ms)
55504eeddc0SDimitry Andric     {
55604eeddc0SDimitry Andric         return chrono::milliseconds(static_cast<chrono::milliseconds::rep>(__ms));
55704eeddc0SDimitry Andric     }
55804eeddc0SDimitry Andric 
559bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double, milli> operator""ms(long double __ms)
56004eeddc0SDimitry Andric     {
56104eeddc0SDimitry Andric         return chrono::duration<long double, milli>(__ms);
56204eeddc0SDimitry Andric     }
56304eeddc0SDimitry Andric 
56404eeddc0SDimitry Andric 
565bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::microseconds operator""us(unsigned long long __us)
56604eeddc0SDimitry Andric     {
56704eeddc0SDimitry Andric         return chrono::microseconds(static_cast<chrono::microseconds::rep>(__us));
56804eeddc0SDimitry Andric     }
56904eeddc0SDimitry Andric 
570bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double, micro> operator""us(long double __us)
57104eeddc0SDimitry Andric     {
57204eeddc0SDimitry Andric         return chrono::duration<long double, micro> (__us);
57304eeddc0SDimitry Andric     }
57404eeddc0SDimitry Andric 
57504eeddc0SDimitry Andric 
576bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::nanoseconds operator""ns(unsigned long long __ns)
57704eeddc0SDimitry Andric     {
57804eeddc0SDimitry Andric         return chrono::nanoseconds(static_cast<chrono::nanoseconds::rep>(__ns));
57904eeddc0SDimitry Andric     }
58004eeddc0SDimitry Andric 
581bdd1243dSDimitry Andric     _LIBCPP_HIDE_FROM_ABI constexpr chrono::duration<long double, nano> operator""ns(long double __ns)
58204eeddc0SDimitry Andric     {
58304eeddc0SDimitry Andric         return chrono::duration<long double, nano> (__ns);
58404eeddc0SDimitry Andric     }
58504eeddc0SDimitry Andric 
58604eeddc0SDimitry Andric } // namespace chrono_literals
58704eeddc0SDimitry Andric } // namespace literals
58804eeddc0SDimitry Andric 
58904eeddc0SDimitry Andric namespace chrono { // hoist the literals into namespace std::chrono
59004eeddc0SDimitry Andric    using namespace literals::chrono_literals;
59104eeddc0SDimitry Andric } // namespace chrono
59204eeddc0SDimitry Andric 
59306c3fb27SDimitry Andric #endif // _LIBCPP_STD_VER >= 14
59404eeddc0SDimitry Andric 
59504eeddc0SDimitry Andric _LIBCPP_END_NAMESPACE_STD
59604eeddc0SDimitry Andric 
59704eeddc0SDimitry Andric _LIBCPP_POP_MACROS
59804eeddc0SDimitry Andric 
599bdd1243dSDimitry Andric #if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES) && _LIBCPP_STD_VER <= 20
600bdd1243dSDimitry Andric #  include <type_traits>
601bdd1243dSDimitry Andric #endif
602bdd1243dSDimitry Andric 
60304eeddc0SDimitry Andric #endif // _LIBCPP___CHRONO_DURATION_H
604