You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

vm.c 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638
  1. #include "vm/vm.h"
  2. #include <string.h>
  3. #include <stdio.h>
  4. #include <stdarg.h>
  5. #include "vm/builtins.h"
  6. static int stdio_inited = 0;
  7. static struct l2_io_file_writer std_output;
  8. static struct l2_io_file_writer std_error;
  9. static l2_word alloc_val(struct l2_vm *vm) {
  10. size_t id = l2_bitset_set_next(&vm->valueset);
  11. if (id + 16 >= vm->valuessize) {
  12. if (id >= vm->valuessize) {
  13. if (vm->valuessize == 0) {
  14. vm->valuessize = 64;
  15. }
  16. while (id >= vm->valuessize) {
  17. vm->valuessize *= 2;
  18. }
  19. vm->values = realloc(vm->values, sizeof(*vm->values) * vm->valuessize);
  20. } else {
  21. vm->gc_scheduled = 1;
  22. }
  23. }
  24. return (l2_word)id;
  25. }
  26. static double u32s_to_double(uint32_t high, uint32_t low) {
  27. double d;
  28. uint64_t num = (uint64_t)high << 32 | (uint64_t)low;
  29. memcpy(&d, &num, sizeof(num));
  30. return d;
  31. }
  32. static void gc_mark_array(struct l2_vm *vm, struct l2_vm_value *val);
  33. static void gc_mark_namespace(struct l2_vm *vm, struct l2_vm_value *val);
  34. static void gc_mark(struct l2_vm *vm, l2_word id) {
  35. struct l2_vm_value *val = &vm->values[id];
  36. if (val->flags & L2_VAL_MARKED) {
  37. return;
  38. }
  39. val->flags |= L2_VAL_MARKED;
  40. int typ = l2_vm_value_type(val);
  41. if (typ == L2_VAL_TYPE_ARRAY) {
  42. gc_mark_array(vm, val);
  43. } else if (typ == L2_VAL_TYPE_NAMESPACE) {
  44. gc_mark_namespace(vm, val);
  45. } else if (typ == L2_VAL_TYPE_FUNCTION) {
  46. gc_mark(vm, val->func.ns);
  47. }
  48. }
  49. static void gc_mark_array(struct l2_vm *vm, struct l2_vm_value *val) {
  50. if (val->array == NULL) {
  51. return;
  52. }
  53. for (size_t i = 0; i < val->array->len; ++i) {
  54. gc_mark(vm, val->array->data[i]);
  55. }
  56. }
  57. static void gc_mark_namespace(struct l2_vm *vm, struct l2_vm_value *val) {
  58. if (val->extra.ns_parent != 0) {
  59. gc_mark(vm, val->extra.ns_parent);
  60. }
  61. if (val->ns == NULL) {
  62. return;
  63. }
  64. for (size_t i = 0; i < val->ns->size; ++i) {
  65. l2_word key = val->ns->data[i];
  66. if (key == 0 || key == ~(l2_word)0) {
  67. continue;
  68. }
  69. gc_mark(vm, val->ns->data[val->ns->size + i]);
  70. }
  71. }
  72. static void gc_free(struct l2_vm *vm, l2_word id) {
  73. struct l2_vm_value *val = &vm->values[id];
  74. l2_bitset_unset(&vm->valueset, id);
  75. // Don't need to do anything more; the next round of GC will free
  76. // whichever values were only referenced by the array
  77. int typ = l2_vm_value_type(val);
  78. if (typ == L2_VAL_TYPE_ARRAY) {
  79. free(val->array);
  80. } else if (typ == L2_VAL_TYPE_BUFFER) {
  81. free(val->buffer);
  82. } else if (typ == L2_VAL_TYPE_NAMESPACE) {
  83. free(val->ns);
  84. } else if (typ == L2_VAL_TYPE_ERROR) {
  85. free(val->error);
  86. }
  87. }
  88. static size_t gc_sweep(struct l2_vm *vm) {
  89. size_t freed = 0;
  90. // Skip ID 0, because that should always exist
  91. for (size_t i = 1; i < vm->valuessize; ++i) {
  92. if (!l2_bitset_get(&vm->valueset, i)) {
  93. continue;
  94. }
  95. struct l2_vm_value *val = &vm->values[i];
  96. if (!(val->flags & L2_VAL_MARKED)) {
  97. l2_bitset_unset(&vm->valueset, i);
  98. gc_free(vm, i);
  99. freed += 1;
  100. } else {
  101. val->flags &= ~L2_VAL_MARKED;
  102. }
  103. }
  104. return freed;
  105. }
  106. const char *l2_value_type_name(enum l2_value_type typ) {
  107. switch (typ) {
  108. case L2_VAL_TYPE_NONE: return "NONE";
  109. case L2_VAL_TYPE_ATOM: return "ATOM";
  110. case L2_VAL_TYPE_REAL: return "REAL";
  111. case L2_VAL_TYPE_BUFFER: return "BUFFER";
  112. case L2_VAL_TYPE_ARRAY: return "ARRAY";
  113. case L2_VAL_TYPE_NAMESPACE: return "NAMESPACE";
  114. case L2_VAL_TYPE_FUNCTION: return "FUNCTION";
  115. case L2_VAL_TYPE_CFUNCTION: return "CFUNCTION";
  116. case L2_VAL_TYPE_ERROR: return "ERROR";
  117. case L2_VAL_TYPE_CONTINUATION: return "CONTINUATION";
  118. }
  119. return "(unknown)";
  120. }
  121. void l2_vm_init(struct l2_vm *vm, l2_word *ops, size_t opcount) {
  122. if (!stdio_inited) {
  123. std_output.w.write = l2_io_file_write;
  124. std_output.f = stdout;
  125. std_error.w.write = l2_io_file_write;
  126. std_error.f = stderr;
  127. stdio_inited = 1;
  128. }
  129. vm->std_output = &std_output.w;
  130. vm->std_error = &std_error.w;
  131. vm->halted = 0;
  132. vm->gc_scheduled = 0;
  133. vm->ops = ops;
  134. vm->opcount = opcount;
  135. vm->iptr = 0;
  136. vm->sptr = 0;
  137. vm->fsptr = 0;
  138. vm->values = NULL;
  139. vm->valuessize = 0;
  140. l2_bitset_init(&vm->valueset);
  141. // It's wasteful to allocate new 'none' variables all the time,
  142. // variable ID 0 should be the only 'none' variable in the system
  143. l2_word none_id = alloc_val(vm);
  144. vm->values[none_id].flags = L2_VAL_TYPE_NONE | L2_VAL_CONST;
  145. // Need to allocate a builtins namespace
  146. l2_word builtins = alloc_val(vm);
  147. vm->values[builtins].extra.ns_parent = 0;
  148. vm->values[builtins].ns = NULL; // Will be allocated on first insert
  149. vm->values[builtins].flags = L2_VAL_TYPE_NAMESPACE;
  150. vm->fstack[vm->fsptr].ns = builtins;
  151. vm->fstack[vm->fsptr].retptr = 0;
  152. vm->fsptr += 1;
  153. // Need to allocate a root namespace
  154. l2_word root = alloc_val(vm);
  155. vm->values[root].extra.ns_parent = builtins;
  156. vm->values[root].ns = NULL;
  157. vm->values[root].flags = L2_VAL_TYPE_NAMESPACE;
  158. vm->fstack[vm->fsptr].ns = root;
  159. vm->fstack[vm->fsptr].retptr = 0;
  160. vm->fsptr += 1;
  161. // Define a C function variable for every builtin
  162. l2_word id;
  163. l2_word key = 1;
  164. #define Y(name, k) \
  165. if (strcmp(#k, "knone") == 0) { \
  166. id = 0; \
  167. } else { \
  168. id = alloc_val(vm); \
  169. vm->values[id].flags = L2_VAL_TYPE_ATOM | L2_VAL_CONST; \
  170. vm->values[id].atom = key; \
  171. } \
  172. vm->k = id; \
  173. l2_vm_namespace_set(&vm->values[builtins], key++, id);
  174. #define X(name, f) \
  175. id = alloc_val(vm); \
  176. vm->values[id].flags = L2_VAL_TYPE_CFUNCTION | L2_VAL_CONST; \
  177. vm->values[id].cfunc = f; \
  178. l2_vm_namespace_set(&vm->values[builtins], key++, id);
  179. #include "builtins.x.h"
  180. #undef X
  181. }
  182. l2_word l2_vm_alloc(struct l2_vm *vm, enum l2_value_type typ, enum l2_value_flags flags) {
  183. l2_word id = alloc_val(vm);
  184. memset(&vm->values[id], 0, sizeof(vm->values[id]));
  185. vm->values[id].flags = typ | flags;
  186. return id;
  187. }
  188. l2_word l2_vm_error(struct l2_vm *vm, const char *fmt, ...) {
  189. l2_word id = alloc_val(vm);
  190. struct l2_vm_value *val = &vm->values[id];
  191. val->flags = L2_VAL_CONST | L2_VAL_TYPE_ERROR;
  192. char buf[256];
  193. va_list va;
  194. va_start(va, fmt);
  195. int n = vsnprintf(buf, sizeof(buf), fmt, va);
  196. if (n < 0) {
  197. const char *message = "Failed to generate error message!";
  198. val->error = malloc(strlen(message) + 1);
  199. strcpy(val->error, message);
  200. va_end(va);
  201. return id;
  202. } else if ((size_t)n + 1 < sizeof(buf)) {
  203. val->error = malloc(n + 1);
  204. strcpy(val->error, buf);
  205. va_end(va);
  206. return id;
  207. }
  208. val->error = malloc(n + 1);
  209. vsnprintf(val->error, n + 1, fmt, va);
  210. va_end(va);
  211. return id;
  212. }
  213. l2_word l2_vm_type_error(struct l2_vm *vm, struct l2_vm_value *val) {
  214. return l2_vm_error(vm, "Unexpected type %s", l2_value_type_name(l2_vm_value_type(val)));
  215. }
  216. void l2_vm_free(struct l2_vm *vm) {
  217. // Skip ID 0, because that's always NONE
  218. for (size_t i = 1; i < vm->valuessize; ++i) {
  219. if (!l2_bitset_get(&vm->valueset, i)) {
  220. continue;
  221. }
  222. gc_free(vm, i);
  223. }
  224. free(vm->values);
  225. l2_bitset_free(&vm->valueset);
  226. }
  227. size_t l2_vm_gc(struct l2_vm *vm) {
  228. for (l2_word sptr = 0; sptr < vm->sptr; ++sptr) {
  229. gc_mark(vm, vm->stack[sptr]);
  230. }
  231. for (l2_word fsptr = 0; fsptr < vm->fsptr; ++fsptr) {
  232. gc_mark(vm, vm->fstack[fsptr].ns);
  233. }
  234. return gc_sweep(vm);
  235. }
  236. void l2_vm_run(struct l2_vm *vm) {
  237. while (!vm->halted) {
  238. l2_vm_step(vm);
  239. }
  240. }
  241. // The 'call_func' function assumes that all relevant values have been popped off
  242. // the stack, so that the return value can be pushed to the top of the stack
  243. // straight away
  244. static void call_func(
  245. struct l2_vm *vm, l2_word func_id,
  246. l2_word argc, l2_word *argv) {
  247. l2_word stack_base = vm->sptr;
  248. struct l2_vm_value *func = &vm->values[func_id];
  249. enum l2_value_type typ = l2_vm_value_type(func);
  250. // C functions are called differently from language functions
  251. if (typ == L2_VAL_TYPE_CFUNCTION) {
  252. // Make this a while loop, because using call_func would
  253. // make the call stack depth unbounded
  254. vm->stack[vm->sptr++] = func->cfunc(vm, argc, argv);
  255. while (1) {
  256. struct l2_vm_value *val = &vm->values[vm->stack[vm->sptr - 1]];
  257. if (l2_vm_value_type(val) != L2_VAL_TYPE_CONTINUATION) {
  258. break;
  259. }
  260. l2_word cont_id = val->cont.call;
  261. struct l2_vm_value *cont = &vm->values[cont_id];
  262. l2_word new_argc;
  263. l2_word new_argv[1];
  264. if (val->cont.arg == 0) {
  265. new_argc = 0;
  266. } else {
  267. new_argc = 1;
  268. new_argv[0] = val->cont.arg;
  269. }
  270. if (l2_vm_value_type(cont) == L2_VAL_TYPE_CFUNCTION) {
  271. vm->stack[vm->sptr - 1] = cont->cfunc(vm, new_argc, new_argv);
  272. } else {
  273. vm->sptr -= 1;
  274. call_func(vm, cont_id, new_argc, new_argv);
  275. break;
  276. }
  277. }
  278. return;
  279. }
  280. // Don't interpret a non-function as a function
  281. if (typ != L2_VAL_TYPE_FUNCTION) {
  282. vm->stack[vm->sptr++] = l2_vm_error(vm, "Attempt to call non-function");
  283. return;
  284. }
  285. l2_word arr_id = alloc_val(vm);
  286. vm->values[arr_id].flags = L2_VAL_TYPE_ARRAY;
  287. vm->values[arr_id].array = malloc(
  288. sizeof(struct l2_vm_array) + sizeof(l2_word) * argc);
  289. struct l2_vm_array *arr = vm->values[arr_id].array;
  290. arr->len = argc;
  291. arr->size = argc;
  292. for (l2_word i = 0; i < argc; ++i) {
  293. arr->data[i] = argv[i];
  294. }
  295. vm->stack[vm->sptr++] = arr_id;
  296. l2_word ns_id = alloc_val(vm);
  297. func = &vm->values[func_id]; // func might be stale after alloc
  298. vm->values[ns_id].extra.ns_parent = func->func.ns;
  299. vm->values[ns_id].ns = NULL;
  300. vm->values[ns_id].flags = L2_VAL_TYPE_NAMESPACE;
  301. vm->fstack[vm->fsptr].ns = ns_id;
  302. vm->fstack[vm->fsptr].retptr = vm->iptr;
  303. vm->fstack[vm->fsptr].sptr = stack_base;
  304. vm->fsptr += 1;
  305. vm->iptr = func->func.pos;
  306. }
  307. void l2_vm_step(struct l2_vm *vm) {
  308. enum l2_opcode opcode = (enum l2_opcode)vm->ops[vm->iptr++];
  309. l2_word word;
  310. switch (opcode) {
  311. case L2_OP_NOP:
  312. break;
  313. case L2_OP_DISCARD:
  314. vm->sptr -= 1;
  315. if (l2_vm_value_type(&vm->values[vm->stack[vm->sptr]]) == L2_VAL_TYPE_ERROR) {
  316. l2_io_printf(vm->std_error, "Error: %s\n", vm->values[vm->stack[vm->sptr]].error);
  317. vm->halted = 1;
  318. }
  319. break;
  320. case L2_OP_SWAP_DISCARD:
  321. vm->stack[vm->sptr - 2] = vm->stack[vm->sptr - 1];
  322. vm->sptr -= 1;
  323. if (l2_vm_value_type(&vm->values[vm->stack[vm->sptr]]) == L2_VAL_TYPE_ERROR) {
  324. l2_io_printf(vm->std_error, "Error: %s\n", vm->values[vm->stack[vm->sptr]].error);
  325. vm->halted = 1;
  326. }
  327. break;
  328. case L2_OP_DUP:
  329. vm->stack[vm->sptr] = vm->ops[vm->sptr - 1];
  330. vm->sptr += 1;
  331. break;
  332. case L2_OP_ADD:
  333. vm->stack[vm->sptr - 2] += vm->stack[vm->sptr - 1];
  334. vm->sptr -= 1;
  335. break;
  336. case L2_OP_FUNC_CALL:
  337. {
  338. l2_word argc = vm->ops[vm->iptr++];
  339. vm->sptr -= argc;
  340. l2_word *argv = vm->stack + vm->sptr;
  341. l2_word func_id = vm->stack[--vm->sptr];
  342. call_func(vm, func_id, argc, argv);
  343. }
  344. break;
  345. case L2_OP_RJMP:
  346. vm->iptr += vm->ops[vm->iptr] + 1;
  347. break;
  348. case L2_OP_STACK_FRAME_LOOKUP:
  349. {
  350. l2_word key = vm->ops[vm->iptr++];
  351. struct l2_vm_value *ns = &vm->values[vm->fstack[vm->fsptr - 1].ns];
  352. vm->stack[vm->sptr++] = l2_vm_namespace_get(vm, ns, key);
  353. }
  354. break;
  355. case L2_OP_STACK_FRAME_SET:
  356. {
  357. l2_word key = vm->ops[vm->iptr++];
  358. l2_word val = vm->stack[vm->sptr - 1];
  359. struct l2_vm_value *ns = &vm->values[vm->fstack[vm->fsptr - 1].ns];
  360. l2_vm_namespace_set(ns, key, val);
  361. }
  362. break;
  363. case L2_OP_STACK_FRAME_REPLACE:
  364. {
  365. l2_word key = vm->ops[vm->iptr++];
  366. l2_word val = vm->stack[vm->sptr - 1];
  367. struct l2_vm_value *ns = &vm->values[vm->fstack[vm->fsptr - 1].ns];
  368. l2_vm_namespace_replace(vm, ns, key, val); // TODO: error if returns -1
  369. }
  370. break;
  371. case L2_OP_RET:
  372. {
  373. l2_word retval = vm->stack[--vm->sptr];
  374. l2_word retptr = vm->fstack[vm->fsptr - 1].retptr;
  375. l2_word sptr = vm->fstack[vm->fsptr - 1].sptr;
  376. vm->fsptr -= 1;
  377. vm->sptr = sptr;
  378. vm->stack[vm->sptr++] = retval;
  379. vm->iptr = retptr;
  380. }
  381. break;
  382. case L2_OP_ALLOC_NONE:
  383. vm->stack[vm->sptr++] = 0;
  384. break;
  385. case L2_OP_ALLOC_ATOM:
  386. word = alloc_val(vm);
  387. vm->values[word].flags = L2_VAL_TYPE_ATOM;
  388. vm->values[word].atom = vm->ops[vm->iptr++];
  389. vm->stack[vm->sptr++] = word;
  390. break;
  391. case L2_OP_ALLOC_REAL:
  392. {
  393. word = alloc_val(vm);
  394. l2_word high = vm->ops[vm->iptr++];
  395. l2_word low = vm->ops[vm->iptr++];
  396. vm->values[word].flags = L2_VAL_TYPE_REAL;
  397. vm->values[word].real = u32s_to_double(high, low);
  398. vm->stack[vm->sptr++] = word;
  399. }
  400. break;
  401. case L2_OP_ALLOC_BUFFER_STATIC:
  402. {
  403. word = alloc_val(vm);
  404. l2_word length = vm->ops[vm->iptr++];
  405. l2_word offset = vm->ops[vm->iptr++];
  406. vm->values[word].flags = L2_VAL_TYPE_BUFFER;
  407. vm->values[word].buffer = malloc(sizeof(struct l2_vm_buffer) + length);
  408. vm->values[word].buffer->len = length;
  409. memcpy(
  410. (unsigned char *)vm->values[word].buffer + sizeof(struct l2_vm_buffer),
  411. vm->ops + offset, length);
  412. vm->stack[vm->sptr] = word;
  413. vm->sptr += 1;
  414. }
  415. break;
  416. case L2_OP_ALLOC_ARRAY:
  417. {
  418. l2_word count = vm->ops[vm->iptr++];
  419. l2_word arr_id = alloc_val(vm);
  420. struct l2_vm_value *arr = &vm->values[arr_id];
  421. arr->flags = L2_VAL_TYPE_ARRAY;
  422. if (count == 0) {
  423. arr->array = NULL;
  424. vm->stack[vm->sptr++] = arr_id;
  425. break;
  426. }
  427. arr->array = malloc(sizeof(struct l2_vm_array) + count * sizeof(l2_word));
  428. arr->array->len = count;
  429. arr->array->size = count;
  430. for (l2_word i = 0; i < count; ++i) {
  431. arr->array->data[count - 1 - i] = vm->stack[--vm->sptr];
  432. }
  433. vm->stack[vm->sptr++] = arr_id;
  434. }
  435. break;
  436. case L2_OP_ALLOC_NAMESPACE:
  437. word = alloc_val(vm);
  438. vm->values[word].flags = L2_VAL_TYPE_NAMESPACE;
  439. vm->values[word].extra.ns_parent = 0;
  440. vm->values[word].ns = NULL; // Will be allocated on first insert
  441. vm->stack[vm->sptr] = word;
  442. vm->sptr += 1;
  443. break;
  444. case L2_OP_ALLOC_FUNCTION:
  445. word = alloc_val(vm);
  446. vm->values[word].flags = L2_VAL_TYPE_FUNCTION;
  447. vm->values[word].func.pos = vm->ops[vm->iptr++];
  448. vm->values[word].func.ns = vm->fstack[vm->fsptr - 1].ns;
  449. vm->stack[vm->sptr] = word;
  450. vm->sptr += 1;
  451. break;
  452. case L2_OP_NAMESPACE_SET:
  453. {
  454. l2_word key = vm->ops[vm->iptr++];
  455. l2_word val = vm->stack[vm->sptr - 1];
  456. l2_word ns = vm->stack[vm->sptr - 2];
  457. l2_vm_namespace_set(&vm->values[ns], key, val);
  458. }
  459. break;
  460. case L2_OP_NAMESPACE_LOOKUP:
  461. {
  462. l2_word key = vm->ops[vm->iptr++];
  463. l2_word ns = vm->stack[--vm->sptr];
  464. vm->stack[vm->sptr++] = l2_vm_namespace_get(vm, &vm->values[ns], key);
  465. }
  466. break;
  467. case L2_OP_ARRAY_LOOKUP:
  468. {
  469. l2_word key = vm->ops[vm->iptr++];
  470. l2_word arr = vm->stack[--vm->sptr];
  471. // TODO: Error if out of bounds or incorrect type
  472. vm->stack[vm->sptr++] = vm->values[arr].array->data[key];
  473. }
  474. break;
  475. case L2_OP_ARRAY_SET:
  476. {
  477. l2_word key = vm->ops[vm->iptr++];
  478. l2_word val = vm->stack[vm->sptr - 1];
  479. l2_word arr = vm->stack[vm->sptr - 2];
  480. // TODO: Error if out of bounds or incorrect type
  481. vm->values[arr].array->data[key] = val;
  482. }
  483. break;
  484. case L2_OP_DYNAMIC_LOOKUP:
  485. {
  486. l2_word key_id = vm->stack[--vm->sptr];
  487. l2_word container_id = vm->stack[--vm->sptr];
  488. struct l2_vm_value *key = &vm->values[key_id];
  489. struct l2_vm_value *container = &vm->values[container_id];
  490. if (
  491. l2_vm_value_type(key) == L2_VAL_TYPE_REAL &&
  492. l2_vm_value_type(container) == L2_VAL_TYPE_ARRAY) {
  493. // TODO: Error if out of bounds
  494. vm->stack[vm->sptr++] = container->array->data[(size_t)key->real];
  495. } else if (
  496. l2_vm_value_type(key) == L2_VAL_TYPE_ATOM &&
  497. l2_vm_value_type(container) == L2_VAL_TYPE_NAMESPACE) {
  498. // TODO: Error if out of bounds
  499. vm->stack[vm->sptr++] = l2_vm_namespace_get(vm, container, key->atom);
  500. } else {
  501. // TODO: error
  502. }
  503. }
  504. break;
  505. case L2_OP_DYNAMIC_SET:
  506. {
  507. l2_word val = vm->stack[--vm->sptr];
  508. l2_word key_id = vm->stack[--vm->sptr];
  509. l2_word container_id = vm->stack[--vm->sptr];
  510. vm->stack[vm->sptr++] = val;
  511. struct l2_vm_value *key = &vm->values[key_id];
  512. struct l2_vm_value *container = &vm->values[container_id];
  513. if (
  514. l2_vm_value_type(key) == L2_VAL_TYPE_REAL &&
  515. l2_vm_value_type(container) == L2_VAL_TYPE_ARRAY) {
  516. // TODO: Error if out of bounds
  517. container->array->data[(size_t)key->real] = val;
  518. } else if (
  519. l2_vm_value_type(key) == L2_VAL_TYPE_ATOM &&
  520. l2_vm_value_type(container) == L2_VAL_TYPE_NAMESPACE) {
  521. // TODO: Error if out of bounds
  522. l2_vm_namespace_set(container, key->atom, val);
  523. } else {
  524. // TODO: error
  525. }
  526. }
  527. break;
  528. case L2_OP_FUNC_CALL_INFIX:
  529. {
  530. l2_word rhs = vm->stack[--vm->sptr];
  531. l2_word func_id = vm->stack[--vm->sptr];
  532. l2_word lhs = vm->stack[--vm->sptr];
  533. l2_word argv[] = {lhs, rhs};
  534. call_func(vm, func_id, 2, argv);
  535. }
  536. break;
  537. case L2_OP_HALT:
  538. vm->halted = 1;
  539. break;
  540. }
  541. if (vm->gc_scheduled) {
  542. l2_vm_gc(vm);
  543. vm->gc_scheduled = 0;
  544. }
  545. }