about summary refs log tree commit diff stats
path: root/cpp/026new
diff options
context:
space:
mode:
authorKartik K. Agaram <vc@akkartik.com>2015-03-27 10:51:27 -0700
committerKartik K. Agaram <vc@akkartik.com>2015-03-27 10:51:27 -0700
commite2240eb4e89721928fef4e694524e350fb0140cb (patch)
treed5ede45295705c9e50e72e46436dfec6e16017e1 /cpp/026new
parent67bd9b1ba1f84c9a6d73378daf764355a09f3c4f (diff)
downloadmu-e2240eb4e89721928fef4e694524e350fb0140cb.tar.gz
987 - c++: memory allocator
Diffstat (limited to 'cpp/026new')
-rw-r--r--cpp/026new52
1 files changed, 52 insertions, 0 deletions
diff --git a/cpp/026new b/cpp/026new
new file mode 100644
index 00000000..edc17b17
--- /dev/null
+++ b/cpp/026new
@@ -0,0 +1,52 @@
+//: A simple memory allocator to create space for new variables at runtime.
+
+:(scenarios run)
+:(scenario "new")
+# call new two times with identical arguments; you should get back different results
+recipe main [
+  1:address:integer/raw <- new integer:type
+  2:address:integer/raw <- new integer:type
+  3:boolean/raw <- equal 1:address:integer/raw, 2:address:integer/raw
+]
++mem: storing 0 in location 3
+
+:(before "End Globals")
+const size_t Alloc_init = 1000;
+:(before "End Routine Fields")
+size_t alloc;
+:(replace{} "routine::routine(recipe_number r)")
+  routine::routine(recipe_number r) :alloc(Alloc_init) {
+    calls.push(call(r));
+  }
+
+//: first handle 'type' operands
+:(before "End Mu Types Initialization")
+Type_number["type"] = 0;
+:(after "Per-recipe Transforms")
+// replace type names with type_numbers
+if (inst.operation == Recipe_number["new"]) {
+  // first arg must be of type 'type'
+  assert(inst.ingredients.size() >= 1);
+  assert(isa_literal(inst.ingredients[0]));
+  assert(inst.ingredients[0].properties[0].second[0] == "type");
+  inst.ingredients[0].value = Type_number[inst.ingredients[0].name];
+  trace("new") << inst.ingredients[0].name << " -> " << inst.ingredients[0].value;
+}
+
+:(before "End Globals")
+// Operator to look at elements of arrays.
+const int NEW = 24;
+:(before "End Primitive Recipe Numbers")
+Recipe_number["new"] = NEW;
+assert(Next_recipe_number == NEW);
+Next_recipe_number++;
+:(before "End Primitive Recipe Implementations")
+case NEW: {
+  vector<int> result;
+  result.push_back(Current_routine.alloc);
+  write_memory(instructions[pc].products[0], result);
+  vector<int> types;
+  types.push_back(instructions[pc].ingredients[0].value);
+  Current_routine.alloc += size_of(types);
+  break;
+}