1 1.1 mrg // <experimental/timer> -*- C++ -*- 2 1.1 mrg 3 1.1.1.3 mrg // Copyright (C) 2015-2022 Free Software Foundation, Inc. 4 1.1 mrg // 5 1.1 mrg // This file is part of the GNU ISO C++ Library. This library is free 6 1.1 mrg // software; you can redistribute it and/or modify it under the 7 1.1 mrg // terms of the GNU General Public License as published by the 8 1.1 mrg // Free Software Foundation; either version 3, or (at your option) 9 1.1 mrg // any later version. 10 1.1 mrg 11 1.1 mrg // This library is distributed in the hope that it will be useful, 12 1.1 mrg // but WITHOUT ANY WARRANTY; without even the implied warranty of 13 1.1 mrg // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 14 1.1 mrg // GNU General Public License for more details. 15 1.1 mrg 16 1.1 mrg // Under Section 7 of GPL version 3, you are granted additional 17 1.1 mrg // permissions described in the GCC Runtime Library Exception, version 18 1.1 mrg // 3.1, as published by the Free Software Foundation. 19 1.1 mrg 20 1.1 mrg // You should have received a copy of the GNU General Public License and 21 1.1 mrg // a copy of the GCC Runtime Library Exception along with this program; 22 1.1 mrg // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see 23 1.1 mrg // <http://www.gnu.org/licenses/>. 24 1.1 mrg 25 1.1 mrg /** @file experimental/timer 26 1.1 mrg * This is a TS C++ Library header. 27 1.1.1.2 mrg * @ingroup networking-ts 28 1.1 mrg */ 29 1.1 mrg 30 1.1 mrg #ifndef _GLIBCXX_EXPERIMENTAL_TIMER 31 1.1 mrg #define _GLIBCXX_EXPERIMENTAL_TIMER 1 32 1.1 mrg 33 1.1 mrg #pragma GCC system_header 34 1.1 mrg 35 1.1 mrg #if __cplusplus >= 201402L 36 1.1 mrg 37 1.1.1.3 mrg #include <bits/chrono.h> 38 1.1 mrg #include <system_error> 39 1.1 mrg #include <thread> 40 1.1 mrg #include <experimental/netfwd> 41 1.1 mrg #include <experimental/io_context> 42 1.1 mrg #include <experimental/bits/net.h> 43 1.1 mrg 44 1.1 mrg namespace std _GLIBCXX_VISIBILITY(default) 45 1.1 mrg { 46 1.1 mrg _GLIBCXX_BEGIN_NAMESPACE_VERSION 47 1.1 mrg namespace experimental 48 1.1 mrg { 49 1.1 mrg namespace net 50 1.1 mrg { 51 1.1 mrg inline namespace v1 52 1.1 mrg { 53 1.1 mrg 54 1.1.1.2 mrg /** @addtogroup networking-ts 55 1.1.1.2 mrg * @{ 56 1.1 mrg */ 57 1.1 mrg 58 1.1 mrg template<typename _Clock> 59 1.1 mrg struct wait_traits 60 1.1 mrg { 61 1.1 mrg static typename _Clock::duration 62 1.1 mrg to_wait_duration(const typename _Clock::duration& __d) 63 1.1 mrg { return __d; } 64 1.1 mrg 65 1.1 mrg static typename _Clock::duration 66 1.1 mrg to_wait_duration(const typename _Clock::time_point& __t) 67 1.1 mrg { 68 1.1 mrg auto __now = _Clock::now(); 69 1.1 mrg auto __diff = __t - __now; 70 1.1 mrg if (__diff > _Clock::duration::max()) 71 1.1 mrg return _Clock::duration::max(); 72 1.1 mrg if (__diff < _Clock::duration::min()) 73 1.1 mrg return _Clock::duration::min(); 74 1.1 mrg return __diff; 75 1.1 mrg } 76 1.1 mrg }; 77 1.1 mrg 78 1.1 mrg template<typename _Clock, typename _WaitTraits> 79 1.1 mrg class basic_waitable_timer 80 1.1 mrg { 81 1.1 mrg public: 82 1.1 mrg // types: 83 1.1 mrg 84 1.1.1.3 mrg using executor_type = io_context::executor_type; 85 1.1.1.3 mrg using clock_type = _Clock; 86 1.1.1.3 mrg using duration = typename clock_type::duration; 87 1.1.1.3 mrg using time_point = typename clock_type::time_point; 88 1.1.1.3 mrg using traits_type = _WaitTraits; 89 1.1 mrg 90 1.1 mrg // construct / copy / destroy: 91 1.1 mrg 92 1.1 mrg explicit 93 1.1 mrg basic_waitable_timer(io_context& __ctx) 94 1.1 mrg : _M_ex(__ctx.get_executor()), _M_expiry() 95 1.1 mrg { } 96 1.1 mrg 97 1.1 mrg basic_waitable_timer(io_context& __ctx, const time_point& __t) 98 1.1 mrg : _M_ex(__ctx.get_executor()), _M_expiry(__t) 99 1.1 mrg { } 100 1.1 mrg 101 1.1 mrg basic_waitable_timer(io_context& __ctx, const duration& __d) 102 1.1 mrg : _M_ex(__ctx.get_executor()), _M_expiry(_Clock::now() + __d) 103 1.1 mrg { } 104 1.1 mrg 105 1.1 mrg basic_waitable_timer(const basic_waitable_timer&) = delete; 106 1.1 mrg 107 1.1 mrg basic_waitable_timer(basic_waitable_timer&& __rhs) 108 1.1 mrg : _M_ex(std::move(__rhs._M_ex)), _M_expiry(__rhs._M_expiry) 109 1.1 mrg { 110 1.1 mrg _M_key.swap(__rhs._M_key); 111 1.1 mrg __rhs._M_expiry = time_point{}; 112 1.1 mrg } 113 1.1 mrg 114 1.1 mrg ~basic_waitable_timer() { cancel(); } 115 1.1 mrg 116 1.1 mrg basic_waitable_timer& operator=(const basic_waitable_timer&) = delete; 117 1.1 mrg 118 1.1 mrg basic_waitable_timer& 119 1.1 mrg operator=(basic_waitable_timer&& __rhs) 120 1.1 mrg { 121 1.1 mrg if (this == std::addressof(__rhs)) 122 1.1 mrg return *this; 123 1.1 mrg cancel(); 124 1.1 mrg _M_ex = std::move(__rhs._M_ex); 125 1.1 mrg _M_expiry = __rhs._M_expiry; 126 1.1 mrg __rhs._M_expiry = time_point{}; 127 1.1 mrg _M_key.swap(__rhs._M_key); 128 1.1 mrg return *this; 129 1.1 mrg } 130 1.1 mrg 131 1.1 mrg // basic_waitable_timer operations: 132 1.1 mrg 133 1.1 mrg executor_type get_executor() noexcept { return _M_ex; } 134 1.1 mrg 135 1.1 mrg size_t cancel() { return _M_ex.context().cancel(*this); } 136 1.1 mrg size_t cancel_one() { return _M_ex.context().cancel_one(*this); } 137 1.1 mrg 138 1.1 mrg time_point expiry() const { return _M_expiry; } 139 1.1 mrg 140 1.1 mrg size_t expires_at(const time_point& __t) 141 1.1 mrg { 142 1.1 mrg size_t __cancelled = cancel(); 143 1.1 mrg _M_expiry = __t; 144 1.1 mrg return __cancelled; 145 1.1 mrg } 146 1.1 mrg 147 1.1 mrg size_t expires_after(const duration& __d) 148 1.1 mrg { return expires_at(_Clock::now() + __d); } 149 1.1 mrg 150 1.1 mrg void wait(); 151 1.1 mrg void wait(error_code& __ec); 152 1.1 mrg 153 1.1 mrg template<typename _CompletionToken> 154 1.1 mrg __deduced_t<_CompletionToken, void(error_code)> 155 1.1 mrg async_wait(_CompletionToken&& __token) 156 1.1 mrg { 157 1.1 mrg async_completion<_CompletionToken, void(error_code)> __init(__token); 158 1.1 mrg _M_ex.context().async_wait(*this, 159 1.1 mrg std::move(__init.completion_handler)); 160 1.1 mrg return __init.result.get(); 161 1.1 mrg } 162 1.1 mrg 163 1.1 mrg private: 164 1.1 mrg executor_type _M_ex; 165 1.1 mrg time_point _M_expiry; 166 1.1 mrg 167 1.1 mrg struct _Key { }; // TODO move _M_expiry into here? 168 1.1 mrg unique_ptr<_Key> _M_key{new _Key}; 169 1.1 mrg 170 1.1 mrg friend class io_context; 171 1.1 mrg }; 172 1.1 mrg 173 1.1.1.3 mrg using system_timer = basic_waitable_timer<chrono::system_clock>; 174 1.1.1.3 mrg using steady_timer = basic_waitable_timer<chrono::steady_clock>; 175 1.1.1.3 mrg using high_resolution_timer 176 1.1.1.3 mrg = basic_waitable_timer<chrono::high_resolution_clock>; 177 1.1 mrg 178 1.1 mrg template<typename _Clock, typename _WaitTraits> 179 1.1 mrg void 180 1.1 mrg basic_waitable_timer<_Clock, _WaitTraits>::wait() 181 1.1 mrg { 182 1.1 mrg _M_ex.dispatch([this] { 183 1.1 mrg while (clock_type::now() < _M_expiry) 184 1.1 mrg this_thread::sleep_for(traits_type::to_wait_duration(_M_expiry)); 185 1.1 mrg }, allocator<void>{}); 186 1.1 mrg } 187 1.1 mrg 188 1.1 mrg template<typename _Clock, typename _WaitTraits> 189 1.1 mrg void 190 1.1 mrg basic_waitable_timer<_Clock, _WaitTraits>::wait(error_code&) 191 1.1 mrg { 192 1.1 mrg _M_ex.dispatch([this] { 193 1.1 mrg while (clock_type::now() < _M_expiry) 194 1.1 mrg this_thread::sleep_for(traits_type::to_wait_duration(_M_expiry)); 195 1.1 mrg }, allocator<void>{}); 196 1.1 mrg } 197 1.1 mrg 198 1.1 mrg /// @} 199 1.1 mrg 200 1.1 mrg } // namespace v1 201 1.1 mrg } // namespace net 202 1.1 mrg } // namespace experimental 203 1.1 mrg _GLIBCXX_END_NAMESPACE_VERSION 204 1.1 mrg } // namespace std 205 1.1 mrg 206 1.1 mrg #endif // C++14 207 1.1 mrg 208 1.1 mrg #endif // _GLIBCXX_EXPERIMENTAL_TIMER 209