uint64_t timer_queue_expire( mpqueue_head_t *queue, uint64_t deadline) { timer_call_t call; DBG("timer_queue_expire(%p,)\n", queue); timer_call_lock_spin(queue); while (!queue_empty(&queue->head)) { call = TIMER_CALL(queue_first(&queue->head)); if (call->soft_deadline <= deadline) { timer_call_func_t func; timer_call_param_t param0, param1; if (!simple_lock_try(&call->lock)) { /* case (2b) lock inversion, dequeue and skip */ timer_queue_expire_lock_skips++; (void) remque(qe(call)); call->async_dequeue = TRUE; continue; } timer_call_entry_dequeue(call); func = CE(call)->func; param0 = CE(call)->param0; param1 = CE(call)->param1; simple_unlock(&call->lock); timer_call_unlock(queue); KERNEL_DEBUG_CONSTANT_IST(KDEBUG_TRACE, DECR_TIMER_CALLOUT | DBG_FUNC_START, VM_KERNEL_UNSLIDE(func), param0, param1, 0, 0); #if CONFIG_DTRACE && (DEVELOPMENT || DEBUG ) DTRACE_TMR3(callout__start, timer_call_func_t, func, timer_call_param_t, param0, timer_call_param_t, param1); #endif (*func)(param0, param1); #if CONFIG_DTRACE && (DEVELOPMENT || DEBUG ) DTRACE_TMR3(callout__end, timer_call_func_t, func, timer_call_param_t, param0, timer_call_param_t, param1); #endif KERNEL_DEBUG_CONSTANT_IST(KDEBUG_TRACE, DECR_TIMER_CALLOUT | DBG_FUNC_END, VM_KERNEL_UNSLIDE(func), param0, param1, 0, 0); timer_call_lock_spin(queue); } else break; } if (!queue_empty(&queue->head)) deadline = CE(call)->deadline; else deadline = UINT64_MAX; timer_call_unlock(queue); return (deadline); }
static void timer_call_interrupt(uint64_t timestamp) { timer_call_t call; queue_t queue; simple_lock(&timer_call_lock); queue = &PROCESSOR_DATA(current_processor(), timer_call_queue); call = TC(queue_first(queue)); while (!queue_end(queue, qe(call))) { if (call->deadline <= timestamp) { timer_call_func_t func; timer_call_param_t param0, param1; _delayed_call_dequeue(call); func = call->func; param0 = call->param0; param1 = call->param1; simple_unlock(&timer_call_lock); KERNEL_DEBUG_CONSTANT(MACHDBG_CODE(DBG_MACH_EXCP_DECI, 2) | DBG_FUNC_START, (unsigned int)func, (unsigned int)param0, (unsigned int)param1, 0, 0); #if CONFIG_DTRACE && (DEVELOPMENT || DEBUG ) DTRACE_TMR3(callout__start, timer_call_func_t, func, timer_call_param_t, param0, timer_call_param_t, param1); #endif (*func)(param0, param1); #if CONFIG_DTRACE && (DEVELOPMENT || DEBUG ) DTRACE_TMR3(callout__end, timer_call_func_t, func, timer_call_param_t, param0, timer_call_param_t, param1); #endif KERNEL_DEBUG_CONSTANT(MACHDBG_CODE(DBG_MACH_EXCP_DECI, 2) | DBG_FUNC_END, (unsigned int)func, (unsigned int)param0, (unsigned int)param1, 0, 0); simple_lock(&timer_call_lock); } else break; call = TC(queue_first(queue)); } if (!queue_end(queue, qe(call))) _set_delayed_call_timer(call); simple_unlock(&timer_call_lock); }