pikapython/src/PikaParser.c

1143 lines
38 KiB
C
Raw Normal View History

2021-11-16 08:46:44 +08:00
/*
* This file is part of the PikaScript project.
* http://github.com/pikastech/pikascript
*
* MIT License
*
2021-11-16 08:49:33 +08:00
* Copyright (c) 2021 lyon liang6516@outlook.com
2021-11-16 08:46:44 +08:00
*
* 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-10-13 18:05:54 +08:00
#include "PikaParser.h"
2021-10-16 19:52:19 +08:00
#include "BaseObj.h"
2021-10-13 18:05:54 +08:00
#include "PikaObj.h"
2021-10-16 17:32:02 +08:00
#include "dataQueue.h"
2021-10-13 18:05:54 +08:00
#include "dataQueueObj.h"
2021-10-16 18:59:46 +08:00
#include "dataStack.h"
2021-10-13 18:05:54 +08:00
#include "dataStrs.h"
2021-11-27 20:49:08 +08:00
char* strsPopTokenWithSkip_byStr(Args* buffs,
char* stmts,
char* str,
char skipStart,
char skipEnd) {
int32_t str_size = strGetSize(str);
2021-10-13 20:07:32 +08:00
int32_t size = strGetSize(stmts);
if (0 == size) {
return NULL;
}
char* strOut = args_getBuff(buffs, size);
int32_t stmtEnd = 0;
uint8_t isGetSign = 0;
int32_t parentheseDeepth = 0;
for (int32_t i = 0; i < size; i++) {
if (skipStart == stmts[i]) {
2021-10-13 20:07:32 +08:00
parentheseDeepth++;
}
if (skipEnd == stmts[i]) {
2021-10-13 20:07:32 +08:00
parentheseDeepth--;
}
if (parentheseDeepth == 0) {
2021-11-27 20:49:08 +08:00
if (0 == strncmp(stmts + i, str, str_size)) {
2021-10-13 20:07:32 +08:00
stmtEnd = i;
isGetSign = 1;
break;
}
}
}
if (!isGetSign) {
stmtEnd = size;
}
for (int32_t i = 0; i < stmtEnd; i++) {
strOut[i] = stmts[i];
}
2021-11-27 20:49:08 +08:00
memmove(stmts, stmts + stmtEnd + str_size, size);
2021-10-13 20:07:32 +08:00
strOut[stmtEnd] = 0;
return strOut;
}
2021-11-27 20:49:08 +08:00
char* strsPopTokenWithSkip(Args* buffs,
char* stmts,
char sign,
char skipStart,
char skipEnd) {
char str_buff[2] = {0};
str_buff[0] = sign;
return strsPopTokenWithSkip_byStr(buffs, stmts, str_buff, skipStart,
skipEnd);
}
2021-11-28 17:02:27 +08:00
char* strsGetCleanCmd(Args* outBuffs, char* cmd) {
int32_t size = strGetSize(cmd);
Args* buffs = New_strBuff();
char* tokens = Lexer_getTokens(buffs, cmd);
uint16_t token_size = strCountSign(tokens, 0x1F) + 1;
char* strOut = args_getBuff(outBuffs, size);
int32_t iOut = 0;
for (uint16_t i = 0; i < token_size; i++) {
char* token = strsPopToken(buffs, tokens, 0x1F);
for (uint16_t k = 0; k < strGetSize(token + 1); k++) {
strOut[iOut] = token[k + 1];
iOut++;
}
}
/* add \0 */
strOut[iOut] = 0;
args_deinit(buffs);
return strOut;
}
2021-11-26 18:59:09 +08:00
enum TokenType {
TOKEN_strEnd = 0,
TOKEN_symbol,
2021-11-26 18:59:41 +08:00
TOKEN_keyword,
2021-11-26 18:59:09 +08:00
TOKEN_operator,
2021-11-27 11:58:15 +08:00
TOKEN_devider,
2021-11-26 18:59:09 +08:00
TOKEN_literal,
};
2021-10-20 20:40:45 +08:00
enum StmtType {
2021-11-26 18:59:09 +08:00
STMT_reference,
STMT_string,
STMT_number,
STMT_method,
STMT_operator,
STMT_none,
2021-10-20 20:40:45 +08:00
};
2021-11-27 20:49:08 +08:00
char* strsDeleteBetween(Args* buffs, char* strIn, char begin, char end) {
2021-10-20 21:06:13 +08:00
int32_t size = strGetSize(strIn);
char* strOut = args_getBuff(buffs, size);
2021-10-20 21:21:03 +08:00
uint8_t deepth = 0;
2021-10-20 21:06:13 +08:00
uint32_t iOut = 0;
for (int i = 0; i < size; i++) {
2021-10-20 21:21:03 +08:00
if (end == strIn[i]) {
deepth--;
}
if (0 == deepth) {
2021-10-20 21:06:13 +08:00
strOut[iOut] = strIn[i];
2021-10-20 21:21:03 +08:00
iOut++;
2021-10-20 21:06:13 +08:00
}
if (begin == strIn[i]) {
2021-10-20 21:21:03 +08:00
deepth++;
2021-10-20 21:06:13 +08:00
}
}
strOut[iOut] = 0;
return strOut;
}
2021-12-10 11:41:33 +08:00
static uint8_t Lexer_isError(char* line) {
Args* buffs = New_strBuff();
uint8_t res = 0; /* not error */
char* tokens = Lexer_getTokens(buffs, line);
if (NULL == tokens) {
res = 1; /* lex error */
goto exit;
}
goto exit;
exit:
args_deinit(buffs);
return res;
}
2021-11-27 21:06:01 +08:00
static enum StmtType Lexer_matchStmtType(char* right) {
2021-10-20 21:06:13 +08:00
Args* buffs = New_strBuff();
2021-11-26 18:59:09 +08:00
enum StmtType stmtType = STMT_none;
2021-11-27 20:49:08 +08:00
char* rightWithoutSubStmt = strsDeleteBetween(buffs, right, '(', ')');
2021-11-27 18:05:32 +08:00
char* tokens = Lexer_getTokens(buffs, rightWithoutSubStmt);
2021-11-28 17:02:27 +08:00
uint16_t token_size = strCountSign(tokens, 0x1F) + 1;
2021-11-27 19:00:17 +08:00
uint8_t is_get_operator = 0;
uint8_t is_get_method = 0;
uint8_t is_get_string = 0;
uint8_t is_get_number = 0;
uint8_t is_get_symbol = 0;
2021-11-28 17:02:27 +08:00
for (int i = 0; i < token_size; i++) {
2021-11-27 18:05:32 +08:00
char* token = strsPopToken(buffs, tokens, 0x1F);
2021-12-10 16:22:19 +08:00
enum TokenType token_type = (enum TokenType)token[0];
2021-11-27 19:00:17 +08:00
/* collect type */
2021-11-27 18:05:32 +08:00
if (token_type == TOKEN_operator) {
2021-11-27 19:00:17 +08:00
is_get_operator = 1;
continue;
2021-11-27 18:05:32 +08:00
}
2021-11-27 19:00:17 +08:00
if (token_type == TOKEN_devider) {
is_get_method = 1;
continue;
}
if (token_type == TOKEN_literal) {
if (token[1] == '\'' || token[1] == '"') {
is_get_string = 1;
continue;
}
is_get_number = 1;
continue;
}
if (token_type == TOKEN_symbol) {
is_get_symbol = 1;
continue;
}
}
if (is_get_operator) {
stmtType = STMT_operator;
goto exit;
2021-10-20 21:06:13 +08:00
}
2021-11-27 19:00:17 +08:00
if (is_get_method) {
2021-11-26 18:59:09 +08:00
stmtType = STMT_method;
2021-10-20 21:06:13 +08:00
goto exit;
2021-10-20 20:40:45 +08:00
}
2021-11-27 19:00:17 +08:00
if (is_get_string) {
2021-11-26 18:59:09 +08:00
stmtType = STMT_string;
2021-10-20 21:06:13 +08:00
goto exit;
2021-10-20 20:40:45 +08:00
}
2021-11-27 19:00:17 +08:00
if (is_get_number) {
2021-11-26 18:59:09 +08:00
stmtType = STMT_number;
2021-10-20 21:06:13 +08:00
goto exit;
2021-10-20 20:40:45 +08:00
}
2021-11-27 19:00:17 +08:00
if (is_get_symbol) {
2021-11-26 18:59:09 +08:00
stmtType = STMT_reference;
2021-10-20 21:06:13 +08:00
goto exit;
2021-10-20 20:40:45 +08:00
}
2021-10-20 21:06:13 +08:00
exit:
args_deinit(buffs);
return stmtType;
2021-10-20 20:40:45 +08:00
}
2021-11-27 20:49:08 +08:00
uint8_t Parser_checkIsDirect(char* str) {
2021-10-25 21:53:35 +08:00
/* include '0' */
uint32_t size = strGetSize(str) + 1;
for (uint32_t i = 1; i + 1 < size; i++) {
2021-11-27 21:25:45 +08:00
if ((str[i - 1] != '%') && (str[i - 1] != '!') && (str[i - 1] != '<') &&
(str[i - 1] != '>') && (str[i - 1] != '=') && (str[i - 1] != '+') &&
(str[i - 1] != '-') && (str[i - 1] != '*') && (str[i - 1] != '/') &&
(str[i + 1] != '=') && (str[i] == '=')) {
2021-10-25 21:53:35 +08:00
return 1;
}
}
return 0;
}
2021-11-26 20:33:35 +08:00
char* Lexer_printTokens(Args* outBuffs, char* tokens) {
/* init */
Args* buffs = New_strBuff();
char* printOut = strsCopy(buffs, "");
/* process */
2021-11-27 17:26:06 +08:00
uint16_t tokenSize = strCountSign(tokens, 0x1F) + 1;
for (uint16_t i = 0; i < tokenSize; i++) {
2021-11-27 17:26:06 +08:00
char* token = strsPopToken(buffs, tokens, 0x1F);
2021-11-26 20:33:35 +08:00
if (token[0] == TOKEN_operator) {
2021-11-27 00:30:21 +08:00
printOut = strsAppend(buffs, printOut, "{opt}");
printOut = strsAppend(buffs, printOut, token + 1);
}
2021-11-27 11:58:15 +08:00
if (token[0] == TOKEN_devider) {
printOut = strsAppend(buffs, printOut, "{dvd}");
2021-11-27 00:30:21 +08:00
printOut = strsAppend(buffs, printOut, token + 1);
}
if (token[0] == TOKEN_symbol) {
printOut = strsAppend(buffs, printOut, "{sym}");
2021-11-26 20:33:35 +08:00
printOut = strsAppend(buffs, printOut, token + 1);
}
2021-11-27 01:13:24 +08:00
if (token[0] == TOKEN_literal) {
printOut = strsAppend(buffs, printOut, "{lit}");
printOut = strsAppend(buffs, printOut, token + 1);
}
2021-11-26 20:33:35 +08:00
}
/* out put */
printOut = strsCopy(outBuffs, printOut);
args_deinit(buffs);
return printOut;
}
Arg* Lexer_setToken(Arg* tokens_arg,
enum TokenType token_type,
char*
operator) {
2021-11-26 20:19:41 +08:00
Args* buffs = New_strBuff();
char token_type_buff[3] = {0};
2021-11-27 17:26:06 +08:00
token_type_buff[0] = 0x1F;
2021-11-26 21:54:34 +08:00
token_type_buff[1] = token_type;
char* tokens = arg_getStr(tokens_arg);
2021-11-26 20:19:41 +08:00
tokens = strsAppend(buffs, tokens, token_type_buff);
tokens = strsAppend(buffs, tokens, operator);
Arg* new_tokens_arg = arg_setStr(tokens_arg, "", tokens);
arg_deinit(tokens_arg);
2021-11-26 20:19:41 +08:00
args_deinit(buffs);
return new_tokens_arg;
2021-11-26 20:19:41 +08:00
}
Arg* Lexer_setSymbel(Arg* tokens_arg,
char* stmt,
int32_t i,
int32_t* symbol_start_index) {
Args* buffs = New_strBuff();
2021-12-10 16:22:19 +08:00
char* symbol_buff = NULL;
/* nothing to add symbel */
2021-11-27 00:30:21 +08:00
if (i == *symbol_start_index) {
*symbol_start_index = -1;
goto exit;
2021-11-27 00:30:21 +08:00
}
2021-12-10 16:22:19 +08:00
symbol_buff = args_getBuff(buffs, i - *symbol_start_index);
2021-12-10 11:41:33 +08:00
__platform_memcpy(symbol_buff, stmt + *symbol_start_index,
i - *symbol_start_index);
2021-11-27 17:26:06 +08:00
/* literal */
if ((symbol_buff[0] == '-') || (symbol_buff[0] == '\'') ||
(symbol_buff[0] == '"') ||
2021-11-27 01:13:24 +08:00
((symbol_buff[0] >= '0') && (symbol_buff[0] <= '9'))) {
tokens_arg = Lexer_setToken(tokens_arg, TOKEN_literal, symbol_buff);
2021-11-27 18:05:32 +08:00
} else {
/* symbol */
tokens_arg = Lexer_setToken(tokens_arg, TOKEN_symbol, symbol_buff);
2021-11-27 01:13:24 +08:00
}
2021-11-27 00:30:21 +08:00
*symbol_start_index = -1;
exit:
args_deinit(buffs);
return tokens_arg;
2021-11-27 00:30:21 +08:00
}
2021-11-26 20:19:41 +08:00
/* tokens is devided by space */
/* a token is [TOKENTYPE|(CONTENT)] */
char* Lexer_getTokens(Args* outBuffs, char* stmt) {
/* init */
Arg* tokens_arg = New_arg(NULL);
tokens_arg = arg_setStr(tokens_arg, "", "");
int32_t size = strGetSize(stmt);
2021-12-10 11:41:33 +08:00
uint8_t bracket_deepth = 0;
2021-11-26 20:19:41 +08:00
uint8_t c0 = 0;
uint8_t c1 = 0;
uint8_t c2 = 0;
uint8_t c3 = 0;
int32_t symbol_start_index = -1;
2021-11-27 17:26:06 +08:00
int is_in_string = 0;
2021-12-10 11:41:33 +08:00
char* tokens;
/* process */
for (int32_t i = 0; i < size; i++) {
2021-11-26 20:19:41 +08:00
/* update char */
c0 = stmt[i];
if (i + 1 < size) {
c1 = stmt[i + 1];
}
if (i + 2 < size) {
c2 = stmt[i + 2];
}
if (i + 3 < size) {
c3 = stmt[i + 3];
}
2021-11-27 00:30:21 +08:00
if (-1 == symbol_start_index) {
symbol_start_index = i;
}
2021-11-27 17:26:06 +08:00
/* solve string */
if (0 == is_in_string) {
if ('\'' == c0) {
/* in ' */
is_in_string = 1;
continue;
}
if ('"' == c0) {
/* in "" */
is_in_string = 2;
continue;
}
}
if (1 == is_in_string) {
if ('\'' == c0) {
is_in_string = 0;
tokens_arg = Lexer_setSymbel(tokens_arg, stmt, i + 1,
&symbol_start_index);
}
continue;
}
if (2 == is_in_string) {
if ('"' == c0) {
is_in_string = 0;
tokens_arg = Lexer_setSymbel(tokens_arg, stmt, i + 1,
&symbol_start_index);
}
continue;
}
2021-11-27 11:58:15 +08:00
/* match devider*/
2021-12-20 15:55:35 +08:00
if (('(' == c0) || (')' == c0) || (',' == c0) || ('[' == c0) ||
2021-12-21 00:04:39 +08:00
(']' == c0) || (':' == c0)) {
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
2021-11-26 23:30:23 +08:00
char content[2] = {0};
content[0] = c0;
2021-11-27 11:58:15 +08:00
tokens_arg = Lexer_setToken(tokens_arg, TOKEN_devider, content);
2021-12-10 11:41:33 +08:00
if (c0 == '(') {
bracket_deepth++;
}
if (c0 == ')') {
bracket_deepth--;
}
2021-11-26 22:18:01 +08:00
continue;
}
2021-11-26 23:30:23 +08:00
/* match operator */
if (('>' == c0) || ('<' == c0) || ('*' == c0) || ('/' == c0) ||
('+' == c0) || ('-' == c0) || ('!' == c0) || ('=' == c0) ||
('%' == c0) || ('&' == c0) || ('|' == c0) || ('^' == c0) ||
('~' == c0)) {
if (('*' == c0) || ('/' == c0)) {
/*
//=, **=
*/
if ((c0 == c1) && ('=' == c2)) {
char content[4] = {0};
content[0] = c0;
content[1] = c1;
content[2] = '=';
tokens_arg = Lexer_setSymbel(tokens_arg, stmt, i,
&symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-26 23:30:23 +08:00
i = i + 2;
continue;
}
2021-11-26 21:56:46 +08:00
}
2021-11-26 23:30:23 +08:00
/*
>>, <<, **, //
*/
if (('>' == c0) || ('<' == c0) || ('*' == c0) || ('/' == c0)) {
if (c0 == c1) {
char content[3] = {0};
content[0] = c0;
content[1] = c1;
tokens_arg = Lexer_setSymbel(tokens_arg, stmt, i,
&symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-26 23:30:23 +08:00
i = i + 1;
continue;
}
2021-11-26 22:18:01 +08:00
}
2021-11-26 23:30:23 +08:00
/*
>=, <=, *=, /=, +=, -=, !=, ==, %=
*/
if (('>' == c0) || ('<' == c0) || ('*' == c0) || ('/' == c0) ||
('+' == c0) || ('-' == c0) || ('!' == c0) || ('=' == c0) ||
('%' == c0)) {
if ('=' == c1) {
char content[3] = {0};
content[0] = c0;
content[1] = c1;
tokens_arg = Lexer_setSymbel(tokens_arg, stmt, i,
&symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-26 23:30:23 +08:00
i = i + 1;
continue;
}
2021-11-26 22:18:01 +08:00
}
2021-11-26 23:30:23 +08:00
/* single */
char content[2] = {0};
content[0] = c0;
2021-11-27 01:13:24 +08:00
if ('-' != c0) {
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-27 01:13:24 +08:00
continue;
}
/* when c0 is '-' */
if (!((c1 >= '0') && (c1 <= '9'))) {
/* is a '-' */
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-27 01:13:24 +08:00
continue;
}
if (i != symbol_start_index) {
/* is a '-' */
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, content);
2021-11-27 01:13:24 +08:00
continue;
}
/* is a symbel */
2021-11-26 22:20:49 +08:00
continue;
}
2021-11-26 22:18:01 +08:00
/* not */
if ('n' == c0) {
if (('o' == c1) && ('t' == c2) && (' ' == c3)) {
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
2021-11-28 17:02:27 +08:00
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, " not ");
2021-11-26 22:18:01 +08:00
i = i + 3;
continue;
}
}
/* and */
if ('a' == c0) {
if (('n' == c1) && ('d' == c2) && (' ' == c3)) {
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
2021-11-28 17:02:27 +08:00
tokens_arg =
Lexer_setToken(tokens_arg, TOKEN_operator, " and ");
2021-11-26 22:18:01 +08:00
i = i + 3;
continue;
}
}
/* or */
if ('o' == c0) {
if (('r' == c1) && (' ' == c2)) {
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
2021-11-28 17:02:27 +08:00
tokens_arg = Lexer_setToken(tokens_arg, TOKEN_operator, " or ");
2021-11-26 22:18:01 +08:00
i = i + 2;
continue;
}
}
2021-11-27 00:30:21 +08:00
/* skip spaces */
2021-11-27 17:26:06 +08:00
if (' ' == c0) {
/* not get symbal */
if (i == symbol_start_index) {
symbol_start_index++;
} else {
/* already get symbal */
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, i, &symbol_start_index);
}
2021-11-27 00:30:21 +08:00
}
2021-11-27 01:13:24 +08:00
if (i == size - 1) {
/* last check symbel */
tokens_arg =
Lexer_setSymbel(tokens_arg, stmt, size, &symbol_start_index);
2021-11-27 01:13:24 +08:00
}
2021-11-26 20:19:41 +08:00
}
2021-12-10 11:41:33 +08:00
if (0 != bracket_deepth) {
/* bracket match error */
tokens = NULL;
goto exit;
}
/* output */
2021-12-10 11:41:33 +08:00
tokens = arg_getStr(tokens_arg);
2021-11-26 20:19:41 +08:00
tokens = strsCopy(outBuffs, tokens);
2021-12-10 11:41:33 +08:00
exit:
arg_deinit(tokens_arg);
2021-11-26 20:19:41 +08:00
return tokens;
}
2021-12-21 00:04:39 +08:00
char* Lexer_popToken(Args* buffs, char* tokens_buff) {
2021-12-20 16:16:13 +08:00
return strsPopToken(buffs, tokens_buff, 0x1F);
}
2021-11-27 20:11:35 +08:00
uint8_t Lexer_isContain(char* tokens, char* operator) {
Args* buffs = New_strBuff();
char* tokens_buff = strsCopy(buffs, tokens);
uint8_t res = 0;
uint16_t token_size = strCountSign(tokens, 0x1F) + 1;
for (int i = 0; i < token_size; i++) {
2021-12-20 16:16:13 +08:00
char* token = Lexer_popToken(buffs, tokens_buff);
2021-11-27 20:11:35 +08:00
if (TOKEN_operator == token[0]) {
if (strEqu(token + 1, operator)) {
res = 1;
goto exit;
}
}
}
exit:
args_deinit(buffs);
return res;
}
char* Lexer_getOperator(Args* outBuffs, char* stmt) {
Args* buffs = New_strBuff();
char* tokens = Lexer_getTokens(buffs, stmt);
char* operator= NULL;
2021-11-28 17:26:17 +08:00
const char operators[][6] = {
2021-11-28 17:02:27 +08:00
"**", "~", "*", "/", "%", "//", "+", "-", ">>", "<<",
"&", "^", "|", "<", "<=", ">", ">=", "!=", "==", "%=",
"/=", "//=", "-=", "+=", "*=", "**=", " not ", " and ", " or "};
for (uint32_t i = 0; i < sizeof(operators) / 6; i++) {
2021-11-27 21:22:19 +08:00
if (Lexer_isContain(tokens, (char*)operators[i])) {
operator= strsCopy(buffs, (char*)operators[i]);
2021-11-27 21:06:01 +08:00
}
2021-11-27 20:11:35 +08:00
}
/* out put */
operator= strsCopy(outBuffs, operator);
args_deinit(buffs);
return operator;
}
2021-10-13 19:41:04 +08:00
AST* AST_parseStmt(AST* ast, char* stmt) {
2021-10-13 18:05:54 +08:00
Args* buffs = New_strBuff();
2021-10-13 18:24:38 +08:00
char* assignment = strsGetFirstToken(buffs, stmt, '(');
2021-10-13 18:05:54 +08:00
char* direct = NULL;
2021-10-13 18:24:38 +08:00
char* method = NULL;
2021-10-13 21:20:45 +08:00
char* ref = NULL;
2021-10-14 09:51:50 +08:00
char* str = NULL;
2021-10-14 15:08:45 +08:00
char* num = NULL;
2021-10-14 15:50:42 +08:00
char* right = NULL;
2021-10-20 20:40:45 +08:00
/* solve direct */
uint8_t directExist = 0;
2021-11-27 20:49:08 +08:00
if (Parser_checkIsDirect(assignment)) {
2021-10-13 21:20:45 +08:00
directExist = 1;
}
if (directExist) {
2021-10-13 18:05:54 +08:00
direct = strsGetFirstToken(buffs, assignment, '=');
2021-10-13 20:07:32 +08:00
obj_setStr(ast, (char*)"direct", direct);
2021-10-13 18:05:54 +08:00
}
2021-10-20 20:40:45 +08:00
/* solve right stmt */
if (directExist) {
right = strsGetLastToken(buffs, stmt, '=');
} else {
right = stmt;
}
2021-11-27 21:06:01 +08:00
enum StmtType stmtType = Lexer_matchStmtType(right);
/* solve operator stmt */
2021-11-26 18:59:09 +08:00
if (STMT_operator == stmtType) {
2021-11-27 20:49:08 +08:00
char* rightWithoutSubStmt = strsDeleteBetween(buffs, right, '(', ')');
2021-11-27 20:11:35 +08:00
char* operator= Lexer_getOperator(buffs, rightWithoutSubStmt);
2021-10-20 21:33:16 +08:00
obj_setStr(ast, (char*)"operator", operator);
2021-10-20 21:53:10 +08:00
char* rightBuff = strsCopy(buffs, right);
char* subStmt1 =
2021-11-27 20:49:08 +08:00
strsPopTokenWithSkip_byStr(buffs, rightBuff, operator, '(', ')');
2021-10-20 21:53:10 +08:00
char* subStmt2 = rightBuff;
2021-10-20 21:33:16 +08:00
queueObj_pushObj(ast, (char*)"stmt");
AST_parseStmt(queueObj_getCurrentObj(ast), subStmt1);
queueObj_pushObj(ast, (char*)"stmt");
AST_parseStmt(queueObj_getCurrentObj(ast), subStmt2);
goto exit;
}
2021-11-27 21:06:01 +08:00
/* solve method stmt */
2021-11-26 18:59:09 +08:00
if (STMT_method == stmtType) {
2021-10-20 20:40:45 +08:00
method = strsGetFirstToken(buffs, right, '(');
2021-10-13 21:20:45 +08:00
obj_setStr(ast, (char*)"method", method);
2021-10-20 20:40:45 +08:00
char* subStmts = strsCut(buffs, stmt, '(', ')');
2021-10-13 20:18:12 +08:00
while (1) {
char* subStmt =
strsPopTokenWithSkip(buffs, subStmts, ',', '(', ')');
2021-10-13 20:18:12 +08:00
if (NULL == subStmt) {
break;
}
queueObj_pushObj(ast, (char*)"stmt");
AST_parseStmt(queueObj_getCurrentObj(ast), subStmt);
2021-10-13 20:07:32 +08:00
}
2021-10-14 09:51:50 +08:00
goto exit;
}
2021-10-20 20:40:45 +08:00
/* solve reference stmt */
2021-11-26 18:59:09 +08:00
if (STMT_reference == stmtType) {
2021-10-14 15:50:42 +08:00
ref = right;
2021-10-13 21:20:45 +08:00
obj_setStr(ast, (char*)"ref", ref);
2021-10-14 09:51:50 +08:00
goto exit;
}
2021-10-20 20:40:45 +08:00
/* solve str stmt */
2021-11-26 18:59:09 +08:00
if (STMT_string == stmtType) {
2021-10-14 15:50:42 +08:00
str = right;
2021-10-14 09:51:50 +08:00
str = strsDeleteChar(buffs, str, '\'');
str = strsDeleteChar(buffs, str, '\"');
2021-11-27 19:08:20 +08:00
obj_setStr(ast, (char*)"string", str);
2021-10-14 09:51:50 +08:00
goto exit;
2021-10-13 20:07:32 +08:00
}
2021-10-20 20:40:45 +08:00
/* solve number stmt */
2021-11-26 18:59:09 +08:00
if (STMT_number == stmtType) {
2021-10-14 15:50:42 +08:00
num = right;
2021-10-14 15:08:45 +08:00
obj_setStr(ast, (char*)"num", num);
goto exit;
}
2021-10-13 18:05:54 +08:00
exit:
args_deinit(buffs);
return ast;
}
2021-11-27 20:49:08 +08:00
static int32_t Parser_getPyLineBlockDeepth(char* line) {
2021-10-16 17:32:02 +08:00
uint32_t size = strGetSize(line);
for (uint32_t i = 0; i < size; i++) {
2021-10-16 17:32:02 +08:00
if (line[i] != ' ') {
uint32_t spaceNum = i;
if (0 == spaceNum % 4) {
return spaceNum / 4;
}
/* space Num is not 4N, error*/
return -1;
}
}
2021-10-19 09:13:10 +08:00
return 0;
2021-10-16 17:32:02 +08:00
}
2021-12-10 20:09:11 +08:00
char* Parser_removeAnnotation(char* line) {
uint8_t is_annotation_exit = 0;
for (uint32_t i = 0; i < strGetSize(line); i++) {
2021-12-10 20:09:11 +08:00
if ('#' == line[i]) {
/* end the line */
line[i] = 0;
is_annotation_exit = 1;
break;
}
}
/* no annotation, exit */
if (!is_annotation_exit) {
return line;
}
/* check empty line */
for (uint32_t i = 0; i < strGetSize(line); i++) {
2021-12-10 20:09:11 +08:00
if (' ' != line[i]) {
return line;
}
}
/* is an emply line */
line[0] = '#';
line[1] = 0;
return line;
}
2021-11-27 20:49:08 +08:00
AST* AST_parseLine(char* line, Stack* blockStack) {
2021-10-13 19:41:04 +08:00
AST* ast = New_queueObj();
Args* buffs = New_strBuff();
2021-10-19 22:26:34 +08:00
line = strsDeleteChar(buffs, line, '\r');
2021-12-10 20:09:11 +08:00
line = Parser_removeAnnotation(line);
2021-12-11 10:19:19 +00:00
uint8_t blockDeepth;
uint8_t blockDeepthLast;
char* lineStart;
char* stmt;
2021-12-10 20:09:11 +08:00
if (strEqu("#", line)) {
obj_setStr(ast, "annotation", "annotation");
goto exit;
}
2021-12-14 19:55:33 +08:00
/* get block deepth */
2021-12-11 10:19:19 +00:00
blockDeepth = Parser_getPyLineBlockDeepth(line);
blockDeepthLast = blockDeepth;
2021-12-10 20:09:11 +08:00
/* in block */
2021-10-16 19:52:19 +08:00
if (NULL != blockStack) {
blockDeepthLast = args_getInt(blockStack, "top");
2021-12-14 19:55:33 +08:00
/* check if exit block */
2021-10-16 19:52:19 +08:00
for (int i = 0; i < blockDeepthLast - blockDeepth; i++) {
QueueObj* exitBlock = obj_getObj(ast, "exitBlock", 0);
if (NULL == exitBlock) {
obj_newObj(ast, "exitBlock", "", New_TinyObj);
2021-10-16 19:52:19 +08:00
exitBlock = obj_getObj(ast, "exitBlock", 0);
queueObj_init(exitBlock);
}
char buff[10] = {0};
char* blockType = stack_popStr(blockStack, buff);
queueObj_pushStr(exitBlock, blockType);
}
}
2021-12-14 19:55:33 +08:00
/* set block deepth */
2021-10-16 17:48:01 +08:00
obj_setInt(ast, "blockDeepth", blockDeepth);
2021-12-11 10:19:19 +00:00
lineStart = line + blockDeepth * 4;
stmt = lineStart;
2021-12-14 19:55:33 +08:00
/* match block start */
2021-10-16 17:48:01 +08:00
if (0 == strncmp(lineStart, (char*)"while ", 6)) {
stmt = strsCut(buffs, lineStart, ' ', ':');
2021-10-31 16:34:41 +08:00
obj_setStr(ast, "block", "while");
2021-10-16 19:09:19 +08:00
if (NULL != blockStack) {
stack_pushStr(blockStack, "while");
}
2021-12-14 19:55:33 +08:00
goto block_matched;
2021-10-16 17:48:01 +08:00
}
if (0 == strncmp(lineStart, (char*)"for ", 4)) {
2021-12-25 14:19:20 +08:00
Args* list_buffs = New_strBuff();
2021-12-29 00:20:46 +08:00
char* line_buff = strsCopy(list_buffs, lineStart + 4);
char* arg_in = strsPopToken(list_buffs, line_buff, ' ');
obj_setStr(ast, "arg_in", arg_in);
strsPopToken(list_buffs, line_buff, ' ');
if (strIsStartWith(line_buff, "range(")) {
2021-12-28 00:49:13 +08:00
obj_setInt(ast, "isRange", 1);
}
2021-12-29 00:20:46 +08:00
char* list_in = strsPopToken(list_buffs, line_buff, ':');
2021-12-27 21:27:07 +08:00
list_in = strsAppend(list_buffs, "iter(", list_in);
list_in = strsAppend(list_buffs, list_in, ")");
list_in = strsCopy(buffs, list_in);
args_deinit(list_buffs);
2021-12-23 10:09:00 +08:00
2021-12-27 21:27:07 +08:00
obj_setStr(ast, "block", "for");
obj_setStr(ast, "list_in", list_in);
if (NULL != blockStack) {
stack_pushStr(blockStack, "for");
2021-12-22 22:01:31 +08:00
}
2021-12-27 21:27:07 +08:00
stmt = list_in;
goto block_matched;
2021-12-22 22:01:31 +08:00
}
2021-10-16 17:48:01 +08:00
if (0 == strncmp(lineStart, (char*)"if ", 3)) {
stmt = strsCut(buffs, lineStart, ' ', ':');
2021-10-31 16:34:41 +08:00
obj_setStr(ast, "block", "if");
2021-10-16 19:09:19 +08:00
if (NULL != blockStack) {
stack_pushStr(blockStack, "if");
}
2021-12-14 19:55:33 +08:00
goto block_matched;
}
if (0 == strncmp(lineStart, (char*)"elif ", 5)) {
stmt = strsCut(buffs, lineStart, ' ', ':');
obj_setStr(ast, "block", "elif");
if (NULL != blockStack) {
stack_pushStr(blockStack, "elif");
}
goto block_matched;
}
if (0 == strncmp(lineStart, (char*)"else", 4)) {
if ((lineStart[4] == ' ') || (lineStart[4] == ':')) {
stmt = "";
2021-12-14 21:17:36 +08:00
obj_setStr(ast, "block", "else");
2021-12-14 19:55:33 +08:00
if (NULL != blockStack) {
2021-12-14 21:17:36 +08:00
stack_pushStr(blockStack, "else");
2021-12-14 19:55:33 +08:00
}
}
goto block_matched;
2021-10-16 17:48:01 +08:00
}
2021-10-31 20:48:34 +08:00
if (strEqu(lineStart, (char*)"return")) {
obj_setStr(ast, "return", "");
2021-10-31 20:59:14 +08:00
stmt = "";
2021-12-14 19:55:33 +08:00
goto block_matched;
2021-10-31 20:48:34 +08:00
}
if (0 == strncmp(lineStart, (char*)"return ", 7)) {
char* lineBuff = strsCopy(buffs, lineStart);
strsPopToken(buffs, lineBuff, ' ');
stmt = lineBuff;
obj_setStr(ast, "return", "");
2021-12-14 19:55:33 +08:00
goto block_matched;
2021-10-31 20:48:34 +08:00
}
2021-10-31 16:48:53 +08:00
if (0 == strncmp(lineStart, (char*)"def ", 4)) {
stmt = "";
char* declear = strsCut(buffs, lineStart, ' ', ':');
declear = strsGetCleanCmd(buffs, declear);
obj_setStr(ast, "block", "def");
obj_setStr(ast, "declear", declear);
if (NULL != blockStack) {
stack_pushStr(blockStack, "def");
}
2021-12-14 19:55:33 +08:00
goto block_matched;
2021-10-31 16:48:53 +08:00
}
2021-12-14 19:55:33 +08:00
block_matched:
2021-10-16 17:48:01 +08:00
stmt = strsGetCleanCmd(buffs, stmt);
2021-10-13 19:41:04 +08:00
ast = AST_parseStmt(ast, stmt);
goto exit;
exit:
args_deinit(buffs);
return ast;
}
2021-10-13 22:02:53 +08:00
2021-11-27 20:49:08 +08:00
char* Parser_LineToAsm(Args* buffs, char* line, Stack* blockStack) {
2021-12-10 11:41:33 +08:00
char* pikaAsm = NULL;
AST* ast = NULL;
if (Lexer_isError(line)) {
pikaAsm = NULL;
goto exit;
}
ast = AST_parseLine(line, blockStack);
2021-12-10 20:09:11 +08:00
if (obj_isArgExist(ast, "annotation")) {
pikaAsm = strsCopy(buffs, "");
goto exit;
}
2021-12-10 11:41:33 +08:00
pikaAsm = AST_toPikaAsm(ast, buffs);
exit:
if (NULL != ast) {
AST_deinit(ast);
}
2021-10-14 10:44:59 +08:00
return pikaAsm;
}
2021-11-27 20:49:08 +08:00
static Arg* ASM_saveSingleAsm(Args* buffs,
Arg* pikaAsmBuff,
char* singleAsm,
uint8_t isToFlash) {
if (isToFlash) {
2021-12-07 19:17:36 +08:00
uint8_t saveErr = __platform_save_pikaAsm(singleAsm);
2021-10-27 21:06:52 +08:00
if (0 == saveErr) {
if (NULL != pikaAsmBuff) {
arg_deinit(pikaAsmBuff);
}
return NULL;
}
}
2021-10-27 21:06:52 +08:00
2021-10-27 15:56:25 +08:00
char* pikaAsm = arg_getStr(pikaAsmBuff);
pikaAsm = strsAppend(buffs, pikaAsm, singleAsm);
arg_deinit(pikaAsmBuff);
pikaAsmBuff = arg_setStr(NULL, "", pikaAsm);
return pikaAsmBuff;
}
2021-11-27 20:49:08 +08:00
static char* ASM_getOutAsm(Args* outBuffs,
Arg* pikaAsmBuff,
uint8_t isToFlash) {
2021-10-27 21:06:52 +08:00
if (isToFlash) {
2021-12-07 19:17:36 +08:00
return __platform_load_pikaAsm();
}
2021-10-27 15:56:25 +08:00
return strsCopy(outBuffs, arg_getStr(pikaAsmBuff));
}
2021-11-27 20:49:08 +08:00
char* Parser_multiLineToAsm(Args* outBuffs, char* multiLine) {
2021-10-17 19:07:33 +08:00
Stack* blockStack = New_Stack();
2021-10-17 19:35:50 +08:00
Arg* pikaAsmBuff = arg_setStr(NULL, "", "");
2021-10-27 15:24:42 +08:00
uint32_t lineOffset = 0;
uint32_t multiLineSize = strGetSize(multiLine);
2021-12-07 19:17:36 +08:00
uint8_t isToFlash = __platform_Asm_is_to_flash(multiLine);
2021-12-10 11:41:33 +08:00
char* outAsm = NULL;
2021-10-17 19:07:33 +08:00
while (1) {
Args* singleRunBuffs = New_strBuff();
2021-10-27 15:56:25 +08:00
char* line =
strsGetFirstToken(singleRunBuffs, multiLine + lineOffset, '\n');
2021-10-27 15:24:42 +08:00
uint32_t lineSize = strGetSize(line);
lineOffset = lineOffset + lineSize + 1;
2021-11-27 20:49:08 +08:00
char* singleAsm = Parser_LineToAsm(singleRunBuffs, line, blockStack);
2021-12-10 11:41:33 +08:00
if (NULL == singleAsm) {
outAsm = NULL;
args_deinit(singleRunBuffs);
goto exit;
}
2021-11-27 20:49:08 +08:00
pikaAsmBuff = ASM_saveSingleAsm(singleRunBuffs, pikaAsmBuff, singleAsm,
isToFlash);
2021-10-17 19:07:33 +08:00
args_deinit(singleRunBuffs);
2021-10-27 15:44:35 +08:00
if (lineOffset >= multiLineSize) {
2021-10-17 19:07:33 +08:00
break;
}
}
2021-10-31 16:34:41 +08:00
if (isToFlash) {
2021-12-07 19:17:36 +08:00
__platform_save_pikaAsm_EOF();
2021-10-27 21:06:52 +08:00
}
2021-12-10 11:41:33 +08:00
outAsm = ASM_getOutAsm(outBuffs, pikaAsmBuff, isToFlash);
goto exit;
exit:
2021-10-27 16:02:15 +08:00
if (NULL != pikaAsmBuff) {
arg_deinit(pikaAsmBuff);
}
2021-10-17 19:07:33 +08:00
stack_deinit(blockStack);
2021-10-27 15:56:25 +08:00
return outAsm;
2021-10-17 19:07:33 +08:00
}
2021-10-13 22:17:40 +08:00
char* AST_appandPikaAsm(AST* ast, AST* subAst, Args* buffs, char* pikaAsm) {
2021-12-07 15:14:21 +08:00
int deepth = obj_getInt(ast, "deepth");
2021-10-13 21:20:45 +08:00
while (1) {
2021-10-13 22:02:53 +08:00
QueueObj* subStmt = queueObj_popObj(subAst);
2021-10-13 21:20:45 +08:00
if (NULL == subStmt) {
break;
}
2021-10-13 22:02:53 +08:00
obj_setInt(ast, "deepth", deepth + 1);
2021-10-13 22:17:40 +08:00
pikaAsm = AST_appandPikaAsm(ast, subStmt, buffs, pikaAsm);
2021-10-13 21:20:45 +08:00
}
2021-10-13 22:02:53 +08:00
char* method = obj_getStr(subAst, "method");
2021-10-20 21:42:32 +08:00
char* operator= obj_getStr(subAst, "operator");
2021-10-13 22:02:53 +08:00
char* ref = obj_getStr(subAst, "ref");
char* direct = obj_getStr(subAst, "direct");
2021-11-27 19:08:20 +08:00
char* str = obj_getStr(subAst, "string");
2021-10-14 15:08:45 +08:00
char* num = obj_getStr(subAst, "num");
2021-10-13 21:20:45 +08:00
if (NULL != ref) {
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d REF %s\n", deepth, ref);
2021-10-13 22:17:40 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
2021-10-13 21:20:45 +08:00
}
2021-10-20 21:42:32 +08:00
if (NULL != operator) {
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d OPT %s\n", deepth, operator);
2021-10-20 21:42:32 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
}
2021-10-13 21:20:45 +08:00
if (NULL != method) {
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d RUN %s\n", deepth, method);
2021-10-13 22:17:40 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
2021-10-13 21:20:45 +08:00
}
2021-10-14 09:51:50 +08:00
if (NULL != str) {
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d STR %s\n", deepth, str);
2021-10-14 09:51:50 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
}
2021-10-14 15:50:42 +08:00
if (NULL != num) {
2021-10-13 21:20:45 +08:00
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d NUM %s\n", deepth, num);
2021-10-13 22:17:40 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
2021-10-13 21:20:45 +08:00
}
2021-10-14 15:50:42 +08:00
if (NULL != direct) {
2021-10-14 15:08:45 +08:00
char buff[32] = {0};
2021-12-07 19:17:36 +08:00
__platform_sprintf(buff, "%d OUT %s\n", deepth, direct);
2021-10-14 15:08:45 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, buff);
}
2021-10-13 22:02:53 +08:00
obj_setInt(ast, "deepth", deepth - 1);
2021-10-13 22:17:40 +08:00
return pikaAsm;
2021-10-13 21:20:45 +08:00
}
2021-11-27 20:49:08 +08:00
static char* ASM_addBlockDeepth(AST* ast,
Args* buffs,
char* pikaAsm,
uint8_t deepthOffset) {
2021-10-16 17:32:02 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"B");
char buff[11];
2021-10-17 16:19:43 +08:00
pikaAsm = strsAppend(
buffs, pikaAsm,
fast_itoa(buff, obj_getInt(ast, "blockDeepth") + deepthOffset));
2021-10-16 17:32:02 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"\n");
2021-10-19 09:13:10 +08:00
return pikaAsm;
2021-10-16 19:52:19 +08:00
}
char* AST_toPikaAsm(AST* ast, Args* buffs) {
Args* runBuffs = New_strBuff();
char* pikaAsm = strsCopy(runBuffs, "");
QueueObj* exitBlock = obj_getObj(ast, "exitBlock", 0);
2021-12-14 19:55:33 +08:00
/* exiting from block */
if (exitBlock != NULL) {
2021-10-16 19:52:19 +08:00
while (1) {
2021-10-17 16:19:43 +08:00
uint8_t blockTypeNum = obj_getInt(exitBlock, "top") -
obj_getInt(exitBlock, "bottom") - 1;
2021-10-16 19:52:19 +08:00
char* blockType = queueObj_popStr(exitBlock);
if (NULL == blockType) {
break;
}
2021-10-17 18:08:14 +08:00
/* goto the while start when exit while block */
2021-12-23 10:26:51 +08:00
if (strEqu(blockType, "while")) {
2021-11-27 20:49:08 +08:00
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, blockTypeNum);
2021-10-16 19:52:19 +08:00
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"0 JMP -1\n");
}
2021-12-23 10:26:51 +08:00
/* goto the while start when exit while block */
if (strEqu(blockType, "for")) {
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, blockTypeNum);
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"0 JMP -1\n");
2021-12-24 13:00:39 +08:00
/* garbage collect for the list */
2021-12-28 00:49:13 +08:00
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, blockTypeNum);
char* __list_x = strsCopy(buffs, "__list");
2021-12-27 21:29:35 +08:00
char block_deepth_str[] = "0";
block_deepth_str[0] += obj_getInt(ast, "blockDeepth");
__list_x = strsAppend(runBuffs, __list_x, block_deepth_str);
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"0 DEL ");
pikaAsm = strsAppend(buffs, pikaAsm, (char*)__list_x);
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"\n");
2021-12-23 10:26:51 +08:00
}
2021-12-25 18:48:23 +08:00
/* goto the while start when exit while block */
if (strEqu(blockType, "for_range")) {
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, blockTypeNum);
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"0 JMP -1\n");
}
2021-10-31 20:11:55 +08:00
/* return when exit method */
if (strEqu(blockType, "def")) {
pikaAsm = strsAppend(buffs, pikaAsm, (char*)"0 RET\n");
}
2021-10-16 19:52:19 +08:00
}
}
2021-12-14 19:55:33 +08:00
/* add block deepth */
/* example: B0 */
2021-11-27 20:49:08 +08:00
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, 0);
2021-12-14 21:32:53 +08:00
/* "deepth" is invoke deepth, not the blockDeepth */
2021-10-13 22:02:53 +08:00
obj_setInt(ast, "deepth", 0);
2021-10-20 21:42:32 +08:00
2021-12-14 21:17:36 +08:00
/* match block */
2021-12-14 22:17:37 +08:00
uint8_t is_block_matched = 0;
2021-12-23 10:09:00 +08:00
if (strEqu(obj_getStr(ast, "block"), "for")) {
2021-12-22 22:01:31 +08:00
/* for "for" iter */
2021-12-23 10:09:00 +08:00
char* arg_in = obj_getStr(ast, "arg_in");
Arg* newAsm_arg = arg_setStr(NULL, "", "");
char __list_x[] = "__listx";
char block_deepth_char = '0';
block_deepth_char += obj_getInt(ast, "blockDeepth");
__list_x[sizeof(__list_x) - 2] = block_deepth_char;
2021-12-23 10:09:00 +08:00
/* init iter */
2021-12-23 10:39:47 +08:00
/* get the iter(__list<x>) */
2021-12-23 10:09:00 +08:00
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
newAsm_arg = arg_strAppend(newAsm_arg, "0 OUT ");
newAsm_arg = arg_strAppend(newAsm_arg, __list_x);
newAsm_arg = arg_strAppend(newAsm_arg, "\n");
2021-12-28 00:49:13 +08:00
if (1 == obj_getInt(ast, "isRange")) {
newAsm_arg = arg_strAppend(newAsm_arg, "0 REF __range_a1\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 REF __range_a2\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 REF __range_a3\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 OUT ");
newAsm_arg = arg_strAppend(newAsm_arg, __list_x);
newAsm_arg = arg_strAppend(newAsm_arg, ".a1\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 OUT ");
newAsm_arg = arg_strAppend(newAsm_arg, __list_x);
newAsm_arg = arg_strAppend(newAsm_arg, ".a2\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 OUT ");
newAsm_arg = arg_strAppend(newAsm_arg, __list_x);
newAsm_arg = arg_strAppend(newAsm_arg, ".a3\n");
2021-12-28 00:49:13 +08:00
}
pikaAsm = strsAppend(runBuffs, pikaAsm, arg_getStr(newAsm_arg));
arg_deinit(newAsm_arg);
newAsm_arg = arg_setStr(NULL, "", "");
2021-12-23 10:09:00 +08:00
/* get next */
/* run next(__list<x>) */
/* check item is exist */
pikaAsm = ASM_addBlockDeepth(ast, buffs, pikaAsm, 0);
newAsm_arg = arg_strAppend(newAsm_arg, "0 RUN ");
newAsm_arg = arg_strAppend(newAsm_arg, __list_x);
newAsm_arg = arg_strAppend(newAsm_arg, ".__next__\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 OUT ");
newAsm_arg = arg_strAppend(newAsm_arg, arg_in);
newAsm_arg = arg_strAppend(newAsm_arg, "\n");
newAsm_arg = arg_strAppend(newAsm_arg, "0 EST ");
newAsm_arg = arg_strAppend(newAsm_arg, arg_in);
newAsm_arg = arg_strAppend(newAsm_arg, "\n0 JEZ 2\n");
pikaAsm = strsAppend(runBuffs, pikaAsm, arg_getStr(newAsm_arg));
arg_deinit(newAsm_arg);
2021-12-23 10:09:00 +08:00
is_block_matched = 1;
goto exit;
}
if (strEqu(obj_getStr(ast, "block"), "while")) {
2021-12-14 22:17:37 +08:00
/* parse stmt ast */
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
2021-10-16 18:39:09 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 JEZ 2\n");
2021-12-14 22:17:37 +08:00
is_block_matched = 1;
goto exit;
2021-10-16 18:13:52 +08:00
}
2021-10-31 16:34:41 +08:00
if (strEqu(obj_getStr(ast, "block"), "if")) {
2021-12-14 22:17:37 +08:00
/* parse stmt ast */
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
2021-10-17 18:08:14 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 JEZ 1\n");
2021-12-14 22:17:37 +08:00
is_block_matched = 1;
goto exit;
2021-12-14 21:17:36 +08:00
}
if (strEqu(obj_getStr(ast, "block"), "else")) {
2021-12-14 22:17:37 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 NEL 1\n");
goto exit;
2021-12-14 21:17:36 +08:00
}
2021-12-14 21:32:53 +08:00
if (strEqu(obj_getStr(ast, "block"), "elif")) {
2021-12-14 21:17:36 +08:00
/* skip if __else is 0 */
2021-12-14 22:17:37 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 NEL 1\n");
/* parse stmt ast */
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
/* skip if stmt is 0 */
2021-12-14 21:17:36 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 JEZ 1\n");
2021-12-14 22:17:37 +08:00
is_block_matched = 1;
goto exit;
2021-10-17 18:08:14 +08:00
}
2021-10-31 16:48:53 +08:00
if (strEqu(obj_getStr(ast, "block"), "def")) {
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 DEF ");
pikaAsm = strsAppend(runBuffs, pikaAsm, obj_getStr(ast, "declear"));
pikaAsm = strsAppend(runBuffs, pikaAsm, "\n");
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 JMP 1\n");
2021-12-14 22:17:37 +08:00
is_block_matched = 1;
goto exit;
2021-10-31 16:48:53 +08:00
}
2021-10-31 20:48:34 +08:00
if (obj_isArgExist(ast, "return")) {
2021-12-14 22:17:37 +08:00
/* parse stmt ast */
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
2021-10-31 20:48:34 +08:00
pikaAsm = strsAppend(runBuffs, pikaAsm, "0 RET\n");
2021-12-14 22:17:37 +08:00
is_block_matched = 1;
goto exit;
2021-10-31 20:48:34 +08:00
}
2021-12-14 22:17:37 +08:00
exit:
if (!is_block_matched) {
/* parse stmt ast */
pikaAsm = AST_appandPikaAsm(ast, ast, runBuffs, pikaAsm);
}
2021-12-14 19:55:33 +08:00
/* output pikaAsm */
2021-10-14 10:05:55 +08:00
pikaAsm = strsCopy(buffs, pikaAsm);
args_deinit(runBuffs);
return pikaAsm;
2021-10-13 18:05:54 +08:00
}
2021-12-25 14:19:20 +08:00
2021-10-13 18:05:54 +08:00
int32_t AST_deinit(AST* ast) {
return obj_deinit(ast);
2021-11-14 18:39:08 +08:00
}