Path: blob/main/contrib/llvm-project/libcxx/include/__cxx03/__chrono/duration.h
213799 views
// -*- C++ -*-1//===----------------------------------------------------------------------===//2//3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.4// See https://llvm.org/LICENSE.txt for license information.5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception6//7//===----------------------------------------------------------------------===//89#ifndef _LIBCPP___CXX03___CHRONO_DURATION_H10#define _LIBCPP___CXX03___CHRONO_DURATION_H1112#include <__cxx03/__config>13#include <__cxx03/__type_traits/common_type.h>14#include <__cxx03/__type_traits/enable_if.h>15#include <__cxx03/__type_traits/is_convertible.h>16#include <__cxx03/__type_traits/is_floating_point.h>17#include <__cxx03/limits>18#include <__cxx03/ratio>1920#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)21# pragma GCC system_header22#endif2324_LIBCPP_PUSH_MACROS25#include <__cxx03/__undef_macros>2627_LIBCPP_BEGIN_NAMESPACE_STD2829namespace chrono {3031template <class _Rep, class _Period = ratio<1> >32class _LIBCPP_TEMPLATE_VIS duration;3334template <class _Tp>35struct __is_duration : false_type {};3637template <class _Rep, class _Period>38struct __is_duration<duration<_Rep, _Period> > : true_type {};3940template <class _Rep, class _Period>41struct __is_duration<const duration<_Rep, _Period> > : true_type {};4243template <class _Rep, class _Period>44struct __is_duration<volatile duration<_Rep, _Period> > : true_type {};4546template <class _Rep, class _Period>47struct __is_duration<const volatile duration<_Rep, _Period> > : true_type {};4849} // namespace chrono5051template <class _Rep1, class _Period1, class _Rep2, class _Period2>52struct _LIBCPP_TEMPLATE_VIS common_type<chrono::duration<_Rep1, _Period1>, chrono::duration<_Rep2, _Period2> > {53typedef chrono::duration<typename common_type<_Rep1, _Rep2>::type, typename __ratio_gcd<_Period1, _Period2>::type>54type;55};5657namespace chrono {5859// duration_cast6061template <class _FromDuration,62class _ToDuration,63class _Period = typename ratio_divide<typename _FromDuration::period, typename _ToDuration::period>::type,64bool = _Period::num == 1,65bool = _Period::den == 1>66struct __duration_cast;6768template <class _FromDuration, class _ToDuration, class _Period>69struct __duration_cast<_FromDuration, _ToDuration, _Period, true, true> {70_LIBCPP_HIDE_FROM_ABI _ToDuration operator()(const _FromDuration& __fd) const {71return _ToDuration(static_cast<typename _ToDuration::rep>(__fd.count()));72}73};7475template <class _FromDuration, class _ToDuration, class _Period>76struct __duration_cast<_FromDuration, _ToDuration, _Period, true, false> {77_LIBCPP_HIDE_FROM_ABI _ToDuration operator()(const _FromDuration& __fd) const {78typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;79return _ToDuration(80static_cast<typename _ToDuration::rep>(static_cast<_Ct>(__fd.count()) / static_cast<_Ct>(_Period::den)));81}82};8384template <class _FromDuration, class _ToDuration, class _Period>85struct __duration_cast<_FromDuration, _ToDuration, _Period, false, true> {86_LIBCPP_HIDE_FROM_ABI _ToDuration operator()(const _FromDuration& __fd) const {87typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;88return _ToDuration(89static_cast<typename _ToDuration::rep>(static_cast<_Ct>(__fd.count()) * static_cast<_Ct>(_Period::num)));90}91};9293template <class _FromDuration, class _ToDuration, class _Period>94struct __duration_cast<_FromDuration, _ToDuration, _Period, false, false> {95_LIBCPP_HIDE_FROM_ABI _ToDuration operator()(const _FromDuration& __fd) const {96typedef typename common_type<typename _ToDuration::rep, typename _FromDuration::rep, intmax_t>::type _Ct;97return _ToDuration(static_cast<typename _ToDuration::rep>(98static_cast<_Ct>(__fd.count()) * static_cast<_Ct>(_Period::num) / static_cast<_Ct>(_Period::den)));99}100};101102template <class _ToDuration, class _Rep, class _Period, __enable_if_t<__is_duration<_ToDuration>::value, int> = 0>103inline _LIBCPP_HIDE_FROM_ABI _ToDuration duration_cast(const duration<_Rep, _Period>& __fd) {104return __duration_cast<duration<_Rep, _Period>, _ToDuration>()(__fd);105}106107template <class _Rep>108struct _LIBCPP_TEMPLATE_VIS treat_as_floating_point : is_floating_point<_Rep> {};109110// clang-format off111template <class _Rep>112struct _LIBCPP_TEMPLATE_VIS duration_values {113public:114_LIBCPP_HIDE_FROM_ABI static _Rep zero() _NOEXCEPT { return _Rep(0); }115_LIBCPP_HIDE_FROM_ABI static _Rep max() _NOEXCEPT { return numeric_limits<_Rep>::max(); }116_LIBCPP_HIDE_FROM_ABI static _Rep min() _NOEXCEPT { return numeric_limits<_Rep>::lowest(); }117};118// clang-format on119120// duration121122template <class _Rep, class _Period>123class _LIBCPP_TEMPLATE_VIS duration {124static_assert(!__is_duration<_Rep>::value, "A duration representation can not be a duration");125static_assert(__is_ratio<_Period>::value, "Second template parameter of duration must be a std::ratio");126static_assert(_Period::num > 0, "duration period must be positive");127128template <class _R1, class _R2>129struct __no_overflow {130private:131static const intmax_t __gcd_n1_n2 = __static_gcd<_R1::num, _R2::num>::value;132static const intmax_t __gcd_d1_d2 = __static_gcd<_R1::den, _R2::den>::value;133static const intmax_t __n1 = _R1::num / __gcd_n1_n2;134static const intmax_t __d1 = _R1::den / __gcd_d1_d2;135static const intmax_t __n2 = _R2::num / __gcd_n1_n2;136static const intmax_t __d2 = _R2::den / __gcd_d1_d2;137static const intmax_t max = -((intmax_t(1) << (sizeof(intmax_t) * CHAR_BIT - 1)) + 1);138139template <intmax_t _Xp, intmax_t _Yp, bool __overflow>140struct __mul // __overflow == false141{142static const intmax_t value = _Xp * _Yp;143};144145template <intmax_t _Xp, intmax_t _Yp>146struct __mul<_Xp, _Yp, true> {147static const intmax_t value = 1;148};149150public:151static const bool value = (__n1 <= max / __d2) && (__n2 <= max / __d1);152typedef ratio<__mul<__n1, __d2, !value>::value, __mul<__n2, __d1, !value>::value> type;153};154155public:156typedef _Rep rep;157typedef typename _Period::type period;158159private:160rep __rep_;161162public:163_LIBCPP_HIDE_FROM_ABI duration() {}164165template <class _Rep2,166__enable_if_t<is_convertible<const _Rep2&, rep>::value &&167(treat_as_floating_point<rep>::value || !treat_as_floating_point<_Rep2>::value),168int> = 0>169_LIBCPP_HIDE_FROM_ABI explicit duration(const _Rep2& __r) : __rep_(__r) {}170171// conversions172template <class _Rep2,173class _Period2,174__enable_if_t<__no_overflow<_Period2, period>::value && (treat_as_floating_point<rep>::value ||175(__no_overflow<_Period2, period>::type::den == 1 &&176!treat_as_floating_point<_Rep2>::value)),177int> = 0>178_LIBCPP_HIDE_FROM_ABI duration(const duration<_Rep2, _Period2>& __d)179: __rep_(chrono::duration_cast<duration>(__d).count()) {}180181// observer182183_LIBCPP_HIDE_FROM_ABI rep count() const { return __rep_; }184185// arithmetic186187_LIBCPP_HIDE_FROM_ABI typename common_type<duration>::type operator+() const {188return typename common_type<duration>::type(*this);189}190_LIBCPP_HIDE_FROM_ABI typename common_type<duration>::type operator-() const {191return typename common_type<duration>::type(-__rep_);192}193_LIBCPP_HIDE_FROM_ABI duration& operator++() {194++__rep_;195return *this;196}197_LIBCPP_HIDE_FROM_ABI duration operator++(int) { return duration(__rep_++); }198_LIBCPP_HIDE_FROM_ABI duration& operator--() {199--__rep_;200return *this;201}202_LIBCPP_HIDE_FROM_ABI duration operator--(int) { return duration(__rep_--); }203204_LIBCPP_HIDE_FROM_ABI duration& operator+=(const duration& __d) {205__rep_ += __d.count();206return *this;207}208_LIBCPP_HIDE_FROM_ABI duration& operator-=(const duration& __d) {209__rep_ -= __d.count();210return *this;211}212213_LIBCPP_HIDE_FROM_ABI duration& operator*=(const rep& __rhs) {214__rep_ *= __rhs;215return *this;216}217_LIBCPP_HIDE_FROM_ABI duration& operator/=(const rep& __rhs) {218__rep_ /= __rhs;219return *this;220}221_LIBCPP_HIDE_FROM_ABI duration& operator%=(const rep& __rhs) {222__rep_ %= __rhs;223return *this;224}225_LIBCPP_HIDE_FROM_ABI duration& operator%=(const duration& __rhs) {226__rep_ %= __rhs.count();227return *this;228}229230// special values231232_LIBCPP_HIDE_FROM_ABI static duration zero() _NOEXCEPT { return duration(duration_values<rep>::zero()); }233_LIBCPP_HIDE_FROM_ABI static duration min() _NOEXCEPT { return duration(duration_values<rep>::min()); }234_LIBCPP_HIDE_FROM_ABI static duration max() _NOEXCEPT { return duration(duration_values<rep>::max()); }235};236237typedef duration<long long, nano> nanoseconds;238typedef duration<long long, micro> microseconds;239typedef duration<long long, milli> milliseconds;240typedef duration<long long > seconds;241typedef duration< long, ratio< 60> > minutes;242typedef duration< long, ratio<3600> > hours;243244// Duration ==245246template <class _LhsDuration, class _RhsDuration>247struct __duration_eq {248_LIBCPP_HIDE_FROM_ABI bool operator()(const _LhsDuration& __lhs, const _RhsDuration& __rhs) const {249typedef typename common_type<_LhsDuration, _RhsDuration>::type _Ct;250return _Ct(__lhs).count() == _Ct(__rhs).count();251}252};253254template <class _LhsDuration>255struct __duration_eq<_LhsDuration, _LhsDuration> {256_LIBCPP_HIDE_FROM_ABI bool operator()(const _LhsDuration& __lhs, const _LhsDuration& __rhs) const {257return __lhs.count() == __rhs.count();258}259};260261template <class _Rep1, class _Period1, class _Rep2, class _Period2>262inline _LIBCPP_HIDE_FROM_ABI bool263operator==(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {264return __duration_eq<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >()(__lhs, __rhs);265}266267// Duration !=268269template <class _Rep1, class _Period1, class _Rep2, class _Period2>270inline _LIBCPP_HIDE_FROM_ABI bool271operator!=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {272return !(__lhs == __rhs);273}274275// Duration <276277template <class _LhsDuration, class _RhsDuration>278struct __duration_lt {279_LIBCPP_HIDE_FROM_ABI bool operator()(const _LhsDuration& __lhs, const _RhsDuration& __rhs) const {280typedef typename common_type<_LhsDuration, _RhsDuration>::type _Ct;281return _Ct(__lhs).count() < _Ct(__rhs).count();282}283};284285template <class _LhsDuration>286struct __duration_lt<_LhsDuration, _LhsDuration> {287_LIBCPP_HIDE_FROM_ABI bool operator()(const _LhsDuration& __lhs, const _LhsDuration& __rhs) const {288return __lhs.count() < __rhs.count();289}290};291292template <class _Rep1, class _Period1, class _Rep2, class _Period2>293inline _LIBCPP_HIDE_FROM_ABI bool294operator<(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {295return __duration_lt<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >()(__lhs, __rhs);296}297298// Duration >299300template <class _Rep1, class _Period1, class _Rep2, class _Period2>301inline _LIBCPP_HIDE_FROM_ABI bool302operator>(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {303return __rhs < __lhs;304}305306// Duration <=307308template <class _Rep1, class _Period1, class _Rep2, class _Period2>309inline _LIBCPP_HIDE_FROM_ABI bool310operator<=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {311return !(__rhs < __lhs);312}313314// Duration >=315316template <class _Rep1, class _Period1, class _Rep2, class _Period2>317inline _LIBCPP_HIDE_FROM_ABI bool318operator>=(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {319return !(__lhs < __rhs);320}321322// Duration +323324template <class _Rep1, class _Period1, class _Rep2, class _Period2>325inline _LIBCPP_HIDE_FROM_ABI typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type326operator+(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {327typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;328return _Cd(_Cd(__lhs).count() + _Cd(__rhs).count());329}330331// Duration -332333template <class _Rep1, class _Period1, class _Rep2, class _Period2>334inline _LIBCPP_HIDE_FROM_ABI typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type335operator-(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {336typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;337return _Cd(_Cd(__lhs).count() - _Cd(__rhs).count());338}339340// Duration *341342template <class _Rep1,343class _Period,344class _Rep2,345__enable_if_t<is_convertible<const _Rep2&, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>346inline _LIBCPP_HIDE_FROM_ABI duration<typename common_type<_Rep1, _Rep2>::type, _Period>347operator*(const duration<_Rep1, _Period>& __d, const _Rep2& __s) {348typedef typename common_type<_Rep1, _Rep2>::type _Cr;349typedef duration<_Cr, _Period> _Cd;350return _Cd(_Cd(__d).count() * static_cast<_Cr>(__s));351}352353template <class _Rep1,354class _Period,355class _Rep2,356__enable_if_t<is_convertible<const _Rep1&, typename common_type<_Rep1, _Rep2>::type>::value, int> = 0>357inline _LIBCPP_HIDE_FROM_ABI duration<typename common_type<_Rep1, _Rep2>::type, _Period>358operator*(const _Rep1& __s, const duration<_Rep2, _Period>& __d) {359return __d * __s;360}361362// Duration /363364template <class _Rep1,365class _Period,366class _Rep2,367__enable_if_t<!__is_duration<_Rep2>::value &&368is_convertible<const _Rep2&, typename common_type<_Rep1, _Rep2>::type>::value,369int> = 0>370inline _LIBCPP_HIDE_FROM_ABI duration<typename common_type<_Rep1, _Rep2>::type, _Period>371operator/(const duration<_Rep1, _Period>& __d, const _Rep2& __s) {372typedef typename common_type<_Rep1, _Rep2>::type _Cr;373typedef duration<_Cr, _Period> _Cd;374return _Cd(_Cd(__d).count() / static_cast<_Cr>(__s));375}376377template <class _Rep1, class _Period1, class _Rep2, class _Period2>378inline _LIBCPP_HIDE_FROM_ABI typename common_type<_Rep1, _Rep2>::type379operator/(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {380typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Ct;381return _Ct(__lhs).count() / _Ct(__rhs).count();382}383384// Duration %385386template <class _Rep1,387class _Period,388class _Rep2,389__enable_if_t<!__is_duration<_Rep2>::value &&390is_convertible<const _Rep2&, typename common_type<_Rep1, _Rep2>::type>::value,391int> = 0>392inline _LIBCPP_HIDE_FROM_ABI duration<typename common_type<_Rep1, _Rep2>::type, _Period>393operator%(const duration<_Rep1, _Period>& __d, const _Rep2& __s) {394typedef typename common_type<_Rep1, _Rep2>::type _Cr;395typedef duration<_Cr, _Period> _Cd;396return _Cd(_Cd(__d).count() % static_cast<_Cr>(__s));397}398399template <class _Rep1, class _Period1, class _Rep2, class _Period2>400inline _LIBCPP_HIDE_FROM_ABI typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type401operator%(const duration<_Rep1, _Period1>& __lhs, const duration<_Rep2, _Period2>& __rhs) {402typedef typename common_type<_Rep1, _Rep2>::type _Cr;403typedef typename common_type<duration<_Rep1, _Period1>, duration<_Rep2, _Period2> >::type _Cd;404return _Cd(static_cast<_Cr>(_Cd(__lhs).count()) % static_cast<_Cr>(_Cd(__rhs).count()));405}406407} // namespace chrono408409_LIBCPP_END_NAMESPACE_STD410411_LIBCPP_POP_MACROS412413#if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES)414# include <__cxx03/type_traits>415#endif416417#endif // _LIBCPP___CXX03___CHRONO_DURATION_H418419420