Diag-Client-Lib
sync_timer.h
Go to the documentation of this file.
1 /* Diagnostic Client library
2 * Copyright (C) 2024 Avijit Dey
3 *
4 * This Source Code Form is subject to the terms of the Mozilla Public
5 * License, v. 2.0. If a copy of the MPL was not distributed with this
6 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7 */
8 #ifndef DIAGNOSTIC_CLIENT_LIB_LIB_UTILITY_UTILITY_SYNC_TIMER_H
9 #define DIAGNOSTIC_CLIENT_LIB_LIB_UTILITY_UTILITY_SYNC_TIMER_H
10 
11 #include <chrono>
12 #include <condition_variable>
13 #include <cstdint>
14 #include <mutex>
15 
16 namespace utility {
17 namespace sync_timer {
18 
24 template<typename ClockType>
25 class SyncTimer final {
26  public:
30  using Clock = ClockType;
31 
35  using TimePoint = std::chrono::time_point<Clock>;
36 
40  enum class TimerState : std::uint8_t { kIdle = 0, kCancelRequested, kTimeout };
41 
46 
51  std::lock_guard<std::mutex> const lck(mutex_lock_);
52  exit_request_ = false;
53  start_running_ = false;
54  cond_var_.notify_all();
55  }
56 
70  template<typename TimeoutCallback, typename CancelCallback>
71  void WaitForTimeout(TimeoutCallback &&timeout_func, CancelCallback &&cancellation_func,
72  std::chrono::milliseconds const timeout) {
73  if (Start(timeout) == TimerState::kTimeout) {
74  timeout_func();
75  } else {
76  cancellation_func();
77  }
78  }
79 
83  auto IsTimerActive() {
84  std::lock_guard<std::mutex> const lck(mutex_lock_);
85  return start_running_;
86  }
87 
91  void CancelWait() { Stop(); }
92 
93  private:
101  auto Start(std::chrono::milliseconds const timeout) noexcept -> TimerState {
102  TimerState timer_state{TimerState::kIdle};
103  std::unique_lock<std::mutex> lck(mutex_lock_);
104  start_running_ = true;
105  TimePoint const expiry_time_point{Clock::now() + timeout};
106  if (cond_var_.wait_until(lck, expiry_time_point, [this, &expiry_time_point, &timer_state]() {
107  bool do_exit_wait{false};
108  // check if exit was requested
109  if (!exit_request_) {
110  TimePoint const current_time{Clock::now()};
111  // check for expiry
112  if (current_time >= expiry_time_point) {
113  // timeout
114  timer_state = TimerState::kTimeout;
115  do_exit_wait = true;
116  } else {
117  // check for cancellation request
118  if (!start_running_) {
119  timer_state = TimerState::kCancelRequested;
120  do_exit_wait = true;
121  } // else - spurious wake-up, do nothing
122  }
123  } else {
124  do_exit_wait = true;
125  }
126  return do_exit_wait;
127  })) {}
128  return timer_state;
129  }
130 
134  void Stop() noexcept {
135  std::lock_guard<std::mutex> const lck(mutex_lock_);
136  start_running_ = false;
137  cond_var_.notify_all();
138  }
139 
140  private:
144  std::condition_variable cond_var_;
145 
149  std::mutex mutex_lock_;
150 
155 
160 };
161 
162 } // namespace sync_timer
163 } // namespace utility
164 
165 #endif // DIAGNOSTIC_CLIENT_LIB_LIB_UTILITY_UTILITY_SYNC_TIMER_H
Timer class for timeout monitoring.
Definition: sync_timer.h:25
SyncTimer()
Construct an instance of SyncTimer.
Definition: sync_timer.h:45
auto Start(std::chrono::milliseconds const timeout) noexcept -> TimerState
Function to start the timeout monitoring.
Definition: sync_timer.h:101
TimerState
Definition of different timer state during timeout monitoring.
Definition: sync_timer.h:40
void Stop() noexcept
Function to stop the current timeout monitoring.
Definition: sync_timer.h:134
std::chrono::time_point< Clock > TimePoint
Type alias for the clock time point.
Definition: sync_timer.h:35
std::mutex mutex_lock_
The mutex for the conditional variable.
Definition: sync_timer.h:149
ClockType Clock
Type alias for the clock type.
Definition: sync_timer.h:30
~SyncTimer()
Destruct an instance of SyncTimer.
Definition: sync_timer.h:50
void CancelWait()
Function to cancel the synchronous wait.
Definition: sync_timer.h:91
auto IsTimerActive()
Function to query if timer is running.
Definition: sync_timer.h:83
void WaitForTimeout(TimeoutCallback &&timeout_func, CancelCallback &&cancellation_func, std::chrono::milliseconds const timeout)
Helper function to wait for response with timeout monitoring.
Definition: sync_timer.h:71
bool start_running_
The flag to stop the current running timer.
Definition: sync_timer.h:159
std::condition_variable cond_var_
The conditional variable needed for synchronizing between start and stop of running timer.
Definition: sync_timer.h:144
bool exit_request_
The flag to terminate the wait.
Definition: sync_timer.h:154