xref: /iperf/src/timer.c (revision aee0b741)
1 /*
2  * iperf, Copyright (c) 2014, The Regents of the University of
3  * California, through Lawrence Berkeley National Laboratory (subject
4  * to receipt of any required approvals from the U.S. Dept. of
5  * Energy).  All rights reserved.
6  *
7  * If you have questions about your rights to use or distribute this
8  * software, please contact Berkeley Lab's Technology Transfer
9  * Department at [email protected].
10  *
11  * NOTICE.  This software is owned by the U.S. Department of Energy.
12  * As such, the U.S. Government has been granted for itself and others
13  * acting on its behalf a paid-up, nonexclusive, irrevocable,
14  * worldwide license in the Software to reproduce, prepare derivative
15  * works, and perform publicly and display publicly.  Beginning five
16  * (5) years after the date permission to assert copyright is obtained
17  * from the U.S. Department of Energy, and subject to any subsequent
18  * five (5) year renewals, the U.S. Government is granted for itself
19  * and others acting on its behalf a paid-up, nonexclusive,
20  * irrevocable, worldwide license in the Software to reproduce,
21  * prepare derivative works, distribute copies to the public, perform
22  * publicly and display publicly, and to permit others to do so.
23  *
24  * This code is distributed under a BSD style license, see the LICENSE
25  * file for complete information.
26  *
27  * Based on timers.c by Jef Poskanzer. Used with permission.
28  */
29 
30 #include <sys/types.h>
31 #include <stdlib.h>
32 
33 #include "timer.h"
34 
35 
36 static Timer* timers = NULL;
37 static Timer* free_timers = NULL;
38 
39 TimerClientData JunkClientData;
40 
41 
42 
43 /* This is an efficiency tweak.  All the routines that need to know the
44 ** current time get passed a pointer to a struct timeval.  If it's non-NULL
45 ** it gets used, otherwise we do our own gettimeofday() to fill it in.
46 ** This lets the caller avoid extraneous gettimeofday()s when efficiency
47 ** is needed, and not bother with the extra code when efficiency doesn't
48 ** matter too much.
49 */
50 static void
51 getnow( struct timeval* nowP, struct timeval* nowP2 )
52 {
53     if ( nowP != NULL )
54 	*nowP2 = *nowP;
55     else
56 	(void) gettimeofday( nowP2, NULL );
57 }
58 
59 
60 static void
61 list_add( Timer* t )
62 {
63     Timer* t2;
64     Timer* t2prev;
65 
66     if ( timers == NULL ) {
67 	/* The list is empty. */
68 	timers = t;
69 	t->prev = t->next = NULL;
70     } else {
71 	if ( t->time.tv_sec < timers->time.tv_sec ||
72 	     ( t->time.tv_sec == timers->time.tv_sec &&
73 	       t->time.tv_usec < timers->time.tv_usec ) ) {
74 	    /* The new timer goes at the head of the list. */
75 	    t->prev = NULL;
76 	    t->next = timers;
77 	    timers->prev = t;
78 	    timers = t;
79 	} else {
80 	    /* Walk the list to find the insertion point. */
81 	    for ( t2prev = timers, t2 = timers->next; t2 != NULL;
82 		  t2prev = t2, t2 = t2->next ) {
83 		if ( t->time.tv_sec < t2->time.tv_sec ||
84 		     ( t->time.tv_sec == t2->time.tv_sec &&
85 		       t->time.tv_usec < t2->time.tv_usec ) ) {
86 		    /* Found it. */
87 		    t2prev->next = t;
88 		    t->prev = t2prev;
89 		    t->next = t2;
90 		    t2->prev = t;
91 		    return;
92 		}
93 	    }
94 	    /* Oops, got to the end of the list.  Add to tail. */
95 	    t2prev->next = t;
96 	    t->prev = t2prev;
97 	    t->next = NULL;
98 	}
99     }
100 }
101 
102 
103 static void
104 list_remove( Timer* t )
105 {
106     if ( t->prev == NULL )
107 	timers = t->next;
108     else
109 	t->prev->next = t->next;
110     if ( t->next != NULL )
111 	t->next->prev = t->prev;
112 }
113 
114 
115 static void
116 list_resort( Timer* t )
117 {
118     /* Remove the timer from the list. */
119     list_remove( t );
120     /* And add it back in, sorted correctly. */
121     list_add( t );
122 }
123 
124 
125 static void
126 add_usecs( struct timeval* t, int64_t usecs )
127 {
128     t->tv_sec += usecs / 1000000L;
129     t->tv_usec += usecs % 1000000L;
130     if ( t->tv_usec >= 1000000L ) {
131 	t->tv_sec += t->tv_usec / 1000000L;
132 	t->tv_usec %= 1000000L;
133     }
134 }
135 
136 
137 Timer*
138 tmr_create(
139     struct timeval* nowP, TimerProc* timer_proc, TimerClientData client_data,
140     int64_t usecs, int periodic )
141 {
142     struct timeval now;
143     Timer* t;
144 
145     getnow( nowP, &now );
146 
147     if ( free_timers != NULL ) {
148 	t = free_timers;
149 	free_timers = t->next;
150     } else {
151 	t = (Timer*) malloc( sizeof(Timer) );
152 	if ( t == NULL )
153 	    return NULL;
154     }
155 
156     t->timer_proc = timer_proc;
157     t->client_data = client_data;
158     t->usecs = usecs;
159     t->periodic = periodic;
160     t->time = now;
161     add_usecs( &t->time, usecs );
162     /* Add the new timer to the active list. */
163     list_add( t );
164 
165     return t;
166 }
167 
168 
169 struct timeval*
170 tmr_timeout( struct timeval* nowP )
171 {
172     struct timeval now;
173     int64_t usecs;
174     static struct timeval timeout;
175 
176     getnow( nowP, &now );
177     /* Since the list is sorted, we only need to look at the first timer. */
178     if ( timers == NULL )
179 	return NULL;
180     usecs = ( timers->time.tv_sec - now.tv_sec ) * 1000000LL +
181 	    ( timers->time.tv_usec - now.tv_usec );
182     if ( usecs <= 0 )
183 	usecs = 0;
184     timeout.tv_sec = usecs / 1000000LL;
185     timeout.tv_usec = usecs % 1000000LL;
186     return &timeout;
187 }
188 
189 
190 void
191 tmr_run( struct timeval* nowP )
192 {
193     struct timeval now;
194     Timer* t;
195     Timer* next;
196 
197     getnow( nowP, &now );
198     for ( t = timers; t != NULL; t = next ) {
199 	next = t->next;
200 	/* Since the list is sorted, as soon as we find a timer
201 	** that isn't ready yet, we are done.
202 	*/
203 	if ( t->time.tv_sec > now.tv_sec ||
204 	     ( t->time.tv_sec == now.tv_sec &&
205 	       t->time.tv_usec > now.tv_usec ) )
206 	    break;
207 	(t->timer_proc)( t->client_data, &now );
208 	if ( t->periodic ) {
209 	    /* Reschedule. */
210 	    add_usecs( &t->time, t->usecs );
211 	    list_resort( t );
212 	} else
213 	    tmr_cancel( t );
214     }
215 }
216 
217 
218 void
219 tmr_reset( struct timeval* nowP, Timer* t )
220 {
221     struct timeval now;
222 
223     getnow( nowP, &now );
224     t->time = now;
225     add_usecs( &t->time, t->usecs );
226     list_resort( t );
227 }
228 
229 
230 void
231 tmr_cancel( Timer* t )
232 {
233     /* Remove it from the active list. */
234     list_remove( t );
235     /* And put it on the free list. */
236     t->next = free_timers;
237     free_timers = t;
238     t->prev = NULL;
239 }
240 
241 
242 void
243 tmr_cleanup( void )
244 {
245     Timer* t;
246 
247     while ( free_timers != NULL ) {
248 	t = free_timers;
249 	free_timers = t->next;
250 	free( (void*) t );
251     }
252 }
253 
254 
255 void
256 tmr_destroy( void )
257 {
258     while ( timers != NULL )
259 	tmr_cancel( timers );
260     tmr_cleanup();
261 }
262