Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

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