*** empty log message ***
[dana/urxvt.git] / src / iom.C
1 /*
2     iom.C -- generic I/O multiplexor
3     Copyright (C) 2003 Marc Lehmann <pcg@goof.com>
4  
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9  
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14  
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc. 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18 */
19
20 #include "../config.h"
21
22 #include <cstdio>
23
24 #include <sys/select.h>
25 #include <sys/time.h>
26
27 #include "iom.h"
28
29 tstamp NOW;
30 bool iom_valid;
31 io_manager iom;
32
33 template<class watcher>
34 void io_manager::reg (watcher *w, simplevec<watcher *> &queue)
35 {
36   if (find (queue.begin (), queue.end (), w) == queue.end ())
37     queue.push_back (w);
38 }
39
40 template<class watcher>
41 void io_manager::unreg (watcher *w, simplevec<watcher *> &queue)
42 {
43   queue.erase (find (queue.begin (), queue.end (), w));
44 }
45
46 #if IOM_IO
47 io_watcher::~io_watcher ()
48 {
49   if (iom_valid)
50     iom.unreg (this);
51 }
52
53 void io_manager::reg (io_watcher *w)
54 {
55   reg (w, iow);
56 }
57
58 void io_manager::unreg (io_watcher *w)
59 {
60   unreg (w, iow);
61 }
62
63 #endif
64
65 #if IOM_TIME
66 void time_watcher::trigger ()
67 {
68   call (*this);
69
70   iom.reg (this);
71 }
72
73 time_watcher::~time_watcher ()
74 {
75   if (iom_valid)
76     iom.unreg (this);
77
78   at = TSTAMP_CANCEL;
79 }
80
81 void io_manager::reg (time_watcher *w)
82 {
83   reg (w, tw);
84 }
85
86 void io_manager::unreg (time_watcher *w)
87 {
88   unreg (w, tw);
89 }
90 #endif
91
92 #if IOM_CHECK
93 check_watcher::~check_watcher ()
94 {
95   if (iom_valid)
96     iom.unreg (this);
97 }
98
99 void io_manager::reg (check_watcher *w)
100 {
101   reg (w, cw);
102 }
103
104 void io_manager::unreg (check_watcher *w)
105 {
106   unreg (w, cw);
107 }
108 #endif
109
110 #if IOM_IDLE
111 idle_watcher::~idle_watcher ()
112 {
113   if (iom_valid)
114     iom.unreg (this);
115 }
116
117 void io_manager::reg (idle_watcher *w)
118 {
119   reg (w, iw);
120 }
121
122 void io_manager::unreg (idle_watcher *w)
123 {
124   unreg (w, iw);
125 }
126 #endif
127
128 #if IOM_TIME
129 inline void set_now (void)
130 {
131   struct timeval tv;
132
133   gettimeofday (&tv, 0);
134
135   NOW = (tstamp)tv.tv_sec + (tstamp)tv.tv_usec / 1000000;
136 #endif
137 }
138
139 void io_manager::loop ()
140 {
141 #if IOM_TIME
142   set_now ();
143 #endif
144
145   for (;;)
146     {
147       struct timeval *to = 0;
148       struct timeval tval;
149
150 #if IOM_IDLE
151       if (iw.size ())
152         {
153           tval.tv_sec  = 0;
154           tval.tv_usec = 0;
155           to = &tval;
156         }
157       else
158 #endif
159         {
160 #if IOM_TIME
161           time_watcher *w;
162
163           for (;tw.size ();)
164             {
165               w = tw[0];
166
167               for (time_watcher **i = tw.begin (); i < tw.end (); ++i)
168                 if ((*i)->at < w->at)
169                   w = *i;
170
171               if (w->at > NOW)
172                 {
173                   double diff = w->at - NOW;
174                   tval.tv_sec  = (int)diff;
175                   tval.tv_usec = (int)((diff - tval.tv_sec) * 1000000);
176                   to = &tval;
177                   break;
178                 }
179               else if (w->at >= 0)
180                 w->call (*w);
181               else
182                 unreg (w);
183             }
184 #endif
185         }
186
187 #if IOM_CHECK
188       for (int i = 0; i < cw.size (); ++i)
189         cw[i]->call (*cw[i]);
190 #endif
191
192 #if IOM_IO
193       fd_set rfd, wfd;
194
195       FD_ZERO (&rfd);
196       FD_ZERO (&wfd);
197
198       int fds = 0;
199
200       for (io_watcher **w = iow.begin (); w < iow.end (); ++w)
201         {
202           if ((*w)->events & EVENT_READ ) FD_SET ((*w)->fd, &rfd);
203           if ((*w)->events & EVENT_WRITE) FD_SET ((*w)->fd, &wfd);
204
205           if ((*w)->fd >= fds) fds = (*w)->fd + 1;
206         }
207
208       if (!to && !fds)
209         break; // no events
210
211       fds = select (fds, &rfd, &wfd, 0, to);
212 # if IOM_TIME
213       set_now ();
214 # endif
215
216       if (fds > 0)
217         for (int i = 0; i < iow.size (); ++i)
218           {
219             io_watcher *w = iow[i];
220
221             short revents = w->events;
222
223             if (!FD_ISSET (w->fd, &rfd)) revents &= ~EVENT_READ;
224             if (!FD_ISSET (w->fd, &wfd)) revents &= ~EVENT_WRITE;
225
226             if (revents)
227               w->call (*w, revents);
228           }
229 #if IOM_IDLE
230       else if (iw.size ())
231         for (int i = 0; i < iw.size (); ++i)
232           iw[i]->call (*iw[i]);
233 #endif
234
235 #elif IOM_TIME
236       if (!to)
237         break;
238
239       select (0, 0, 0, 0, &to);
240       set_now ();
241 #else
242       break;
243 #endif
244     }
245 }
246
247 io_manager::io_manager ()
248 {
249 #if IOM_TIME
250   set_now ();
251 #endif
252
253   iom_valid = true;
254 }
255
256 io_manager::~io_manager ()
257 {
258   iom_valid = false;
259 }
260