https://github.com/akkartik/mu1/blob/master/034address.cc
  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 :(before "End Globals")
 43 long long Next_alloc_id = 0;
 44 :(before "End Reset")
 45 Next_alloc_id = 0;
 46 
 47 
 48 
 49 
 50 
 51 
 52 
 53 
 54 
 55 :(scenario new)
 56 
 57 
 58 def main [
 59   10:&:num <- new num:type
 60   12:&:num <- new num:type
 61   20:bool <- equal 10:&:num, 12:&:num
 62 ]
 63 +mem: storing 1000 in location 11
 64 +mem: storing 0 in location 20
 65 
 66 :(scenario new_array)
 67 
 68 def main [
 69   10:&:@:num <- new num:type, 5
 70   12:&:num <- new num:type
 71   20:num/alloc2, 21:num/alloc1 <- deaddress 10:&:@:num, 12:&:num
 72   30:num <- subtract 21:num/alloc2, 20:num/alloc1
 73 ]
 74 +run: {10: ("address" "array" "number")} <- new {num: "type"}, {5: "literal"}
 75 +mem: array length is 5
 76 
 77 +mem: storing 1000 in location 11
 78 
 79 +mem: storing 7 in location 30
 80 
 81 :(scenario dilated_reagent_with_new)
 82 def main [
 83   10:&:&:num <- new {(& num): type}
 84 ]
 85 +new: size of '(& num)' is 2
 86 
 87 
 88 :(before "End Mu Types Initialization")
 89 put(Type_ordinal, "type", 0);
 90 :(code)
 91 bool is_mu_type_literal(const reagent& r) {
 92   return is_literal(r) && r.type && r.type->name == "type";
 93 }
 94 
 95 :(before "End Primitive Recipe Declarations")
 96 NEW,
 97 :(before "End Primitive Recipe Numbers")
 98 put(Recipe_ordinal, "new", NEW);
 99 :(before "End Primitive Recipe Checks")
