xref: /aosp_15_r20/external/openscreen/util/alarm.cc (revision 3f982cf4871df8771c9d4abe6e9a6f8d829b2736)
1*3f982cf4SFabien Sanglard // Copyright 2019 The Chromium Authors. All rights reserved.
2*3f982cf4SFabien Sanglard // Use of this source code is governed by a BSD-style license that can be
3*3f982cf4SFabien Sanglard // found in the LICENSE file.
4*3f982cf4SFabien Sanglard 
5*3f982cf4SFabien Sanglard #include "util/alarm.h"
6*3f982cf4SFabien Sanglard 
7*3f982cf4SFabien Sanglard #include "util/osp_logging.h"
8*3f982cf4SFabien Sanglard 
9*3f982cf4SFabien Sanglard namespace openscreen {
10*3f982cf4SFabien Sanglard 
11*3f982cf4SFabien Sanglard class Alarm::CancelableFunctor {
12*3f982cf4SFabien Sanglard  public:
CancelableFunctor(Alarm * alarm)13*3f982cf4SFabien Sanglard   explicit CancelableFunctor(Alarm* alarm) : alarm_(alarm) {
14*3f982cf4SFabien Sanglard     OSP_DCHECK(alarm_);
15*3f982cf4SFabien Sanglard     OSP_DCHECK(!alarm_->queued_fire_);
16*3f982cf4SFabien Sanglard     alarm_->queued_fire_ = this;
17*3f982cf4SFabien Sanglard   }
18*3f982cf4SFabien Sanglard 
~CancelableFunctor()19*3f982cf4SFabien Sanglard   ~CancelableFunctor() { Cancel(); }
20*3f982cf4SFabien Sanglard 
CancelableFunctor(CancelableFunctor && other)21*3f982cf4SFabien Sanglard   CancelableFunctor(CancelableFunctor&& other) : alarm_(other.alarm_) {
22*3f982cf4SFabien Sanglard     other.alarm_ = nullptr;
23*3f982cf4SFabien Sanglard     if (alarm_) {
24*3f982cf4SFabien Sanglard       OSP_DCHECK_EQ(alarm_->queued_fire_, &other);
25*3f982cf4SFabien Sanglard       alarm_->queued_fire_ = this;
26*3f982cf4SFabien Sanglard     }
27*3f982cf4SFabien Sanglard   }
28*3f982cf4SFabien Sanglard 
operator =(CancelableFunctor && other)29*3f982cf4SFabien Sanglard   CancelableFunctor& operator=(CancelableFunctor&& other) {
30*3f982cf4SFabien Sanglard     Cancel();
31*3f982cf4SFabien Sanglard     alarm_ = other.alarm_;
32*3f982cf4SFabien Sanglard     other.alarm_ = nullptr;
33*3f982cf4SFabien Sanglard     if (alarm_) {
34*3f982cf4SFabien Sanglard       OSP_DCHECK_EQ(alarm_->queued_fire_, &other);
35*3f982cf4SFabien Sanglard       alarm_->queued_fire_ = this;
36*3f982cf4SFabien Sanglard     }
37*3f982cf4SFabien Sanglard     return *this;
38*3f982cf4SFabien Sanglard   }
39*3f982cf4SFabien Sanglard 
operator ()()40*3f982cf4SFabien Sanglard   void operator()() noexcept {
41*3f982cf4SFabien Sanglard     if (alarm_) {
42*3f982cf4SFabien Sanglard       OSP_DCHECK_EQ(alarm_->queued_fire_, this);
43*3f982cf4SFabien Sanglard       alarm_->queued_fire_ = nullptr;
44*3f982cf4SFabien Sanglard       alarm_->TryInvoke();
45*3f982cf4SFabien Sanglard       alarm_ = nullptr;
46*3f982cf4SFabien Sanglard     }
47*3f982cf4SFabien Sanglard   }
48*3f982cf4SFabien Sanglard 
Cancel()49*3f982cf4SFabien Sanglard   void Cancel() {
50*3f982cf4SFabien Sanglard     if (alarm_) {
51*3f982cf4SFabien Sanglard       OSP_DCHECK_EQ(alarm_->queued_fire_, this);
52*3f982cf4SFabien Sanglard       alarm_->queued_fire_ = nullptr;
53*3f982cf4SFabien Sanglard       alarm_ = nullptr;
54*3f982cf4SFabien Sanglard     }
55*3f982cf4SFabien Sanglard   }
56*3f982cf4SFabien Sanglard 
57*3f982cf4SFabien Sanglard  private:
58*3f982cf4SFabien Sanglard   Alarm* alarm_;
59*3f982cf4SFabien Sanglard };
60*3f982cf4SFabien Sanglard 
Alarm(ClockNowFunctionPtr now_function,TaskRunner * task_runner)61*3f982cf4SFabien Sanglard Alarm::Alarm(ClockNowFunctionPtr now_function, TaskRunner* task_runner)
62*3f982cf4SFabien Sanglard     : now_function_(now_function), task_runner_(task_runner) {
63*3f982cf4SFabien Sanglard   OSP_DCHECK(now_function_);
64*3f982cf4SFabien Sanglard   OSP_DCHECK(task_runner_);
65*3f982cf4SFabien Sanglard }
66*3f982cf4SFabien Sanglard 
~Alarm()67*3f982cf4SFabien Sanglard Alarm::~Alarm() {
68*3f982cf4SFabien Sanglard   if (queued_fire_) {
69*3f982cf4SFabien Sanglard     queued_fire_->Cancel();
70*3f982cf4SFabien Sanglard     OSP_DCHECK(!queued_fire_);
71*3f982cf4SFabien Sanglard   }
72*3f982cf4SFabien Sanglard }
73*3f982cf4SFabien Sanglard 
Cancel()74*3f982cf4SFabien Sanglard void Alarm::Cancel() {
75*3f982cf4SFabien Sanglard   scheduled_task_ = TaskRunner::Task();
76*3f982cf4SFabien Sanglard }
77*3f982cf4SFabien Sanglard 
ScheduleWithTask(TaskRunner::Task task,Clock::time_point desired_alarm_time)78*3f982cf4SFabien Sanglard void Alarm::ScheduleWithTask(TaskRunner::Task task,
79*3f982cf4SFabien Sanglard                              Clock::time_point desired_alarm_time) {
80*3f982cf4SFabien Sanglard   OSP_DCHECK(task.valid());
81*3f982cf4SFabien Sanglard 
82*3f982cf4SFabien Sanglard   scheduled_task_ = std::move(task);
83*3f982cf4SFabien Sanglard 
84*3f982cf4SFabien Sanglard   const Clock::time_point now = now_function_();
85*3f982cf4SFabien Sanglard   alarm_time_ = std::max(now, desired_alarm_time);
86*3f982cf4SFabien Sanglard 
87*3f982cf4SFabien Sanglard   // Ensure that a later firing will occur, and not too late.
88*3f982cf4SFabien Sanglard   if (queued_fire_) {
89*3f982cf4SFabien Sanglard     if (next_fire_time_ <= alarm_time_) {
90*3f982cf4SFabien Sanglard       return;
91*3f982cf4SFabien Sanglard     }
92*3f982cf4SFabien Sanglard     queued_fire_->Cancel();
93*3f982cf4SFabien Sanglard     OSP_DCHECK(!queued_fire_);
94*3f982cf4SFabien Sanglard   }
95*3f982cf4SFabien Sanglard   InvokeLater(now, alarm_time_);
96*3f982cf4SFabien Sanglard }
97*3f982cf4SFabien Sanglard 
InvokeLater(Clock::time_point now,Clock::time_point fire_time)98*3f982cf4SFabien Sanglard void Alarm::InvokeLater(Clock::time_point now, Clock::time_point fire_time) {
99*3f982cf4SFabien Sanglard   OSP_DCHECK(!queued_fire_);
100*3f982cf4SFabien Sanglard   next_fire_time_ = fire_time;
101*3f982cf4SFabien Sanglard   // Note: Instantiating the CancelableFunctor below sets |this->queued_fire_|.
102*3f982cf4SFabien Sanglard   task_runner_->PostTaskWithDelay(CancelableFunctor(this), fire_time - now);
103*3f982cf4SFabien Sanglard }
104*3f982cf4SFabien Sanglard 
TryInvoke()105*3f982cf4SFabien Sanglard void Alarm::TryInvoke() {
106*3f982cf4SFabien Sanglard   if (!scheduled_task_.valid()) {
107*3f982cf4SFabien Sanglard     return;  // This Alarm was canceled in the meantime.
108*3f982cf4SFabien Sanglard   }
109*3f982cf4SFabien Sanglard 
110*3f982cf4SFabien Sanglard   // If this is an early firing, re-schedule for later. This happens if
111*3f982cf4SFabien Sanglard   // Schedule() was called again before this firing had occurred.
112*3f982cf4SFabien Sanglard   const Clock::time_point now = now_function_();
113*3f982cf4SFabien Sanglard   if (now < alarm_time_) {
114*3f982cf4SFabien Sanglard     InvokeLater(now, alarm_time_);
115*3f982cf4SFabien Sanglard     return;
116*3f982cf4SFabien Sanglard   }
117*3f982cf4SFabien Sanglard 
118*3f982cf4SFabien Sanglard   // Move the client Task to the stack before executing, just in case the task
119*3f982cf4SFabien Sanglard   // itself: a) calls any Alarm methods re-entrantly, or b) causes the
120*3f982cf4SFabien Sanglard   // destruction of this Alarm instance.
121*3f982cf4SFabien Sanglard   // WARNING: |this| is not valid after here!
122*3f982cf4SFabien Sanglard   TaskRunner::Task task = std::move(scheduled_task_);
123*3f982cf4SFabien Sanglard   task();
124*3f982cf4SFabien Sanglard }
125*3f982cf4SFabien Sanglard 
126*3f982cf4SFabien Sanglard // static
127*3f982cf4SFabien Sanglard constexpr Clock::time_point Alarm::kImmediately;
128*3f982cf4SFabien Sanglard 
129*3f982cf4SFabien Sanglard }  // namespace openscreen
130