1 // excerpts from http://code.google.com/p/muduo/
2 //
3 // Use of this source code is governed by a BSD-style license
4 // that can be found in the License file.
5 //
6 // Author: Shuo Chen (chenshuo at chenshuo dot com)
7 
8 #ifndef MUDUO_NET_EVENTLOOP_H
9 #define MUDUO_NET_EVENTLOOP_H
10 
11 #include "datetime/Timestamp.h"
12+#include "thread/Mutex.h"
13 #include "thread/Thread.h"
14 #include "Callbacks.h"
15 #include "TimerId.h"
16 
17 #include <boost/scoped_ptr.hpp>
18 #include <vector>
19 
20 namespace muduo
21 {
22 
23 class Channel;
24 class Poller;
25 class TimerQueue;
26 
27 class EventLoop : boost::noncopyable
28 {
29  public:
30+  typedef boost::function<void()> Functor;
31 
32   EventLoop();
33 
34   // force out-line dtor, for scoped_ptr members.
35   ~EventLoop();
36 
37   ///
38   /// Loops forever.
39   ///
40   /// Must be called in the same thread as creation of the object.
41   ///
42   void loop();
43 
44   void quit();
45 
46   ///
47   /// Time when poll returns, usually means data arrivial.
48   ///
49   Timestamp pollReturnTime() const { return pollReturnTime_; }
50 
51+  /// Runs callback immediately in the loop thread.
52+  /// It wakes up the loop, and run the cb.
53+  /// If in the same loop thread, cb is run within the function.
54+  /// Safe to call from other threads.
55+  void runInLoop(const Functor& cb);
56+  /// Queues callback in the loop thread.
57+  /// Runs after finish pooling.
58+  /// Safe to call from other threads.
59+  void queueInLoop(const Functor& cb);
60+
61   // timers
62 
63   ///
64   /// Runs callback at 'time'.
65+  /// Safe to call from other threads.
66   ///
67   TimerId runAt(const Timestamp& time, const TimerCallback& cb);
68   ///
69   /// Runs callback after @c delay seconds.
70+  /// Safe to call from other threads.
71   ///
72   TimerId runAfter(double delay, const TimerCallback& cb);
73   ///
74   /// Runs callback every @c interval seconds.
75+  /// Safe to call from other threads.
76   ///
77   TimerId runEvery(double interval, const TimerCallback& cb);
78 
79   // void cancel(TimerId timerId);
80 
81   // internal use only
82+  void wakeup();
83   void updateChannel(Channel* channel);
84   // void removeChannel(Channel* channel);
85 
86   void assertInLoopThread()
87   {
88     if (!isInLoopThread())
89     {
90       abortNotInLoopThread();
91     }
92   }
93 
94   bool isInLoopThread() const { return threadId_ == CurrentThread::tid(); }
95 
96  private:
97 
98   void abortNotInLoopThread();
99+  void handleRead();  // waked up
100+  void doPendingFunctors();
101 
102   typedef std::vector<Channel*> ChannelList;
103 
104   bool looping_; /* atomic */
105   bool quit_; /* atomic */
106+  bool callingPendingFunctors_; /* atomic */
107   const pid_t threadId_;
108   Timestamp pollReturnTime_;
109   boost::scoped_ptr<Poller> poller_;
110   boost::scoped_ptr<TimerQueue> timerQueue_;
111+  int wakeupFd_;
112+  // unlike in TimerQueue, which is an internal class,
113+  // we don't expose Channel to client.
114+  boost::scoped_ptr<Channel> wakeupChannel_;
115   ChannelList activeChannels_;
116+  MutexLock mutex_;
117+  std::vector<Functor> pendingFunctors_; // @GuardedBy mutex_
118 };
119 
120 }
121 
122 #endif  // MUDUO_NET_EVENTLOOP_H
123