1 | #include <fstream>
|
---|
2 | #include <kernel>
|
---|
3 | #include <monitor>
|
---|
4 | #include <stdlib>
|
---|
5 | #include <thread>
|
---|
6 |
|
---|
7 | static const unsigned long N = 50_000ul;
|
---|
8 |
|
---|
9 | #ifndef PREEMPTION_RATE
|
---|
10 | #define PREEMPTION_RATE 10_000ul
|
---|
11 | #endif
|
---|
12 |
|
---|
13 | unsigned int default_preemption() {
|
---|
14 | return 0;
|
---|
15 | }
|
---|
16 | enum state_t { WAIT, SIGNAL, BARGE };
|
---|
17 |
|
---|
18 | monitor global_t {};
|
---|
19 |
|
---|
20 | monitor global_data_t {
|
---|
21 | volatile bool done;
|
---|
22 | int counter;
|
---|
23 | state_t state;
|
---|
24 |
|
---|
25 | unsigned short do_signal;
|
---|
26 | unsigned short do_wait2;
|
---|
27 | unsigned short do_wait1;
|
---|
28 | };
|
---|
29 |
|
---|
30 | void ?{} ( global_data_t * this ) {
|
---|
31 | this->done = false;
|
---|
32 | this->counter = 0;
|
---|
33 | this->state = BARGE;
|
---|
34 |
|
---|
35 | this->do_signal = 6;
|
---|
36 | this->do_wait1 = 1;
|
---|
37 | this->do_wait2 = 3;
|
---|
38 | }
|
---|
39 |
|
---|
40 | void ^?{} ( global_data_t * this ) {}
|
---|
41 |
|
---|
42 | global_t globalA;
|
---|
43 | global_t globalB;
|
---|
44 | global_data_t globalC;
|
---|
45 |
|
---|
46 | condition cond;
|
---|
47 |
|
---|
48 | thread Threads {};
|
---|
49 |
|
---|
50 | bool logicC( global_t * mutex a, global_t * mutex b, global_data_t * mutex c ) {
|
---|
51 | c->counter++;
|
---|
52 |
|
---|
53 | if( (c->counter % 1000) == 0 ) sout | c->counter | endl;
|
---|
54 |
|
---|
55 | int action = c->counter % 10;
|
---|
56 |
|
---|
57 | if( action == 0 ) {
|
---|
58 | c->do_signal = max( ((unsigned)rand48()) % 10, 1);
|
---|
59 | c->do_wait1 = ((unsigned)rand48()) % (c->do_signal);
|
---|
60 | c->do_wait2 = ((unsigned)rand48()) % (c->do_signal);
|
---|
61 |
|
---|
62 | if(c->do_wait1 == c->do_wait2) sout | "Same" | endl;
|
---|
63 | }
|
---|
64 |
|
---|
65 | if( action == c->do_wait1 || action == c->do_wait2 ) {
|
---|
66 | c->state = WAIT;
|
---|
67 | wait( &cond );
|
---|
68 |
|
---|
69 | if(c->state != SIGNAL) {
|
---|
70 | sout | "ERROR Barging detected" | c->counter | endl;
|
---|
71 | abort();
|
---|
72 | }
|
---|
73 | }
|
---|
74 | else if( action == c->do_signal ) {
|
---|
75 | c->state = SIGNAL;
|
---|
76 |
|
---|
77 | signal( &cond );
|
---|
78 | signal( &cond );
|
---|
79 | }
|
---|
80 | else {
|
---|
81 | c->state = BARGE;
|
---|
82 | }
|
---|
83 |
|
---|
84 | if( c->counter >= N ) c->done = true;
|
---|
85 | return !c->done;
|
---|
86 | }
|
---|
87 |
|
---|
88 | bool logicB( global_t * mutex a, global_t * mutex b ) {
|
---|
89 | return logicC(a, b, &globalC);
|
---|
90 | }
|
---|
91 |
|
---|
92 | bool logicA( global_t * mutex a ) {
|
---|
93 | return logicB(a, &globalB);
|
---|
94 | }
|
---|
95 |
|
---|
96 | void main( Threads* this ) {
|
---|
97 | while( logicA(&globalA) ) { yield(); };
|
---|
98 | }
|
---|
99 |
|
---|
100 | static thread_desc * volatile the_threads;
|
---|
101 |
|
---|
102 | int main(int argc, char* argv[]) {
|
---|
103 | rand48seed(0);
|
---|
104 | processor p;
|
---|
105 | {
|
---|
106 | Threads t[17];
|
---|
107 | the_threads = (thread_desc*)t;
|
---|
108 | }
|
---|
109 | }
|
---|