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