-
Notifications
You must be signed in to change notification settings - Fork 65
/
Copy pathmultimer.c
235 lines (210 loc) · 8.92 KB
/
multimer.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
/********************************************************************
* *
* Voice Terminal (VT) *
* June, 1991 *
* *
* Written at USC/Information Sciences Institute from an earlier *
* version developed by Bolt Beranek and Newman Inc. *
* *
* Copyright (c) 1991 University of Southern California. *
* All rights reserved. *
* *
* Redistribution and use in source and binary forms are permitted *
* provided that the above copyright notice and this paragraph are *
* duplicated in all such forms and that any documentation, *
* advertising materials, and other materials related to such *
* distribution and use acknowledge that the software was *
* developed by the University of Southern California, Information *
* Sciences Institute. The name of the University may not be used *
* to endorse or promote products derived from this software *
* without specific prior written permission. *
* THIS SOFTWARE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR *
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED *
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR *
* PURPOSE. *
* *
*******************************************************************/
/****************************************************************************/
/****** ******/
/****** Multiple Timer Package ******/
/****** ******/
/****************************************************************************/
/* */
/* These routines manage an ordered queue of interval timers so */
/* that a single process may have multiple, independent timers */
/* pending. Each timer is identified by an opaque client handle. */
/* These routines are intended to multiplex the timers onto the */
/* timeout mechanism of the select() call, or onto the single */
/* interval timer provided by setitimer(). */
/* */
/****************************************************************************/
#include <sys/types.h>
#include <assert.h>
#include <stdlib.h>
#include <stdio.h>
#ifndef WIN32
#include <sys/time.h>
#endif
#include "sysdep.h"
#include "notify.h"
#include "multimer.h"
typedef struct TQE {
struct TQE *link; /* link to next timer */
struct timeval time; /* expiration time */
struct timeval interval; /* next interval */
Notify_func func; /* function to be invoked */
Notify_client client;
int which; /* type; currently always ITIMER_REAL */
} TQE;
/* active timer queue, in time order */
static TQE *timerQ = (TQE *)0;
/* queue of free Timer Queue Elements */
static TQE *freeTQEQ = (TQE *)0;
#ifndef timeradd
void timeradd(struct timeval *a, struct timeval *b,
struct timeval *sum)
{
sum->tv_usec = a->tv_usec + b->tv_usec;
if (sum->tv_usec >= 1000000L) {
sum->tv_sec = a->tv_sec + b->tv_sec + 1;
sum->tv_usec -= 1000000L;
}
else {
sum->tv_sec = a->tv_sec + b->tv_sec;
}
} /* timeradd */
#endif
/*
* Return 1 if a < b, 0 otherwise.
*/
static int timerless(struct timeval *a, struct timeval *b)
{
if (a->tv_sec < b->tv_sec ||
(a->tv_sec == b->tv_sec && a->tv_usec < b->tv_usec)) return 1;
return 0;
} /* timerless */
static void timer_check(void)
{
register struct TQE *np;
for (np = timerQ; np; np = np->link) {
assert(np->time.tv_usec < 1000000);
assert(np->interval.tv_usec < 1000000);
}
} /* timer_check */
/*
* This routine sets a timer event for the specified client. The client
* pointer is opaque to this routine but must be unique among all clients.
* Each client may have only one timer pending. If the interval specified
* is zero, the pending timer, if any, for this client will be cancelled.
* Otherwise, a timer event will be created for the requested amount of
* time in the future, and will be inserted in chronological order
* into the queue of all clients' timers.
* interval: in: time interval
* func: in: function to be called when time expires
* client: in: first argument for the handler function
* relative: in: flag; set relative to current time
*/
struct timeval *timer_set(struct timeval *interval,
Notify_func func, Notify_client client, int relative)
{
register struct TQE *np, *op, *tp; /* To scan the timer queue */
/* scan the timer queue to see if client has pending timer */
op = (struct TQE *)&timerQ; /* Fudge OK since link is first */
for (np = timerQ; np; op = np, np = np->link)
if (np->client == client) {
op->link = np->link; /* Yes, remove the timer from Q */
break; /* and stop the search */
}
/* if the requested interval is zero, just free the timer */
if (interval == 0) {
if (np) { /* If we found a timer, */
np->link = freeTQEQ; /* link TQE at head of free Q */
freeTQEQ = np;
}
return 0; /* return, no timer set */
}
/* nonzero interval, calculate new expiration time */
if (!(tp = np)) { /* If no previous timer, get a TQE */
/* allocate timer */
if (!freeTQEQ) {
freeTQEQ = (TQE *)malloc(sizeof(TQE));
freeTQEQ->link = (TQE *)0;
freeTQEQ->interval.tv_usec = 0;
freeTQEQ->interval.tv_sec = 0;
}
tp = freeTQEQ;
freeTQEQ = tp->link;
}
/* calculate expiration time */
if (relative) {
(void) gettimeofday(&(tp->time), NULL);
timeradd(&(tp->time), interval, &(tp->time));
assert(tp->time.tv_usec < 1000000);
}
else tp->time = *interval;
#ifdef DEBUG
printf("timer_set(): %d.%06d\n", tp->time.tv_sec, tp->time.tv_usec);
#endif
tp->func = func;
tp->client = client;
tp->which = ITIMER_REAL;
/* insert new timer into timer queue */
op = (struct TQE *)&timerQ; /* fudge OK since link is first */
for (np = timerQ; np; op = np, np=np->link) {
if (timerless(&tp->time, &np->time)) break;
}
op->link = tp; /* point prev TQE to new one */
tp->link = np; /* point new TQE to next one */
timer_check(); /*DEBUG*/
return &(tp->interval);
} /* timer_set */
/*
* This routine returns a timeout value suitable for use in a select() call.
* Before returning, all timer events that have expired are removed from the
* queue and processed. If no timer events remain, a NULL pointer is returned
* so the select() will just block. Otherwise, the supplied timeval struct is
* filled with the timeout interval until the next timer expires.
* Note: This routine may be called recursively if the timer event handling
* routine leads to another select() call! Therefore, we just take one timer
* at a time, and don't use static variables.
*/
struct timeval *timer_get(struct timeval *timeout)
{
register struct TQE *tp; /* to scan the timer queue */
struct timeval now; /* current time */
timer_check(); /*DEBUG*/
for (;;) {
/* return null pointer if there is no timer pending. */
if (!timerQ) return (struct timeval *)0;
/* check head of timer queue to see if timer has expired */
(void) gettimeofday(&now, NULL);
if (timerless(&now, &timerQ->time)) { /* unexpired, calc timeout */
timeout->tv_sec = timerQ->time.tv_sec - now.tv_sec;
timeout->tv_usec = timerQ->time.tv_usec - now.tv_usec;
if (timeout->tv_usec < 0) {
timeout->tv_usec += 1000000L;
--timeout->tv_sec;
}
assert(timeout->tv_usec < 1000000);
return timeout; /* timeout until timer expires */
} else { /* head timer has expired, */
tp = timerQ; /* so remove it from the */
timerQ = tp->link; /* timer queue, */
tp->link = freeTQEQ;
freeTQEQ = tp;
/* restart timer (absolute) */
if (tp->interval.tv_sec || tp->interval.tv_usec) {
timeradd(&tp->interval, &tp->time, &tp->time);
timer_set(&tp->time, tp->func, tp->client, 0);
}
(*(tp->func))(tp->client); /* call the event handler */
}
} /* loop to see if another timer expired */
} /* timer_get */
/*
* Return 1 if the timer queue is not empty.
*/
int timer_pending(void)
{
return timerQ != 0;
} /* timer_pending */