From 9a9e5cd3e001acc2b49c41c10fe2b47c92b356df Mon Sep 17 00:00:00 2001 From: Jacob Date: Sat, 4 Jan 2025 09:53:39 -0500 Subject: [PATCH] Fixed parsing of negative numbers. --- src/exec.c | 1 - src/grammar.y | 100 ++++++++++++++++---------------------------------- src/lexer.c | 2 +- 3 files changed, 33 insertions(+), 70 deletions(-) diff --git a/src/exec.c b/src/exec.c index ae2b956..d0744dd 100644 --- a/src/exec.c +++ b/src/exec.c @@ -22,7 +22,6 @@ ASTNumData exec_call(AST* ast) { fflush(stdout); ASTCallData* calldata = (ASTCallData*)ast->data; if (!strcmp(calldata->to, "sum") && calldata->argc == 2) { - ASTNumData n1 = exec_expr(calldata->argv[0]); ASTNumData n2 = exec_expr(calldata->argv[1]); diff --git a/src/grammar.y b/src/grammar.y index 878bc37..c3697cd 100644 --- a/src/grammar.y +++ b/src/grammar.y @@ -28,13 +28,19 @@ %token CALL %token NUM -%token NEG +%token SUB %token PLUS %token MULT %token DIV %token NL -%type exp + +%left DIV PLUS MULT SUB + +%precedence NEG + +%type num; +%type exp; %% @@ -43,89 +49,47 @@ input: | exp { root = $1; } ; -exp: - NUM { $$ = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); } - | NEG NUM { $$ = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); } +num: + NUM { $$ = $1; } + | SUB NUM { $$ = -$2; } %prec NEG + ; - | CALL LGROUP NUM SEP NUM RGROUP { +exp: + num { $$ = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); } + + // name(thing, thing) + | CALL LGROUP exp SEP exp RGROUP { AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($3)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($5)); + argv[0] = $3; + argv[1] = $5; $$ = ast_init(AST_TYPE_CALL, ast_call_data_init($1, 2, argv)); } - | NUM PLUS NUM { + | exp PLUS exp { AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($3)); + argv[0] = $1; + argv[1] = $3; $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("sum", 2, argv)); } - | NEG NUM PLUS NUM { + + | exp SUB exp { AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($4)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("sum", 2, argv)); - } - | NUM NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($3)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("sub", 2, argv)); - } - | NEG NUM NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($4)); + argv[0] = $1; + argv[1] = $3; $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("sub", 2, argv)); } - | NUM MULT NUM { + | exp MULT exp { AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($3)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("mul", 2, argv)); - } - | NEG NUM MULT NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($4)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("mul", 2, argv)); - } - | NEG NUM MULT NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$5)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("mul", 2, argv)); - } - | NUM MULT NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$4)); + argv[0] = $1; + argv[1] = $3; $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("mul", 2, argv)); } - | NUM DIV NUM { + | exp DIV exp { AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($3)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("div", 2, argv)); - } - | NEG NUM DIV NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init($4)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("div", 2, argv)); - } - | NEG NUM DIV NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$2)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$5)); - $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("div", 2, argv)); - } - | NUM DIV NEG NUM { - AST** argv = calloc(2, sizeof(AST*)); - argv[0] = ast_init(AST_TYPE_NUM, ast_num_data_init($1)); - argv[1] = ast_init(AST_TYPE_NUM, ast_num_data_init(-$4)); + argv[0] = $1; + argv[1] = $3; $$ = ast_init(AST_TYPE_CALL, ast_call_data_init("div", 2, argv)); } %% diff --git a/src/lexer.c b/src/lexer.c index 579b155..eea5788 100644 --- a/src/lexer.c +++ b/src/lexer.c @@ -72,7 +72,7 @@ int yylex() { switch (c) { case '+': return PLUS; case '\n': return NL; - case '-': return NEG; + case '-': return SUB; case '*': return MULT; case '/': return DIV; case '(': return LGROUP;