|  | #undef exits | 
|  | #undef _exits | 
|  |  | 
|  | extern int __isthreaded; | 
|  |  | 
|  | /* | 
|  | * spin locks | 
|  | */ | 
|  | extern int _tas(int*); | 
|  |  | 
|  | void | 
|  | _threadunlock(Lock *l, ulong pc) | 
|  | { | 
|  | USED(pc); | 
|  |  | 
|  | l->held = 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | _threadlock(Lock *l, int block, ulong pc) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | USED(pc); | 
|  |  | 
|  | /* once fast */ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | if(!block) | 
|  | return 0; | 
|  |  | 
|  | /* a thousand times pretty fast */ | 
|  | for(i=0; i<1000; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | sleep(0); | 
|  | } | 
|  | /* increasingly slow */ | 
|  | for(i=0; i<10; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(1); | 
|  | } | 
|  | for(i=0; i<10; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(10); | 
|  | } | 
|  | for(i=0; i<10; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(100); | 
|  | } | 
|  | for(i=0; i<10; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(1000); | 
|  | } | 
|  | for(i=0; i<10; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(10*1000); | 
|  | } | 
|  | /* now nice and slow */ | 
|  | for(i=0; i<1000; i++){ | 
|  | if(!_tas(&l->held)) | 
|  | return 1; | 
|  | usleep(100*1000); | 
|  | } | 
|  | /* take your time */ | 
|  | while(_tas(&l->held)) | 
|  | usleep(1000*1000); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * For libc. | 
|  | */ | 
|  |  | 
|  | typedef struct { | 
|  | volatile long	access_lock; | 
|  | volatile long	lock_owner; | 
|  | volatile char	*fname; | 
|  | volatile int	lineno; | 
|  | } spinlock_t; | 
|  |  | 
|  | void | 
|  | _spinlock(spinlock_t *lk) | 
|  | { | 
|  | lock((Lock*)&lk->access_lock); | 
|  | } | 
|  |  | 
|  | void | 
|  | _spinunlock(spinlock_t *lk) | 
|  | { | 
|  | unlock((Lock*)&lk->access_lock); | 
|  | } | 
|  |  | 
|  |  | 
|  |  | 
|  | /* | 
|  | * sleep and wakeup | 
|  | */ | 
|  | static void | 
|  | ign(int x) | 
|  | { | 
|  | USED(x); | 
|  | } | 
|  |  | 
|  | static void /*__attribute__((constructor))*/ | 
|  | ignusr1(int restart) | 
|  | { | 
|  | struct sigaction sa; | 
|  |  | 
|  | memset(&sa, 0, sizeof sa); | 
|  | sa.sa_handler = ign; | 
|  | sigemptyset(&sa.sa_mask); | 
|  | sigaddset(&sa.sa_mask, SIGUSR1); | 
|  | if(restart) | 
|  | sa.sa_flags = SA_RESTART; | 
|  | sigaction(SIGUSR1, &sa, nil); | 
|  | } | 
|  |  | 
|  | void | 
|  | _procsleep(_Procrendez *r) | 
|  | { | 
|  | sigset_t mask; | 
|  |  | 
|  | /* | 
|  | * Go to sleep. | 
|  | * | 
|  | * Block USR1, set the handler to interrupt system calls, | 
|  | * unlock the vouslock so our waker can wake us, | 
|  | * and then suspend. | 
|  | */ | 
|  | again: | 
|  | r->asleep = 1; | 
|  | r->pid = getpid(); | 
|  |  | 
|  | sigprocmask(SIG_SETMASK, nil, &mask); | 
|  | sigaddset(&mask, SIGUSR1); | 
|  | sigprocmask(SIG_SETMASK, &mask, nil); | 
|  | ignusr1(0); | 
|  | unlock(r->l); | 
|  | sigdelset(&mask, SIGUSR1); | 
|  | sigsuspend(&mask); | 
|  |  | 
|  | /* | 
|  | * We're awake.  Make USR1 not interrupt system calls. | 
|  | */ | 
|  | lock(r->l); | 
|  | ignusr1(1); | 
|  | if(r->asleep && r->pid == getpid()){ | 
|  | /* Didn't really wake up - signal from something else */ | 
|  | goto again; | 
|  | } | 
|  | } | 
|  |  | 
|  | void | 
|  | _procwakeup(_Procrendez *r) | 
|  | { | 
|  | if(r->asleep){ | 
|  | r->asleep = 0; | 
|  | assert(r->pid >= 1); | 
|  | kill(r->pid, SIGUSR1); | 
|  | } | 
|  | } | 
|  |  | 
|  | void | 
|  | _procwakeupandunlock(_Procrendez *r) | 
|  | { | 
|  | _procwakeup(r); | 
|  | unlock(r->l); | 
|  | } | 
|  |  | 
|  |  | 
|  | /* | 
|  | * process creation and exit | 
|  | */ | 
|  | typedef struct Stackfree Stackfree; | 
|  | struct Stackfree | 
|  | { | 
|  | Stackfree	*next; | 
|  | int	pid; | 
|  | }; | 
|  | static Lock stacklock; | 
|  | static Stackfree *stackfree; | 
|  |  | 
|  | static void | 
|  | delayfreestack(uchar *stk) | 
|  | { | 
|  | Stackfree *sf; | 
|  |  | 
|  | sf = (Stackfree*)stk; | 
|  | sf->pid = getpid(); | 
|  | lock(&stacklock); | 
|  | sf->next = stackfree; | 
|  | stackfree = sf; | 
|  | unlock(&stacklock); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dofreestacks(void) | 
|  | { | 
|  | Stackfree *sf, *last, *next; | 
|  |  | 
|  | if(stackfree==nil || !canlock(&stacklock)) | 
|  | return; | 
|  |  | 
|  | for(last=nil,sf=stackfree; sf; last=sf,sf=next){ | 
|  | next = sf->next; | 
|  | if(sf->pid >= 1 && kill(sf->pid, 0) < 0 && errno == ESRCH){ | 
|  | free(sf); | 
|  | if(last) | 
|  | last->next = next; | 
|  | else | 
|  | stackfree = next; | 
|  | sf = last; | 
|  | } | 
|  | } | 
|  | unlock(&stacklock); | 
|  | } | 
|  |  | 
|  | static int | 
|  | startprocfn(void *v) | 
|  | { | 
|  | void **a; | 
|  | uchar *stk; | 
|  | void (*fn)(void*); | 
|  | Proc *p; | 
|  |  | 
|  | a = (void**)v; | 
|  | fn = a[0]; | 
|  | p = a[1]; | 
|  | stk = a[2]; | 
|  | free(a); | 
|  | p->osprocid = getpid(); | 
|  |  | 
|  | (*fn)(p); | 
|  |  | 
|  | delayfreestack(stk); | 
|  | _exit(0); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | void | 
|  | _procstart(Proc *p, void (*fn)(Proc*)) | 
|  | { | 
|  | void **a; | 
|  | uchar *stk; | 
|  | int pid; | 
|  |  | 
|  | dofreestacks(); | 
|  | a = malloc(3*sizeof a[0]); | 
|  | if(a == nil) | 
|  | sysfatal("_procstart malloc: %r"); | 
|  | stk = malloc(65536); | 
|  | if(stk == nil) | 
|  | sysfatal("_procstart malloc stack: %r"); | 
|  |  | 
|  | a[0] = fn; | 
|  | a[1] = p; | 
|  | a[2] = stk; | 
|  |  | 
|  | pid = rfork_thread(RFPROC|RFMEM|RFNOWAIT, stk+65536-64, startprocfn, a); | 
|  | if(pid < 0){ | 
|  | fprint(2, "_procstart rfork_thread: %r\n"); | 
|  | abort(); | 
|  | } | 
|  | } | 
|  |  | 
|  | static char *threadexitsmsg; | 
|  | void | 
|  | sigusr2handler(int s) | 
|  | { | 
|  | /*	fprint(2, "%d usr2 %d\n", time(0), getpid()); */ | 
|  | if(threadexitsmsg) | 
|  | _exits(threadexitsmsg); | 
|  | } | 
|  |  | 
|  | void | 
|  | threadexitsall(char *msg) | 
|  | { | 
|  | static int pid[1024]; | 
|  | int i, npid, mypid; | 
|  | Proc *p; | 
|  |  | 
|  | if(msg == nil) | 
|  | msg = ""; | 
|  |  | 
|  | /* | 
|  | * Only one guy, ever, gets to run this. | 
|  | * If two guys do it, inevitably they end up | 
|  | * tripping over each other in the underlying | 
|  | * C library exit() implementation, which is | 
|  | * trying to run the atexit handlers and apparently | 
|  | * not thread safe.  This has been observed on | 
|  | * both Linux and OpenBSD.  Sigh. | 
|  | */ | 
|  | { | 
|  | static Lock onelock; | 
|  | if(!canlock(&onelock)) | 
|  | _exits(threadexitsmsg); | 
|  | threadexitsmsg = msg; | 
|  | } | 
|  |  | 
|  | if(msg == nil) | 
|  | msg = ""; | 
|  | mypid = getpid(); | 
|  | lock(&_threadprocslock); | 
|  | threadexitsmsg = msg; | 
|  | npid = 0; | 
|  | for(p=_threadprocs; p; p=p->next) | 
|  | if(p->osprocid != mypid && p->osprocid >= 1) | 
|  | pid[npid++] = p->osprocid; | 
|  | for(i=0; i<npid; i++) | 
|  | kill(pid[i], SIGUSR2); | 
|  | unlock(&_threadprocslock); | 
|  | exits(msg); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * per-process data, indexed by pid | 
|  | */ | 
|  | typedef struct Perproc Perproc; | 
|  | struct Perproc | 
|  | { | 
|  | int		pid; | 
|  | Proc	*proc; | 
|  | }; | 
|  |  | 
|  | static Lock perlock; | 
|  | static Perproc perproc[1024]; | 
|  | #define P ((Proc*)-1) | 
|  |  | 
|  | static Perproc* | 
|  | myperproc(void) | 
|  | { | 
|  | int i, pid, h; | 
|  | Perproc *p; | 
|  |  | 
|  | pid = getpid(); | 
|  | h = pid%nelem(perproc); | 
|  | for(i=0; i<nelem(perproc); i++){ | 
|  | p = &perproc[(i+h)%nelem(perproc)]; | 
|  | if(p->pid == pid) | 
|  | return p; | 
|  | if(p->pid == 0){ | 
|  | print("found 0 at %d (h=%d)\n", (i+h)%nelem(perproc), h); | 
|  | break; | 
|  | } | 
|  | } | 
|  | fprint(2, "myperproc %d: cannot find self\n", pid); | 
|  | abort(); | 
|  | return nil; | 
|  | } | 
|  |  | 
|  | static Perproc* | 
|  | newperproc(void) | 
|  | { | 
|  | int i, pid, h; | 
|  | Perproc *p; | 
|  |  | 
|  | lock(&perlock); | 
|  | pid = getpid(); | 
|  | h = pid%nelem(perproc); | 
|  | for(i=0; i<nelem(perproc); i++){ | 
|  | p = &perproc[(i+h)%nelem(perproc)]; | 
|  | if(p->pid == pid || p->pid == -1 || p->pid == 0){ | 
|  | p->pid = pid; | 
|  | unlock(&perlock); | 
|  | return p; | 
|  | } | 
|  | } | 
|  | fprint(2, "newperproc %d: out of procs\n", pid); | 
|  | abort(); | 
|  | return nil; | 
|  | } | 
|  |  | 
|  | Proc* | 
|  | _threadproc(void) | 
|  | { | 
|  | return myperproc()->proc; | 
|  | } | 
|  |  | 
|  | void | 
|  | _threadsetproc(Proc *p) | 
|  | { | 
|  | Perproc *pp; | 
|  |  | 
|  | if(p) | 
|  | p->osprocid = getpid(); | 
|  | pp = newperproc(); | 
|  | pp->proc = p; | 
|  | if(p == nil) | 
|  | pp->pid = -1; | 
|  | } | 
|  |  | 
|  | void | 
|  | _threadpexit(void) | 
|  | { | 
|  | _exit(0); | 
|  | } | 
|  |  |