ref: a54fcac01609be9438f6fb05b4da4be01c303383
dir: /sys/src/libthread/sched.c/
#include <u.h> #include <libc.h> #include <thread.h> #include "threadimpl.h" static char *_psstate[] = { "Dead", "Running", "Ready", "Rendezvous", }; static char* psstate(int s) { if(s < 0 || s >= nelem(_psstate)) return "unknown"; return _psstate[s]; } static void unlinkproc(Proc *p) { Proc **l; lock(&_threadpq.lock); for(l=&_threadpq.head; *l; l=&(*l)->next){ if(*l == p){ *l = p->next; if(*l == nil) _threadpq.tail = l; break; } } unlock(&_threadpq.lock); } void _schedinit(void) { Proc *p; Thread *t, **l; p = _threadgetproc(); p->pid = getpid(); while(setjmp(p->sched)) ; _threaddebug(DBGSCHED, "top of schedinit, _threadexitsallstatus=%p", _threadexitsallstatus); if(_threadexitsallstatus) exits(_threadexitsallstatus); lock(&p->lock); if((t=p->thread) != nil){ p->thread = nil; if(t->moribund){ t->state = Dead; for(l=&p->threads.head; *l; l=&(*l)->nextt) if(*l == t){ *l = t->nextt; if(*l==nil) p->threads.tail = l; p->nthreads--; break; } unlock(&p->lock); if(t->inrendez){ _threadflagrendez(t); _threadbreakrendez(); } free(t->stk); free(t->cmdname); free(t); /* XXX how do we know there are no references? */ t = nil; _sched(); } if(p->needexec){ t->ret = _schedexec(&p->exec); p->needexec = 0; } if(p->newproc){ Thread *x = p->newproc->threads.head; if(x->moribund){ x->proc = p; _threadready(x); unlinkproc(p->newproc); free(p->newproc); p->newproc = nil; } else if(_schedfork(p->newproc) != -1) p->newproc = nil; } t->state = t->nextstate; if(t->state == Ready) _threadready(t); } unlock(&p->lock); _sched(); } void needstack(int n) { int x; Proc *p; Thread *t; p = _threadgetproc(); t = p->thread; if((uchar*)&x - n < (uchar*)t->stk){ fprint(2, "%s %lud: &x=%p n=%d t->stk=%p\n", argv0, (ulong)p->pid, &x, n, t->stk); fprint(2, "%s %lud: stack overflow\n", argv0, (ulong)p->pid); abort(); } } static Thread* runthread(Proc *p) { Thread *t; Tqueue *q; if(p->nthreads==0) return nil; q = &p->ready; lock(&p->readylock); if(q->head == nil){ q->asleep = 1; _threaddebug(DBGSCHED, "sleeping for more work"); unlock(&p->readylock); while(rendezvous(q, 0) == (void*)~0){ if(_threadexitsallstatus) exits(_threadexitsallstatus); } /* lock picked up from _threadready */ } t = q->head; q->head = t->next; unlock(&p->readylock); return t; } void _sched(void) { Proc *p; Thread *t; Resched: p = _threadgetproc(); if((t = p->thread) != nil){ needstack(128); _threaddebug(DBGSCHED, "pausing, state=%s", psstate(t->state)); if(setjmp(t->sched)==0) longjmp(p->sched, 1); return; }else{ t = runthread(p); if(t == nil){ _threaddebug(DBGSCHED, "all threads gone; exiting"); unlinkproc(p); _schedexit(p); /* frees proc */ } _threaddebug(DBGSCHED, "running %d.%d", t->proc->pid, t->id); p->thread = t; if(t->moribund){ _threaddebug(DBGSCHED, "%d.%d marked to die", t->proc->pid, t->id); goto Resched; } t->state = Running; t->nextstate = Ready; longjmp(t->sched, 1); } } void _threadready(Thread *t) { Tqueue *q; assert(t->state == Ready); _threaddebug(DBGSCHED, "readying %d.%d", t->proc->pid, t->id); q = &t->proc->ready; lock(&t->proc->readylock); t->next = nil; if(q->head==nil) q->head = t; else *q->tail = t; q->tail = &t->next; if(q->asleep){ q->asleep = 0; /* lock passes to runthread */ _threaddebug(DBGSCHED, "waking process %d", t->proc->pid); while(rendezvous(q, 0) == (void*)~0){ if(_threadexitsallstatus) exits(_threadexitsallstatus); } }else unlock(&t->proc->readylock); } void yield(void) { _sched(); }