100 case NEW: {
101   const recipe& caller = get(Recipe, r);
102   if (inst.ingredients.empty() || SIZE(inst.ingredients) > 2) {
103     raise << maybe(caller.name) << "'new' requires one or two ingredients, but got '" << to_original_string(inst) << "'\n" << end();
104     break;
105   }
106   
107   const reagent& type = inst.ingredients.at(0);
108   if (!is_mu_type_literal(type)) {
109     raise << maybe(caller.name) << "first ingredient of 'new' should be a type, but got '" << type.original_string << "'\n" << end();
110     break;
111   }
112   if (SIZE(inst.ingredients) > 1 && !is_mu_number(inst.ingredients.at(1))) {
113     raise << maybe(caller.name) << "second ingredient of 'new' should be a number (array length), but got '" << type.original_string << "'\n" << end();
114     break;
115   }
116   if (inst.products.empty()) {
117     raise << maybe(caller.name) << "result of 'new' should never be ignored\n" << end();
118     break;
119   }
120   if (!product_of_new_is_valid(inst)) {
121     raise << maybe(caller.name) << "product of 'new' has incorrect type: '" << to_original_string(inst) << "'\n" << end();
122     break;
123   }
124   break;
125 }
126 :(code)
127 bool product_of_new_is_valid(const instruction& inst) {
128   reagent product = inst.products.at(0);
129   
130   if (!product.type || product.type->atom || product.type->left->value != Address_type_ordinal)
131     return false;
132   drop_from_type(product, "address");
133   if (SIZE(inst.ingredients) > 1) {
134     
135     if (!product.type || product.type->atom || product.type->left->value != Array_type_ordinal)
136       return false;
137     drop_from_type(product, "array");
138   }
139   reagent expected_product(new_type_tree(inst.ingredients.at(0).name));
140   return types_strictly_match(product, expected_product);
141 }
142 
143 void drop_from_type(reagent& r, string expected_type) {
144   assert(!r.type->atom);
145   if (r.type->left->name != expected_type) {
146     raise << "can't drop2 " << expected_type << " from '" << to_string(r) << "'\n" << end();
147     return;
148   }
149   
150   type_tree* tmp = r.type;
151   r.type = tmp->right;
152   tmp->right = NULL;
153   delete tmp;
154   
155   assert(!r.type->atom);
156   if (r.type->right) return;
157   tmp = r.type;
158   r.type = tmp->left;
159   tmp->left = NULL;
160   delete tmp;
161 }
162 
163 :(scenario new_returns_incorrect_type)
164 % Hide_errors = true;
165 def main [
166   1:bool <- new num:type
167 ]
168 +error: main: product of 'new' has incorrect type: '1:bool <- new num:type'
169 
170 :(scenario new_discerns_singleton_list_from_atom_container)
171 % Hide_errors = true;
172 def main [
173   1:&:num <- new {(num): type}  
174 ]
175 +error: main: product of 'new' has incorrect type: '1:&:num <- new {(num): type}'
176 
177 :(scenario new_with_type_abbreviation)
178 def main [
179   1:&:num <- new num:type
180 ]
181 $error: 0
182 
183 :(scenario new_with_type_abbreviation_inside_compound)
184 def main [
185   {1: (address address number), raw: ()} <- new {(& num): type}
186 ]
187 $error: 0
188 
189 :(scenario equal_result_of_new_with_null)
190 def main [
191   1:&:num <- new num:type
192   10:bool <- equal 1:&:num, null
193 ]
194 +mem: storing 0 in location 10
195 
196 
197 
198 
199 
200 
201 
202 :(before "End Primitive Recipe Checks")
203 case ALLOCATE: {
204   raise << "never call 'allocate' directly'; always use 'new'\n" << end();
205   break;
206 }
207 :(before "End Primitive Recipe Implementations")
208 case NEW: {
209   raise << "no implementation for 'new'; why wasn't it translated to 'allocate'? Please save a copy of your program and send it to Kartik.\n" << end();
210   break;
211 }
212 
213 :(after "Transform.push_back(check_instruction)")  
214 Transform.push_back(transform_new_to_allocate);  
215 
216 :(code)
217 void transform_new_to_allocate(const recipe_ordinal r) {
218   trace(9991, "transform") << "--- convert 'new' to 'allocate' for recipe " << get(Recipe, r).name << end();
219   for (int i = 0;  i < SIZE(get(Recipe, r).steps);  ++i) {
220     instruction& inst = get(Recipe, r).steps.at(i);
221     
222     if (inst.name == "new") {
223       if (inst.ingredients.empty()) return;  
224       inst.operation = ALLOCATE;
225       type_tree* type = new_type_tree(inst.ingredients.at(0).name);
226       inst.ingredients.at(0).set_value(size_of(type));
227       trace(9992, "new") << "size of '" << inst.ingredients.at(0).name << "' is " << inst.ingredients.at(0).value << end();
228       delete type;
229     }
230   }
231 }
232 
233 
234 
235 :(before "End Globals")
236 extern const int Reserved_for_tests = 1000;
237 int Memory_allocated_until = Reserved_for_tests;
238 int Initial_memory_per_routine = 100000;
239 :(before "End Reset")
240 Memory_allocated_until = Reserved_for_tests;
241 Initial_memory_per_routine = 100000;
242 :(before "End routine Fields")
243 int alloc, alloc_max;
244 :(before "End routine Constructor")
245 alloc = Memory_allocated_until;
246 Memory_allocated_until += Initial_memory_per_routine;
247 alloc_max = Memory_allocated_until;
248 trace("new") << "routine allocated memory from " << alloc << " to " << alloc_max << end();
249 
250 :(before "End Primitive Recipe Declarations")
251 ALLOCATE,
252 :(before "End Primitive Recipe Numbers")
253 put(Recipe_ordinal, "allocate", ALLOCATE);
254 :(before "End Primitive Recipe Implementations")
255 case ALLOCATE: {
256   
257   int size = ingredients.at(0).at(0);
258   int alloc_id = Next_alloc_id;
259   Next_alloc_id++;
260   if (SIZE(ingredients) > 1) {
261     
262     trace("mem") << "array length is " << ingredients.at(1).at(0) << end();
263     size = 1 + size*ingredients.at(1).at(0);
264   }
265   int result = allocate(size);
266   
267   trace("mem") << "storing alloc-id " << alloc_id << " in location " << result << end();
268   put(Memory, result, alloc_id);
269   if (SIZE(current_instruction().ingredients) > 1) {
270     
271     trace("mem") << "storing array length " << ingredients.at(1).at(0) << " in location " << result+1 << end();
272     put(Memory, result+1, ingredients.at(1).at(0));
273   }
274   products.resize(1);
275   products.at(0).push_back(alloc_id);
276   products.at(0).push_back(result);
277   break;
278 }
279 :(code)
280 int allocate(int size) {
281   
282   ++size;
283   trace("mem") << "allocating size " << size << end();
284 
285 
286   
287   
288   
289   ensure_space(size);
290   const int result = Current_routine->alloc;
291   trace("mem") << "new alloc: " << result << end();
292   
293   for (int address = result;  address < result+size;  ++address) {
294     trace("mem") << "storing 0 in location " << address << end();
295     put(Memory, address, 0);
296   }
297   Current_routine->alloc += size;
298   
299   assert(Current_routine->alloc <= Current_routine->alloc_max);
300   return result;
301 }
302 
303 
304 
305 
306 
307 
308 
309 
310 
311 
312 
313 
314 
315 
316 
317 :(code)
318 void ensure_space(int size) {
319   if (size > Initial_memory_per_routine) {
320     cerr << "can't allocate " << size << " locations, that's too much compared to " << Initial_memory_per_routine << ".\n";
321     exit(1);
322   }
323   if (Current_routine->alloc + size > Current_routine->alloc_max) {
324     
325     Current_routine->alloc = Memory_allocated_until;
326     Memory_allocated_until += Initial_memory_per_routine;
327     Current_routine->alloc_max = Memory_allocated_until;
328     trace("new") << "routine allocated memory from " << Current_routine->alloc << " to " << Current_routine->alloc_max << end();
329   }
330 }
331 
332 :(scenario new_initializes)
333 % Memory_allocated_until = 10;
334 % put(Memory, Memory_allocated_until, 1);
335 def main [
336   1:&:num <- new num:type
337 ]
338 +mem: storing 0 in location 10
339 +mem: storing 0 in location 11
340 +mem: storing 10 in location 2
341 
342 :(scenario new_initializes_alloc_id)
343 % Memory_allocated_until = 10;
344 % put(Memory, Memory_allocated_until, 1);
345 % Next_alloc_id = 23;
346 def main [
347   1:&:num <- new num:type
348 ]
349 
350 +mem: storing 0 in location 10
351 +mem: storing 0 in location 11
352 
353 +mem: storing alloc-id 23 in location 10
354 
355 +mem: storing 23 in location 1
356 
357 :(scenario new_size)
358 def main [
359   10:&:num <- new num:type
360   12:&:num <- new num:type
361   20:num/alloc1, 21:num/alloc2 <- deaddress 10:&:num, 12:&:num
362   30:num <- subtract 21:num/alloc2, 20:num/alloc1
363 ]
364 
365 +mem: storing 2 in location 30
366 
367 :(scenario new_array_size)
368 def main [
369   10:&:@:num <- new num:type, 5
370   12:&:num <- new num:type
371   20:num/alloc1, 21:num/alloc2 <- deaddress 10:&:num, 12:&:num
372   30:num <- subtract 21:num/alloc2, 20:num/alloc1
373 ]
374 
375 +mem: storing 7 in location 30
376 
377 :(scenario new_empty_array)
378 def main [
379   10:&:@:num <- new num:type, 0
380   12:&:num <- new num:type
381   20:num/alloc1, 21:num/alloc2 <- deaddress 10:&:@:num, 12:&:num
382   30:num <- subtract 21:num/alloc2, 20:num/alloc1
383 ]
384 +run: {10: ("address" "array" "number")} <- new {num: "type"}, {0: "literal"}
385 +mem: array length is 0
386 
387 +mem: storing 2 in location 30
388 
389 
390 :(scenario new_overflow)
391 % Initial_memory_per_routine = 3;  // barely enough room for point allocation below
392 def main [
393   10:&:num <- new num:type
394   12:&:point <- new point:type  
395 ]
396 +new: routine allocated memory from 1000 to 1003
397 +new: routine allocated memory from 1003 to 1006
398 
399 :(scenario new_without_ingredient)
400 % Hide_errors = true;
401 def main [
402   1:&:num <- new  
403 ]
404 +error: main: 'new' requires one or two ingredients, but got '1:&:num <- new'
405 
406 
407 
408 :(before "End Primitive Recipe Declarations")
409 DEADDRESS,
410 :(before "End Primitive Recipe Numbers")
411 put(Recipe_ordinal, "deaddress", DEADDRESS);
412 :(before "End Primitive Recipe Checks")
413 case DEADDRESS: {
414   
415   for (int i = 0;  i < SIZE(inst.ingredients);  ++i) {
416     if (!is_mu_address(inst.ingredients.at(i))) {
417       raise << maybe(get(Recipe, r).name) << "'deaddress' requires address ingredients, but got '" << inst.ingredients.at(i).original_string << "'\n" << end();
418       goto finish_checking_instruction;
419     }
420   }
421   if (SIZE(inst.products) > SIZE(inst.ingredients)) {
422     raise << maybe(get(Recipe, r).name) << "too many products in '" << to_original_string(inst) << "'\n" << end();
423     break;
424   }
425   for (int i = 0;  i < SIZE(inst.products);  ++i) {
426     if (!is_real_mu_number(inst.products.at(i))) {
427       raise << maybe(get(Recipe, r).name) << "'deaddress' requires number products, but got '" << inst.products.at(i).original_string << "'\n" << end();
428       goto finish_checking_instruction;
429     }
430   }
431   break;
432 }
433 :(before "End Primitive Recipe Implementations")
434 case DEADDRESS: {
435   products.resize(SIZE(ingredients));
436   for (int i = 0;  i < SIZE(ingredients);  ++i) {
437     products.at(i).push_back(ingredients.at(i).at(1));
438   }
439   break;
440 }