Commit fe6bc8c5 authored by Rusty Russell's avatar Rusty Russell

timer: change timers_expire() to return a single timer.

The linked list method was problematic, especially if timers delete
other expired timers (eg. the next one in the expired list: the timer_del
will delete it from expired, but that's a bit unexpected).
Signed-off-by: default avatarRusty Russell <rusty@rustcorp.com.au>
parent 2ecae4f3
...@@ -28,7 +28,7 @@ ...@@ -28,7 +28,7 @@
* { * {
* struct timers timers; * struct timers timers;
* struct list_head strings; * struct list_head strings;
* struct list_head expired; * struct timer *t;
* struct timed_string *s; * struct timed_string *s;
* *
* timers_init(&timers, time_now()); * timers_init(&timers, time_now());
...@@ -48,9 +48,8 @@ ...@@ -48,9 +48,8 @@
* struct timeabs now = time_now(); * struct timeabs now = time_now();
* list_for_each(&strings, s, node) * list_for_each(&strings, s, node)
* printf("%s", s->string); * printf("%s", s->string);
* timers_expire(&timers, now, &expired); * while ((t = timers_expire(&timers, now)) != NULL) {
* while ((s = list_pop(&expired, struct timed_string, * s = container_of(t, struct timed_string, timer);
* timer.list)) != NULL) {
* list_del_from(&strings, &s->node); * list_del_from(&strings, &s->node);
* free(s); * free(s);
* } * }
......
...@@ -34,10 +34,10 @@ int main(int argc, char *argv[]) ...@@ -34,10 +34,10 @@ int main(int argc, char *argv[])
curr = time_add(curr, time_from_msec(1)); curr = time_add(curr, time_from_msec(1));
if (check) if (check)
timers_check(&timers, NULL); timers_check(&timers, NULL);
timers_expire(&timers, curr, &expired); if (timers_expire(&timers, curr))
abort();
if (check) if (check)
timers_check(&timers, NULL); timers_check(&timers, NULL);
assert(list_empty(&expired));
if (i >= PER_CONN_TIME) { if (i >= PER_CONN_TIME) {
timer_del(&timers, &t[i%PER_CONN_TIME]); timer_del(&timers, &t[i%PER_CONN_TIME]);
......
...@@ -7,7 +7,6 @@ int main(void) ...@@ -7,7 +7,6 @@ int main(void)
{ {
struct timers timers; struct timers timers;
struct timer t; struct timer t;
struct list_head list;
/* This is how many tests you plan to run */ /* This is how many tests you plan to run */
plan_tests(7); plan_tests(7);
...@@ -17,12 +16,10 @@ int main(void) ...@@ -17,12 +16,10 @@ int main(void)
timer_add(&timers, &t, grains_to_time(1364984761003398ULL)); timer_add(&timers, &t, grains_to_time(1364984761003398ULL));
ok1(t.time == 1364984761003398ULL); ok1(t.time == 1364984761003398ULL);
ok1(timers.first == 1364984761003398ULL); ok1(timers.first == 1364984761003398ULL);
timers_expire(&timers, grains_to_time(1364984760903444ULL), &list); ok1(!timers_expire(&timers, grains_to_time(1364984760903444ULL)));
ok1(timers_check(&timers, NULL)); ok1(timers_check(&timers, NULL));
ok1(list_pop(&list, struct timer, list) == NULL); ok1(!timers_expire(&timers, grains_to_time(1364984761002667ULL)));
timers_expire(&timers, grains_to_time(1364984761002667ULL), &list);
ok1(timers_check(&timers, NULL)); ok1(timers_check(&timers, NULL));
ok1(list_pop(&list, struct timer, list) == NULL);
timers_cleanup(&timers); timers_cleanup(&timers);
......
...@@ -12,20 +12,17 @@ static struct timeabs timeabs_from_usec(unsigned long long usec) ...@@ -12,20 +12,17 @@ static struct timeabs timeabs_from_usec(unsigned long long usec)
int main(void) int main(void)
{ {
struct timers timers; struct timers timers;
struct timer t; struct timer t, *expired;
struct list_head expired;
/* This is how many tests you plan to run */ /* This is how many tests you plan to run */
plan_tests(3); plan_tests(3);
timers_init(&timers, timeabs_from_usec(1364726722653919ULL)); timers_init(&timers, timeabs_from_usec(1364726722653919ULL));
timer_add(&timers, &t, timeabs_from_usec(1364726722703919ULL)); timer_add(&timers, &t, timeabs_from_usec(1364726722703919ULL));
timers_expire(&timers, timeabs_from_usec(1364726722653920ULL), &expired); ok1(!timers_expire(&timers, timeabs_from_usec(1364726722653920ULL)));
ok1(list_empty(&expired)); expired = timers_expire(&timers, timeabs_from_usec(1364726725454187ULL));
timers_expire(&timers, timeabs_from_usec(1364726725454187ULL), &expired); ok1(expired == &t);
ok1(!list_empty(&expired)); ok1(!timers_expire(&timers, timeabs_from_usec(1364726725454187ULL)));
ok1(list_top(&expired, struct timer, list) == &t);
timers_cleanup(&timers); timers_cleanup(&timers);
/* This exits depending on whether all tests passed */ /* This exits depending on whether all tests passed */
......
...@@ -13,7 +13,6 @@ int main(void) ...@@ -13,7 +13,6 @@ int main(void)
{ {
struct timers timers; struct timers timers;
struct timer t[64]; struct timer t[64];
struct list_head expired;
struct timeabs earliest; struct timeabs earliest;
uint64_t i; uint64_t i;
struct timeabs epoch = { { 0, 0 } }; struct timeabs epoch = { { 0, 0 } };
...@@ -69,13 +68,11 @@ int main(void) ...@@ -69,13 +68,11 @@ int main(void)
struct timer *t1, *t2; struct timer *t1, *t2;
ok1(timer_earliest(&timers, &earliest)); ok1(timer_earliest(&timers, &earliest));
timers_expire(&timers, earliest, &expired); t1 = timers_expire(&timers, earliest);
t1 = list_pop(&expired, struct timer, list);
ok1(t1); ok1(t1);
t2 = list_pop(&expired, struct timer, list); t2 = timers_expire(&timers, earliest);
ok1(t2); ok1(t2);
ok1(list_empty(&expired)); ok1(!timers_expire(&timers, earliest));
ok1(t1 == &t[i*2] || t1 == &t[i*2+1]); ok1(t1 == &t[i*2] || t1 == &t[i*2+1]);
ok1(t2 != t1 && (t2 == &t[i*2] || t2 == &t[i*2+1])); ok1(t2 != t1 && (t2 == &t[i*2] || t2 == &t[i*2+1]));
......
...@@ -259,37 +259,35 @@ static void timer_fast_forward(struct timers *timers, uint64_t time) ...@@ -259,37 +259,35 @@ static void timer_fast_forward(struct timers *timers, uint64_t time)
timer_add_raw(timers, i); timer_add_raw(timers, i);
} }
/* Fills list of expired timers. */ /* Returns an expired timer. */
void timers_expire(struct timers *timers, struct timer *timers_expire(struct timers *timers, struct timeabs expire)
struct timeabs expire,
struct list_head *list)
{ {
uint64_t now = time_to_grains(expire); uint64_t now = time_to_grains(expire);
unsigned int off; unsigned int off;
struct timer *t;
assert(now >= timers->base); assert(now >= timers->base);
list_head_init(list);
if (!timers->level[0]) { if (!timers->level[0]) {
if (list_empty(&timers->far)) if (list_empty(&timers->far))
return; return NULL;
add_level(timers, 0); add_level(timers, 0);
} }
do { do {
if (timers->first > now) { if (timers->first > now) {
timer_fast_forward(timers, now); timer_fast_forward(timers, now);
break; return NULL;
} }
timer_fast_forward(timers, timers->first); timer_fast_forward(timers, timers->first);
off = timers->base % PER_LEVEL; off = timers->base % PER_LEVEL;
list_append_list(list, &timers->level[0]->list[off]); /* This *may* be NULL, if we deleted the first timer */
if (timers->base == now) t = list_pop(&timers->level[0]->list[off], struct timer, list);
break; } while (!t && update_first(timers));
} while (update_first(timers));
return t;
} }
static bool timer_list_check(const struct list_head *l, static bool timer_list_check(const struct list_head *l,
......
...@@ -89,32 +89,29 @@ void timer_del(struct timers *timers, struct timer *timer); ...@@ -89,32 +89,29 @@ void timer_del(struct timers *timers, struct timer *timer);
bool timer_earliest(struct timers *timers, struct timeabs *first); bool timer_earliest(struct timers *timers, struct timeabs *first);
/** /**
* timers_expire - update timers structure and remove expired timers. * timers_expire - update timers structure and remove one expire timer.
* @timers: the struct timers * @timers: the struct timers
* @expire: the current time * @expire: the current time
* @list: the list for expired timers.
* *
* @list will be initialized to the empty list, then all timers added * A timers added with a @when arg less than or equal to @expire will be
* with a @when arg less than or equal to @expire will be added to it in * returned (within TIMER_GRANULARITY nanosecond precision). If
* expiry order (within TIMER_GRANULARITY nanosecond precision). * there are no timers due to expire, NULL is returned.
* *
* After this, @expire is considered the current time, and adding any * After this returns NULL, @expire is considered the current time,
* timers with @when before this value will be silently changed to * and adding any timers with @when before this value will be silently
* adding them with immediate expiration. * changed to adding them with immediate expiration.
* *
* You should not move @expire backwards, though it need not move * You should not move @expire backwards, though it need not move
* forwards. * forwards.
* *
* Example: * Example:
* struct list_head expired; * struct timer *expired;
* *
* timers_expire(&timeouts, time_now(), &expired); * while ((expired = timers_expire(&timeouts, time_now())) != NULL)
* if (!list_empty(&expired))
* printf("Timer expired!\n"); * printf("Timer expired!\n");
*
*/ */
void timers_expire(struct timers *timers, struct timer *timers_expire(struct timers *timers, struct timeabs expire);
struct timeabs expire,
struct list_head *list);
/** /**
* timers_check - check timer structure for consistency * timers_check - check timer structure for consistency
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment