/* * This file is part of the PikaScript project. * http://github.com/pikastech/pikascript * * MIT License * * Copyright (c) 2021 lyon 李昂 liang6516@outlook.com * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include "PikaObj.h" #include "BaseObj.h" #include "PikaCompiler.h" #include "PikaPlatform.h" #include "dataArgs.h" #include "dataMemory.h" #include "dataString.h" #include "dataStrs.h" static const uint64_t __talbe_fast_atoi[][10] = { {0, 1e0, 2e0, 3e0, 4e0, 5e0, 6e0, 7e0, 8e0, 9e0}, {0, 1e1, 2e1, 3e1, 4e1, 5e1, 6e1, 7e1, 8e1, 9e1}, {0, 1e2, 2e2, 3e2, 4e2, 5e2, 6e2, 7e2, 8e2, 9e2}, {0, 1e3, 2e3, 3e3, 4e3, 5e3, 6e3, 7e3, 8e3, 9e3}, {0, 1e4, 2e4, 3e4, 4e4, 5e4, 6e4, 7e4, 8e4, 9e4}, {0, 1e5, 2e5, 3e5, 4e5, 5e5, 6e5, 7e5, 8e5, 9e5}, {0, 1e6, 2e6, 3e6, 4e6, 5e6, 6e6, 7e6, 8e6, 9e6}, {0, 1e7, 2e7, 3e7, 4e7, 5e7, 6e7, 7e7, 8e7, 9e7}, {0, 1e8, 2e8, 3e8, 4e8, 5e8, 6e8, 7e8, 8e8, 9e8}, {0, 1e9, 2e9, 3e9, 4e9, 5e9, 6e9, 7e9, 8e9, 9e9}, }; int fast_atoi(char* src) { const char* p = src; uint16_t size = strGetSize(src); p = p + size - 1; if (*p) { int s = 0; const uint64_t* n = __talbe_fast_atoi[0]; while (p != src) { s += n[(*p - '0')]; n += 10; p--; } if (*p == '-') { return -s; } return s + n[(*p - '0')]; } return 0; } static uint16_t const str100p[100] = { 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 0x3630, 0x3730, 0x3830, 0x3930, 0x3031, 0x3131, 0x3231, 0x3331, 0x3431, 0x3531, 0x3631, 0x3731, 0x3831, 0x3931, 0x3032, 0x3132, 0x3232, 0x3332, 0x3432, 0x3532, 0x3632, 0x3732, 0x3832, 0x3932, 0x3033, 0x3133, 0x3233, 0x3333, 0x3433, 0x3533, 0x3633, 0x3733, 0x3833, 0x3933, 0x3034, 0x3134, 0x3234, 0x3334, 0x3434, 0x3534, 0x3634, 0x3734, 0x3834, 0x3934, 0x3035, 0x3135, 0x3235, 0x3335, 0x3435, 0x3535, 0x3635, 0x3735, 0x3835, 0x3935, 0x3036, 0x3136, 0x3236, 0x3336, 0x3436, 0x3536, 0x3636, 0x3736, 0x3836, 0x3936, 0x3037, 0x3137, 0x3237, 0x3337, 0x3437, 0x3537, 0x3637, 0x3737, 0x3837, 0x3937, 0x3038, 0x3138, 0x3238, 0x3338, 0x3438, 0x3538, 0x3638, 0x3738, 0x3838, 0x3938, 0x3039, 0x3139, 0x3239, 0x3339, 0x3439, 0x3539, 0x3639, 0x3739, 0x3839, 0x3939, }; char* fast_itoa(char* buf, uint32_t val) { char* p = &buf[10]; *p = '\0'; while (val >= 100) { uint32_t const old = val; p -= 2; val /= 100; __platform_memcpy(p, &str100p[old - (val * 100)], sizeof(uint16_t)); } p -= 2; __platform_memcpy(p, &str100p[val], sizeof(uint16_t)); return &p[val < 10]; } static int32_t obj_deinit_no_del(PikaObj* self) { /* free the list */ args_deinit(self->list); /* free the pointer */ pikaFree(self, sizeof(PikaObj)); self = NULL; return 0; } int32_t obj_deinit(PikaObj* self) { Arg* del = obj_getMethodArg(self, "__del__"); if (NULL != del) { const uint8_t bytes[] = { 0x04, 0x00, /* instruct array size */ 0x00, 0x82, 0x01, 0x00, /* instruct array */ 0x09, 0x00, /* const pool size */ 0x00, 0x5f, 0x5f, 0x64, 0x65, 0x6c, 0x5f, 0x5f, 0x00, /* const pool */ }; pikaVM_runByteCode(self, (uint8_t*)bytes); arg_deinit(del); } return obj_deinit_no_del(self); } PIKA_RES obj_setInt(PikaObj* self, char* argPath, int64_t val) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { /* [error] object no found */ return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setInt(obj->list, name, val); return PIKA_RES_OK; } PIKA_RES obj_setPtr(PikaObj* self, char* argPath, void* pointer) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { /* [error] object no found */ return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setPtr(obj->list, name, pointer); return PIKA_RES_OK; } PIKA_RES obj_setRef(PikaObj* self, char* argPath, void* pointer) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { /* [error] object no found */ return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setRef(obj->list, name, pointer); return PIKA_RES_OK; } PIKA_RES obj_setFloat(PikaObj* self, char* argPath, double value) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { /* [error] object no found */ return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setFloat(obj->list, name, value); return PIKA_RES_OK; } PIKA_RES obj_setStr(PikaObj* self, char* argPath, char* str) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setStr(obj->list, name, str); return PIKA_RES_OK; } PIKA_RES obj_setBytes(PikaObj* self, char* argPath, uint8_t* src, size_t size) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return PIKA_RES_ERR_ARG_NO_FOUND; } char* name = strPointToLastToken(argPath, '.'); args_setBytes(obj->list, name, src, size); return PIKA_RES_OK; } int64_t obj_getInt(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return -999999999; } char* argName = strPointToLastToken(argPath, '.'); int res = args_getInt(obj->list, argName); return res; } Arg* obj_getArg(PikaObj* self, char* argPath) { PIKA_BOOL isClass = PIKA_FALSE; PikaObj* obj = obj_getHostObjWithIsClass(self, argPath, &isClass); if (NULL == obj) { return NULL; } char* argName = strPointToLastToken(argPath, '.'); Arg* res = args_getArg(obj->list, argName); if (isClass) { obj_setArg(self, "_buf", res); res = obj_getArg(self, "_buf"); obj_deinit(obj); } return res; } uint8_t* obj_getBytes(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return NULL; } char* argName = strPointToLastToken(argPath, '.'); return args_getBytes(obj->list, argName); } size_t obj_getBytesSize(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return 0; } char* argName = strPointToLastToken(argPath, '.'); return args_getBytesSize(obj->list, argName); } size_t obj_loadBytes(PikaObj* self, char* argPath, uint8_t* out_buff) { size_t size_mem = obj_getBytesSize(self, argPath); void* src = obj_getBytes(self, argPath); if (0 == size_mem) { return 0; } if (NULL == src) { return 0; } __platform_memcpy(out_buff, src, size_mem); return size_mem; } static PIKA_RES __obj_setArg(PikaObj* self, char* argPath, Arg* arg, uint8_t is_copy) { /* setArg would copy arg */ PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { /* object no found */ return PIKA_RES_ERR_ARG_NO_FOUND; } char* argName = strPointToLastToken(argPath, '.'); Arg* newArg; if (is_copy) { newArg = arg_copy(arg); } else { newArg = arg; } newArg = arg_setName(newArg, argName); args_setArg(obj->list, newArg); return PIKA_RES_OK; } PIKA_RES obj_setArg(PikaObj* self, char* argPath, Arg* arg) { return __obj_setArg(self, argPath, arg, 1); }; PIKA_RES obj_setArg_noCopy(PikaObj* self, char* argPath, Arg* arg) { return __obj_setArg(self, argPath, arg, 0); } void* obj_getPtr(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return NULL; } char* argName = strPointToLastToken(argPath, '.'); void* res = args_getPtr(obj->list, argName); return res; } double obj_getFloat(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return -999.999; } char* argName = strPointToLastToken(argPath, '.'); double res = args_getFloat(obj->list, argName); return res; } char* obj_getStr(PikaObj* self, char* argPath) { PikaObj* obj = obj_getHostObj(self, argPath); if (NULL == obj) { return NULL; } char* argName = strPointToLastToken(argPath, '.'); char* res = args_getStr(obj->list, argName); return res; } PikaObj* obj_getClassObjByNewFun(PikaObj* context, char* name, NewFun newClassFun) { Args* initArgs = New_args(NULL); PikaObj* thisClass = newClassFun(initArgs); thisClass->constructor = newClassFun; thisClass->refcnt = 0; args_deinit(initArgs); return thisClass; } Arg* _obj_getMethodArg(PikaObj* obj, char* methodPath, Arg* arg_reg) { Arg* method = NULL; char* methodName = strPointToLastToken(methodPath, '.'); method = obj_getArg(obj, methodName); PikaObj* methodHostClass; if (NULL != method) { method = arg_copy_noalloc(method, arg_reg); goto exit; } methodHostClass = obj_getClassObj(obj); if (NULL == methodHostClass) { method = NULL; goto exit; } method = arg_copy_noalloc(obj_getArg(methodHostClass, methodName), arg_reg); obj_deinit_no_del(methodHostClass); exit: return method; } Arg* obj_getMethodArg(PikaObj* obj, char* methodPath) { return _obj_getMethodArg(obj, methodPath, NULL); } Arg* obj_getMethodArg_noalloc(PikaObj* obj, char* methodPath, Arg* arg_reg) { return _obj_getMethodArg(obj, methodPath, arg_reg); } NewFun obj_getClass(PikaObj* obj) { return (NewFun)obj->constructor; } PikaObj* obj_getClassObj(PikaObj* obj) { NewFun classPtr = obj_getClass(obj); if (NULL == classPtr) { return NULL; } PikaObj* classObj = obj_getClassObjByNewFun(obj, "", classPtr); return classObj; } void* getNewClassObjFunByName(PikaObj* obj, char* name) { char* classPath = name; /* init the subprocess */ void* newClassObjFun = args_getPtr(obj->list, classPath); return newClassObjFun; } int32_t __foreach_removeMethodInfo(Arg* argNow, Args* argList) { if (arg_getType(argNow) == ARG_TYPE_METHOD_NATIVE) { args_removeArg(argList, argNow); return 0; } return 0; } PikaObj* removeMethodInfo(PikaObj* thisClass) { #if PIKA_METHOD_CACHE_ENABLE #else args_foreach(thisClass->list, __foreach_removeMethodInfo, thisClass->list); #endif return thisClass; } PikaObj* newNormalObj(NewFun newObjFun) { PikaObj* thisClass = obj_getClassObjByNewFun(NULL, "", newObjFun); obj_refcntInc(thisClass); return removeMethodInfo(thisClass); } extern PikaObj* __pikaMain; PikaObj* newRootObj(char* name, NewFun newObjFun) { #if PIKA_POOL_ENABLE mem_pool_init(); #endif PikaObj* newObj = newNormalObj(newObjFun); __pikaMain = newObj; return newObj; } Arg* arg_newMetaObj(NewFun new_obj_fun) { Arg* arg_new = New_arg(NULL); /* m means meta-object */ arg_new = arg_setPtr(arg_new, "", ARG_TYPE_OBJECT_META, (void*)new_obj_fun); return arg_new; } Arg* arg_newDirectObj(NewFun new_obj_fun) { PikaObj* newObj = newNormalObj(new_obj_fun); Arg* arg_new = arg_newPtr(ARG_TYPE_OBJECT_NEW, newObj); return arg_new; } Arg* obj_newObjInPackage(NewFun new_obj_fun) { return arg_newDirectObj(new_obj_fun); } static PikaObj* __obj_initSubObj(PikaObj* obj, char* name) { PikaObj* res = NULL; NewFun constructor = (NewFun)getNewClassObjFunByName(obj, name); Args buffs = {0}; PikaObj* thisClass; PikaObj* new_obj; if (NULL == constructor) { /* no such object */ res = NULL; goto exit; } thisClass = obj_getClassObjByNewFun(obj, name, constructor); new_obj = removeMethodInfo(thisClass); obj_refcntInc(new_obj); obj_runNativeMethod(new_obj, "__init__", NULL); args_setPtrWithType(obj->list, name, ARG_TYPE_OBJECT, new_obj); res = obj_getPtr(obj, name); goto exit; exit: strsDeinit(&buffs); return res; } static PikaObj* __obj_getObjDirect(PikaObj* self, char* name, PIKA_BOOL* pIsClass) { *pIsClass = PIKA_FALSE; if (NULL == self) { return NULL; } /* finded object, check type*/ ArgType type = args_getType(self->list, name); /* found meta Object */ if (type == ARG_TYPE_OBJECT_META) { return __obj_initSubObj(self, name); } /* found Objcet */ if (argType_isObject(type)) { return args_getPtr(self->list, name); } /* found class */ if (type == ARG_TYPE_METHOD_NATIVE_CONSTRUCTOR || type == ARG_TYPE_METHOD_CONSTRUCTOR) { *pIsClass = PIKA_TRUE; PikaObj* method_args_obj = New_TinyObj(NULL); Arg* cls_obj_arg = obj_runMethodArg(self, method_args_obj, args_getArg(self->list, name)); obj_deinit(method_args_obj); if (type == ARG_TYPE_METHOD_NATIVE_CONSTRUCTOR) { obj_runNativeMethod(arg_getPtr(cls_obj_arg), "__init__", NULL); } PikaObj* res = arg_getPtr(cls_obj_arg); arg_deinit(cls_obj_arg); return res; } return NULL; } static PikaObj* __obj_getObjWithKeepDeepth(PikaObj* self, char* objPath, PIKA_BOOL* pIsClass, int32_t keepDeepth) { char objPath_buff[PIKA_PATH_BUFF_SIZE]; __platform_memcpy(objPath_buff, objPath, strGetSize(objPath) + 1); char token_buff[PIKA_NAME_BUFF_SIZE] = {0}; int32_t token_num = strGetTokenNum(objPath, '.'); PikaObj* obj = self; for (int32_t i = 0; i < token_num - keepDeepth; i++) { char* token = strPopToken(token_buff, objPath_buff, '.'); obj = __obj_getObjDirect(obj, token, pIsClass); if (obj == NULL) { goto exit; } } goto exit; exit: return obj; } PikaObj* obj_getObj(PikaObj* self, char* objPath) { PIKA_BOOL isClass = PIKA_FALSE; return __obj_getObjWithKeepDeepth(self, objPath, &isClass, 0); } PikaObj* obj_getHostObj(PikaObj* self, char* objPath) { PIKA_BOOL isClass = PIKA_FALSE; return __obj_getObjWithKeepDeepth(self, objPath, &isClass, 1); } PikaObj* obj_getHostObjWithIsClass(PikaObj* self, char* objPath, PIKA_BOOL* pIsClass) { return __obj_getObjWithKeepDeepth(self, objPath, pIsClass, 1); } Method methodArg_getPtr(Arg* method_arg) { uint32_t size_ptr = sizeof(void*); void* info = arg_getContent(method_arg); void* ptr = NULL; __platform_memcpy(&ptr, info, size_ptr); return (Method)ptr; } char* methodArg_getTypeList(Arg* method_arg, char* buffs, size_t size) { char* method_dec = strCopy(buffs, methodArg_getDec(method_arg)); if (strGetSize(method_dec) > size) { return NULL; } return strCut(buffs, method_dec, '(', ')'); } char* methodArg_getName(Arg* method_arg, char* buffs, size_t size) { char* method_dec = strCopy(buffs, methodArg_getDec(method_arg)); char res[PIKA_NAME_BUFF_SIZE] = {0}; if (strGetSize(method_dec) > size) { return NULL; } strPopToken(res, method_dec, '('); strCopy(buffs, res); return buffs; } Method obj_getNativeMethod(PikaObj* self, char* method_name) { Arg* method_arg = obj_getMethodArg(self, method_name); if (NULL == method_arg) { return NULL; } Method res = methodArg_getPtr(method_arg); arg_deinit(method_arg); return res; } ByteCodeFrame* methodArg_getBytecodeFrame(Arg* method_arg) { uint32_t size_ptr = sizeof(void*); void* info = arg_getContent(method_arg) + size_ptr; ByteCodeFrame* ptr = NULL; __platform_memcpy(&ptr, info, size_ptr); return ptr; } char* methodArg_getDec(Arg* method_arg) { uint32_t size_ptr = sizeof(void*); void* info = arg_getContent(method_arg); return (char*)((uintptr_t)info + 3 * size_ptr); } PikaObj* methodArg_getDefContext(Arg* method_arg) { uint32_t size_ptr = sizeof(void*); void* info = arg_getContent(method_arg) + 2 * size_ptr; PikaObj* context = NULL; __platform_memcpy(&context, info, size_ptr); return context; } static void obj_saveMethodInfo(PikaObj* self, MethodInfo* method_info) { Args buffs = {0}; method_info->pars = method_info->dec; Arg* arg = New_arg(NULL); uint32_t size_pars = strGetSize(method_info->pars); uintptr_t method_info_bytecode_frame = (uintptr_t)method_info->bytecode_frame; uintptr_t method_info_def_context = (uintptr_t)method_info->def_context; /* the first arg_value */ arg = arg_setPtr(arg, method_info->name, method_info->type, method_info->ptr); /* the seconed */ arg = arg_append(arg, &(method_info_bytecode_frame), sizeof(method_info_bytecode_frame)); arg = arg_append(arg, &(method_info_def_context), sizeof(method_info_def_context)); arg = arg_append(arg, method_info->pars, size_pars + 1); args_setArg(self->list, arg); strsDeinit(&buffs); } static int32_t __class_defineMethodWithType(PikaObj* self, char* declearation, Method method_ptr, ArgType method_type, PikaObj* def_context, ByteCodeFrame* bytecode_frame) { int32_t size = strGetSize(declearation); int32_t res = 0; Args buffs = {0}; char* clean_declearation = strDeleteChar(args_getBuff(&buffs, size), declearation, ' '); char* method_path = strGetFirstToken(args_getBuff(&buffs, size), clean_declearation, '('); PikaObj* method_host = obj_getHostObj(self, method_path); MethodInfo method_info = {0}; char* method_name; if (NULL == method_host) { /* no found method object */ res = 1; goto exit; } method_name = strPointToLastToken(method_path, '.'); method_info.dec = clean_declearation; method_info.name = method_name; method_info.ptr = (void*)method_ptr; method_info.type = method_type; method_info.def_context = def_context; method_info.bytecode_frame = bytecode_frame; obj_saveMethodInfo(method_host, &method_info); res = 0; goto exit; exit: strsDeinit(&buffs); return res; } /* define a constructor method */ int32_t class_defineConstructor(PikaObj* self, char* declearation, Method methodPtr) { return __class_defineMethodWithType(self, declearation, methodPtr, ARG_TYPE_METHOD_NATIVE_CONSTRUCTOR, NULL, NULL); } /* define a native method as default */ int32_t class_defineMethod(PikaObj* self, char* declearation, Method methodPtr) { return __class_defineMethodWithType(self, declearation, methodPtr, ARG_TYPE_METHOD_NATIVE, NULL, NULL); } /* define object method, object method is which startwith (self) */ int32_t class_defineRunTimeConstructor(PikaObj* self, char* declearation, Method methodPtr, PikaObj* def_context, ByteCodeFrame* bytecode_frame) { return __class_defineMethodWithType(self, declearation, methodPtr, ARG_TYPE_METHOD_CONSTRUCTOR, def_context, bytecode_frame); } /* define object method, object method is which startwith (self) */ int32_t class_defineObjectMethod(PikaObj* self, char* declearation, Method methodPtr, PikaObj* def_context, ByteCodeFrame* bytecode_frame) { return __class_defineMethodWithType(self, declearation, methodPtr, ARG_TYPE_METHOD_OBJECT, def_context, bytecode_frame); } /* define a static method as default */ int32_t class_defineStaticMethod(PikaObj* self, char* declearation, Method methodPtr, PikaObj* def_context, ByteCodeFrame* bytecode_frame) { return __class_defineMethodWithType(self, declearation, methodPtr, ARG_TYPE_METHOD_STATIC, def_context, bytecode_frame); } VMParameters* obj_runDirect(PikaObj* self, char* cmd) { return pikaVM_run(self, cmd); } int32_t obj_removeArg(PikaObj* self, char* argPath) { PikaObj* objHost = obj_getHostObj(self, argPath); char* argName; int32_t res; int32_t err = 0; if (NULL == objHost) { /* [error] object no found */ err = 1; goto exit; } argName = strPointToLastToken(argPath, '.'); res = args_removeArg(objHost->list, args_getArg(objHost->list, argName)); if (1 == res) { /*[error] not found arg*/ err = 2; goto exit; } goto exit; exit: return err; } int32_t obj_isArgExist(PikaObj* self, char* argPath) { PikaObj* obj_host = obj_getHostObj(self, argPath); int32_t res = 0; char* argName; Arg* arg; if (NULL == obj_host) { /* [error] object no found */ res = 1; goto exit; } argName = strPointToLastToken(argPath, '.'); arg = args_getArg(obj_host->list, argName); if (NULL == arg) { /* no found arg */ res = 0; goto exit; } /* found arg */ res = 1; goto exit; exit: return res; } void obj_runNoRes(PikaObj* slef, char* cmd) { /* unsafe, nothing would happend when error occord */ obj_runDirect(slef, cmd); } void obj_run(PikaObj* self, char* cmd) { obj_runDirect(self, cmd); } PIKA_RES obj_runNativeMethod(PikaObj* self, char* method_name, Args* args) { Method native_method = obj_getNativeMethod(self, method_name); if (NULL == native_method) { return PIKA_RES_ERR_ARG_NO_FOUND; } native_method(self, args); return PIKA_RES_OK; } static void __clearBuff(char* buff, int size) { for (int i = 0; i < size; i++) { buff[i] = 0; } } static void __obj_runCharBeforeRun(PikaObj* self) { struct shell_config* cfg = args_getStruct(self->list, "__shcfg"); /* create the line buff for the first time */ obj_setBytes(self, "__shbuf", NULL, PIKA_LINE_BUFF_SIZE); obj_setInt(self, "__shinb", 0); /* print out the prefix when first entry */ __platform_printf(cfg->prefix); } enum shell_state obj_runChar(PikaObj* self, char inputChar) { struct shell_config* cfg = args_getStruct(self->list, "__shcfg"); __obj_shellLineHandler_t __lineHandler_fun = (__obj_shellLineHandler_t)obj_getPtr(self, "__shhdl"); char* rxBuff = (char*)obj_getBytes(self, "__shbuf"); char* input_line = NULL; int is_in_block = obj_getInt(self, "__shinb"); #ifndef __linux __platform_printf("%c", inputChar); #endif if ((inputChar == '\b') || (inputChar == 127)) { uint32_t size = strGetSize(rxBuff); if (size == 0) { __platform_printf(" "); return SHELL_STATE_CONTINUE; } __platform_printf(" \b"); rxBuff[size - 1] = 0; return SHELL_STATE_CONTINUE; } if (inputChar != '\r' && inputChar != '\n') { strAppendWithSize(rxBuff, &inputChar, 1); return SHELL_STATE_CONTINUE; } if ((inputChar == '\r') || (inputChar == '\n')) { #ifndef __linux __platform_printf("\r\n"); #endif /* still in block */ if (is_in_block) { /* load new line into buff */ Args buffs = {0}; char _n = '\n'; strAppendWithSize(rxBuff, &_n, 1); char* shell_buff_new = strsAppend(&buffs, obj_getStr(self, "shell_buff"), rxBuff); obj_setStr(self, "shell_buff", shell_buff_new); strsDeinit(&buffs); /* go out from block */ if ((rxBuff[0] != ' ') && (rxBuff[0] != '\t')) { obj_setInt(self, "__shinb", 0); input_line = obj_getStr(self, "shell_buff"); enum shell_state state = __lineHandler_fun(self, input_line); __platform_printf(">>> "); return state; } else { __platform_printf("... "); } __clearBuff(rxBuff, PIKA_LINE_BUFF_SIZE); return SHELL_STATE_CONTINUE; } if (0 != strGetSize(rxBuff)) { /* go in block */ if (rxBuff[strGetSize(rxBuff) - 1] == ':') { obj_setInt(self, "__shinb", 1); char _n = '\n'; strAppendWithSize(rxBuff, &_n, 1); obj_setStr(self, "shell_buff", rxBuff); __clearBuff(rxBuff, PIKA_LINE_BUFF_SIZE); __platform_printf("... "); return SHELL_STATE_CONTINUE; } } input_line = rxBuff; enum shell_state state = __lineHandler_fun(self, input_line); __platform_printf(cfg->prefix); __clearBuff(rxBuff, PIKA_LINE_BUFF_SIZE); return state; } return SHELL_STATE_CONTINUE; } static void obj_shellConfig(PikaObj* self, __obj_shellLineHandler_t __lineHandler_fun, struct shell_config* cfg) { struct shell_config cfg_stack = {0}; __platform_memcpy(&cfg_stack, cfg, sizeof(cfg_stack)); args_setStruct(self->list, "__shcfg", cfg_stack); obj_setPtr(self, "__shhdl", __lineHandler_fun); } void obj_shellLineProcess(PikaObj* self, __obj_shellLineHandler_t __lineHandler_fun, struct shell_config* cfg) { /* config the shell */ obj_shellConfig(self, __lineHandler_fun, cfg); /* init the shell */ __obj_runCharBeforeRun(self); /* getchar and run */ while (1) { char inputChar = __platform_getchar(); if (SHELL_STATE_EXIT == obj_runChar(self, inputChar)) { break; } } } static enum shell_state __obj_shellLineHandler_REPL(PikaObj* self, char* input_line) { /* exit */ if (strEqu("exit()", input_line)) { /* exit pika shell */ return SHELL_STATE_EXIT; } /* run single line */ obj_run(self, input_line); return SHELL_STATE_CONTINUE; } void obj_runCharInit(PikaObj* self) { struct shell_config cfg = { .prefix = ">>> ", }; obj_shellConfig(self, __obj_shellLineHandler_REPL, &cfg); /* init the shell */ __obj_runCharBeforeRun(self); } void pikaScriptShell(PikaObj* self) { struct shell_config cfg = { .prefix = ">>> ", }; obj_shellLineProcess(self, __obj_shellLineHandler_REPL, &cfg); } void obj_setErrorCode(PikaObj* self, int32_t errCode) { obj_setInt(self, "__errCode", errCode); } int32_t obj_getErrorCode(PikaObj* self) { if (!obj_isArgExist(self, "__errCode")) { return 0; } return obj_getInt(self, "__errCode"); } int32_t args_getErrorCode(Args* args) { if (!args_isArgExist(args, "__errCode")) { return 0; } return args_getInt(args, "__errCode"); } void obj_setSysOut(PikaObj* self, char* str) { args_setSysOut(self->list, str); } char* obj_getSysOut(PikaObj* self) { return obj_getStr(self, "__sysOut"); } void args_setSysOut(Args* args, char* str) { // args_setStr(args, "__sysOut", str); if (NULL == str) { return; } if (strEqu("", str)) { return; } __platform_printf("%s\r\n", str); } void method_returnBytes(Args* args, uint8_t* val) { args_setBytes(args, "return", val, PIKA_BYTES_DEFAULT_SIZE); } void method_returnStr(Args* args, char* val) { args_setStr(args, "return", val); } void method_returnInt(Args* args, int32_t val) { args_setInt(args, "return", val); } void method_returnFloat(Args* args, double val) { args_setFloat(args, "return", val); } void method_returnPtr(Args* args, void* val) { args_setPtr(args, "return", val); } void method_returnObj(Args* args, void* val) { if (NULL == val) { return; } args_setPtrWithType(args, "return", ARG_TYPE_OBJECT_NEW, val); } void method_returnArg(Args* args, Arg* arg) { arg = arg_setName(arg, "return"); args_setArg(args, arg); } int32_t method_getInt(Args* args, char* argName) { return args_getInt(args, argName); } double method_getFloat(Args* args, char* argName) { return args_getFloat(args, argName); } char* method_getStr(Args* args, char* argName) { return args_getStr(args, argName); } PikaObj* New_PikaObj(void) { PikaObj* self = pikaMalloc(sizeof(PikaObj)); /* List */ self->list = New_args(NULL); self->refcnt = 0; self->constructor = NULL; return self; } Arg* arg_setRef(Arg* self, char* name, PikaObj* obj) { obj_refcntInc(obj); return arg_setPtr(self, name, ARG_TYPE_OBJECT, obj); } int32_t obj_newDirectObj(PikaObj* self, char* objName, NewFun newFunPtr) { Arg* new_obj = arg_newDirectObj(newFunPtr); new_obj = arg_setName(new_obj, objName); new_obj = arg_setType(new_obj, ARG_TYPE_OBJECT); args_setArg(self->list, new_obj); return 0; } int32_t obj_newMetaObj(PikaObj* self, char* objName, NewFun newFunPtr) { /* add meta Obj, no inited */ Arg* new_obj = arg_newMetaObj(newFunPtr); new_obj = arg_setName(new_obj, objName); args_setArg(self->list, new_obj); return 0; } int32_t obj_newObj(PikaObj* self, char* objName, char* className, NewFun newFunPtr) { return obj_newMetaObj(self, objName, newFunPtr); } PikaObj* obj_importModuleWithByteCode(PikaObj* self, char* name, uint8_t* byteCode) { PikaObj* New_PikaStdLib_SysObj(Args * args); obj_newDirectObj(self, name, New_PikaStdLib_SysObj); pikaVM_runByteCode(obj_getObj(self, name), (uint8_t*)byteCode); return self; } PikaObj* obj_importModuleWithByteCodeFrame(PikaObj* self, char* name, ByteCodeFrame* byteCode_frame) { PikaObj* New_PikaStdLib_SysObj(Args * args); obj_newDirectObj(self, name, New_PikaStdLib_SysObj); pikaVM_runByteCodeFrame(obj_getObj(self, name), byteCode_frame); return self; } PikaObj* Obj_linkLibraryFile(PikaObj* self, char* input_file_name) { obj_newMetaObj(self, "__lib", New_LibObj); LibObj* lib = obj_getObj(self, "__lib"); LibObj_loadLibraryFile(lib, input_file_name); return self; } PikaObj* obj_linkLibrary(PikaObj* self, uint8_t* library_bytes) { obj_newMetaObj(self, "__lib", New_LibObj); LibObj* lib = obj_getObj(self, "__lib"); LibObj_loadLibrary(lib, library_bytes); return self; } PikaObj* obj_linkLibObj(PikaObj* self, LibObj* library) { obj_setPtr(self, "__lib", library); return self; } uint8_t* obj_getByteCodeFromModule(PikaObj* self, char* module_name) { /* exit when no found '__lib' */ if (!obj_isArgExist(self, "__lib")) { return NULL; } /* find module from the library */ LibObj* lib = obj_getPtr(self, "__lib"); PikaObj* module = obj_getObj(lib, module_name); /* exit when no module in '__lib' */ if (NULL == module) { return NULL; } return obj_getPtr(module, "bytecode"); } int obj_runModule(PikaObj* self, char* module_name) { uint8_t* bytecode = obj_getByteCodeFromModule(self, module_name); if (NULL == bytecode) { return 1; } pikaVM_runByteCode(self, bytecode); return 0; } int obj_importModule(PikaObj* self, char* module_name) { /* import bytecode of the module */ uint8_t* bytecode = obj_getByteCodeFromModule(__pikaMain, module_name); if (NULL == bytecode) { return 1; } obj_importModuleWithByteCode(self, module_name, bytecode); return 0; } char* obj_toStr(PikaObj* self) { /* clang-format off */ PIKA_PYTHON( __res = __str__() ) /* clang-format on */ /* check method arg */ Arg* method_arg = obj_getMethodArg(self, "__str__"); if (NULL != method_arg) { arg_deinit(method_arg); const uint8_t bytes[] = { 0x08, 0x00, /* instruct array size */ 0x00, 0x82, 0x01, 0x00, 0x00, 0x04, 0x09, 0x00, /* instruct array */ 0x0f, 0x00, /* const pool size */ 0x00, 0x5f, 0x5f, 0x73, 0x74, 0x72, 0x5f, 0x5f, 0x00, 0x5f, 0x5f, 0x72, 0x65, 0x73, 0x00, /* const pool */ }; pikaVM_runByteCode(self, (uint8_t*)bytes); char* str_res = obj_getStr(self, "__res"); return str_res; } /* normal object */ Args buffs = {0}; char* str_res = strsFormat(&buffs, PIKA_SPRINTF_BUFF_SIZE, "", self); obj_setStr(self, "__res", str_res); strsDeinit(&buffs); return obj_getStr(self, "__res"); } void pks_eventLicener_registEvent(PikaEventListener* self, uint32_t eventId, PikaObj* eventHandleObj) { Args buffs = {0}; char* event_name = strsFormat(&buffs, PIKA_SPRINTF_BUFF_SIZE, "%ld", eventId); obj_newDirectObj(self, event_name, New_TinyObj); PikaObj* event_item = obj_getObj(self, event_name); obj_setRef(event_item, "eventHandleObj", eventHandleObj); strsDeinit(&buffs); } void pks_eventLicener_removeEvent(PikaEventListener* self, uint32_t eventId) { Args buffs = {0}; char* event_name = strsFormat(&buffs, PIKA_SPRINTF_BUFF_SIZE, "%ld", eventId); obj_removeArg(self, event_name); strsDeinit(&buffs); } PikaObj* pks_eventLisener_getEventHandleObj(PikaEventListener* self, uint32_t eventId) { Args buffs = {0}; char* event_name = strsFormat(&buffs, PIKA_SPRINTF_BUFF_SIZE, "%ld", eventId); PikaObj* event_item = obj_getObj(self, event_name); PikaObj* eventHandleObj = obj_getPtr(event_item, "eventHandleObj"); strsDeinit(&buffs); return eventHandleObj; } void pks_eventLisener_init(PikaEventListener** p_self) { *p_self = newNormalObj(New_TinyObj); } void pks_eventLisener_deinit(PikaEventListener** p_self) { if (NULL != *p_self) { obj_deinit(*p_self); *p_self = NULL; } } void pks_eventLisener_sendSignal(PikaEventListener* self, uint32_t eventId, int eventSignal) { PikaObj* eventHandleObj = pks_eventLisener_getEventHandleObj(self, eventId); if (NULL == eventHandleObj) { __platform_printf( "Error: can not find event handler by id: [0x%02x]\r\n", eventId); return; } obj_setInt(eventHandleObj, "eventSignal", eventSignal); /* clang-format off */ PIKA_PYTHON( eventCallBack(eventSignal) ) /* clang-format on */ const uint8_t bytes[] = { 0x08, 0x00, /* instruct array size */ 0x10, 0x81, 0x01, 0x00, 0x00, 0x02, 0x0d, 0x00, /* instruct array */ 0x1b, 0x00, /* const pool size */ 0x00, 0x65, 0x76, 0x65, 0x6e, 0x74, 0x53, 0x69, 0x67, 0x6e, 0x61, 0x6c, 0x00, 0x65, 0x76, 0x65, 0x6e, 0x74, 0x43, 0x61, 0x6c, 0x6c, 0x42, 0x61, 0x63, 0x6b, 0x00, /* const pool */ }; pikaVM_runByteCode(eventHandleObj, (uint8_t*)bytes); } /* print major version info */ void pks_printVersion(void) { __platform_printf("pikascript-core==v%d.%d.%d (%s)\r\n", PIKA_VERSION_MAJOR, PIKA_VERSION_MINOR, PIKA_VERSION_MICRO, PIKA_EDIT_TIME); } void pks_getVersion(char *buff) { __platform_sprintf(buff, "%d.%d.%d", PIKA_VERSION_MAJOR, PIKA_VERSION_MINOR, PIKA_VERSION_MICRO); } void* obj_getStruct(PikaObj* self, char* name) { return args_getStruct(self->list, name); } char* obj_cacheStr(PikaObj* self, char* str) { return args_cacheStr(self->list, str); }