| /* |
| * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #ifndef API_NETEQ_TICK_TIMER_H_ |
| #define API_NETEQ_TICK_TIMER_H_ |
| |
| #include <stdint.h> |
| |
| #include <memory> |
| |
| #include "rtc_base/checks.h" |
| |
| namespace webrtc { |
| |
| // Implements a time counter. The counter is advanced with the Increment() |
| // methods, and is queried with the ticks() accessor. It is assumed that one |
| // "tick" of the counter corresponds to 10 ms. |
| // A TickTimer object can provide two types of associated time-measuring |
| // objects: Stopwatch and Countdown. |
| class TickTimer { |
| public: |
| // Stopwatch measures time elapsed since it was started, by querying the |
| // associated TickTimer for the current time. The intended use is to request a |
| // new Stopwatch object from a TickTimer object with the GetNewStopwatch() |
| // method. Note: since the Stopwatch object contains a reference to the |
| // TickTimer it is associated with, it cannot outlive the TickTimer. |
| class Stopwatch { |
| public: |
| explicit Stopwatch(const TickTimer& ticktimer); |
| |
| uint64_t ElapsedTicks() const { return ticktimer_.ticks() - starttick_; } |
| |
| uint64_t ElapsedMs() const { |
| const uint64_t elapsed_ticks = ticktimer_.ticks() - starttick_; |
| const int ms_per_tick = ticktimer_.ms_per_tick(); |
| return elapsed_ticks < UINT64_MAX / ms_per_tick |
| ? elapsed_ticks * ms_per_tick |
| : UINT64_MAX; |
| } |
| |
| private: |
| const TickTimer& ticktimer_; |
| const uint64_t starttick_; |
| }; |
| |
| // Countdown counts down from a given start value with each tick of the |
| // associated TickTimer, until zero is reached. The Finished() method will |
| // return true if zero has been reached, false otherwise. The intended use is |
| // to request a new Countdown object from a TickTimer object with the |
| // GetNewCountdown() method. Note: since the Countdown object contains a |
| // reference to the TickTimer it is associated with, it cannot outlive the |
| // TickTimer. |
| class Countdown { |
| public: |
| Countdown(const TickTimer& ticktimer, uint64_t ticks_to_count); |
| |
| ~Countdown(); |
| |
| bool Finished() const { |
| return stopwatch_->ElapsedTicks() >= ticks_to_count_; |
| } |
| |
| private: |
| const std::unique_ptr<Stopwatch> stopwatch_; |
| const uint64_t ticks_to_count_; |
| }; |
| |
| TickTimer() : TickTimer(10) {} |
| explicit TickTimer(int ms_per_tick) : ms_per_tick_(ms_per_tick) { |
| RTC_DCHECK_GT(ms_per_tick_, 0); |
| } |
| |
| TickTimer(const TickTimer&) = delete; |
| TickTimer& operator=(const TickTimer&) = delete; |
| |
| void Increment() { ++ticks_; } |
| |
| // Mainly intended for testing. |
| void Increment(uint64_t x) { ticks_ += x; } |
| |
| uint64_t ticks() const { return ticks_; } |
| |
| int ms_per_tick() const { return ms_per_tick_; } |
| |
| // Returns a new Stopwatch object, based on the current TickTimer. Note that |
| // the new Stopwatch object contains a reference to the current TickTimer, |
| // and must therefore not outlive the TickTimer. |
| std::unique_ptr<Stopwatch> GetNewStopwatch() const { |
| return std::unique_ptr<Stopwatch>(new Stopwatch(*this)); |
| } |
| |
| // Returns a new Countdown object, based on the current TickTimer. Note that |
| // the new Countdown object contains a reference to the current TickTimer, |
| // and must therefore not outlive the TickTimer. |
| std::unique_ptr<Countdown> GetNewCountdown(uint64_t ticks_to_count) const { |
| return std::unique_ptr<Countdown>(new Countdown(*this, ticks_to_count)); |
| } |
| |
| private: |
| uint64_t ticks_ = 0; |
| const int ms_per_tick_; |
| }; |
| |
| } // namespace webrtc |
| #endif // API_NETEQ_TICK_TIMER_H_ |