/* * Copyright (C) Igor Sysoev * Copyright (C) NGINX, Inc. */ #include /* * Timer operations are batched in the changes array to improve instruction * and data cache locality of rbtree operations. * * nxt_timer_add() adds or modify a timer. * * nxt_timer_disable() disables a timer. * * nxt_timer_delete() deletes a timer. It returns 1 if there are pending * changes in the changes array or 0 otherwise. */ static intptr_t nxt_timer_rbtree_compare(nxt_rbtree_node_t *node1, nxt_rbtree_node_t *node2); static void nxt_timer_change(nxt_event_engine_t *engine, nxt_timer_t *timer, nxt_timer_operation_t change, nxt_msec_t time); static void nxt_timer_changes_commit(nxt_event_engine_t *engine); static void nxt_timer_handler(nxt_task_t *task, void *obj, void *data); nxt_int_t nxt_timers_init(nxt_timers_t *timers, nxt_uint_t mchanges) { nxt_rbtree_init(&timers->tree, nxt_timer_rbtree_compare); if (mchanges > NXT_TIMER_MAX_CHANGES) { mchanges = NXT_TIMER_MAX_CHANGES; } timers->mchanges = mchanges; timers->changes = nxt_malloc(sizeof(nxt_timer_change_t) * mchanges); if (nxt_fast_path(timers->changes != NULL)) { return NXT_OK; } return NXT_ERROR; } static intptr_t nxt_timer_rbtree_compare(nxt_rbtree_node_t *node1, nxt_rbtree_node_t *node2) { nxt_timer_t *timer1, *timer2; timer1 = (nxt_timer_t *) node1; timer2 = (nxt_timer_t *) node2; /* * Timer values are distributed in small range, usually several minutes * and overflow every 49 days if nxt_msec_t is stored in 32 bits. * This signed comparison takes into account that overflow. */ /* timer1->time < timer2->time */ return nxt_msec_diff(timer1->time , timer2->time); } void nxt_timer_add(nxt_event_engine_t *engine, nxt_timer_t *timer, nxt_msec_t timeout) { int32_t diff; uint32_t time; time = engine->timers.now + timeout; nxt_debug(timer->task, "timer add: %M %M:%M", timer->time, timeout, time); timer->enabled = 1; if (nxt_timer_is_in_tree(timer)) { diff = nxt_msec_diff(time, timer->time); /* * Use the previous timer if difference between it and the * new timer is less than required precision milliseconds: this * decreases number of rbtree operations for fast connections. */ if (nxt_abs(diff) < timer->precision) { nxt_debug(timer->task, "timer previous: %M", time); nxt_timer_change(engine, timer, NXT_TIMER_NOPE, 0); return; } } nxt_timer_change(engine, timer, NXT_TIMER_ADD, time); } nxt_bool_t nxt_timer_delete(nxt_event_engine_t *engine, nxt_timer_t *timer) { nxt_debug(timer->task, "timer delete: %M", timer->time); timer->enabled = 0; if (nxt_timer_is_in_tree(timer)) { nxt_timer_change(engine, timer, NXT_TIMER_DELETE, 0); return 1; } nxt_timer_change(engine, timer, NXT_TIMER_NOPE, 0); return (timer->queued || timer->change != NXT_TIMER_NO_CHANGE); } static void nxt_timer_change(nxt_event_engine_t *engine, nxt_timer_t *timer, nxt_timer_operation_t change, nxt_msec_t time) { nxt_timers_t *timers; nxt_timer_change_t *ch; timers = &engine->timers; if (timer->change == NXT_TIMER_NO_CHANGE) { if (change == NXT_TIMER_NOPE) { return; } if (timers->nchanges >= timers->mchanges) { nxt_timer_changes_commit(engine); } timers->nchanges++; timer->change = timers->nchanges; } nxt_debug(timer->task, "timer change: %M:%d", time, change); ch = &timers->changes[timer->change - 1]; ch->change = change; ch->time = time; ch->timer = timer; } static void nxt_timer_changes_commit(nxt_event_engine_t *engine) { nxt_timer_t *timer, **add, **add_end; nxt_timers_t *timers; nxt_timer_change_t *ch, *end; timers = &engine->timers; nxt_debug(&engine->task, "timers changes: %ui", timers->nchanges); ch = timers->changes; end = ch + timers->nchanges; add = (nxt_timer_t **) ch; add_end = add; while (ch < end) { timer = ch->timer; switch (ch->change) { case NXT_TIMER_NOPE: break; case NXT_TIMER_ADD: timer->time = ch->time; *add_end++ = timer; if (!nxt_timer_is_in_tree(timer)) { break; } /* Fall through. */ case NXT_TIMER_DELETE: nxt_debug(timer->task, "timer rbtree delete: %M", timer->time); nxt_rbtree_delete(&timers->tree, &timer->node); nxt_timer_in_tree_clear(timer); break; } timer->change = NXT_TIMER_NO_CHANGE; ch++; } while (add < add_end) { timer = *add; nxt_debug(timer->task, "timer rbtree insert: %M", timer->time); nxt_rbtree_insert(&timers->tree, &timer->node); nxt_timer_in_tree_set(timer); add++; } timers->nchanges = 0; } nxt_msec_t nxt_timer_find(nxt_event_engine_t *engine) { int32_t delta; nxt_msec_t time; nxt_timer_t *timer; nxt_timers_t *timers; nxt_rbtree_t *tree; nxt_rbtree_node_t *node, *next; timers = &engine->timers; if (timers->nchanges != 0) { nxt_timer_changes_commit(engine); } tree = &timers->tree; for (node = nxt_rbtree_min(tree); nxt_rbtree_is_there_successor(tree, node); node = next) { next = nxt_rbtree_node_successor(tree, node); timer = (nxt_timer_t *) node; /* * Disabled timers are not deleted here since the minimum active * timer may be larger than a disabled timer, but event poll may * return much earlier and the disabled timer can be reactivated. */ if (timer->enabled) { time = timer->time; timers->minimum = time; nxt_debug(timer->task, "timer found minimum: %M:%M", time, timers->now); delta = nxt_msec_diff(time, timers->now); return (nxt_msec_t) nxt_max(delta, 0); } } /* Set minimum time one day ahead. */ timers->minimum = timers->now + 24 * 60 * 60 * 1000; return NXT_INFINITE_MSEC; } void nxt_timer_expire(nxt_event_engine_t *engine, nxt_msec_t now) { nxt_timer_t *timer; nxt_timers_t *timers; nxt_rbtree_t *tree; nxt_rbtree_node_t *node, *next; timers = &engine->timers; timers->now = now; nxt_debug(&engine->task, "timer expire minimum: %M:%M", timers->minimum, now); /* timers->minimum > now */ if (nxt_msec_diff(timers->minimum , now) > 0) { return; } tree = &timers->tree; for (node = nxt_rbtree_min(tree); nxt_rbtree_is_there_successor(tree, node); node = next) { timer = (nxt_timer_t *) node; /* timer->time > now */ if (nxt_msec_diff(timer->time , now) > 0) { return; } next = nxt_rbtree_node_successor(tree, node); nxt_debug(timer->task, "timer expire delete: %M", timer->time); nxt_rbtree_delete(tree, &timer->node); nxt_timer_in_tree_clear(timer); if (timer->enabled) { timer->queued = 1; nxt_work_queue_add(timer->work_queue, nxt_timer_handler, timer->task, timer, NULL); } } } static void nxt_timer_handler(nxt_task_t *task, void *obj, void *data) { nxt_timer_t *timer; timer = obj; timer->queued = 0; if (timer->enabled && timer->change == NXT_TIMER_NO_CHANGE) { timer->enabled = 0; timer->handler(task, timer, NULL); } }