#include #include #include "semlock.h" enum { Nsleep = 8, Nloop = 1000, }; extern void pause(void); void lock(Lock *l) { int i, j; for(i = 0; i < Nsleep; i++){ for(j = 0; j < Nloop; j++){ if(l->key == 0) goto acq; // pause(); } sleep(0); } acq: if(ainc(&l->key) == 1) return; /* changed from 0 -> 1: we hold lock */ /* otherwise wait in kernel */ while(semacquire(&l->sem, 1) < 0){ /* interrupted; try again */ } } void unlock(Lock *l) { if(adec(&l->key) == 0) return; /* changed from 1 -> 0: no contention */ semrelease(&l->sem, 1); } int canlock(Lock *l) { if(ainc(&l->key) == 1) return 1; /* changed from 0 -> 1: success */ /* Undo increment (but don't miss wakeup) */ if(adec(&l->key) == 0) return 0; /* changed from 1 -> 0: no contention */ semrelease(&l->sem, 1); return 0; }