mu/073scheduler.cc

623 lines
18 KiB
C++
Raw Normal View History

2015-05-06 16:15:08 +00:00
//: Run a second routine concurrently using 'start-running', without any
//: guarantees on how the operations in each are interleaved with each other.
2015-04-25 07:00:49 +00:00
:(scenario scheduler)
def f1 [
start-running f2
2015-05-10 12:42:56 +00:00
# wait for f2 to run
{
2016-09-17 17:28:25 +00:00
jump-unless 1:num, -1
2015-05-10 12:42:56 +00:00
}
2015-04-25 04:39:09 +00:00
]
def f2 [
2016-09-17 17:28:25 +00:00
1:num <- copy 1
2015-04-25 04:39:09 +00:00
]
+schedule: f1
+schedule: f2
2015-04-25 04:39:09 +00:00
//: first, add a deadline to run(routine)
:(before "End Globals")
int Scheduling_interval = 500;
:(before "End routine Fields")
int instructions_run_this_scheduling_slice;
:(before "End routine Constructor")
instructions_run_this_scheduling_slice = 0;
2017-03-20 23:53:36 +00:00
:(after "Running One Instruction")
++Current_routine->instructions_run_this_scheduling_slice;
:(replace{} "bool should_continue_running(const routine* current_routine)")
bool should_continue_running(const routine* current_routine) {
assert(current_routine == Current_routine); // argument passed in just to make caller readable above
return Current_routine->state == RUNNING
&& Current_routine->instructions_run_this_scheduling_slice < Scheduling_interval;
}
:(after "stop_running_current_routine:")
// Reset instructions_run_this_scheduling_slice
Current_routine->instructions_run_this_scheduling_slice = 0;
2015-04-25 04:39:09 +00:00
//: 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;
2015-04-25 04:39:09 +00:00
:(before "End Globals")
vector<routine*> Routines;
int Current_routine_index = 0;
2017-07-09 21:34:17 +00:00
:(before "End Reset")
2015-04-25 07:00:49 +00:00
Scheduling_interval = 500;
for (int i = 0; i < SIZE(Routines); ++i)
delete Routines.at(i);
Routines.clear();
Current_routine = NULL;
2016-10-22 23:10:23 +00:00
:(replace{} "void run(const recipe_ordinal r)")
void run(const recipe_ordinal r) {
run(new routine(r));
}
:(code)
void run(routine* rr) {
Routines.push_back(rr);
Current_routine_index = 0, Current_routine = Routines.at(0);
while (!all_routines_done()) {
skip_to_next_routine();
assert(Current_routine);
assert(Current_routine->state == RUNNING);
2015-10-29 18:56:10 +00:00
trace(9990, "schedule") << current_routine_label() << end();
run_current_routine();
// Scheduler State Transitions
2015-04-26 05:30:20 +00:00
if (Current_routine->completed())
Current_routine->state = COMPLETED;
// End Scheduler State Transitions
// Scheduler Cleanup
// End Scheduler Cleanup
}
// End Run Routine
}
bool all_routines_done() {
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->state == RUNNING)
return false;
}
return true;
}
// skip Current_routine_index past non-RUNNING routines
void skip_to_next_routine() {
assert(!Routines.empty());
2015-05-17 09:22:41 +00:00
assert(Current_routine_index < SIZE(Routines));
for (int i = (Current_routine_index+1)%SIZE(Routines); i != Current_routine_index; i = (i+1)%SIZE(Routines)) {
if (Routines.at(i)->state == RUNNING) {
Current_routine_index = i;
Current_routine = Routines.at(i);
return;
}
2015-04-25 04:39:09 +00:00
}
}
string current_routine_label() {
2016-08-28 01:05:11 +00:00
return routine_label(Current_routine);
}
string routine_label(routine* r) {
ostringstream result;
2016-08-28 01:05:11 +00:00
const call_stack& calls = r->calls;
2016-10-20 05:10:35 +00:00
for (call_stack::const_iterator p = calls.begin(); p != calls.end(); ++p) {
if (p != calls.begin()) result << '/';
result << get(Recipe, p->running_recipe).name;
}
return result.str();
}
//: special case for the very first routine
:(replace{} "void run_main(int argc, char* argv[])")
void run_main(int argc, char* argv[]) {
2015-11-28 01:49:30 +00:00
recipe_ordinal r = get(Recipe_ordinal, "main");
assert(r);
routine* main_routine = new routine(r);
// pass in commandline args as ingredients to main
// todo: test this
Current_routine = main_routine;
2016-10-20 05:10:35 +00:00
for (int i = 1; i < argc; ++i) {
vector<double> arg;
2016-09-17 06:52:15 +00:00
arg.push_back(new_mu_text(argv[i]));
2016-11-06 07:48:31 +00:00
assert(get(Memory, arg.back()) == 0);
current_call().ingredient_atoms.push_back(arg);
}
run(main_routine);
}
2015-05-06 16:15:08 +00:00
//:: To schedule new routines to run, call 'start-running'.
2015-05-06 16:15:08 +00:00
//: 'start-running' will return a unique id for the routine that was created.
//: routine id is a number, but don't do any arithmetic on it
:(before "End routine Fields")
int id;
:(before "End Globals")
int Next_routine_id = 1;
2017-07-09 21:34:17 +00:00
:(before "End Reset")
Next_routine_id = 1;
:(before "End routine Constructor")
id = Next_routine_id;
++Next_routine_id;
//: routines save the routine that spawned them
:(before "End routine Fields")
// todo: really should be routine_id, but that's less efficient.
int parent_index; // only < 0 if there's no parent_index
:(before "End routine Constructor")
parent_index = -1;
2015-04-25 04:39:09 +00:00
:(before "End Primitive Recipe Declarations")
2015-05-01 22:25:47 +00:00
START_RUNNING,
2015-04-25 04:39:09 +00:00
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "start-running", START_RUNNING);
2015-10-02 00:30:14 +00:00
:(before "End Primitive Recipe Checks")
2015-05-01 22:25:47 +00:00
case START_RUNNING: {
2015-10-02 00:30:14 +00:00
if (inst.ingredients.empty()) {
2016-02-26 21:04:55 +00:00
raise << maybe(get(Recipe, r).name) << "'start-running' requires at least one ingredient: the recipe to start running\n" << end();
break;
}
2015-10-07 07:22:49 +00:00
if (!is_mu_recipe(inst.ingredients.at(0))) {
raise << maybe(get(Recipe, r).name) << "first ingredient of 'start-running' should be a recipe, but got '" << to_string(inst.ingredients.at(0)) << "'\n" << end();
break;
}
2015-10-02 00:30:14 +00:00
break;
}
:(before "End Primitive Recipe Implementations")
case START_RUNNING: {
routine* new_routine = new routine(ingredients.at(0).at(0));
new_routine->parent_index = Current_routine_index;
// populate ingredients
2017-09-10 17:57:19 +00:00
for (int i = /*skip callee*/1; i < SIZE(current_instruction().ingredients); ++i) {
new_routine->calls.front().ingredient_atoms.push_back(ingredients.at(i));
2016-05-06 07:46:39 +00:00
reagent/*copy*/ ingredient = current_instruction().ingredients.at(i);
new_routine->calls.front().ingredients.push_back(ingredient);
// End Populate start-running Ingredient
}
Routines.push_back(new_routine);
products.resize(1);
products.at(0).push_back(new_routine->id);
2015-04-25 04:39:09 +00:00
break;
}
2015-04-25 07:00:49 +00:00
:(scenario scheduler_runs_single_routine)
% Scheduling_interval = 1;
def f1 [
2016-09-17 17:28:25 +00:00
1:num <- copy 0
2:num <- copy 0
]
+schedule: f1
+run: {1: "number"} <- copy {0: "literal"}
+schedule: f1
+run: {2: "number"} <- copy {0: "literal"}
2015-04-25 07:00:49 +00:00
:(scenario scheduler_interleaves_routines)
% Scheduling_interval = 1;
def f1 [
start-running f2
2016-09-17 17:28:25 +00:00
1:num <- copy 0
2:num <- copy 0
2015-04-25 07:00:49 +00:00
]
def f2 [
2016-09-17 17:28:25 +00:00
3:num <- copy 0
4:num <- copy 0
2015-04-25 07:00:49 +00:00
]
+schedule: f1
+run: start-running {f2: "recipe-literal"}
2015-04-25 07:00:49 +00:00
+schedule: f2
+run: {3: "number"} <- copy {0: "literal"}
2015-04-25 07:00:49 +00:00
+schedule: f1
+run: {1: "number"} <- copy {0: "literal"}
2015-04-25 07:00:49 +00:00
+schedule: f2
+run: {4: "number"} <- copy {0: "literal"}
2015-04-25 07:00:49 +00:00
+schedule: f1
+run: {2: "number"} <- copy {0: "literal"}
2015-10-28 20:05:01 +00:00
:(scenario start_running_takes_ingredients)
def f1 [
start-running f2, 3
2015-05-10 12:42:56 +00:00
# wait for f2 to run
{
2016-09-17 17:28:25 +00:00
jump-unless 1:num, -1
2015-05-10 12:42:56 +00:00
}
]
def f2 [
2016-09-17 17:28:25 +00:00
1:num <- next-ingredient
2:num <- add 1:num, 1
]
+mem: storing 4 in location 2
//: type-checking for 'start-running'
:(scenario start_running_checks_types)
% Hide_errors = true;
def f1 [
start-running f2, 3
]
def f2 n:&:num [
]
+error: f1: ingredient 0 has the wrong type at 'start-running f2, 3'
// 'start-running' only uses the ingredients of the callee, not its products
:(before "End is_indirect_call_with_ingredients Special-cases")
if (r == START_RUNNING) return true;
//: back to testing 'start-running'
:(scenario start_running_returns_routine_id)
def f1 [
2016-09-17 17:28:25 +00:00
1:num <- start-running f2
]
def f2 [
2016-09-17 17:28:25 +00:00
12:num <- copy 44
]
+mem: storing 2 in location 1
//: this scenario will require some careful setup in escaped C++
//: (straining our tangle capabilities to near-breaking point)
:(scenario scheduler_skips_completed_routines)
2016-09-17 17:28:25 +00:00
% recipe_ordinal f1 = load("recipe f1 [\n1:num <- copy 0\n]\n").front();
% recipe_ordinal f2 = load("recipe f2 [\n2:num <- copy 0\n]\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
# must have at least one routine without escaping
def f3 [
2016-09-17 17:28:25 +00:00
3:num <- copy 0
]
# 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;
def f1 [
2016-09-17 17:28:25 +00:00
1:num <- copy 0
2:num <- copy 0
]
+schedule: f1
-run: idle
//:: Errors in a routine cause it to terminate.
:(scenario scheduler_terminates_routines_after_errors)
% Hide_errors = true;
% Scheduling_interval = 2;
def f1 [
start-running f2
2016-09-17 17:28:25 +00:00
1:num <- copy 0
2:num <- copy 0
]
def f2 [
# divide by 0 twice
2016-09-17 17:28:25 +00:00
3:num <- divide-with-remainder 4, 0
4:num <- divide-with-remainder 4, 0
]
# f2 should stop after first divide by 0
2016-09-17 17:28:25 +00:00
+error: f2: divide by zero in '3:num <- divide-with-remainder 4, 0'
-error: f2: divide by zero in '4:num <- divide-with-remainder 4, 0'
:(after "operator<<(ostream& os, end /*unused*/)")
if (Trace_stream && Trace_stream->curr_label == "error" && Current_routine) {
Current_routine->state = COMPLETED;
}
//:: Routines are marked completed when their parent completes.
2015-05-10 12:42:56 +00:00
:(scenario scheduler_kills_orphans)
def main [
start-running f1
2015-05-10 12:42:56 +00:00
# f1 never actually runs because its parent completes without waiting for it
]
def f1 [
2016-09-17 17:28:25 +00:00
1:num <- copy 0
2015-05-10 12:42:56 +00:00
]
-schedule: f1
:(before "End Scheduler Cleanup")
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->state == COMPLETED) continue;
if (Routines.at(i)->parent_index < 0) continue; // root thread
2016-08-16 22:26:52 +00:00
// structured concurrency: http://250bpm.com/blog:71
2015-05-10 12:42:56 +00:00
if (has_completed_parent(i)) {
Routines.at(i)->state = COMPLETED;
}
}
:(code)
bool has_completed_parent(int routine_index) {
2016-10-20 05:10:35 +00:00
for (int j = routine_index; j >= 0; j = Routines.at(j)->parent_index) {
if (Routines.at(j)->state == COMPLETED)
return true;
}
return false;
}
//:: 'routine-state' can tell if a given routine id is running
:(scenario routine_state_test)
% Scheduling_interval = 2;
def f1 [
2016-09-17 17:28:25 +00:00
1:num/child-id <- start-running f2
12:num <- copy 0 # race condition since we don't care about location 12
# thanks to Scheduling_interval, f2's one instruction runs in between here and completes
2016-09-17 17:28:25 +00:00
2:num/state <- routine-state 1:num/child-id
]
def f2 [
2016-09-17 17:28:25 +00:00
12:num <- copy 0
# trying to run a second instruction marks routine as completed
]
# recipe f2 should be in state COMPLETED
+mem: storing 1 in location 2
:(before "End Primitive Recipe Declarations")
ROUTINE_STATE,
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "routine-state", ROUTINE_STATE);
2015-10-02 00:30:14 +00:00
:(before "End Primitive Recipe Checks")
case ROUTINE_STATE: {
2015-10-02 00:30:14 +00:00
if (SIZE(inst.ingredients) != 1) {
2017-05-26 23:43:18 +00:00
raise << maybe(get(Recipe, r).name) << "'routine-state' requires exactly one ingredient, but got '" << to_original_string(inst) << "'\n" << end();
break;
}
2015-10-07 07:22:49 +00:00
if (!is_mu_number(inst.ingredients.at(0))) {
raise << maybe(get(Recipe, r).name) << "first ingredient of 'routine-state' should be a routine id generated by 'start-running', but got '" << inst.ingredients.at(0).original_string << "'\n" << end();
break;
}
2015-10-02 00:30:14 +00:00
break;
}
:(before "End Primitive Recipe Implementations")
case ROUTINE_STATE: {
int id = ingredients.at(0).at(0);
int result = -1;
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->id == id) {
result = Routines.at(i)->state;
break;
}
}
products.resize(1);
products.at(0).push_back(result);
break;
}
//:: miscellaneous helpers
:(before "End Primitive Recipe Declarations")
STOP,
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "stop", STOP);
2015-10-02 00:30:14 +00:00
:(before "End Primitive Recipe Checks")
case STOP: {
2015-10-02 00:30:14 +00:00
if (SIZE(inst.ingredients) != 1) {
2017-05-26 23:43:18 +00:00
raise << maybe(get(Recipe, r).name) << "'stop' requires exactly one ingredient, but got '" << to_original_string(inst) << "'\n" << end();
break;
}
2015-10-07 07:22:49 +00:00
if (!is_mu_number(inst.ingredients.at(0))) {
raise << maybe(get(Recipe, r).name) << "first ingredient of 'stop' should be a routine id generated by 'start-running', but got '" << inst.ingredients.at(0).original_string << "'\n" << end();
break;
}
2015-10-02 00:30:14 +00:00
break;
}
:(before "End Primitive Recipe Implementations")
case STOP: {
int id = ingredients.at(0).at(0);
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->id == id) {
Routines.at(i)->state = COMPLETED;
break;
}
}
break;
}
2015-05-08 15:21:36 +00:00
:(before "End Primitive Recipe Declarations")
_DUMP_ROUTINES,
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "$dump-routines", _DUMP_ROUTINES);
2015-10-02 00:30:14 +00:00
:(before "End Primitive Recipe Checks")
case _DUMP_ROUTINES: {
break;
}
2015-05-08 15:21:36 +00:00
:(before "End Primitive Recipe Implementations")
case _DUMP_ROUTINES: {
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
2015-05-10 12:42:56 +00:00
cerr << i << ": " << Routines.at(i)->id << ' ' << Routines.at(i)->state << ' ' << Routines.at(i)->parent_index << '\n';
2015-05-08 15:21:36 +00:00
}
break;
}
2015-08-17 03:12:14 +00:00
//: support for stopping routines after some number of cycles
:(scenario routine_discontinues_past_limit)
% Scheduling_interval = 2;
def f1 [
2016-09-17 17:28:25 +00:00
1:num/child-id <- start-running f2
limit-time 1:num/child-id, 10
# padding loop just to make sure f2 has time to completed
2016-09-17 17:28:25 +00:00
2:num <- copy 20
2:num <- subtract 2:num, 1
jump-if 2:num, -2:offset
2015-08-17 03:12:14 +00:00
]
def f2 [
jump -1:offset # run forever
$print [should never get here], 10/newline
2015-08-17 03:12:14 +00:00
]
# f2 terminates
+schedule: discontinuing routine 2
:(before "End routine States")
DISCONTINUED,
:(before "End Scheduler State Transitions")
if (Current_routine->limit >= 0) {
if (Current_routine->limit <= Scheduling_interval) {
trace("schedule") << "discontinuing routine " << Current_routine->id << end();
2015-08-17 03:12:14 +00:00
Current_routine->state = DISCONTINUED;
Current_routine->limit = 0;
}
else {
Current_routine->limit -= Scheduling_interval;
}
}
:(before "End Test Teardown")
if (Passed && any_routines_with_error())
raise << "some routines died with errors\n" << end();
:(before "End Mu Test Teardown")
if (Passed && any_routines_with_error())
raise << Current_scenario->name << ": some routines died with errors\n" << end();
:(code)
bool any_routines_with_error() {
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->state == DISCONTINUED)
return true;
}
return false;
}
2015-08-17 03:12:14 +00:00
:(before "End routine Fields")
int limit;
2015-08-17 03:12:14 +00:00
:(before "End routine Constructor")
limit = -1; /* no limit */
:(before "End Primitive Recipe Declarations")
LIMIT_TIME,
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "limit-time", LIMIT_TIME);
2015-10-02 00:30:14 +00:00
:(before "End Primitive Recipe Checks")
2015-08-17 03:12:14 +00:00
case LIMIT_TIME: {
2015-10-02 00:30:14 +00:00
if (SIZE(inst.ingredients) != 2) {
2017-05-26 23:43:18 +00:00
raise << maybe(get(Recipe, r).name) << "'limit-time' requires exactly two ingredient, but got '" << to_original_string(inst) << "'\n" << end();
2015-08-17 03:12:14 +00:00
break;
}
2015-10-07 07:22:49 +00:00
if (!is_mu_number(inst.ingredients.at(0))) {
raise << maybe(get(Recipe, r).name) << "first ingredient of 'limit-time' should be a routine id generated by 'start-running', but got '" << inst.ingredients.at(0).original_string << "'\n" << end();
2015-08-17 03:12:14 +00:00
break;
}
2015-10-07 07:22:49 +00:00
if (!is_mu_number(inst.ingredients.at(1))) {
raise << maybe(get(Recipe, r).name) << "second ingredient of 'limit-time' should be a number (of instructions to run for), but got '" << inst.ingredients.at(1).original_string << "'\n" << end();
2015-08-17 03:12:14 +00:00
break;
}
2015-10-02 00:30:14 +00:00
break;
}
:(before "End Primitive Recipe Implementations")
case LIMIT_TIME: {
int id = ingredients.at(0).at(0);
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
2015-08-17 03:12:14 +00:00
if (Routines.at(i)->id == id) {
Routines.at(i)->limit = ingredients.at(1).at(0);
break;
}
}
break;
}
:(before "End routine Fields")
int instructions_run;
:(before "End routine Constructor")
instructions_run = 0;
:(before "Reset instructions_run_this_scheduling_slice")
Current_routine->instructions_run += Current_routine->instructions_run_this_scheduling_slice;
:(before "End Primitive Recipe Declarations")
NUMBER_OF_INSTRUCTIONS,
:(before "End Primitive Recipe Numbers")
put(Recipe_ordinal, "number-of-instructions", NUMBER_OF_INSTRUCTIONS);
:(before "End Primitive Recipe Checks")
case NUMBER_OF_INSTRUCTIONS: {
if (SIZE(inst.ingredients) != 1) {
2017-05-26 23:43:18 +00:00
raise << maybe(get(Recipe, r).name) << "'number-of-instructions' requires exactly one ingredient, but got '" << to_original_string(inst) << "'\n" << end();
break;
}
if (!is_mu_number(inst.ingredients.at(0))) {
raise << maybe(get(Recipe, r).name) << "first ingredient of 'number-of-instructions' should be a routine id generated by 'start-running', but got '" << inst.ingredients.at(0).original_string << "'\n" << end();
break;
}
break;
}
:(before "End Primitive Recipe Implementations")
case NUMBER_OF_INSTRUCTIONS: {
int id = ingredients.at(0).at(0);
int result = -1;
2016-10-20 05:10:35 +00:00
for (int i = 0; i < SIZE(Routines); ++i) {
if (Routines.at(i)->id == id) {
result = Routines.at(i)->instructions_run;
break;
}
}
products.resize(1);
products.at(0).push_back(result);
break;
}
2016-07-25 06:25:59 +00:00
:(scenario number_of_instructions)
def f1 [
2016-09-17 17:28:25 +00:00
10:num/child-id <- start-running f2
{
2016-09-17 17:28:25 +00:00
loop-unless 20:num
}
2016-09-17 17:28:25 +00:00
11:num <- number-of-instructions 10:num
]
def f2 [
2016-07-25 06:25:59 +00:00
# 2 instructions worth of work
2016-09-17 17:28:25 +00:00
1:num <- copy 34
20:num <- copy 1
]
2016-07-25 06:25:59 +00:00
# f2 runs an extra instruction for the implicit return added by the
# fill_in_return_ingredients transform
2016-07-25 06:25:59 +00:00
+mem: storing 3 in location 11
2016-07-25 06:25:59 +00:00
:(scenario number_of_instructions_across_multiple_scheduling_intervals)
% Scheduling_interval = 1;
def f1 [
2016-09-17 17:28:25 +00:00
10:num/child-id <- start-running f2
{
2016-09-17 17:28:25 +00:00
loop-unless 20:num
}
2016-09-17 17:28:25 +00:00
11:num <- number-of-instructions 10:num
]
def f2 [
# 4 instructions worth of work
2016-09-17 17:28:25 +00:00
1:num <- copy 34
2:num <- copy 1
2:num <- copy 3
20:num <- copy 1
]
# f2 runs an extra instruction for the implicit return added by the
# fill_in_return_ingredients transform
+mem: storing 5 in location 11
2016-07-25 06:25:59 +00:00
//:: make sure that each routine gets a different alloc to start
:(scenario new_concurrent)
def f1 [
start-running f2
2016-09-17 19:55:10 +00:00
1:&:num/raw <- new number:type
2016-07-25 06:25:59 +00:00
# wait for f2 to complete
{
2016-09-17 17:28:25 +00:00
loop-unless 4:num/raw
2016-07-25 06:25:59 +00:00
}
]
def f2 [
2016-09-17 19:55:10 +00:00
2:&:num/raw <- new number:type
2016-07-25 06:25:59 +00:00
# hack: assumes scheduler implementation
2016-09-17 19:55:10 +00:00
3:bool/raw <- equal 1:&:num/raw, 2:&:num/raw
2016-07-25 06:25:59 +00:00
# signal f2 complete
2016-09-17 17:28:25 +00:00
4:num/raw <- copy 1
2016-07-25 06:25:59 +00:00
]
+mem: storing 0 in location 3