/*- * 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 #include #include #include #include #ifdef KTRACE #include #include #endif #include #include #include #include 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); /* * Setup per-cpu idle process contexts. The AP's shouldn't be running or * accessing their idle processes at this point, so don't bother with * locking. */ static void idle_setup(void *dummy) { #ifdef SMP struct globaldata *gd; #endif struct proc *p; int error; #ifdef SMP SLIST_FOREACH(gd, &cpuhead, gd_allcpu) { error = kthread_create(idle_proc, NULL, &p, RFSTOPPED | RFHIGHPID, "idle: cpu%d", gd->gd_cpuid); gd->gd_idleproc = p; if (gd->gd_curproc == NULL) gd->gd_curproc = p; #else error = kthread_create(idle_proc, NULL, &p, RFSTOPPED | RFHIGHPID, "idle"); PCPU_SET(idleproc, p); #endif if (error) panic("idle_setup: kthread_create error %d\n", error); p->p_flag |= P_NOLOAD; p->p_stat = SRUN; #ifdef SMP } #endif } /* * idle process context */ static void idle_proc(void *dummy) { #ifdef DIAGNOSTIC int count; #endif for (;;) { mtx_assert(&Giant, MA_NOTOWNED); #ifdef DIAGNOSTIC count = 0; while (count >= 0 && procrunnable() == 0) { #else while (procrunnable() == 0) { #endif /* * This is a good place to put things to be done in * the background, including sanity checks. */ #ifdef DIAGNOSTIC if (count++ < 0) CTR0(KTR_PROC, "idle_proc: timed out waiting" " for a process"); #endif #if 0 if (vm_page_zero_idle() != 0) continue; #endif #ifdef __i386__ cpu_idle(); #endif } mtx_lock_spin(&sched_lock); curproc->p_stats->p_ru.ru_nvcsw++; mi_switch(); mtx_unlock_spin(&sched_lock); } }