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.

print.c 7.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. #include "vm/print.h"
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <stdint.h>
  5. static l2_word read_u4le(unsigned char *ops, size_t *ptr) {
  6. unsigned char *data = &ops[*ptr];
  7. l2_word ret =
  8. (l2_word)data[0] |
  9. (l2_word)data[1] << 8 |
  10. (l2_word)data[2] << 16 |
  11. (l2_word)data[3] << 24;
  12. *ptr += 4;
  13. return ret;
  14. }
  15. static l2_word read_u1le(unsigned char *ops, size_t *ptr) {
  16. return ops[(*ptr)++];
  17. }
  18. static double read_d8le(unsigned char *ops, size_t *ptr) {
  19. unsigned char *data = &ops[*ptr];
  20. uint64_t integer = 0 |
  21. (uint64_t)data[0] |
  22. (uint64_t)data[1] << 8 |
  23. (uint64_t)data[2] << 16 |
  24. (uint64_t)data[3] << 24 |
  25. (uint64_t)data[4] << 32 |
  26. (uint64_t)data[5] << 40 |
  27. (uint64_t)data[6] << 48 |
  28. (uint64_t)data[7] << 56;
  29. double num;
  30. memcpy(&num, &integer, 8);
  31. *ptr += 8;
  32. return num;
  33. }
  34. void l2_vm_print_val(struct l2_vm_value *val) {
  35. switch (l2_value_get_type(val)) {
  36. case L2_VAL_TYPE_NONE:
  37. printf("NONE\n");
  38. break;
  39. case L2_VAL_TYPE_ATOM:
  40. printf("ATOM %u\n", val->atom);
  41. break;
  42. case L2_VAL_TYPE_REAL:
  43. printf("REAL %f\n", val->real);
  44. break;
  45. case L2_VAL_TYPE_ARRAY:
  46. {
  47. if (val->array == NULL) {
  48. printf("ARRAY, empty\n");
  49. return;
  50. }
  51. printf("ARRAY, len %u\n", val->extra.arr_length);
  52. for (size_t i = 0; i < val->extra.arr_length; ++i) {
  53. printf(" %zu: %u\n", i, val->array->data[i]);
  54. }
  55. }
  56. break;
  57. case L2_VAL_TYPE_BUFFER:
  58. {
  59. if (val->buffer == NULL) {
  60. printf("BUFFER, empty\n");
  61. return;
  62. }
  63. printf("BUFFER, len %u\n", val->extra.buf_length);
  64. for (size_t i = 0; i < val->extra.buf_length; ++i) {
  65. printf(" %zu: %c\n", i, val->buffer[i]);
  66. }
  67. }
  68. break;
  69. case L2_VAL_TYPE_NAMESPACE:
  70. {
  71. if (val->ns == NULL) {
  72. printf("NAMESPACE, empty, parent %u\n", val->extra.ns_parent);
  73. return;
  74. }
  75. printf("NAMESPACE, len %zu, parent %u\n", val->ns->len, val->extra.ns_parent);
  76. for (size_t i = 0; i < val->ns->size; ++i) {
  77. l2_word key = val->ns->data[i];
  78. l2_word v = val->ns->data[val->ns->size + i];
  79. if (key == 0 || key == ~(l2_word)0) continue;
  80. printf(" %u: %u\n", key, v);
  81. }
  82. }
  83. break;
  84. case L2_VAL_TYPE_FUNCTION:
  85. printf("FUNCTION, pos %u, ns %u\n", val->func.pos, val->func.ns);
  86. break;
  87. case L2_VAL_TYPE_CFUNCTION:
  88. // ISO C doesn't let you cast a function pointer to void*.
  89. printf("C FUNCTION, %8jx\n", (uintmax_t)val->cfunc);
  90. break;
  91. case L2_VAL_TYPE_ERROR:
  92. printf("ERROR, %s\n", val->error);
  93. break;
  94. case L2_VAL_TYPE_CONTINUATION:
  95. printf("CONTINUATION, call %u, cont %08jx\n",
  96. val->extra.cont_call, (uintmax_t)val->cont);
  97. break;
  98. }
  99. }
  100. void l2_vm_print_state(struct l2_vm *vm) {
  101. printf("Stack:\n");
  102. l2_vm_print_stack(vm);
  103. printf("Heap:\n");
  104. l2_vm_print_heap(vm);
  105. printf("Frame Stack:\n");
  106. l2_vm_print_fstack(vm);
  107. }
  108. void l2_vm_print_heap(struct l2_vm *vm) {
  109. printf(" Root: ");
  110. l2_vm_print_val(&vm->values[2]);
  111. printf(" 0-%u: (builtins)\n", vm->gc_start - 1);
  112. for (l2_word i = vm->gc_start; i < vm->valuessize; ++i) {
  113. if (l2_bitset_get(&vm->valueset, i)) {
  114. printf(" %u: ", i);
  115. l2_vm_print_val(&vm->values[i]);
  116. }
  117. }
  118. }
  119. void l2_vm_print_stack(struct l2_vm *vm) {
  120. for (l2_word i = 0; i < vm->sptr; ++i) {
  121. printf(" %i: %i\n", i, vm->stack[i]);
  122. }
  123. }
  124. void l2_vm_print_fstack(struct l2_vm *vm) {
  125. for (l2_word i = 0; i < vm->fsptr; ++i) {
  126. printf(" %i: %i, ret %i, stack base %i\n",
  127. i, vm->fstack[i].ns, vm->fstack[i].retptr, vm->fstack[i].sptr);
  128. }
  129. }
  130. void l2_vm_print_op(unsigned char *ops, size_t opcount, size_t *ptr) {
  131. enum l2_opcode opcode = (enum l2_opcode)ops[(*ptr)++];
  132. switch (opcode) {
  133. case L2_OP_NOP:
  134. printf("NOP\n");
  135. return;
  136. case L2_OP_DISCARD:
  137. printf("DISCARD\n");
  138. return;
  139. case L2_OP_SWAP_DISCARD:
  140. printf("SWAP_DISCARD\n");
  141. return;
  142. case L2_OP_DUP:
  143. printf("DUP\n");
  144. return;
  145. case L2_OP_ADD:
  146. printf("ADD\n");
  147. return;
  148. case L2_OP_FUNC_CALL_U4:
  149. printf("FUNC_CALL %u\n", read_u4le(ops, ptr));
  150. return;
  151. case L2_OP_FUNC_CALL_U1:
  152. printf("FUNC_CALL %u\n", read_u1le(ops, ptr));
  153. return;
  154. case L2_OP_FUNC_CALL_INFIX:
  155. printf("FUNC_CALL_INFIX\n");
  156. return;
  157. case L2_OP_RJMP_U4:
  158. printf("RJMP %u\n", read_u4le(ops, ptr));
  159. return;
  160. case L2_OP_RJMP_U1:
  161. printf("RJMP %u\n", read_u1le(ops, ptr));
  162. return;
  163. case L2_OP_STACK_FRAME_LOOKUP_U4:
  164. printf("STACK_FRAME_LOOKUP %u\n", read_u4le(ops, ptr));
  165. return;
  166. case L2_OP_STACK_FRAME_LOOKUP_U1:
  167. printf("STACK_FRAME_LOOKUP %u\n", read_u1le(ops, ptr));
  168. return;
  169. case L2_OP_STACK_FRAME_SET_U4:
  170. printf("STACK_FRAME_SET %u\n", read_u4le(ops, ptr));
  171. return;
  172. case L2_OP_STACK_FRAME_SET_U1:
  173. printf("STACK_FRAME_SET %u\n", read_u1le(ops, ptr));
  174. return;
  175. case L2_OP_STACK_FRAME_REPLACE_U4:
  176. printf("STACK_FRAME_REPLACE %u\n", read_u4le(ops, ptr));
  177. return;
  178. case L2_OP_STACK_FRAME_REPLACE_U1:
  179. printf("STACK_FRAME_REPLACE %u\n", read_u1le(ops, ptr));
  180. return;
  181. case L2_OP_RET:
  182. printf("RET\n");
  183. return;
  184. case L2_OP_ALLOC_NONE:
  185. printf("ALLOC_NONE\n");
  186. return;
  187. case L2_OP_ALLOC_ATOM_U4:
  188. printf("ALLOC_ATOM %u\n", read_u4le(ops, ptr));
  189. return;
  190. case L2_OP_ALLOC_ATOM_U1:
  191. printf("ALLOC_ATOM %u\n", read_u1le(ops, ptr));
  192. return;
  193. case L2_OP_ALLOC_REAL_D8:
  194. printf("ALLOC_REAL %f\n", read_d8le(ops, ptr));
  195. return;
  196. case L2_OP_ALLOC_BUFFER_STATIC_U4:
  197. {
  198. l2_word w1 = read_u4le(ops, ptr);
  199. l2_word w2 = read_u4le(ops, ptr);;
  200. printf("ALLOC_BUFFER_STATIC %u %u\n", w1, w2);
  201. }
  202. return;
  203. case L2_OP_ALLOC_BUFFER_STATIC_U1:
  204. {
  205. l2_word w1 = read_u1le(ops, ptr);
  206. l2_word w2 = read_u1le(ops, ptr);;
  207. printf("ALLOC_BUFFER_STATIC %u %u\n", w1, w2);
  208. }
  209. return;
  210. case L2_OP_ALLOC_ARRAY_U4:
  211. printf("ALLOC_ARRAY %u\n", read_u4le(ops, ptr));
  212. return;
  213. case L2_OP_ALLOC_ARRAY_U1:
  214. printf("ALLOC_ARRAY %u\n", read_u1le(ops, ptr));
  215. return;
  216. case L2_OP_ALLOC_NAMESPACE:
  217. printf("ALLOC_NAMESPACE\n");
  218. return;
  219. case L2_OP_ALLOC_FUNCTION_U4:
  220. printf("ALLOC_FUNCTION %u\n", read_u4le(ops, ptr));
  221. return;
  222. case L2_OP_ALLOC_FUNCTION_U1:
  223. printf("ALLOC_FUNCTION %u\n", read_u1le(ops, ptr));
  224. return;
  225. case L2_OP_NAMESPACE_SET_U4:
  226. printf("NAMESPACE_SET %u\n", read_u4le(ops, ptr));
  227. return;
  228. case L2_OP_NAMESPACE_SET_U1:
  229. printf("NAMESPACE_SET %u\n", read_u1le(ops, ptr));
  230. return;
  231. case L2_OP_NAMESPACE_LOOKUP_U4:
  232. printf("NAMESPACE_LOOKUP %u\n", read_u4le(ops, ptr));
  233. return;
  234. case L2_OP_NAMESPACE_LOOKUP_U1:
  235. printf("NAMESPACE_LOOKUP %u\n", read_u1le(ops, ptr));
  236. return;
  237. case L2_OP_ARRAY_LOOKUP_U4:
  238. printf("ARRAY_LOOKUP %u\n", read_u4le(ops, ptr));
  239. return;
  240. case L2_OP_ARRAY_LOOKUP_U1:
  241. printf("ARRAY_LOOKUP %u\n", read_u1le(ops, ptr));
  242. return;
  243. case L2_OP_ARRAY_SET_U4:
  244. printf("ARRAY_SET %u\n", read_u4le(ops, ptr));
  245. return;
  246. case L2_OP_ARRAY_SET_U1:
  247. printf("ARRAY_SET %u\n", read_u1le(ops, ptr));
  248. return;
  249. case L2_OP_DYNAMIC_LOOKUP:
  250. printf("DYNAMIC_LOOKUP\n");
  251. return;
  252. case L2_OP_DYNAMIC_SET:
  253. printf("DYNAMIC_SET\n");
  254. return;
  255. case L2_OP_HALT:
  256. printf("HALT\n");
  257. return;
  258. }
  259. printf("? 0x%02x\n", opcode);
  260. }
  261. void l2_vm_print_bytecode(unsigned char *ops, size_t opcount) {
  262. size_t ptr = 0;
  263. while (ptr < opcount) {
  264. printf("%04zu ", ptr);
  265. l2_vm_print_op(ops, opcount, &ptr);
  266. }
  267. }