forked from miking-lang/miking
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy paththread.mc
97 lines (74 loc) · 2.49 KB
/
thread.mc
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
90
91
92
93
94
95
96
97
include "atomic.mc"
include "seq.mc"
-- Multicore threads.
type Thread a
-- 'threadSpawn f' spawns a new thread, which will execute 'f' in parallel with
-- the current thread.
external externalThreadSpawn ! : all a. (() -> a) -> Thread a
let threadSpawn = lam f.
externalThreadSpawn f
-- 'threadJoin t' blocks until the thread 't' runs to completion. Returns the
-- value returned by running 't'.
-- [NOTE]: should be called exactly once per each spawned thread.
external externalThreadJoin ! : all a. Thread a -> a
let threadJoin = lam t.
externalThreadJoin t
-- 'threadGetID t' returns the ID of the thread 't'
external externalThreadGetID ! : all a. Thread a -> Int
let threadGetID = lam t.
externalThreadGetID t
-- 'threadSelf ()' returns the ID of the current thread
external externalThreadSelf ! : () -> Int
let threadSelf = lam u.
externalThreadSelf u
-- 'threadCPURelax ()' may improve performance during busy waiting.
external externalThreadCPURelax ! : () -> ()
let threadCPURelax = lam u.
externalThreadCPURelax u
mexpr
utest
-- Threads --
let ps = create 10 (lam. threadSpawn (lam. threadSelf ())) in
let tids = map threadJoin ps in
-- We expect the thread IDs to be unique.
utest length (distinct eqi tids) with length tids in
-- Threaded atomic operations --
-- increase/decrease an atomic in different threads
let incr = lam a. atomicFetchAndAdd a 1 in
let decr = lam a. atomicFetchAndAdd a (subi 0 1) in
let a = atomicMake 0 in
recursive let work : (ARef Int -> Int) -> Int -> () = lam op. lam n.
match n with 0 then ()
else
op a;
work op (subi n 1)
in
let nIncr = 10000 in
let nDecr = 1000 in
let nSpawns = 8 in
let threads = create nSpawns (lam. threadSpawn (lam. work incr nIncr)) in
work decr nDecr;
iter (lam t. threadJoin t; ()) threads;
utest atomicGet a with subi (muli nIncr nSpawns) nDecr in
-- Locksteps using CAS --
-- use integer tids to make physical comparison in CAS possible
let me = threadSelf () in
let tid = atomicMake me in
-- Wait for friend to take a step before each step.
recursive let loopf : Int -> Int -> () = lam n. lam friend.
match n with 0 then ()
else
match atomicCAS tid friend (threadSelf ()) with true then
loopf (subi n 1) friend
else
threadCPURelax ();
loopf n friend
in
let n = 100 in
let t = threadSpawn (lam. loopf n me) in
loopf n (threadGetID t);
-- Does not loop forever = the test has passed!
threadJoin t;
()
with () in
()