566 lines
15 KiB
C
Raw Normal View History

2021-11-16 21:29:27 +08:00
/*
2021-11-16 21:41:57 +08:00
* 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.
*/
2021-11-16 21:29:27 +08:00
#include "PikaObj.h"
#include <stdarg.h>
2021-11-16 21:41:57 +08:00
#include "BaseObj.h"
#include "PikaVM.h"
2021-11-16 21:29:27 +08:00
#include "dataArgs.h"
#include "dataMemory.h"
#include "dataString.h"
#include "dataStrs.h"
int32_t deinitEachSubObj(Arg* argEach, Args* handleArgs) {
2021-11-16 21:41:57 +08:00
if (NULL != handleArgs) {
/* error: tOhis handle not need handle args */
return 1;
}
ArgType type = arg_getType(argEach);
/* deinit sub object */
if (type == TYPE_OBJECT) {
PikaObj* subObj = arg_getPtr(argEach);
obj_deinit(subObj);
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
return 0;
2021-11-16 21:29:27 +08:00
}
void deinitAllSubObj(PikaObj* self) {
2021-11-16 21:41:57 +08:00
Args* args = self->list;
args_foreach(args, deinitEachSubObj, NULL);
2021-11-16 21:29:27 +08:00
}
int32_t obj_deinit(PikaObj* self) {
2021-11-16 21:41:57 +08:00
deinitAllSubObj(self);
args_deinit(self->list);
// DynMemPut(self->mem);
pikaFree(self, sizeof(PikaObj));
self = NULL;
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_enable(PikaObj* self) {
2021-11-16 21:41:57 +08:00
obj_setInt(self, "isEnable", 1);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_disable(PikaObj* self) {
2021-11-16 21:41:57 +08:00
obj_setInt(self, "isEnable", 0);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_setInt(PikaObj* self, char* argPath, int64_t val) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
/* [error] object no found */
return 1;
}
Args* buffs = New_strBuff();
char* name = strsGetLastToken(buffs, argPath, '.');
args_setInt(obj->list, name, val);
args_deinit(buffs);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_setPtr(PikaObj* self, char* argPath, void* pointer) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return 1;
}
Args* buffs = New_strBuff();
char* name = strsGetLastToken(buffs, argPath, '.');
args_setPtr(obj->list, name, pointer);
args_deinit(buffs);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_setFloat(PikaObj* self, char* argPath, float value) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return 1;
}
Args* buffs = New_strBuff();
char* name = strsGetLastToken(buffs, argPath, '.');
args_setFloat(obj->list, name, value);
args_deinit(buffs);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_setStr(PikaObj* self, char* argPath, char* str) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return 1;
}
Args* buffs = New_strBuff();
char* name = strsGetLastToken(buffs, argPath, '.');
args_setStr(obj->list, name, str);
args_deinit(buffs);
return 0;
2021-11-16 21:29:27 +08:00
}
int64_t obj_getInt(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return -999999999;
}
Args* buffs = New_strBuff();
char* argName = strsGetLastToken(buffs, argPath, '.');
int res = args_getInt(obj->list, argName);
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
Arg* obj_getArg(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return NULL;
}
Args* buffs = New_strBuff();
char* argName = strsGetLastToken(buffs, argPath, '.');
Arg* res = args_getArg(obj->list, argName);
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
int32_t obj_setArg(PikaObj* self, char* argPath, Arg* arg) {
2021-11-16 21:41:57 +08:00
/* setArg would copy arg */
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
/* object no found */
return 1;
}
args_copyArg(obj->list, arg);
return 0;
2021-11-16 21:29:27 +08:00
}
void* obj_getPtr(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return NULL;
}
Args* buffs = New_strBuff();
char* argName = strsGetLastToken(buffs, argPath, '.');
void* res = args_getPtr(obj->list, argName);
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
float obj_getFloat(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return -999.999;
}
Args* buffs = New_strBuff();
char* argName = strsGetLastToken(buffs, argPath, '.');
float res = args_getFloat(obj->list, argName);
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
char* obj_getStr(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
if (NULL == obj) {
return NULL;
}
Args* buffs = New_strBuff();
char* argName = strsGetLastToken(buffs, argPath, '.');
char* res = args_getStr(obj->list, argName);
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
int32_t obj_load(PikaObj* self, Args* args, char* name) {
2021-11-16 21:41:57 +08:00
args_copyArgByName(args, name, self->list);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_addOther(PikaObj* self, char* subObjectName, void* new_ObjectFun) {
2021-11-16 21:41:57 +08:00
Args* initArgs = New_args(NULL);
void* (*new_Object)(Args * initArgs) = (void* (*)(Args*))new_ObjectFun;
void* subObject = new_Object(initArgs);
obj_setPtr(self, subObjectName, subObject);
args_deinit(initArgs);
return 0;
2021-11-16 21:29:27 +08:00
}
int32_t obj_freeObj(PikaObj* self, char* objPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getPtr(self, objPath);
obj_deinit(obj);
return 0;
2021-11-16 21:29:27 +08:00
}
char* obj_print(PikaObj* self, char* name) {
2021-11-16 21:41:57 +08:00
if (NULL == self) {
return NULL;
}
return args_print(self->list, name);
2021-11-16 21:29:27 +08:00
}
PikaObj* obj_getClassObjByNewFun(PikaObj* context,
char* name,
NewFun newClassFun) {
2021-11-16 21:41:57 +08:00
Args* initArgs = New_args(NULL);
PikaObj* thisClass = newClassFun(initArgs);
obj_setPtr(thisClass, "_clsptr", newClassFun);
args_deinit(initArgs);
return thisClass;
}
Arg* obj_getMethod(PikaObj* obj, char* methodPath) {
Arg* method = NULL;
Args* buffs = New_strBuff();
char* methodName = strsGetLastToken(buffs, methodPath, '.');
method = obj_getArg(obj, methodName);
PikaObj* methodHostClass;
if (NULL != method) {
method = arg_copy(method);
goto exit;
}
methodHostClass = obj_getClassObj(obj);
method = arg_copy(obj_getArg(methodHostClass, methodName));
obj_deinit(methodHostClass);
exit:
args_deinit(buffs);
return method;
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
PikaObj* obj_getClassObj(PikaObj* obj) {
Args* buffs = New_strBuff();
NewFun classPtr = (NewFun)obj_getPtr(obj, "_clsptr");
char* classObjName = strsAppend(buffs, "_cls-", "");
PikaObj* classObj = obj_getClassObjByNewFun(obj, classObjName, classPtr);
args_deinit(buffs);
return classObj;
2021-11-16 21:29:27 +08:00
}
void* getNewClassObjFunByName(PikaObj* obj, char* name) {
2021-11-16 21:41:57 +08:00
char* classPath = name;
/* init the subprocess */
void* newClassObjFun = args_getPtr(obj->list, classPath);
return newClassObjFun;
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
int32_t __foreach_removeMethodInfo(Arg* argNow, Args* argList) {
if (arg_getType(argNow) == TYPE_METHOD) {
args_removeArg(argList, argNow);
return 0;
}
return 0;
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
PikaObj* removeMethodInfo(PikaObj* thisClass) {
args_foreach(thisClass->list, __foreach_removeMethodInfo, thisClass->list);
return thisClass;
2021-11-16 21:29:27 +08:00
}
PikaObj* newRootObj(char* name, NewFun newObjFun) {
2021-11-16 21:41:57 +08:00
PikaObj* thisClass = obj_getClassObjByNewFun(NULL, name, newObjFun);
PikaObj* newObj = removeMethodInfo(thisClass);
return newObj;
2021-11-16 21:29:27 +08:00
}
PikaObj* initObj(PikaObj* obj, char* name) {
2021-11-16 21:41:57 +08:00
PikaObj* res = NULL;
NewFun newObjFun = (NewFun)getNewClassObjFunByName(obj, name);
Args* buffs = New_args(NULL);
PikaObj* thisClass;
PikaObj* newObj;
if (NULL == newObjFun) {
/* no such object */
res = NULL;
goto exit;
}
thisClass = obj_getClassObjByNewFun(obj, name, newObjFun);
newObj = removeMethodInfo(thisClass);
args_setPtrWithType(obj->list, name, TYPE_OBJECT, newObj);
res = obj_getPtr(obj, name);
2021-11-16 21:29:27 +08:00
goto exit;
exit:
2021-11-16 21:41:57 +08:00
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
PikaObj* obj_getObjDirect(PikaObj* self, char* name) {
2021-11-16 21:41:57 +08:00
if (NULL == self) {
return NULL;
}
/* finded object, check type*/
ArgType type = args_getType(self->list, name);
/* found mate Object */
if (type == TYPE_MATE_OBJECT) {
return initObj(self, name);
}
/* found Objcet */
if (type == TYPE_OBJECT) {
return obj_getPtr(self, name);
}
2021-11-16 21:29:27 +08:00
return NULL;
}
PikaObj* obj_getObj(PikaObj* self, char* objPath, int32_t keepDeepth) {
2021-11-16 21:41:57 +08:00
Args* buffs = New_strBuff();
char* objPathBuff = strsCopy(buffs, objPath);
int32_t tokenNum = strGetTokenNum(objPath, '.');
PikaObj* obj = self;
for (int32_t i = 0; i < tokenNum - keepDeepth; i++) {
char* token = strsPopToken(buffs, objPathBuff, '.');
obj = obj_getObjDirect(obj, token);
if (obj == NULL) {
goto exit;
}
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
goto exit;
2021-11-16 21:29:27 +08:00
exit:
2021-11-16 21:41:57 +08:00
args_deinit(buffs);
return obj;
}
void* methodArg_getPtr(Arg* method_arg) {
uint32_t size_ptr = sizeof(void*);
void* info = arg_getContent(method_arg);
void* ptr = NULL;
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*)((uint64_t)info + size_ptr);
}
void obj_saveMethodInfo(PikaObj* self,
char* method_name,
char* method_dec,
void* method_ptr) {
Args* buffs = New_strBuff();
char* pars = strsRemovePrefix(buffs, method_dec, method_name);
Arg* arg = New_arg(NULL);
uint32_t size_ptr = sizeof(void*);
uint32_t size_pars = strGetSize(pars);
uint32_t size_info = size_ptr + size_pars + 1;
void* info = args_getBuff(buffs, size_info);
memcpy(info, &method_ptr, size_ptr);
/* +1 to add \0 */
memcpy((void*)((uint64_t)info + size_ptr), pars, size_pars + 1);
arg = arg_setName(arg, method_name);
arg = arg_setType(arg, TYPE_METHOD);
arg = arg_setContent(arg, info, size_info);
args_setArg(self->list, arg);
args_deinit(buffs);
2021-11-16 21:29:27 +08:00
}
int32_t class_defineMethod(PikaObj* self,
char* declearation,
2021-11-16 21:41:57 +08:00
Method methodPtr) {
int32_t size = strGetSize(declearation);
int32_t res = 0;
Args* buffs = New_strBuff();
char* cleanDeclearation =
strDeleteChar(args_getBuff(buffs, size), declearation, ' ');
char* methodPath =
strGetFirstToken(args_getBuff(buffs, size), cleanDeclearation, '(');
PikaObj* methodHost = obj_getObj(self, methodPath, 1);
char* methodName;
if (NULL == methodHost) {
/* no found method object */
res = 1;
goto exit;
}
methodName = strsGetLastToken(buffs, methodPath, '.');
2021-11-16 21:29:27 +08:00
2021-11-16 21:41:57 +08:00
obj_saveMethodInfo(methodHost, methodName, cleanDeclearation, methodPtr);
res = 0;
2021-11-16 21:29:27 +08:00
goto exit;
exit:
2021-11-16 21:41:57 +08:00
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
2021-11-16 21:41:57 +08:00
PIKA_WEAK int __runExtern_contral(PikaObj* self, char* cmd) {
return 0;
}
2021-11-16 21:29:27 +08:00
2021-11-16 21:41:57 +08:00
Parameters* obj_runDirect(PikaObj* self, char* cmd) {
Parameters* globals = NULL;
2021-11-16 21:29:27 +08:00
2021-11-16 21:41:57 +08:00
globals = pikaVM_run(self, cmd);
2021-11-16 21:29:27 +08:00
goto exit;
exit:
2021-11-16 21:41:57 +08:00
return globals;
2021-11-16 21:29:27 +08:00
}
int32_t obj_removeArg(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* objHost = obj_getObj(self, argPath, 1);
PikaObj* obj = obj_getObj(self, argPath, 0);
Args* buffs = New_strBuff();
char* argName;
int32_t res;
if (NULL != obj) {
obj_deinit(obj);
}
int32_t err = 0;
if (NULL == objHost) {
/* [error] object no found */
err = 1;
goto exit;
}
argName = strsGetLastToken(buffs, argPath, '.');
res = args_removeArg(objHost->list, args_getArg(objHost->list, argName));
if (1 == res) {
/*[error] not found arg*/
err = 2;
goto exit;
}
2021-11-16 21:29:27 +08:00
goto exit;
exit:
2021-11-16 21:41:57 +08:00
args_deinit(buffs);
return err;
2021-11-16 21:29:27 +08:00
}
int32_t obj_isArgExist(PikaObj* self, char* argPath) {
2021-11-16 21:41:57 +08:00
PikaObj* obj = obj_getObj(self, argPath, 1);
Args* buffs = New_strBuff();
int32_t res = 0;
char* argName;
Arg* arg;
if (NULL == obj) {
/* [error] object no found */
res = 1;
goto exit;
}
argName = strsGetLastToken(buffs, argPath, '.');
arg = args_getArg(obj->list, argName);
if (NULL == arg) {
/* no found arg */
res = 0;
goto exit;
}
/* found arg */
2021-11-16 21:29:27 +08:00
res = 1;
goto exit;
exit:
2021-11-16 21:41:57 +08:00
args_deinit(buffs);
return res;
2021-11-16 21:29:27 +08:00
}
void obj_runNoRes(PikaObj* slef, char* cmd) {
2021-11-16 21:41:57 +08:00
/* unsafe, nothing would happend when error occord */
obj_deinit(obj_runDirect(slef, cmd));
2021-11-16 21:29:27 +08:00
}
void obj_run(PikaObj* self, char* cmd) {
2021-11-16 21:41:57 +08:00
/* safe, stop when error occord and error info would be print32_t */
Parameters* globals = obj_runDirect(self, cmd);
obj_deinit(globals);
2021-11-16 21:29:27 +08:00
}
void obj_setErrorCode(PikaObj* self, int32_t errCode) {
2021-11-16 21:41:57 +08:00
obj_setInt(self, "__errCode", errCode);
2021-11-16 21:29:27 +08:00
}
int32_t obj_getErrorCode(PikaObj* self) {
2021-11-16 21:41:57 +08:00
if (!obj_isArgExist(self, "__errCode")) {
return 0;
}
return obj_getInt(self, "__errCode");
2021-11-16 21:29:27 +08:00
}
void args_setErrorCode(Args* args, int32_t errCode) {
2021-11-16 21:41:57 +08:00
args_setInt(args, "__errCode", errCode);
2021-11-16 21:29:27 +08:00
}
int32_t args_getErrorCode(Args* args) {
2021-11-16 21:41:57 +08:00
if (!args_isArgExist(args, "__errCode")) {
return 0;
}
return args_getInt(args, "__errCode");
2021-11-16 21:29:27 +08:00
}
void obj_setSysOut(PikaObj* self, char* str) {
2021-11-16 21:41:57 +08:00
obj_setStr(self, "__sysOut", str);
2021-11-16 21:29:27 +08:00
}
char* obj_getSysOut(PikaObj* self) {
2021-11-16 21:41:57 +08:00
return obj_getStr(self, "__sysOut");
2021-11-16 21:29:27 +08:00
}
char* args_getSysOut(Args* args) {
2021-11-16 21:41:57 +08:00
return args_getStr(args, "__sysOut");
2021-11-16 21:29:27 +08:00
}
void args_setSysOut(Args* args, char* str) {
2021-11-16 21:41:57 +08:00
args_setStr(args, "__sysOut", str);
2021-11-16 21:29:27 +08:00
}
void obj_sysPrintf(PikaObj* self, char* fmt, ...) {
2021-11-16 21:41:57 +08:00
va_list args;
va_start(args, fmt);
char sysOut[128] = {0};
vsprintf(sysOut, fmt, args);
obj_setSysOut(self, sysOut);
va_end(args);
}
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, float val) {
args_setFloat(args, "return", val);
}
void method_returnPtr(Args* args, void* val) {
args_setPtr(args, "return", 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);
}
float 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);
return self;
2021-11-16 21:29:27 +08:00
}