#include "test_common.h" extern "C" { extern char log_buff[LOG_BUFF_MAX][LOG_SIZE]; /* test head */ VMParameters* pikaVM_runAsmWithPars(PikaObj* self, VMParameters* locals, VMParameters* globals, char* pikaAsm); char* Parser_LineToAsm(Args* buffs, char* line, Stack* blockStack); char* Parser_linesToAsm(Args* outBuffs, char* multiLine); int32_t __clearInvokeStackes(VMParameters* locals); int32_t pikaVM_runAsmLine(PikaObj* self, VMParameters* locals, VMParameters* globals, char* pikaAsm, int32_t lineAddr); } TEST(VM, num1) { char* line = "1"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); EXPECT_STREQ(log_buff[1], "BEGIN\r\n"); EXPECT_STREQ(log_buff[0], "1\r\n"); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_1) { char* line = "a = 1"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); pikaVM_runAsm(self, pikaAsm); ASSERT_EQ(args_getInt(self->list, "a"), 1); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_1d1) { char* line = "a = 1.1"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); ASSERT_FLOAT_EQ(args_getFloat(globals->list, "a"), 1.1); obj_deinit(self); // obj_deinit(globals); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, str_xy) { char* line = "a = 'xy'"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); ASSERT_STREQ(args_getStr(globals->list, "a"), "xy"); obj_deinit(self); // obj_deinit(globals); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, str_xy_space) { char* line = "a = 'xy '"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); ASSERT_STREQ(args_getStr(globals->list, "a"), "xy "); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, ref_a_b) { PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); Args* buffs = New_strBuff(); pikaVM_runAsm(self, Parser_LineToAsm(buffs, "a = 'xy '", NULL)); args_deinit(buffs); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, Run_add) { PikaObj* self = newRootObj("root", New_PikaMath_Operator); Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, "a = plusInt(1,2)", NULL); __platform_printf("%s", pikaAsm); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); args_deinit(buffs); int a = args_getInt(globals->list, "a"); ASSERT_EQ(a, 3); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, Run_add_multy) { PikaObj* self = newRootObj("root", New_PikaMath_Operator); Args* buffs = New_strBuff(); pikaVM_runAsm(self, Parser_LineToAsm(buffs, "b = 2", NULL)); args_deinit(buffs); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, Run_add_1_2_3) { PikaObj* self = newRootObj("root", New_PikaMath_Operator); Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, "a = plusInt(1, plusInt(2,3) )", NULL); __platform_printf("%s", pikaAsm); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); args_deinit(buffs); int a = args_getInt(globals->list, "a"); ASSERT_EQ(a, 6); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } extern PikaMemInfo pikaMemInfo; TEST(VM, WHILE) { pikaMemInfo.heapUsedMax = 0; Args* buffs = New_strBuff(); char* lines =(char *) "a = 1\n" "b = 0\n" "while a:\n" " b = 1\n" " a = 0\n" "\n"; printf("%s", lines); char* pikaAsm = Parser_linesToAsm(buffs, lines); printf("%s", pikaAsm); pikaMemInfo.heapUsedMax = 0; PikaObj* self = New_TinyObj(NULL); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); EXPECT_EQ(args_getInt(globals->list, "a"), 0); EXPECT_EQ(args_getInt(globals->list, "b"), 1); // obj_deinit(globals); args_deinit(buffs); obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_1_1) { char* line = "a = 1 + 1"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int res = args_getInt(globals->list, "a"); obj_deinit(self); args_deinit(buffs); ASSERT_EQ(res, 2); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_1_1d1) { char* line = "a = 1 + 1.1"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); float res = args_getFloat(globals->list, "a"); obj_deinit(self); // obj_deinit(globals); args_deinit(buffs); ASSERT_FLOAT_EQ(res, 2.1); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_jjcc) { char* line = "a = (1 + 1.1) * 3 - 2 /4.0"; printf("%s\r\n", line); Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); float res = args_getFloat(globals->list, "a"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(res, 5.8); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, while_a_1to10) { char* line = "a = 0\n" "while a < 10:\n" " a += 1\n" " print(a)\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int res = args_getInt(globals->list, "a"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(res, 10); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, mem_x) { char* line = "mem = PikaStdLib.MemChecker()\n" "mem.x = 1\n" "print(mem.x)\n" "mem.x = 2\n" "print(mem.x)\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("", New_PikaMain); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int res = obj_getInt(globals, "mem.x"); EXPECT_EQ(2, res); obj_deinit(self); // obj_deinit(globals); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RET_instruct) { char* pikaAsm = "B1\n" "0 NUM 13\n" "0 RET \n" "0 NUM 2\n" "0 RET \n" "B0\n"; PikaObj* self = New_TinyObj(NULL); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); Arg* return_arg = args_getArg(globals->list, "return"); int num = arg_getInt(return_arg); EXPECT_EQ(num, 13); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_DEF) { char* pikaAsm = "B0\n" "0 DEF test()\n" "0 JMP 1\n" "B1\n" "0 NUM 1\n" "0 RET \n" "B0\n" "0 RUN test\n" "0 OUT a\n"; PikaObj* self = New_TinyObj(NULL); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int num = obj_getInt(globals, "a"); EXPECT_EQ(num, 1); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_global) { char* pikaAsm = "B0\n" "0 NUM 1\n" "0 OUT a\n" "B0\n" "0 DEF test()\n" "0 JMP 1\n" "B1\n" "0 REF a\n" "0 RET \n" "B0\n" "0 RUN test\n" "0 OUT b\n"; PikaObj* self = New_TinyObj(NULL); pikaVM_runAsm(self, pikaAsm); int a = obj_getInt(self, "a"); int b = obj_getInt(self, "b"); EXPECT_EQ(a, 1); EXPECT_EQ(b, 1); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_local_b) { char* pikaAsm = "B0\n" "0 NUM 1\n" "0 OUT a\n" "B0\n" "0 DEF test()\n" "0 JMP 1\n" "B1\n" "0 REF a\n" "0 OUT b\n" "1 REF b\n" "1 REF a\n" "0 OPT +\n" "0 RET \n" "B0\n" "0 RUN test\n" "0 OUT c\n"; PikaObj* self = newRootObj("", New_BaseObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int a = obj_getInt(globals, "a"); int b = obj_getInt(globals, "b"); int c = obj_getInt(globals, "c"); EXPECT_EQ(a, 1); /* b is local, should not be exist in globals */ EXPECT_EQ(b, -999999999); EXPECT_EQ(c, 2); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_DEF_add) { char* pikaAsm = "B0\n" "0 DEF add(a,b)\n" "0 JMP 1\n" "B1\n" "1 REF b\n" "1 REF a\n" "0 OPT +\n" "0 RET \n" "B0\n" "1 NUM 1\n" "1 NUM 2\n" "0 RUN add\n" "0 OUT c\n"; PikaObj* self = newRootObj("", New_BaseObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int a = obj_getInt(globals, "a"); int b = obj_getInt(globals, "b"); int c = obj_getInt(globals, "c"); /* a is local, should not be exist in globals */ EXPECT_EQ(a, -999999999); /* b is local, should not be exist in globals */ EXPECT_EQ(b, -999999999); EXPECT_EQ(c, 3); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, run_def_add) { char* line = "def add(a, b):\n" " return a + b\n" "c = add(1,3)\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int c = args_getInt(globals->list, "c"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(c, 4); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, equ) { char* line = "if -1 == -1:\n" " a = 1\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int c = args_getInt(globals->list, "a"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(c, 1); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, if_elif) { char* line = "a = 2\n" "b = 0\n" "if a > 1:\n" " b = 1\n" "elif a > 0:\n" " b = 2\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int b = args_getInt(globals->list, "b"); obj_deinit(self); args_deinit(buffs); ASSERT_FLOAT_EQ(b, 1); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, if_else) { char* line = "a = 0\n" "b = 0\n" "if a > 1:\n" " b = 1\n" "else:\n" " b = 2\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int b = args_getInt(globals->list, "b"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(b, 2); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, if_else_) { char* line = "a = 2\n" "b = 0\n" "if a > 1:\n" " b = 1\n" "else:\n" " b = 2\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); VMParameters* globals = pikaVM_runAsm(self, pikaAsm); int b = args_getInt(globals->list, "b"); obj_deinit(self); args_deinit(buffs); // obj_deinit(globals); ASSERT_FLOAT_EQ(b, 1); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, EST) { char* pikaAsm = "B0\n" "0 EST a\n" "0 OUT b\n"; PikaObj* self = newRootObj("", New_BaseObj); pikaVM_runAsm(self, pikaAsm); int a = obj_getInt(self, "a"); int b = obj_getInt(self, "b"); /* a is local, should not be exist in globals */ EXPECT_EQ(a, -999999999); /* b is local, should not be exist in globals */ EXPECT_EQ(b, 0); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_AS) { char* pikaAsm = "B0\n" "0 RUN TinyObj\n" "0 OUT as\n" "B0\n" "0 RAS as\n" "B0\n" "0 NUM 1\n" "0 OUT x\n" "B0\n" "0 RAS $origin\n" "B0\n" "0 NUM 2\n" "0 OUT x\n"; PikaObj* self = newRootObj("", New_PikaMain); pikaVM_runAsm(self, pikaAsm); PikaObj* as = obj_getObj(self, "as"); int x_as_ = obj_getInt(as, "x"); int x_as = obj_getInt(self, "as.x"); int x_origin = obj_getInt(self, "x"); EXPECT_EQ(x_as_, 1); EXPECT_EQ(x_as, 1); EXPECT_EQ(x_origin, 2); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_NEW) { char* pikaAsm = "B0\n" "0 RUN TinyObj\n" "0 OUT newObj\n" "0 NUM 1\n" "0 OUT newObj.x\n" "B0\n" "0 NEW newObj\n" "0 OUT outObj\n"; PikaObj* self = newRootObj("", New_PikaMain); pikaVM_runAsm(self, pikaAsm); Arg* newObj = obj_getArg(self, "newObj"); Arg* outObj = obj_getArg(self, "outObj"); void* newObj_ptr = arg_getPtr(newObj); void* outObj_ptr = arg_getPtr(outObj); EXPECT_EQ(newObj_ptr, outObj_ptr); int x = obj_getInt(self, "outObj.x"); ArgType newObj_type = arg_getType(newObj); ArgType outObj_type = arg_getType(outObj); EXPECT_EQ(x, 1); EXPECT_EQ(newObj_type, ARG_TYPE_OBJECT); EXPECT_EQ(outObj_type, ARG_TYPE_OBJECT); obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, RUN_DEF_NEW) { char* pikaAsm = "B0\n" "0 DEF testnew()\n" "0 JMP 1\n" "B1\n" "0 RUN TinyObj\n" "0 OUT newObj\n" "B1\n" "0 NUM 1\n" "0 OUT newObj.x\n" "B1\n" "0 NEW newObj\n" "0 RET \n" "B1\n" "0 RET \n" "B0\n" "0 RUN testnew\n" "0 OUT outobj\n"; PikaObj* self = newRootObj("", New_PikaMain); pikaVM_runAsm(self, pikaAsm); /* assert */ PikaObj* outobj = obj_getObj(self, "outobj"); int x = obj_getInt(outobj, "x"); EXPECT_EQ(x, 1); /* deinit */ obj_deinit(self); // obj_deinit(globals); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, class_x_1) { char* line = "class Test(TinyObj):\n" " x = 1\n" "test = Test()\n" "print('test.x: ' + str(test.x))\n" "\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("", New_PikaMain); pikaVM_runAsm(self, pikaAsm); PikaObj* test = obj_getObj(self, "test"); Arg* test_arg = obj_getArg(self, "test"); ArgType test_arg_type = arg_getType(test_arg); EXPECT_EQ(test_arg_type, ARG_TYPE_OBJECT); int x = obj_getInt(test, "x"); EXPECT_EQ(x, 1); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, nag_a) { char* line = "a = 1\n" "print(-a)\n" "b = 0.5\n" "print(-b)\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("", New_PikaMain); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); /* assert */ EXPECT_STREQ(log_buff[2], "BEGIN\r\n"); EXPECT_STREQ(log_buff[1], "-1\r\n"); EXPECT_STREQ(log_buff[0], "-0.500000\r\n"); /* deinit */ obj_deinit(self); args_deinit(buffs); /* check mem */ EXPECT_EQ(pikaMemNow(), 0); } TEST(InstructUnit, base) { __platform_printf("BEGIN\r\n"); InstructUnit ins_unit; instructUnit_init(&ins_unit); instructUnit_setBlockDeepth(&ins_unit, 2); instructUnit_setIsNewLine(&ins_unit, 1); instructUnit_setInvokeDeepth(&ins_unit, 3); instructUnit_setInstruct(&ins_unit, (Instruct)4); instructUnit_setConstPoolIndex(&ins_unit, 12); EXPECT_EQ(instructUnit_getBlockDeepth(&ins_unit), 2); EXPECT_EQ(instructUnit_getIsNewLine(&ins_unit), 1); EXPECT_EQ(instructUnit_getInvokeDeepth(&ins_unit), 3); EXPECT_EQ(instructUnit_getInstruct(&ins_unit), 4); EXPECT_EQ(instructUnit_getConstPoolIndex(&ins_unit), 12); instructUnit_print(&ins_unit); EXPECT_STREQ(log_buff[2], "BEGIN\r\n"); EXPECT_STREQ(log_buff[1], "B2\r\n"); EXPECT_STREQ(log_buff[0], "3 OUT #12\r\n"); EXPECT_EQ(pikaMemNow(), 0); } TEST(ConstPool, get) { __platform_printf("BEGIN\r\n"); ConstPool cp; constPool_init(&cp); char* next_content; uint16_t next_offset; next_offset = constPool_getLastOffset(&cp); constPool_append(&cp, "hello"); next_content = constPool_getByOffset(&cp, next_offset); EXPECT_STREQ(next_content, "hello"); next_offset = constPool_getLastOffset(&cp); constPool_append(&cp, "world"); next_content = constPool_getByOffset(&cp, next_offset); EXPECT_STREQ(next_content, "world"); char* first = constPool_getNow(&cp); char* second = constPool_getNext(&cp); char* third = constPool_getNext(&cp); char* forth = constPool_getNext(&cp); EXPECT_STREQ(first, ""); EXPECT_STREQ(second, "hello"); EXPECT_STREQ(third, "world"); EXPECT_EQ((uintptr_t)forth, (uintptr_t)NULL); EXPECT_STREQ(constPool_getByIndex(&cp, 0), ""); EXPECT_STREQ(constPool_getByIndex(&cp, 1), "hello"); EXPECT_STREQ(constPool_getByIndex(&cp, 2), "world"); EXPECT_EQ((uintptr_t)constPool_getByIndex(&cp, 3), (uintptr_t)NULL); constPool_print(&cp); EXPECT_STREQ(log_buff[2], "BEGIN\r\n"); EXPECT_STREQ(log_buff[1], "1: hello\r\n"); EXPECT_STREQ(log_buff[0], "7: world\r\n"); constPool_deinit(&cp); EXPECT_EQ(pikaMemNow(), 0); } TEST(InstructArray, set) { __platform_printf("BEGIN\r\n"); InstructUnit ins_unit; instructUnit_init(&ins_unit); instructUnit_setBlockDeepth(&ins_unit, 2); instructUnit_setIsNewLine(&ins_unit, 1); instructUnit_setInvokeDeepth(&ins_unit, 3); instructUnit_setInstruct(&ins_unit, (Instruct)4); instructUnit_setConstPoolIndex(&ins_unit, 12); InstructArray ins_array; instructArray_init(&ins_array); instructArray_append(&ins_array, &ins_unit); instructArray_print(&ins_array); instructArray_deinit(&ins_array); EXPECT_STREQ(log_buff[2], "BEGIN\r\n"); EXPECT_STREQ(log_buff[1], "B2\r\n"); EXPECT_STREQ(log_buff[0], "3 OUT #12\r\n"); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, bytecode_jjcc) { char* line = "a = (1 + 1.1) * 3 - 2 /4.0"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); ByteCodeFrame byte_frame; byteCodeFrame_init(&byte_frame); byteCodeFrame_appendFromAsm(&byte_frame, pikaAsm); byteCodeFrame_print(&byte_frame); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); // pikaVM_runAsm(self, pikaAsm); pikaVM_runByteCodeFrame(self, &byte_frame); float res = obj_getFloat(self, "a"); obj_deinit(self); args_deinit(buffs); byteCodeFrame_deinit(&byte_frame); ASSERT_FLOAT_EQ(res, 5.8); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, WHILE_byte) { pikaMemInfo.heapUsedMax = 0; Args* buffs = New_strBuff(); char* lines =(char *) "a = 1\n" "b = 0\n" "while a:\n" " b = 1\n" " a = 0\n" "\n"; printf("%s", lines); char* pikaAsm = Parser_linesToAsm(buffs, lines); printf("%s", pikaAsm); pikaMemInfo.heapUsedMax = 0; PikaObj* self = New_TinyObj(NULL); pikaVM_run(self, lines); EXPECT_EQ(obj_getInt(self, "a"), 0); EXPECT_EQ(obj_getInt(self, "b"), 1); args_deinit(buffs); obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, for_break_byte) { pikaMemInfo.heapUsedMax = 0; Args* buffs = New_strBuff(); char* lines =(char *) "a = 0\n" "for i in range(0, 10):\n" " if i == 5:\n" " break\n" " a = a + i\n" "\n"; printf("%s", lines); char* pikaAsm = Parser_linesToAsm(buffs, lines); printf("%s", pikaAsm); pikaMemInfo.heapUsedMax = 0; PikaObj* self = newRootObj("pikaMain", New_PikaMain); pikaVM_run(self, lines); /* assert */ int a = obj_getInt(self, "a"); EXPECT_EQ(a, 10); /* deinit */ args_deinit(buffs); obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, load_static_bytes) { const uint8_t bytes[] = { 0x08, 0x02, /* instruct array size */ 0x00, 0x82, 0x01, 0x00, 0x00, 0x04, 0x0d, 0x00, 0x00, 0x83, 0x11, 0x00, 0x00, 0x04, 0x16, 0x00, 0x00, 0x85, 0x1b, 0x00, 0x00, 0x04, 0x1d, 0x00, 0x20, 0x85, 0x1b, 0x00, 0x20, 0x01, 0x0d, 0x00, 0x10, 0x02, 0x28, 0x00, 0x00, 0x02, 0x2e, 0x00, 0x00, 0x04, 0x33, 0x00, 0x00, 0x01, 0x37, 0x00, 0x00, 0x01, 0x3b, 0x00, 0x00, 0x01, 0x3f, 0x00, 0x00, 0x04, 0x43, 0x00, 0x00, 0x04, 0x4a, 0x00, 0x00, 0x04, 0x51, 0x00, 0x00, 0x82, 0x58, 0x00, 0x00, 0x04, 0x65, 0x00, 0x00, 0x0d, 0x65, 0x00, 0x00, 0x07, 0x67, 0x00, 0x11, 0x81, 0x0d, 0x00, 0x11, 0x05, 0x1b, 0x00, 0x01, 0x08, 0x69, 0x00, 0x01, 0x07, 0x6c, 0x00, 0x02, 0x8e, 0x6c, 0x00, 0x11, 0x81, 0x1d, 0x00, 0x11, 0x05, 0x1b, 0x00, 0x01, 0x08, 0x69, 0x00, 0x01, 0x07, 0x6c, 0x00, 0x12, 0x81, 0x6e, 0x00, 0x12, 0x01, 0x65, 0x00, 0x02, 0x02, 0x76, 0x00, 0x02, 0x04, 0x16, 0x00, 0x02, 0x85, 0x6c, 0x00, 0x02, 0x04, 0x1d, 0x00, 0x01, 0x8b, 0x6c, 0x00, 0x11, 0x01, 0x1d, 0x00, 0x11, 0x05, 0x6c, 0x00, 0x01, 0x08, 0x69, 0x00, 0x01, 0x07, 0x6c, 0x00, 0x12, 0x81, 0x16, 0x00, 0x12, 0x03, 0x7e, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x13, 0x81, 0x6e, 0x00, 0x13, 0x01, 0x65, 0x00, 0x03, 0x02, 0x76, 0x00, 0x03, 0x04, 0x85, 0x00, 0x03, 0x82, 0x85, 0x00, 0x03, 0x85, 0x1b, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x02, 0x8b, 0x6c, 0x00, 0x12, 0x01, 0x16, 0x00, 0x12, 0x03, 0x8a, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x13, 0x81, 0x6e, 0x00, 0x13, 0x01, 0x65, 0x00, 0x03, 0x02, 0x76, 0x00, 0x03, 0x04, 0x8a, 0x00, 0x03, 0x85, 0x67, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x02, 0x8b, 0x6c, 0x00, 0x12, 0x01, 0x16, 0x00, 0x12, 0x03, 0x8f, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x13, 0x81, 0x6e, 0x00, 0x13, 0x01, 0x65, 0x00, 0x03, 0x02, 0x76, 0x00, 0x03, 0x04, 0x8f, 0x00, 0x03, 0x85, 0x67, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x01, 0x8b, 0x6c, 0x00, 0x11, 0x01, 0x1d, 0x00, 0x11, 0x05, 0x67, 0x00, 0x01, 0x08, 0x69, 0x00, 0x01, 0x07, 0x6c, 0x00, 0x12, 0x81, 0x16, 0x00, 0x12, 0x03, 0x8a, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x03, 0x82, 0x8a, 0x00, 0x03, 0x07, 0x6c, 0x00, 0x14, 0x81, 0x6e, 0x00, 0x14, 0x01, 0x65, 0x00, 0x04, 0x02, 0x76, 0x00, 0x04, 0x04, 0x85, 0x00, 0x04, 0x82, 0x85, 0x00, 0x03, 0x85, 0x1b, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x02, 0x8b, 0x6c, 0x00, 0x12, 0x01, 0x16, 0x00, 0x12, 0x03, 0x8f, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x13, 0x81, 0x6e, 0x00, 0x13, 0x01, 0x65, 0x00, 0x03, 0x02, 0x76, 0x00, 0x03, 0x04, 0x85, 0x00, 0x03, 0x85, 0x99, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x01, 0x8b, 0x6c, 0x00, 0x11, 0x01, 0x1d, 0x00, 0x11, 0x05, 0x99, 0x00, 0x01, 0x08, 0x69, 0x00, 0x01, 0x07, 0x6c, 0x00, 0x12, 0x81, 0x16, 0x00, 0x12, 0x03, 0x8f, 0x00, 0x02, 0x08, 0x69, 0x00, 0x02, 0x07, 0x6c, 0x00, 0x13, 0x81, 0x9b, 0x00, 0x23, 0x01, 0x6e, 0x00, 0x23, 0x01, 0x65, 0x00, 0x13, 0x02, 0x76, 0x00, 0x03, 0x08, 0xa2, 0x00, 0x03, 0x07, 0x6c, 0x00, 0x04, 0x82, 0x85, 0x00, 0x14, 0x81, 0x6e, 0x00, 0x14, 0x01, 0x65, 0x00, 0x24, 0x01, 0x9b, 0x00, 0x24, 0x01, 0x8f, 0x00, 0x14, 0x08, 0xa4, 0x00, 0x14, 0x03, 0x6e, 0x00, 0x04, 0x02, 0xa6, 0x00, 0x03, 0x85, 0x1b, 0x00, 0x03, 0x04, 0x1d, 0x00, 0x00, 0x86, 0xae, 0x00, 0x00, 0x8c, 0x33, 0x00, /* instruct array */ 0xb1, 0x00, /* const pool size */ 0x00, 0x5f, 0x5f, 0x63, 0x61, 0x6c, 0x6c, 0x73, 0x2e, 0x6c, 0x65, 0x6e, 0x00, 0x6c, 0x65, 0x6e, 0x00, 0x6e, 0x6f, 0x6e, 0x65, 0x00, 0x6d, 0x6f, 0x64, 0x65, 0x00, 0x30, 0x00, 0x69, 0x6e, 0x66, 0x6f, 0x5f, 0x69, 0x6e, 0x64, 0x65, 0x78, 0x00, 0x72, 0x61, 0x6e, 0x67, 0x65, 0x00, 0x69, 0x74, 0x65, 0x72, 0x00, 0x5f, 0x6c, 0x30, 0x00, 0x5f, 0x72, 0x31, 0x00, 0x5f, 0x72, 0x32, 0x00, 0x5f, 0x72, 0x33, 0x00, 0x5f, 0x6c, 0x30, 0x2e, 0x61, 0x31, 0x00, 0x5f, 0x6c, 0x30, 0x2e, 0x61, 0x32, 0x00, 0x5f, 0x6c, 0x30, 0x2e, 0x61, 0x33, 0x00, 0x5f, 0x6c, 0x30, 0x2e, 0x5f, 0x5f, 0x6e, 0x65, 0x78, 0x74, 0x5f, 0x5f, 0x00, 0x69, 0x00, 0x32, 0x00, 0x3d, 0x3d, 0x00, 0x31, 0x00, 0x5f, 0x5f, 0x63, 0x61, 0x6c, 0x6c, 0x73, 0x00, 0x5f, 0x5f, 0x67, 0x65, 0x74, 0x5f, 0x5f, 0x00, 0x61, 0x6c, 0x77, 0x61, 0x79, 0x73, 0x00, 0x74, 0x6f, 0x64, 0x6f, 0x00, 0x77, 0x68, 0x65, 0x6e, 0x00, 0x70, 0x65, 0x72, 0x69, 0x6f, 0x64, 0x5f, 0x6d, 0x73, 0x00, 0x33, 0x00, 0x5f, 0x5f, 0x74, 0x69, 0x63, 0x6b, 0x00, 0x3e, 0x00, 0x2b, 0x00, 0x5f, 0x5f, 0x73, 0x65, 0x74, 0x5f, 0x5f, 0x00, 0x2d, 0x31, 0x00, /* const pool */ }; ByteCodeFrame bytecode_frame = {.const_pool = { .arg_buff = NULL, .content_offset_now = 0, .size = 0, .content_start = NULL, .output_redirect_fun = NULL, .output_f = NULL, }, .instruct_array = { .arg_buff = NULL, .content_offset_now = 0, .size = 0, .content_start = NULL, .output_redirect_fun = NULL, .output_f = NULL, }}; byteCodeFrame_loadByteCode(&bytecode_frame, (uint8_t*)bytes); byteCodeFrame_print(&bytecode_frame); EXPECT_EQ(instructArray_getSize(&(bytecode_frame.instruct_array)), 520); EXPECT_EQ(bytecode_frame.const_pool.size, 177); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, multi_jian) { pikaMemInfo.heapUsedMax = 0; Args* buffs = New_strBuff(); char* lines = "a = (3-4) - 4\n"; printf("%s", lines); char* pikaAsm = Parser_linesToAsm(buffs, lines); printf("%s", pikaAsm); pikaMemInfo.heapUsedMax = 0; PikaObj* self = newRootObj("pikaMain", New_PikaMain); pikaVM_run(self, lines); /* assert */ int a = obj_getInt(self, "a"); EXPECT_EQ(a, -5); /* deinit */ args_deinit(buffs); obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, _3_p_3) { char* line = "-3+3"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); EXPECT_STREQ(log_buff[1], "BEGIN\r\n"); EXPECT_STREQ(log_buff[0], "0\r\n"); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, _3_3) { char* line = "-3-3"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); EXPECT_STREQ(log_buff[1], "BEGIN\r\n"); EXPECT_STREQ(log_buff[0], "-6\r\n"); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, a_3) { char* line = "a-3"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); EXPECT_STREQ(log_buff[3], "BEGIN\r\n"); EXPECT_STREQ(log_buff[2], "NameError: name 'a' is not defined\r\n"); obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, hex_str) { char* line = "a = '\\x33\\x35'"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_LineToAsm(buffs, line, NULL); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); /* collect */ char* a = obj_getStr(self, "a"); /* assert */ EXPECT_STREQ(a, "35"); /* deinit */ obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, hex_bytes) { char* line = "a = b'\\x03\\x05'\n" "a\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); /* collect */ /* assert */ EXPECT_STREQ(log_buff[2], "\\x03"); EXPECT_STREQ(log_buff[1], "\\x05"); /* deinit */ obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, bytes_equ) { char* line = "a = b'\\x03\\x05'\n" "b = b'\\x03\\x05'\n" "if a == b:\n" " c = 1\n" "b = b'\\x02\\x05'\n" "if a != b:\n" " d = 1\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); /* collect */ /* assert */ int c = obj_getInt(self, "c"); int d = obj_getInt(self, "d"); EXPECT_EQ(c, 1); EXPECT_EQ(d, 1); /* deinit */ obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } #if PIKA_BUILTIN_STRUCT_ENABLE TEST(VM, tuple_literal) { char* line = "(1,2,3)\n"; Args* buffs = New_strBuff(); char* pikaAsm = Parser_linesToAsm(buffs, line); printf("%s", pikaAsm); PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); __platform_printf("BEGIN\r\n"); pikaVM_runAsm(self, pikaAsm); /* collect */ /* assert */ EXPECT_STREQ(log_buff[0], "(1, 2, 3)\r\n"); EXPECT_STREQ(log_buff[1], "BEGIN\r\n"); /* deinit */ obj_deinit(self); args_deinit(buffs); EXPECT_EQ(pikaMemNow(), 0); } #endif TEST(VM, dvd_opt) { char* line = "a = 10%4\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ int a = obj_getInt(self, "a"); /* assert */ EXPECT_EQ(a, 2); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, del_) { char* line = "a = 1\n" "print(a)\n" "del a\n" "print(a)\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ /* assert */ EXPECT_EQ(obj_isArgExist(self, "a"), 0); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, is) { char* line = "a = 1\n" "b = a\n" "res1 = a is b\n" "res2 = 1 is 1\n" "c = TinyObj()\n" "d = c\n" "res3 = c is d\n" "res4 = c is None\n"; ; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ /* assert */ EXPECT_EQ(obj_getInt(self, "res1"), 1); EXPECT_EQ(obj_getInt(self, "res2"), 1); EXPECT_EQ(obj_getInt(self, "res3"), 1); EXPECT_EQ(obj_getInt(self, "res4"), 0); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, in) { char* line = "res1 = 'a' in 'aaa'\n" "res2 = 'aaa' in 'a'\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ /* assert */ EXPECT_EQ(obj_getInt(self, "res1"), 1); EXPECT_EQ(obj_getInt(self, "res2"), 0); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, assert) { char* line = "assert True\n" "assert 1 == 1, 'testparser'\n" "assert 1 == 2, 'testparser'\n" "res1 = 0\n" "try:\n" " assert False\n" "except:\n" " res1 = 1\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ int res1 = obj_getInt(self, "res1"); /* assert */ EXPECT_EQ(res1, 1); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, issue_I5LHJG) { char* line = "res = (a, 2, 3)\n" "[a, 2, 3]\n" "{'x': a, 'y': 2, 'z': 3}\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ /* assert */ EXPECT_EQ(obj_isArgExist(self, "res"), 0); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); } TEST(VM, vars_runtime) { char* line = "def testvars(a, *b):\n" " sum = 0\n" " for i in b:\n" " sum += i\n" " return a * sum\n" "res = testvars(6, 2, 3, 4, 5)\n"; PikaObj* self = newRootObj("root", New_PikaStdLib_SysObj); obj_run(self, line); /* collect */ int res = obj_getInt(self, "res"); /* assert */ EXPECT_EQ(res, 84); /* deinit */ obj_deinit(self); EXPECT_EQ(pikaMemNow(), 0); }