mirror of
https://gitee.com/Lyon1998/pikapython.git
synced 2025-01-15 17:02:53 +08:00
402 lines
12 KiB
C
402 lines
12 KiB
C
/*
|
|
* 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 "dataArg.h"
|
|
#include "dataArgs.h"
|
|
#include "dataMemory.h"
|
|
#include "dataString.h"
|
|
#include "stdlib.h"
|
|
|
|
void arg_deinit(Arg* self) {
|
|
arg_freeContent(self);
|
|
}
|
|
|
|
uint16_t arg_getTotleSize(Arg* self) {
|
|
return content_totleSize(self);
|
|
}
|
|
|
|
uint16_t content_sizeOffset(uint8_t* self) {
|
|
const uint8_t nextLength = sizeof(uint8_t*);
|
|
return nextLength;
|
|
}
|
|
|
|
uint16_t content_getSize(uint8_t* self) {
|
|
uint16_t size = 0;
|
|
size += self[content_sizeOffset(self) + 1];
|
|
size = (size << 8);
|
|
size += self[content_sizeOffset(self)];
|
|
return size;
|
|
}
|
|
|
|
void content_setNext(uint8_t* self, uint8_t* next) {
|
|
uint8_t* nextDir = self + content_nextOffset(self);
|
|
uint64_t pointerTemp = (long)next;
|
|
for (uint32_t i = 0; i < sizeof(uint8_t*); i++) {
|
|
// aboid \0
|
|
nextDir[i] = pointerTemp;
|
|
pointerTemp = pointerTemp >> 8;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* time33 hash
|
|
*/
|
|
Hash hash_time33(char* str) {
|
|
Hash hash = 5381;
|
|
while (*str) {
|
|
hash += (hash << 5) + (*str++);
|
|
}
|
|
return (hash & 0x7FFFFFFF);
|
|
}
|
|
|
|
uint8_t* content_init_hash(Hash nameHash,
|
|
ArgType type,
|
|
uint8_t* content,
|
|
uint16_t size,
|
|
uint8_t* next) {
|
|
const uint8_t nextLength = sizeof(uint8_t*);
|
|
const uint8_t sizeLength = sizeof(uint16_t);
|
|
uint16_t nameSize = sizeof(Hash); // use hash
|
|
uint16_t typeSize = sizeof(ArgType); // use enum
|
|
uint8_t* self = (uint8_t*)pikaMalloc(nextLength + sizeLength + nameSize +
|
|
size + typeSize);
|
|
|
|
uint8_t* nextDir = self;
|
|
uint8_t* sizeDir = nextDir + nextLength;
|
|
uint8_t* nameDir = sizeDir + sizeLength;
|
|
uint8_t* contentDir = nameDir + nameSize;
|
|
uint8_t* typeDir = contentDir + size;
|
|
|
|
__platform_memcpy(nameDir, &nameHash, nameSize); // use hash
|
|
__platform_memcpy(typeDir, &type, typeSize);
|
|
sizeDir[0] = size;
|
|
sizeDir[1] = size >> 8;
|
|
if (NULL != content) {
|
|
__platform_memcpy(contentDir, content, size);
|
|
} else {
|
|
__platform_memset(contentDir, 0, size);
|
|
}
|
|
|
|
uint64_t pointerTemp = (long)next;
|
|
for (uint32_t i = 0; i < sizeof(uint8_t*); i++) {
|
|
// aboid \0
|
|
nextDir[i] = pointerTemp;
|
|
pointerTemp = pointerTemp >> 8;
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
uint8_t* content_init(char* name,
|
|
ArgType type,
|
|
uint8_t* content,
|
|
uint16_t size,
|
|
uint8_t* next) {
|
|
Hash nameHash = hash_time33(name);
|
|
return content_init_hash(nameHash, type, content, size, next);
|
|
}
|
|
|
|
uint16_t content_totleSize(uint8_t* self) {
|
|
const uint8_t size_size = sizeof(uint16_t);
|
|
const uint8_t size_next = sizeof(uint8_t*);
|
|
const uint8_t size_type = sizeof(ArgType);
|
|
const uint8_t size_hash = sizeof(Hash);
|
|
uint16_t size_content = content_getSize(self);
|
|
return size_content + size_hash + size_type + size_size + size_next;
|
|
}
|
|
|
|
void arg_freeContent(Arg* self) {
|
|
if (NULL != self) {
|
|
content_deinit(self);
|
|
}
|
|
}
|
|
|
|
uint8_t content_nameOffset(uint8_t* self) {
|
|
const uint8_t nextLength = sizeof(uint8_t*);
|
|
const uint8_t sizeLength = sizeof(uint16_t);
|
|
return nextLength + sizeLength;
|
|
}
|
|
|
|
Hash content_getNameHash(uint8_t* self) {
|
|
uint8_t* nameHashDir = (uint8_t*)self + content_nameOffset(self);
|
|
Hash nameHash = 0;
|
|
__platform_memcpy(&nameHash, nameHashDir, sizeof(Hash));
|
|
return nameHash;
|
|
}
|
|
|
|
uint8_t* content_deinit(uint8_t* self) {
|
|
uint16_t totleSize = content_totleSize(self);
|
|
pikaFree(self, totleSize);
|
|
return 0;
|
|
}
|
|
|
|
uint8_t* content_setContent(uint8_t* self, uint8_t* content, uint16_t size) {
|
|
if (NULL == self) {
|
|
return content_init("", TYPE_NONE, content, size, NULL);
|
|
}
|
|
Hash nameHash = content_getNameHash(self);
|
|
ArgType type = content_getType(self);
|
|
uint8_t* next = content_getNext(self);
|
|
uint8_t* newContent =
|
|
content_init_hash(nameHash, type, content, size, next);
|
|
content_deinit(self);
|
|
return newContent;
|
|
}
|
|
|
|
uint8_t* content_setNameHash(uint8_t* self, Hash nameHash) {
|
|
if (NULL == self) {
|
|
return content_init_hash(nameHash, TYPE_NONE, NULL, 0, NULL);
|
|
}
|
|
ArgType type = content_getType(self);
|
|
uint8_t* content = content_getContent(self);
|
|
uint16_t size = content_getSize(self);
|
|
uint8_t* next = content_getNext(self);
|
|
uint8_t* newContent =
|
|
content_init_hash(nameHash, type, content, size, next);
|
|
content_deinit(self);
|
|
return newContent;
|
|
}
|
|
|
|
uint8_t* content_setName(uint8_t* self, char* name) {
|
|
if (NULL == self) {
|
|
return content_init(name, TYPE_NONE, NULL, 0, NULL);
|
|
}
|
|
ArgType type = content_getType(self);
|
|
uint8_t* content = content_getContent(self);
|
|
uint16_t size = content_getSize(self);
|
|
uint8_t* next = content_getNext(self);
|
|
uint8_t* newContent = content_init(name, type, content, size, next);
|
|
content_deinit(self);
|
|
return newContent;
|
|
}
|
|
|
|
uint8_t* content_setType(uint8_t* self, ArgType type) {
|
|
if (NULL == self) {
|
|
return content_init("", type, NULL, 0, NULL);
|
|
}
|
|
Hash nameHash = content_getNameHash(self);
|
|
uint8_t* content = content_getContent(self);
|
|
uint16_t size = content_getSize(self);
|
|
uint8_t* next = content_getNext(self);
|
|
uint8_t* newContent =
|
|
content_init_hash(nameHash, type, content, size, next);
|
|
content_deinit(self);
|
|
return newContent;
|
|
}
|
|
|
|
Arg* arg_newContent(Arg* self, uint32_t size) {
|
|
uint8_t* newContent = content_init("", TYPE_NONE, NULL, size, NULL);
|
|
arg_freeContent(self);
|
|
return newContent;
|
|
}
|
|
|
|
Arg* arg_setContent(Arg* self, uint8_t* content, uint32_t size) {
|
|
return content_setContent(self, content, size);
|
|
}
|
|
|
|
Arg* arg_setName(Arg* self, char* name) {
|
|
return content_setName(self, name);
|
|
}
|
|
|
|
Arg* arg_setNameHash(Arg* self, Hash nameHash) {
|
|
return content_setNameHash(self, nameHash);
|
|
}
|
|
|
|
Arg* arg_setType(Arg* self, ArgType type) {
|
|
return content_setType(self, type);
|
|
}
|
|
|
|
ArgType content_getType(uint8_t* self) {
|
|
void* type_ptr = (uint8_t*)self + content_typeOffset(self);
|
|
ArgType type;
|
|
__platform_memcpy(&type, type_ptr, sizeof(ArgType));
|
|
return type;
|
|
}
|
|
|
|
uint16_t content_contentOffset(uint8_t* self) {
|
|
const uint8_t nextLength = sizeof(uint8_t*);
|
|
const uint8_t sizeLength = sizeof(uint16_t);
|
|
return nextLength + sizeLength + sizeof(Hash);
|
|
}
|
|
|
|
uint16_t content_nextOffset(uint8_t* self) {
|
|
return 0;
|
|
}
|
|
|
|
uint8_t* content_getNext(uint8_t* self) {
|
|
uint8_t* nextDir = self + content_nextOffset(self);
|
|
uint8_t* next = NULL;
|
|
uint64_t pointerTemp = 0;
|
|
|
|
for (int32_t i = sizeof(uint8_t*); i > -1; i--) {
|
|
// avoid \0
|
|
uint8_t val = nextDir[i];
|
|
pointerTemp = (pointerTemp << 8);
|
|
pointerTemp += val;
|
|
}
|
|
next = (uint8_t*)(long)pointerTemp;
|
|
return next;
|
|
}
|
|
|
|
uint8_t* content_getContent(uint8_t* self) {
|
|
return self + content_contentOffset(self);
|
|
}
|
|
|
|
uint8_t* arg_getContent(Arg* self) {
|
|
return content_getContent(self);
|
|
}
|
|
|
|
Arg* arg_setInt(Arg* self, char* name, int64_t val) {
|
|
int64_t int64Temp = val;
|
|
uint8_t contentBuff[8];
|
|
for (uint32_t i = 0; i < 4; i++) {
|
|
// add 0x30 to void \0
|
|
contentBuff[i] = int64Temp;
|
|
int64Temp = int64Temp >> 8;
|
|
}
|
|
return content_init(name, TYPE_INT, contentBuff, 4, NULL);
|
|
}
|
|
|
|
Arg* arg_setNull(Arg* self) {
|
|
return content_init("", TYPE_NULL, NULL, 0, NULL);
|
|
}
|
|
|
|
Arg* arg_setFloat(Arg* self, char* name, float val) {
|
|
uint8_t contentBuff[4];
|
|
uint8_t* valPtr = (uint8_t*)&val;
|
|
for (uint32_t i = 0; i < 4; i++) {
|
|
// add 0x30 to void \0
|
|
contentBuff[i] = valPtr[i];
|
|
}
|
|
return content_init(name, TYPE_FLOAT, contentBuff, 4, NULL);
|
|
}
|
|
|
|
float arg_getFloat(Arg* self) {
|
|
if (NULL == arg_getContent(self)) {
|
|
return -999.999;
|
|
}
|
|
|
|
float valOut = 0;
|
|
uint8_t* valOutPtr = (uint8_t*)(&valOut);
|
|
uint8_t* valPtr = arg_getContent(self);
|
|
for (uint32_t i = 0; i < 4; i++) {
|
|
valOutPtr[i] = valPtr[i];
|
|
}
|
|
return valOut;
|
|
}
|
|
|
|
Arg* arg_setPtr(Arg* self, char* name, ArgType type, void* pointer) {
|
|
uint64_t pointerTemp = (long)pointer;
|
|
uint8_t contentBuff[8];
|
|
for (uint32_t i = 0; i < sizeof(uint8_t*); i++) {
|
|
// aboid \0
|
|
contentBuff[i] = pointerTemp;
|
|
pointerTemp = pointerTemp >> 8;
|
|
}
|
|
return content_init(name, type, contentBuff, sizeof(uint8_t*), NULL);
|
|
}
|
|
|
|
Arg* arg_setStr(Arg* self, char* name, char* string) {
|
|
return content_init(name, TYPE_STRING, (uint8_t*)string,
|
|
strGetSize(string) + 1, NULL);
|
|
}
|
|
|
|
int64_t arg_getInt(Arg* self) {
|
|
if (NULL == arg_getContent(self)) {
|
|
return -999999;
|
|
}
|
|
int64_t int64Temp = 0;
|
|
for (int32_t i = 3; i > -1; i--) {
|
|
// add 0x30 to avoid 0
|
|
int64Temp = (int64Temp << 8);
|
|
int64Temp += arg_getContent(self)[i];
|
|
}
|
|
return int64Temp;
|
|
}
|
|
|
|
void* arg_getPtr(Arg* self) {
|
|
void* pointer = NULL;
|
|
uint64_t pointerTemp = 0;
|
|
if (NULL == arg_getContent(self)) {
|
|
return NULL;
|
|
}
|
|
uint8_t* content = arg_getContent(self);
|
|
for (int32_t i = sizeof(uint8_t*) - 1; i > -1; i--) {
|
|
// avoid \0
|
|
uint8_t val = content[i];
|
|
pointerTemp = (pointerTemp << 8);
|
|
pointerTemp += val;
|
|
}
|
|
pointer = (void*)(long)pointerTemp;
|
|
return pointer;
|
|
}
|
|
char* arg_getStr(Arg* self) {
|
|
return (char*)arg_getContent(self);
|
|
}
|
|
|
|
uint16_t content_typeOffset(uint8_t* self) {
|
|
const uint8_t nextLength = sizeof(uint8_t*);
|
|
const uint8_t sizeLength = 2;
|
|
uint16_t size = content_getSize(self);
|
|
uint16_t nameSize = sizeof(Hash);
|
|
return nextLength + sizeLength + nameSize + size;
|
|
}
|
|
|
|
Hash arg_getNameHash(Arg* self) {
|
|
if (NULL == self) {
|
|
return 999999;
|
|
}
|
|
return content_getNameHash(self);
|
|
}
|
|
|
|
ArgType arg_getType(Arg* self) {
|
|
if (NULL == self) {
|
|
return TYPE_NONE;
|
|
}
|
|
return content_getType(self);
|
|
}
|
|
|
|
uint16_t arg_getContentSize(Arg* self) {
|
|
return content_getSize(self);
|
|
}
|
|
|
|
Arg* New_arg(void* voidPointer) {
|
|
return NULL;
|
|
}
|
|
|
|
Arg* arg_copy(Arg* argToBeCopy) {
|
|
if (NULL == argToBeCopy) {
|
|
return NULL;
|
|
}
|
|
Arg* argCopied = New_arg(NULL);
|
|
argCopied = arg_setContent(argCopied, arg_getContent(argToBeCopy),
|
|
arg_getContentSize(argToBeCopy));
|
|
argCopied = arg_setNameHash(argCopied, arg_getNameHash(argToBeCopy));
|
|
argCopied = arg_setType(argCopied, arg_getType(argToBeCopy));
|
|
return argCopied;
|
|
}
|