1a1bde736SShuo Chen// excerpts from http://code.google.com/p/muduo/
2a1bde736SShuo Chen//
3a1bde736SShuo Chen// Use of this source code is governed by a BSD-style license
4a1bde736SShuo Chen// that can be found in the License file.
5a1bde736SShuo Chen//
6a1bde736SShuo Chen// Author: Shuo Chen (chenshuo at chenshuo dot com)
7a1bde736SShuo Chen
8a1bde736SShuo Chen#define __STDC_LIMIT_MACROS
9a1bde736SShuo Chen#include "TimerQueue.h"
10a1bde736SShuo Chen
11a1bde736SShuo Chen#include "logging/Logging.h"
12a1bde736SShuo Chen#include "EventLoop.h"
13a1bde736SShuo Chen#include "Timer.h"
14a1bde736SShuo Chen#include "TimerId.h"
15a1bde736SShuo Chen
16a1bde736SShuo Chen#include <boost/bind.hpp>
17a1bde736SShuo Chen#include <boost/foreach.hpp>
18a1bde736SShuo Chen
19a1bde736SShuo Chen#include <sys/timerfd.h>
20a1bde736SShuo Chen
21a1bde736SShuo Chennamespace muduo
22a1bde736SShuo Chen{
23a1bde736SShuo Chennamespace detail
24a1bde736SShuo Chen{
25a1bde736SShuo Chen
26a1bde736SShuo Chenint createTimerfd()
27a1bde736SShuo Chen{
28a1bde736SShuo Chen  int timerfd = ::timerfd_create(CLOCK_MONOTONIC,
29a1bde736SShuo Chen                                 TFD_NONBLOCK | TFD_CLOEXEC);
30a1bde736SShuo Chen  if (timerfd < 0)
31a1bde736SShuo Chen  {
32a1bde736SShuo Chen    LOG_SYSFATAL << "Failed in timerfd_create";
33a1bde736SShuo Chen  }
34a1bde736SShuo Chen  return timerfd;
35a1bde736SShuo Chen}
36a1bde736SShuo Chen
37a1bde736SShuo Chenstruct timespec howMuchTimeFromNow(Timestamp when)
38a1bde736SShuo Chen{
39a1bde736SShuo Chen  int64_t microseconds = when.microSecondsSinceEpoch()
40a1bde736SShuo Chen                         - Timestamp::now().microSecondsSinceEpoch();
41a1bde736SShuo Chen  if (microseconds < 100)
42a1bde736SShuo Chen  {
43a1bde736SShuo Chen    microseconds = 100;
44a1bde736SShuo Chen  }
45a1bde736SShuo Chen  struct timespec ts;
46a1bde736SShuo Chen  ts.tv_sec = static_cast<time_t>(
47a1bde736SShuo Chen      microseconds / Timestamp::kMicroSecondsPerSecond);
48a1bde736SShuo Chen  ts.tv_nsec = static_cast<long>(
49a1bde736SShuo Chen      (microseconds % Timestamp::kMicroSecondsPerSecond) * 1000);
50a1bde736SShuo Chen  return ts;
51a1bde736SShuo Chen}
52a1bde736SShuo Chen
53a1bde736SShuo Chenvoid readTimerfd(int timerfd, Timestamp now)
54a1bde736SShuo Chen{
55a1bde736SShuo Chen  uint64_t howmany;
56a1bde736SShuo Chen  ssize_t n = ::read(timerfd, &howmany, sizeof howmany);
57a1bde736SShuo Chen  LOG_TRACE << "TimerQueue::handleRead() " << howmany << " at " << now.toString();
58a1bde736SShuo Chen  if (n != sizeof howmany)
59a1bde736SShuo Chen  {
60a1bde736SShuo Chen    LOG_ERROR << "TimerQueue::handleRead() reads " << n << " bytes instead of 8";
61a1bde736SShuo Chen  }
62a1bde736SShuo Chen}
63a1bde736SShuo Chen
64a1bde736SShuo Chenvoid resetTimerfd(int timerfd, Timestamp expiration)
65a1bde736SShuo Chen{
66a1bde736SShuo Chen  // wake up loop by timerfd_settime()
67a1bde736SShuo Chen  struct itimerspec newValue;
68a1bde736SShuo Chen  struct itimerspec oldValue;
69a1bde736SShuo Chen  bzero(&newValue, sizeof newValue);
70a1bde736SShuo Chen  bzero(&oldValue, sizeof oldValue);
71a1bde736SShuo Chen  newValue.it_value = howMuchTimeFromNow(expiration);
72a1bde736SShuo Chen  int ret = ::timerfd_settime(timerfd, 0, &newValue, &oldValue);
73a1bde736SShuo Chen  if (ret)
74a1bde736SShuo Chen  {
75a1bde736SShuo Chen    LOG_SYSERR << "timerfd_settime()";
76a1bde736SShuo Chen  }
77a1bde736SShuo Chen}
78a1bde736SShuo Chen
79a1bde736SShuo Chen}
80a1bde736SShuo Chen}
81a1bde736SShuo Chen
82a1bde736SShuo Chenusing namespace muduo;
83a1bde736SShuo Chenusing namespace muduo::detail;
84a1bde736SShuo Chen
85a1bde736SShuo ChenTimerQueue::TimerQueue(EventLoop* loop)
86a1bde736SShuo Chen  : loop_(loop),
87a1bde736SShuo Chen    timerfd_(createTimerfd()),
88a1bde736SShuo Chen    timerfdChannel_(loop, timerfd_),
89a1bde736SShuo Chen    timers_(),
90a1bde736SShuo Chen    callingExpiredTimers_(false)
91a1bde736SShuo Chen{
92a1bde736SShuo Chen  timerfdChannel_.setReadCallback(
93a1bde736SShuo Chen      boost::bind(&TimerQueue::handleRead, this));
94a1bde736SShuo Chen  // we are always reading the timerfd, we disarm it with timerfd_settime.
95a1bde736SShuo Chen  timerfdChannel_.enableReading();
96a1bde736SShuo Chen}
97a1bde736SShuo Chen
98a1bde736SShuo ChenTimerQueue::~TimerQueue()
99a1bde736SShuo Chen{
100a1bde736SShuo Chen  ::close(timerfd_);
101a1bde736SShuo Chen  // do not remove channel, since we're in EventLoop::dtor();
102a1bde736SShuo Chen  for (TimerList::iterator it = timers_.begin();
103a1bde736SShuo Chen      it != timers_.end(); ++it)
104a1bde736SShuo Chen  {
105a1bde736SShuo Chen    delete it->second;
106a1bde736SShuo Chen  }
107a1bde736SShuo Chen}
108a1bde736SShuo Chen
109a1bde736SShuo ChenTimerId TimerQueue::addTimer(const TimerCallback& cb,
110a1bde736SShuo Chen                             Timestamp when,
111a1bde736SShuo Chen                             double interval)
112a1bde736SShuo Chen{
113a1bde736SShuo Chen  Timer* timer = new Timer(cb, when, interval);
114a1bde736SShuo Chen  loop_->runInLoop(
115a1bde736SShuo Chen      boost::bind(&TimerQueue::addTimerInLoop, this, timer));
116a1bde736SShuo Chen  return TimerId(timer, timer->sequence());
117a1bde736SShuo Chen}
118a1bde736SShuo Chen
119a1bde736SShuo Chenvoid TimerQueue::cancel(TimerId timerId)
120a1bde736SShuo Chen{
121a1bde736SShuo Chen  loop_->runInLoop(
122a1bde736SShuo Chen      boost::bind(&TimerQueue::cancelInLoop, this, timerId));
123a1bde736SShuo Chen}
124a1bde736SShuo Chen
125a1bde736SShuo Chenvoid TimerQueue::addTimerInLoop(Timer* timer)
126a1bde736SShuo Chen{
127a1bde736SShuo Chen  loop_->assertInLoopThread();
128a1bde736SShuo Chen  bool earliestChanged = insert(timer);
129a1bde736SShuo Chen
130a1bde736SShuo Chen  if (earliestChanged)
131a1bde736SShuo Chen  {
132a1bde736SShuo Chen    resetTimerfd(timerfd_, timer->expiration());
133a1bde736SShuo Chen  }
134a1bde736SShuo Chen}
135a1bde736SShuo Chen
136a1bde736SShuo Chenvoid TimerQueue::cancelInLoop(TimerId timerId)
137a1bde736SShuo Chen{
138a1bde736SShuo Chen  loop_->assertInLoopThread();
139a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
140fdb5c17cSShuo Chen  ActiveTimer timer(timerId.timer_, timerId.sequence_);
141a1bde736SShuo Chen  ActiveTimerSet::iterator it = activeTimers_.find(timer);
142a1bde736SShuo Chen  if (it != activeTimers_.end())
143a1bde736SShuo Chen  {
144a1bde736SShuo Chen    size_t n = timers_.erase(Entry(it->first->expiration(), it->first));
145a1bde736SShuo Chen    assert(n == 1); (void)n;
146a1bde736SShuo Chen    delete it->first; // FIXME: no delete please
147a1bde736SShuo Chen    activeTimers_.erase(it);
148a1bde736SShuo Chen  }
149a1bde736SShuo Chen  else if (callingExpiredTimers_)
150a1bde736SShuo Chen  {
151a1bde736SShuo Chen    cancelingTimers_.insert(timer);
152a1bde736SShuo Chen  }
153a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
154a1bde736SShuo Chen}
155a1bde736SShuo Chen
156a1bde736SShuo Chenvoid TimerQueue::handleRead()
157a1bde736SShuo Chen{
158a1bde736SShuo Chen  loop_->assertInLoopThread();
159a1bde736SShuo Chen  Timestamp now(Timestamp::now());
160a1bde736SShuo Chen  readTimerfd(timerfd_, now);
161a1bde736SShuo Chen
162a1bde736SShuo Chen  std::vector<Entry> expired = getExpired(now);
163a1bde736SShuo Chen
164a1bde736SShuo Chen  callingExpiredTimers_ = true;
165a1bde736SShuo Chen  cancelingTimers_.clear();
166a1bde736SShuo Chen  // safe to callback outside critical section
167a1bde736SShuo Chen  for (std::vector<Entry>::iterator it = expired.begin();
168a1bde736SShuo Chen      it != expired.end(); ++it)
169a1bde736SShuo Chen  {
170a1bde736SShuo Chen    it->second->run();
171a1bde736SShuo Chen  }
172a1bde736SShuo Chen  callingExpiredTimers_ = false;
173a1bde736SShuo Chen
174a1bde736SShuo Chen  reset(expired, now);
175a1bde736SShuo Chen}
176a1bde736SShuo Chen
177a1bde736SShuo Chenstd::vector<TimerQueue::Entry> TimerQueue::getExpired(Timestamp now)
178a1bde736SShuo Chen{
179a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
180a1bde736SShuo Chen  std::vector<Entry> expired;
181a1bde736SShuo Chen  Entry sentry = std::make_pair(now, reinterpret_cast<Timer*>(UINTPTR_MAX));
182a1bde736SShuo Chen  TimerList::iterator it = timers_.lower_bound(sentry);
183a1bde736SShuo Chen  assert(it == timers_.end() || now < it->first);
184a1bde736SShuo Chen  std::copy(timers_.begin(), it, back_inserter(expired));
185a1bde736SShuo Chen  timers_.erase(timers_.begin(), it);
186a1bde736SShuo Chen
187a1bde736SShuo Chen  BOOST_FOREACH(Entry entry, expired)
188a1bde736SShuo Chen  {
189a1bde736SShuo Chen    ActiveTimer timer(entry.second, entry.second->sequence());
190a1bde736SShuo Chen    size_t n = activeTimers_.erase(timer);
191a1bde736SShuo Chen    assert(n == 1); (void)n;
192a1bde736SShuo Chen  }
193a1bde736SShuo Chen
194a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
195a1bde736SShuo Chen  return expired;
196a1bde736SShuo Chen}
197a1bde736SShuo Chen
198a1bde736SShuo Chenvoid TimerQueue::reset(const std::vector<Entry>& expired, Timestamp now)
199a1bde736SShuo Chen{
200a1bde736SShuo Chen  Timestamp nextExpire;
201a1bde736SShuo Chen
202a1bde736SShuo Chen  for (std::vector<Entry>::const_iterator it = expired.begin();
203a1bde736SShuo Chen      it != expired.end(); ++it)
204a1bde736SShuo Chen  {
205a1bde736SShuo Chen    ActiveTimer timer(it->second, it->second->sequence());
206a1bde736SShuo Chen    if (it->second->repeat()
207a1bde736SShuo Chen        && cancelingTimers_.find(timer) == cancelingTimers_.end())
208a1bde736SShuo Chen    {
209a1bde736SShuo Chen      it->second->restart(now);
210a1bde736SShuo Chen      insert(it->second);
211a1bde736SShuo Chen    }
212a1bde736SShuo Chen    else
213a1bde736SShuo Chen    {
214a1bde736SShuo Chen      // FIXME move to a free list
215a1bde736SShuo Chen      delete it->second;
216a1bde736SShuo Chen    }
217a1bde736SShuo Chen  }
218a1bde736SShuo Chen
219a1bde736SShuo Chen  if (!timers_.empty())
220a1bde736SShuo Chen  {
221a1bde736SShuo Chen    nextExpire = timers_.begin()->second->expiration();
222a1bde736SShuo Chen  }
223a1bde736SShuo Chen
224a1bde736SShuo Chen  if (nextExpire.valid())
225a1bde736SShuo Chen  {
226a1bde736SShuo Chen    resetTimerfd(timerfd_, nextExpire);
227a1bde736SShuo Chen  }
228a1bde736SShuo Chen}
229a1bde736SShuo Chen
230a1bde736SShuo Chenbool TimerQueue::insert(Timer* timer)
231a1bde736SShuo Chen{
232a1bde736SShuo Chen  loop_->assertInLoopThread();
233a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
234a1bde736SShuo Chen  bool earliestChanged = false;
235a1bde736SShuo Chen  Timestamp when = timer->expiration();
236a1bde736SShuo Chen  TimerList::iterator it = timers_.begin();
237a1bde736SShuo Chen  if (it == timers_.end() || when < it->first)
238a1bde736SShuo Chen  {
239a1bde736SShuo Chen    earliestChanged = true;
240a1bde736SShuo Chen  }
241a1bde736SShuo Chen
242a1bde736SShuo Chen  {
243a1bde736SShuo Chen    std::pair<TimerList::iterator, bool> result
244a1bde736SShuo Chen      = timers_.insert(Entry(when, timer));
245a1bde736SShuo Chen    assert(result.second); (void)result;
246a1bde736SShuo Chen  }
247a1bde736SShuo Chen  {
248a1bde736SShuo Chen    std::pair<ActiveTimerSet::iterator, bool> result
249a1bde736SShuo Chen      = activeTimers_.insert(ActiveTimer(timer, timer->sequence()));
250a1bde736SShuo Chen    assert(result.second); (void)result;
251a1bde736SShuo Chen  }
252a1bde736SShuo Chen
253a1bde736SShuo Chen  assert(timers_.size() == activeTimers_.size());
254a1bde736SShuo Chen  return earliestChanged;
255a1bde736SShuo Chen}
256a1bde736SShuo Chen
257