From ea7549540f2bce42dd4768aead7533bd38f2dae8 Mon Sep 17 00:00:00 2001 From: John Baldwin Date: Fri, 18 May 2001 00:08:38 +0000 Subject: [PATCH] - Use a timeout for the tsleep in scheduler() instead of having vmmeter() wakeup proc0 by hand to enforce the timeout. - When swapping out a process, keep the process locked via the proc lock from the first checks up until we clear PS_INMEM and set PS_SWAPPING in swapout(). The swapout() function now must be called with the proc lock held and releases it before returning. - Comment out the code to attempt to lock a process' VM structures before swapping out. It is broken in that it releases the lock after obtaining it. If it does grab the lock, it needs to hand it off to swapout() instead of releasing it. This can be revisisted when the VM is locked as this is a valid test to perform. It also causes a lock order reversal for the time being, which is the immediate cause for temporarily disabling it. --- sys/vm/vm_glue.c | 23 +++++++++++++++++++++-- sys/vm/vm_meter.c | 4 +--- 2 files changed, 22 insertions(+), 5 deletions(-) diff --git a/sys/vm/vm_glue.c b/sys/vm/vm_glue.c index f59a4610952..a180ae344bb 100644 --- a/sys/vm/vm_glue.c +++ b/sys/vm/vm_glue.c @@ -93,6 +93,8 @@ #include +extern int maxslp; + /* * System initialization * @@ -395,7 +397,7 @@ loop: * Nothing to do, back to sleep. */ if ((p = pp) == NULL) { - tsleep(&proc0, PVM, "sched", 0); + tsleep(&proc0, PVM, "sched", maxslp * hz / 2); mtx_lock(&Giant); goto loop; } @@ -464,16 +466,17 @@ retry: continue; } vm = p->p_vmspace; - PROC_UNLOCK(p); mtx_lock_spin(&sched_lock); if ((p->p_sflag & (PS_INMEM|PS_SWAPPING)) != PS_INMEM) { mtx_unlock_spin(&sched_lock); + PROC_UNLOCK(p); continue; } switch (p->p_stat) { default: mtx_unlock_spin(&sched_lock); + PROC_UNLOCK(p); continue; case SSLEEP: @@ -483,6 +486,7 @@ retry: */ if (PRI_IS_REALTIME(p->p_pri.pri_class)) { mtx_unlock_spin(&sched_lock); + PROC_UNLOCK(p); continue; } @@ -494,6 +498,7 @@ retry: if (((p->p_pri.pri_level) < PSOCK) || (p->p_slptime < swap_idle_threshold1)) { mtx_unlock_spin(&sched_lock); + PROC_UNLOCK(p); continue; } @@ -506,10 +511,19 @@ retry: (((action & VM_SWAP_IDLE) == 0) || (p->p_slptime < swap_idle_threshold2))) { mtx_unlock_spin(&sched_lock); + PROC_UNLOCK(p); continue; } mtx_unlock_spin(&sched_lock); +#if 0 + /* + * XXX: This is broken. We release the lock we + * acquire before calling swapout, so we could + * still deadlock if another CPU locks this process' + * VM data structures after we release the lock but + * before we call swapout(). + */ ++vm->vm_refcnt; /* * do not swapout a process that is waiting for VM @@ -519,9 +533,11 @@ retry: LK_EXCLUSIVE | LK_NOWAIT, (void *)0, curproc)) { vmspace_free(vm); + PROC_UNLOCK(p); continue; } vm_map_unlock(&vm->vm_map); +#endif /* * If the process has been asleep for awhile and had * most of its pages taken away already, swap it out. @@ -534,6 +550,7 @@ retry: didswap++; goto retry; } + PROC_UNLOCK(p); } } sx_sunlock(&allproc_lock); @@ -550,6 +567,7 @@ swapout(p) register struct proc *p; { + PROC_LOCK_ASSERT(p, MA_OWNED); #if defined(SWAP_DEBUG) printf("swapping out %d\n", p->p_pid); #endif @@ -562,6 +580,7 @@ swapout(p) mtx_lock_spin(&sched_lock); p->p_sflag &= ~PS_INMEM; p->p_sflag |= PS_SWAPPING; + PROC_UNLOCK_NOSWITCH(p); if (p->p_stat == SRUN) remrunqueue(p); mtx_unlock_spin(&sched_lock); diff --git a/sys/vm/vm_meter.c b/sys/vm/vm_meter.c index 5f35e63b6fb..8dcb9067bbb 100644 --- a/sys/vm/vm_meter.c +++ b/sys/vm/vm_meter.c @@ -58,7 +58,7 @@ struct loadavg averunnable = struct vmmeter cnt; -static int maxslp = MAXSLP; +int maxslp = MAXSLP; /* * Constants for averages over 1, 5, and 15 minutes @@ -108,8 +108,6 @@ vmmeter() if (time_second % 5 == 0) loadav(&averunnable); - if (proc0.p_slptime > maxslp / 2) - wakeup(&proc0); } SYSCTL_UINT(_vm, VM_V_FREE_MIN, v_free_min,