blob: 9a8d103e980cd6422bd0ccb1a9d240f3abfe402a (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
#include <u.h>
#include <libc.h>
static long lockinit(long);
/*
* barrier is called from atom.s and tas.s assembly
* to execute memory barrier.
*/
long (*_barrier)(long) = lockinit;
static int
cpus(void)
{
char buf[256], *p;
int f, n;
f = open("#c/sysstat", OREAD|OCEXEC);
if(f < 0)
return -1;
n = read(f, buf, sizeof(buf)-1);
close(f);
if(n <= 0)
return -1;
buf[n] = '\0';
n = 0;
p = buf;
while(*p != '\0'){
if(*p == '\n')
n++;
p++;
}
return n;
}
long _dmb(long);
static long
_nop(long r0)
{
return r0;
}
static long
lockinit(long r0)
{
if(cpus() > 1)
_barrier = _dmb;
else
_barrier = _nop;
return (*_barrier)(r0);
}
void
lock(Lock *lk)
{
int i;
/* once fast */
if(!_tas(&lk->val))
return;
/* a thousand times pretty fast */
for(i=0; i<1000; i++){
if(!_tas(&lk->val))
return;
sleep(0);
}
/* now nice and slow */
for(i=0; i<1000; i++){
if(!_tas(&lk->val))
return;
sleep(100);
}
/* take your time */
while(_tas(&lk->val))
sleep(1000);
}
int
canlock(Lock *lk)
{
return _tas(&lk->val) == 0;
}
void
unlock(Lock *lk)
{
lk->val = (*_barrier)(0);
}
|