| #include "threadimpl.h" |
| |
| Rgrp _threadrgrp; |
| static int isdirty; |
| int _threadhighnrendez; |
| int _threadnrendez; |
| static int nrendez; |
| |
| static ulong |
| finish(Thread *t, ulong val) |
| { |
| ulong ret; |
| |
| ret = t->rendval; |
| t->rendval = val; |
| while(t->state == Running) |
| sleep(0); |
| lock(&t->proc->lock); |
| if(t->state == Rendezvous){ /* not always true: might be Dead */ |
| t->state = Ready; |
| _threadready(t); |
| } |
| unlock(&t->proc->lock); |
| return ret; |
| } |
| |
| ulong |
| _threadrendezvous(ulong tag, ulong val) |
| { |
| ulong ret; |
| Thread *t, **l; |
| |
| lock(&_threadrgrp.lock); |
| _threadnrendez++; |
| l = &_threadrgrp.hash[tag%nelem(_threadrgrp.hash)]; |
| for(t=*l; t; l=&t->rendhash, t=*l){ |
| if(t->rendtag==tag){ |
| _threaddebug(DBGREND, "Rendezvous with thread %d.%d", t->proc->pid, t->id); |
| *l = t->rendhash; |
| ret = finish(t, val); |
| --nrendez; |
| unlock(&_threadrgrp.lock); |
| return ret; |
| } |
| } |
| |
| /* Going to sleep here. */ |
| t = _threadgetproc()->thread; |
| t->rendbreak = 0; |
| t->inrendez = 1; |
| t->rendtag = tag; |
| t->rendval = val; |
| t->rendhash = *l; |
| *l = t; |
| ++nrendez; |
| if(nrendez > _threadhighnrendez) |
| _threadhighnrendez = nrendez; |
| _threaddebug(DBGREND, "Rendezvous for tag %lud (m=%d)", t->rendtag, t->moribund); |
| unlock(&_threadrgrp.lock); |
| t->nextstate = Rendezvous; |
| _sched(); |
| t->inrendez = 0; |
| _threaddebug(DBGREND, "Woke after rendezvous; val is %lud", t->rendval); |
| return t->rendval; |
| } |
| |
| /* |
| * This is called while holding _threadpq.lock and p->lock, |
| * so we can't lock _threadrgrp.lock. Instead our caller has |
| * to call _threadbreakrendez after dropping those locks. |
| */ |
| void |
| _threadflagrendez(Thread *t) |
| { |
| t->rendbreak = 1; |
| isdirty = 1; |
| } |
| |
| void |
| _threadbreakrendez(void) |
| { |
| int i; |
| Thread *t, **l; |
| |
| if(isdirty == 0) |
| return; |
| lock(&_threadrgrp.lock); |
| if(isdirty == 0){ |
| unlock(&_threadrgrp.lock); |
| return; |
| } |
| isdirty = 0; |
| for(i=0; i<nelem(_threadrgrp.hash); i++){ |
| l = &_threadrgrp.hash[i]; |
| for(t=*l; t; t=*l){ |
| if(t->rendbreak){ |
| *l = t->rendhash; |
| finish(t, ~0); |
| }else |
| l=&t->rendhash; |
| } |
| } |
| unlock(&_threadrgrp.lock); |
| } |