OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
timer.h
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 // Timer implementation using ASIO and Task infrastructure.
6 // Registers an ASIO timer. On ASIO timer expiry, a task will be created to
7 // run the timer. Supports user specified task-id.
8 //
9 // Operations supported
10 // - Create a timer by allocating an object of type Timer
11 // - Start a timer
12 // If a timer is already running, it is a no op. In order to auto-restart
13 // a timer, return 'tru'e from timer callback, false otherwise
14 //
15 // There can be atmost one "Task" outstanding for the timer.
16 //
17 // - Cancel a timer
18 // Cancels a running timer. Both ASIO registerations and task spawned will
19 // be cancelled.
20 //
21 // If timer is already fired, "Cancel" api will fail and return 'false'
22 //
23 // - Delete a timer
24 // Cancels the timer and triggers deletion of the timer. Application should
25 // not access the timer after its deleted
26 //
27 // Concurrency aspects:
28 // - Timer is allocated by application
29 // - Applications must call TimerManager::DeleteTimer() to delete the timer
30 // - All operations on timer are protected by mutex
31 // - When timer is running, it can have references from ASIO and Task.
32 // Timer class will keep of reference from ASIO and Task. Timer will
33 // be deleted when both the references go away. (via intrusive pointer)
34 //
35 
36 #ifndef TIMER_H_
37 #define TIMER_H_
38 
39 #include <tbb/mutex.h>
40 
41 #include <boost/asio/placeholders.hpp>
42 #include <boost/bind.hpp>
43 #include <boost/intrusive_ptr.hpp>
44 #include <boost/scoped_ptr.hpp>
45 #include <boost/function.hpp>
46 #include <boost/asio.hpp>
47 #include <boost/system/error_code.hpp>
48 #include <set>
49 
50 #include <base/task.h>
51 
52 class TimerImpl;
53 
54 class Timer {
55 private:
56  // Task used to fire the timer
57  class TimerTask;
58 
59 public:
60  typedef boost::function<bool(void)> Handler;
61  typedef boost::function<void(std::string, std::string, std::string)>
63 
64  Timer(boost::asio::io_context &service, const std::string &name,
65  int task_id, int task_instance, bool delete_on_completion = false);
66  virtual ~Timer();
67 
68  // Start a timer
69  // If timer already started, try cancelling and restart
70  //
71  // Timer callback routine handler should return true to start the timer
72  // again, false otherwise
73  //
74  // XXX Timer cannot be Start()ed from timer callback routine itself.
75  // Return true from the callback in order to post the timer again. This
76  // would use the same time, the timer was initially started with
77  //
78  bool Start(int time, Handler handler, ErrorHandler error_handler = NULL);
79 
80  //Can be called only from callback
81  bool Reschedule(int time);
82 
83  // Cancel a running timer
84  bool Cancel();
85 
86  bool running() const {
87  tbb::mutex::scoped_lock lock(mutex_);
88  return (state_ == Running);
89  }
90 
91  bool fired() const {
92  tbb::mutex::scoped_lock lock(mutex_);
93  return (state_ == Fired);
94  }
95 
96  int time() const {
97  return time_;
98  }
99 
100  bool cancelled() const {
101  tbb::mutex::scoped_lock lock(mutex_);
102  return (state_ == Cancelled);
103  }
104 
105  bool Idle() const {
106  tbb::mutex::scoped_lock lock(mutex_);
107  return (state_ == Cancelled || state_ == Init);
108  }
109 
110  bool IsDeleteOnCompletion() const {
111  return delete_on_completion_;
112  }
113 
114  int64_t GetElapsedTime() const;
115 
116  // Only for state machine test
117  // XXX: Don't use in production code
118  void Fire() {
119  tbb::mutex::scoped_lock lock(mutex_);
120  if (handler_ && !handler_.empty()) {
121  SetState(Fired);
122  handler_();
123  SetState(Init);
124  }
125  }
126 
127  virtual std::string Description() {
128  return name_;
129  }
130 private:
131  friend class TimerManager;
132  friend class TimerTest;
133 
134  friend void intrusive_ptr_add_ref(Timer *timer);
135  friend void intrusive_ptr_release(Timer *timer);
136  typedef boost::intrusive_ptr<Timer> TimerPtr;
137 
138  enum TimerState {
139  Init = 0,
140  Running = 1,
141  Fired = 2,
143  };
144 
145  // ASIO callback on timer expiry. Start a task to serve the timer
146  void StartTimerTask(TimerPtr reference,
147  int time, uint32_t seq_no,
148  const boost::system::error_code &ec);
149 
150  void SetState(TimerState s) { state_ = s; }
151  static int GetTimerInstanceId() { return -1; }
152  static int GetTimerTaskId() {
153  static int timer_task_id = -1;
154 
155  if (timer_task_id == -1) {
157  timer_task_id = scheduler->GetTaskId("timer::TimerTask");
158  }
159  return timer_task_id;
160  }
161 
162  boost::scoped_ptr<TimerImpl> impl_;
163  std::string name_;
166  mutable tbb::mutex mutex_;
169  int time_;
170  int task_id_;
172  uint32_t seq_no_;
174  tbb::atomic<int> refcount_;
175 };
176 
177 inline void intrusive_ptr_add_ref(Timer *timer) {
178  timer->refcount_.fetch_and_increment();
179 }
180 
181 inline void intrusive_ptr_release(Timer *timer) {
182  int prev = timer->refcount_.fetch_and_decrement();
183  if (prev == 1) {
184  delete timer;
185  }
186 }
187 
188 //
189 // TimerManager is the place holder for all the Timer objects
190 // instantiated in the life time of a process
191 //
192 // Timer objects are help in TimerSet until all the cleanup is complete
193 // and only then should they be deleted via DeleteTimer() API
194 //
195 // Since Timer objects are also held by boost::asio routines, they are
196 // protected using intrusive pointers
197 //
198 // This is similar to how TcpSession objects are managed via TcpSessionPtr
199 //
201 public:
202  static Timer *CreateTimer(boost::asio::io_context &service,
203  const std::string &name,
204  int task_id = Timer::GetTimerTaskId(),
205  int task_instance = Timer::GetTimerInstanceId(),
206  bool delete_on_completion = false);
207  static bool DeleteTimer(Timer *Timer);
208 
209 private:
210  friend class TimerTest;
211 
212  typedef boost::intrusive_ptr<Timer> TimerPtr;
213  struct TimerPtrCmp {
214  bool operator()(const TimerPtr &lhs,
215  const TimerPtr &rhs) const {
216  return lhs.get() < rhs.get();
217  }
218  };
219  typedef std::set<TimerPtr, TimerPtrCmp> TimerSet;
220  static void AddTimer(Timer *Timer);
221 
222  static tbb::mutex mutex_;
224 };
225 
226 #endif /* TIMER_H_ */
int intrusive_ptr_add_ref(const AsPath *cpath)
Definition: bgp_aspath.h:147
tbb::atomic< int > refcount_
Definition: timer.h:174
boost::intrusive_ptr< Timer > TimerPtr
Definition: timer.h:136
bool operator()(const TimerPtr &lhs, const TimerPtr &rhs) const
Definition: timer.h:214
The TaskScheduler keeps track of what tasks are currently schedulable. When a task is enqueued it is ...
Definition: task.h:178
friend class TimerTest
Definition: timer.h:132
std::set< TimerPtr, TimerPtrCmp > TimerSet
Definition: timer.h:219
void Fire()
Definition: timer.h:118
virtual ~Timer()
Definition: timer.cc:99
Handler handler_
Definition: timer.h:164
int task_instance_
Definition: timer.h:171
boost::function< bool(void)> Handler
Definition: timer.h:57
uint32_t seq_no_
Definition: timer.h:172
bool IsDeleteOnCompletion() const
Definition: timer.h:110
virtual std::string Description()
Definition: timer.h:127
static int GetTimerTaskId()
Definition: timer.h:152
TimerTask * timer_task_
Definition: timer.h:168
int GetTaskId(const std::string &name)
Definition: task.cc:856
tbb::mutex mutex_
Definition: timer.h:166
boost::scoped_ptr< TimerImpl > impl_
Definition: timer.h:162
bool fired() const
Definition: timer.h:91
ErrorHandler error_handler_
Definition: timer.h:165
static TaskScheduler * GetInstance()
Definition: task.cc:547
void SetState(TimerState s)
Definition: timer.h:150
int time_
Definition: timer.h:169
Timer(boost::asio::io_context &service, const std::string &name, int task_id, int task_instance, bool delete_on_completion=false)
Definition: timer.cc:83
bool Idle() const
Definition: timer.h:105
TimerState
Definition: timer.h:138
friend class TimerTest
Definition: timer.h:210
static Timer * CreateTimer(boost::asio::io_context &service, const std::string &name, int task_id=Timer::GetTimerTaskId(), int task_instance=Timer::GetTimerInstanceId(), bool delete_on_completion=false)
Definition: timer.cc:201
static tbb::mutex mutex_
Definition: timer.h:222
friend void intrusive_ptr_add_ref(Timer *timer)
Definition: timer.h:177
friend void intrusive_ptr_release(Timer *timer)
Definition: timer.h:181
bool Cancel()
Definition: timer.cc:150
boost::intrusive_ptr< Timer > TimerPtr
Definition: timer.h:212
bool cancelled() const
Definition: timer.h:100
boost::function< void(std::string, std::string, std::string)> ErrorHandler
Definition: timer.h:62
bool Start(int time, Handler handler, ErrorHandler error_handler=NULL)
Definition: timer.cc:108
static TimerSet timer_ref_
Definition: timer.h:223
TimerState state_
Definition: timer.h:167
void intrusive_ptr_release(const AsPath *cpath)
Definition: bgp_aspath.h:155
bool running() const
Definition: timer.h:86
bool Reschedule(int time)
Definition: timer.cc:137
Definition: timer.h:54
std::string name_
Definition: timer.h:163
int64_t GetElapsedTime() const
Definition: timer.cc:235
int task_id_
Definition: timer.h:170
bool delete_on_completion_
Definition: timer.h:173
static int GetTimerInstanceId()
Definition: timer.h:151
int time() const
Definition: timer.h:96
static bool DeleteTimer(Timer *Timer)
Definition: timer.cc:222
static void AddTimer(Timer *Timer)
Definition: timer.cc:210
void StartTimerTask(TimerPtr reference, int time, uint32_t seq_no, const boost::system::error_code &ec)
Definition: timer.cc:172