test_opcodes.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. #include <assert.h>
  2. #include <stdint.h>
  3. #include <stdio.h>
  4. #include "fvm.h"
  5. char code[4096];
  6. char stack[4096];
  7. int code_cursor = 0;
  8. struct fvm vm;
  9. static void reset() {
  10. vm.pc = (int64_t)code;
  11. vm.sp = (int64_t)stack + 4095;
  12. vm.bp = (int64_t)stack + 4095;
  13. vm.rv = 0;
  14. code_cursor = 0;
  15. }
  16. static void emit_op(enum fvm_op op) {
  17. code[code_cursor] = (char)op;
  18. code_cursor++;
  19. }
  20. static void emit_num(int64_t x) {
  21. *(int64_t*)(code + code_cursor) = x;
  22. code_cursor += 8;
  23. }
  24. #define OP(op) emit_op(FVM_OP_##op)
  25. #define IMM(num) \
  26. emit_op(FVM_OP_IMM); \
  27. emit_num(num);
  28. #define EXPECT(n) assert(fvm_execute(&vm) == (n));
  29. #define BEGIN(name) void test_##name () { printf("[TEST] " #name "\n");
  30. #define END(name) printf("[PASS] " #name "\n"); }
  31. BEGIN(imm)
  32. reset();
  33. IMM(3);
  34. OP(SRV);OP(EXIT);
  35. EXPECT(3);
  36. END(imm)
  37. BEGIN(sp)
  38. reset();
  39. int64_t expected = vm.sp;
  40. OP(SP);
  41. OP(SRV);OP(EXIT);
  42. EXPECT(expected);
  43. END(sp)
  44. BEGIN(ssp)
  45. int64_t expected = vm.sp + 4;
  46. //===
  47. reset();
  48. OP(SP);
  49. IMM(4);
  50. OP(ADD);
  51. OP(SSP);
  52. OP(SP);
  53. OP(SRV);OP(EXIT);
  54. EXPECT(expected);
  55. END(ssp)
  56. //===
  57. BEGIN(ld)
  58. int64_t e1 = 0xfefefefefefefefe;
  59. int32_t e2 = 0xfdfdfdfd;
  60. int16_t e3 = 0xfcfc;
  61. int8_t e4 = 0xfb;
  62. //===
  63. reset();
  64. IMM((int64_t)(&e1));
  65. OP(LD);
  66. OP(SRV);OP(EXIT);
  67. EXPECT(e1);
  68. //===
  69. reset();
  70. IMM((int64_t)(&e2));
  71. OP(LD32);
  72. OP(SRV);OP(EXIT);
  73. EXPECT(e2);
  74. //===
  75. reset();
  76. IMM((int64_t)(&e3));
  77. OP(LD16);
  78. OP(SRV);OP(EXIT);
  79. EXPECT(e3);
  80. //===
  81. reset();
  82. IMM((int64_t)(&e4));
  83. OP(LD8);
  84. OP(SRV);OP(EXIT);
  85. EXPECT(e4);
  86. END(ld)
  87. BEGIN(st)
  88. int64_t loc = 0;
  89. int32_t loc32 = 0;
  90. int16_t loc16 = 0;
  91. int8_t loc8 = 0;
  92. int64_t e1 = 0xfefefefefefefefe;
  93. int32_t e2 = 0xfdfdfdfd;
  94. int16_t e3 = 0xfcfc;
  95. int8_t e4 = 0xfb;
  96. //===
  97. reset();
  98. IMM(42);
  99. IMM(e1);
  100. IMM((int64_t)(&loc));
  101. OP(ST);
  102. OP(SRV);OP(EXIT);
  103. assert(fvm_execute(&vm) == 42);
  104. assert(loc == e1);
  105. //===
  106. reset();
  107. IMM(42);
  108. IMM(e2);
  109. IMM((int64_t)(&loc32));
  110. OP(ST32);
  111. OP(SRV);OP(EXIT);
  112. assert(fvm_execute(&vm) == 42);
  113. assert(loc32 == e2);
  114. //===
  115. reset();
  116. IMM(42);
  117. IMM(e3);
  118. IMM((int64_t)(&loc16));
  119. OP(ST16);
  120. OP(SRV);OP(EXIT);
  121. assert(fvm_execute(&vm) == 42);
  122. assert(loc16 == e3);
  123. //===
  124. reset();
  125. IMM(42);
  126. IMM(e4);
  127. IMM((int64_t)(&loc8));
  128. OP(ST8);
  129. OP(SRV);OP(EXIT);
  130. assert(fvm_execute(&vm) == 42);
  131. assert(loc8 == e4);
  132. END(st)
  133. BEGIN(dup)
  134. reset();
  135. IMM(42);
  136. OP(DUP);
  137. OP(SRV);OP(EXIT);
  138. EXPECT(42);
  139. //===
  140. reset();
  141. IMM(42);
  142. OP(DUP);
  143. OP(POP);
  144. OP(SRV);OP(EXIT);
  145. EXPECT(42);
  146. //===
  147. reset();
  148. IMM(3);
  149. IMM(42);
  150. OP(DUP);
  151. OP(POP);
  152. OP(POP);
  153. OP(SRV);OP(EXIT);
  154. EXPECT(3);
  155. END(dup)
  156. BEGIN(swap)
  157. reset();
  158. IMM(1);
  159. IMM(2);
  160. OP(SWAP);
  161. OP(SRV);OP(EXIT);
  162. EXPECT(1);
  163. //===
  164. reset();
  165. IMM(1);
  166. IMM(2);
  167. OP(SWAP);
  168. OP(POP);
  169. OP(SRV);OP(EXIT);
  170. EXPECT(2);
  171. END(swap)
  172. BEGIN(over)
  173. reset();
  174. IMM(1);
  175. IMM(2);
  176. OP(OVER);
  177. OP(SRV);OP(EXIT);
  178. EXPECT(1);
  179. //===
  180. reset();
  181. IMM(1);
  182. IMM(2);
  183. OP(OVER);
  184. OP(POP);
  185. OP(SRV);OP(EXIT);
  186. EXPECT(2);
  187. //===
  188. reset();
  189. IMM(1);
  190. IMM(2);
  191. OP(OVER);
  192. OP(POP);
  193. OP(POP);
  194. OP(SRV);OP(EXIT);
  195. EXPECT(1);
  196. END(over)
  197. BEGIN(add)
  198. reset();
  199. IMM(1);
  200. IMM(2);
  201. OP(ADD);
  202. OP(SRV);OP(EXIT);
  203. EXPECT(3);
  204. //===
  205. reset();
  206. IMM(42);
  207. IMM(1);
  208. IMM(2);
  209. OP(ADD);
  210. OP(POP);
  211. OP(SRV);OP(EXIT);
  212. EXPECT(42);
  213. END(add)
  214. BEGIN(sub)
  215. reset();
  216. IMM(1);
  217. IMM(2);
  218. OP(SUB);
  219. OP(SRV);OP(EXIT);
  220. EXPECT(-1);
  221. //===
  222. reset();
  223. IMM(42);
  224. IMM(1);
  225. IMM(2);
  226. OP(ADD);
  227. OP(POP);
  228. OP(SRV);OP(EXIT);
  229. EXPECT(42);
  230. END(sub)
  231. BEGIN(jmp)
  232. reset();
  233. IMM((int64_t)code + 14);
  234. OP(JMP);
  235. OP(SRV);OP(EXIT);
  236. OP(SRV);OP(EXIT);
  237. IMM(42);
  238. OP(SRV);OP(EXIT);
  239. EXPECT(42);
  240. END(jmp)
  241. BEGIN(jz)
  242. reset();
  243. IMM(0);
  244. IMM((int64_t)code + 32);
  245. OP(JZ);
  246. IMM(21);
  247. OP(SRV);OP(EXIT);
  248. OP(SRV);OP(EXIT);
  249. IMM(42);
  250. OP(SRV);OP(EXIT);
  251. EXPECT(42);
  252. //===
  253. reset();
  254. IMM(-1);
  255. IMM((int64_t)code + 30);
  256. OP(JZ);
  257. IMM(21);
  258. OP(SRV);OP(EXIT);
  259. OP(SRV);OP(EXIT);
  260. IMM(42);
  261. OP(SRV);OP(EXIT);
  262. EXPECT(21);
  263. END(jz)
  264. BEGIN(jnz)
  265. reset();
  266. IMM(0);
  267. IMM((int64_t)code + 32);
  268. OP(JNZ);
  269. IMM(21);
  270. OP(SRV);OP(EXIT);
  271. OP(SRV);OP(EXIT);
  272. IMM(42);
  273. OP(SRV);OP(EXIT);
  274. EXPECT(21);
  275. //===
  276. reset();
  277. IMM(-1);
  278. IMM((int64_t)code + 32);
  279. OP(JNZ);
  280. IMM(21);
  281. OP(SRV);OP(EXIT);
  282. OP(SRV);OP(EXIT);
  283. IMM(42);
  284. OP(SRV);OP(EXIT);
  285. EXPECT(42);
  286. END(jnz)
  287. int main()
  288. {
  289. test_imm();
  290. test_sp();
  291. test_ssp();
  292. test_ld();
  293. test_st();
  294. test_dup();
  295. test_swap();
  296. test_over();
  297. test_add();
  298. test_sub();
  299. test_jmp();
  300. test_jz();
  301. test_jnz();
  302. return 0;
  303. }