blob: 6e3f74b8176544a20be0721deb3f860fedfd8ef1 (
plain) (
blame)
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
|
//: Run a second routine concurrently using fork, without any guarantees on
//: how the operations in each are interleaved with each other.
:(scenario scheduler)
recipe f1 [
start-running f2:recipe
1:integer <- copy 3:literal
]
recipe f2 [
2:integer <- copy 4:literal
]
+schedule: f1
+schedule: f2
//: first, add a deadline to run(routine)
//: these changes are ugly and brittle; just close your nose and get through the next few lines
:(replace "void run_current_routine()")
void run_current_routine(size_t time_slice)
:(replace "while (!Current_routine->completed())" following "void run_current_routine(size_t time_slice)")
size_t ninstrs = 0;
while (Current_routine->state == RUNNING && ninstrs < time_slice)
:(after "Running One Instruction")
ninstrs++;
//: now the rest of the scheduler is clean
:(before "struct routine")
enum routine_state {
RUNNING,
COMPLETED,
// End routine States
};
:(before "End routine Fields")
enum routine_state state;
:(before "End routine Constructor")
state = RUNNING;
:(before "End Globals")
vector<routine*> Routines;
size_t Current_routine_index = 0;
size_t Scheduling_interval = 500;
:(before "End Setup")
Scheduling_interval = 500;
:(replace{} "void run(recipe_number r)")
void run(recipe_number r) {
Routines.push_back(new routine(r));
Current_routine_index = 0, Current_routine = Routines[0];
while (!all_routines_done()) {
skip_to_next_routine();
//? cout << "scheduler: " << Current_routine_index << '\n'; //? 1
assert(Current_routine);
assert(Current_routine->state == RUNNING);
trace("schedule") << current_recipe_name();
run_current_routine(Scheduling_interval);
if (Current_routine->completed())
Current_routine->state = COMPLETED;
// End Scheduler State Transitions
}
//? cout << "done with run\n"; //? 1
}
:(code)
bool all_routines_done() {
for (size_t i = 0; i < Routines.size(); ++i) {
//? cout << "routine " << i << ' ' << Routines[i]->state << '\n'; //? 1
if (Routines[i]->state == RUNNING) {
return false;
}
}
return true;
}
// skip Current_routine_index past non-RUNNING routines
void skip_to_next_routine() {
assert(!Routines.empty());
assert(Current_routine_index < Routines.size());
for (size_t i = (Current_routine_index+1)%Routines.size(); i != Current_routine_index; i = (i+1)%Routines.size()) {
if (Routines[i]->state == RUNNING) {
//? cout << "switching to " << i << '\n'; //? 1
Current_routine_index = i;
Current_routine = Routines[i];
return;
}
}
//? cout << "all done\n"; //? 1
}
:(before "End Teardown")
for (size_t i = 0; i < Routines.size(); ++i)
delete Routines[i];
Routines.clear();
:(before "End Primitive Recipe Declarations")
START_RUNNING,
:(before "End Primitive Recipe Numbers")
Recipe_number["start-running"] = START_RUNNING;
:(before "End Primitive Recipe Implementations")
case START_RUNNING: {
trace("run") << "ingredient 0 is " << current_instruction().ingredients[0].name;
assert(!current_instruction().ingredients[0].initialized);
Routines.push_back(new routine(Recipe_number[current_instruction().ingredients[0].name]));
break;
}
:(scenario scheduler_runs_single_routine)
% Scheduling_interval = 1;
recipe f1 [
1:integer <- copy 0:literal
2:integer <- copy 0:literal
]
+schedule: f1
+run: instruction f1/0
+schedule: f1
+run: instruction f1/1
:(scenario scheduler_interleaves_routines)
% Scheduling_interval = 1;
recipe f1 [
start-running f2:recipe
1:integer <- copy 0:literal
2:integer <- copy 0:literal
]
recipe f2 [
3:integer <- copy 4:literal
4:integer <- copy 4:literal
]
+schedule: f1
+run: instruction f1/0
+schedule: f2
+run: instruction f2/0
+schedule: f1
+run: instruction f1/1
+schedule: f2
+run: instruction f2/1
+schedule: f1
+run: instruction f1/2
:(scenario scheduler_skips_completed_routines)
# this scenario will require some careful setup in escaped C++
# (straining our tangle capabilities to near-breaking point)
% recipe_number f1 = load("recipe f1 [\n1:integer <- copy 0:literal\n]").front();
% recipe_number f2 = load("recipe f2 [\n2:integer <- copy 0:literal\n]").front();
% Routines.push_back(new routine(f1)); // f1 meant to run
% Routines.push_back(new routine(f2));
% Routines.back()->state = COMPLETED; // f2 not meant to run
#? % Trace_stream->dump_layer = "all";
# must have at least one routine without escaping
recipe f3 [
3:integer <- copy 0:literal
]
# by interleaving '+' lines with '-' lines, we allow f1 and f3 to run in any order
+schedule: f1
+mem: storing 0 in location 1
-schedule: f2
-mem: storing 0 in location 2
+schedule: f3
+mem: storing 0 in location 3
:(scenario scheduler_starts_at_middle_of_routines)
% Routines.push_back(new routine(COPY));
% Routines.back()->state = COMPLETED;
recipe f1 [
1:integer <- copy 0:literal
2:integer <- copy 0:literal
]
+schedule: f1
-run: idle
|