/*- * Copyright (c) 2000, All rights reserved. See /usr/src/COPYRIGHT * * $FreeBSD$ */ #include "opt_ktrace.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef KTRACE #include #include #endif #include #include #include #include #include #include #ifdef SMP_DEBUG #include #include #include #endif static void idle_setup(void *dummy); SYSINIT(idle_setup, SI_SUB_SCHED_IDLE, SI_ORDER_FIRST, idle_setup, NULL) static void idle_proc(void *dummy); EVENTHANDLER_FAST_DEFINE(idle_event, idle_eventhandler_t); /* * setup per-cpu idle process contexts */ static void idle_setup(void *dummy) { struct globaldata *gd; int error; SLIST_FOREACH(gd, &cpuhead, gd_allcpu) { #ifdef SMP error = kthread_create(idle_proc, NULL, &gd->gd_idleproc, RFSTOPPED|RFHIGHPID, "idle: cpu%d", gd->gd_cpuid); #else error = kthread_create(idle_proc, NULL, &gd->gd_idleproc, RFSTOPPED|RFHIGHPID, "idle"); #endif if (error) panic("idle_setup: kthread_create error %d\n", error); gd->gd_idleproc->p_flag |= P_NOLOAD; gd->gd_idleproc->p_stat = SRUN; } } /* * idle process context */ static void idle_proc(void *dummy) { int count; for (;;) { mtx_assert(&Giant, MA_NOTOWNED); count = 0; while (count >= 0 && procrunnable() == 0) { /* * This is a good place to put things to be done in * the background, including sanity checks. */ if (count++ < 0) CTR0(KTR_PROC, "idle_proc: timed out waiting" " for a process"); /* call out to any cpu-becoming-idle events */ EVENTHANDLER_FAST_INVOKE(idle_event, count); } mtx_enter(&sched_lock, MTX_SPIN); mi_switch(); mtx_exit(&sched_lock, MTX_SPIN); spl0(); } }