update

This commit is contained in:
ValKmjolnir 2023-06-19 00:10:14 +08:00
parent 78b7bec786
commit bcf274aed4
4 changed files with 406 additions and 224 deletions

View File

@ -9,9 +9,7 @@
enum class expr_type:u32 {
ast_null=0, // null node
ast_expr, // expression node
ast_root, // mark the root node of ast
ast_block, // expression block
ast_file, // used to store which file the sub-tree is on, only used in main block
ast_nil, // nil keyword
ast_num, // number, basic value type
ast_str, // string, basic value type
@ -26,11 +24,11 @@ enum class expr_type:u32 {
ast_callv, // id[index]
ast_callf, // id()
ast_subvec, // id[index:index]
ast_params, // mark a sub-tree of function parameters
ast_default, // default parameter
ast_dynamic, // dynamic parameter
ast_param, // function parameter
ast_and, // and keyword
ast_or, // or keyword
ast_equal, // =
ast_addeq, // +=
ast_subeq, // -=
@ -40,37 +38,18 @@ enum class expr_type:u32 {
ast_btandeq, // &=
ast_btoreq, // |=
ast_btxoreq, // ^=
ast_cmpeq, // ==
ast_neq, // !=
ast_less, // <
ast_leq, // <=
ast_grt, // >
ast_geq, // >=
ast_add, // +
ast_sub, // -
ast_mult, // *
ast_div, // /
ast_link, // ~
ast_neg, // unary -
ast_lnot, // unary !
ast_bnot, // unary ~ bitwise not
ast_bitor, // bitwise or
ast_bitxor, // bitwise xor
ast_bitand, // bitwise and
ast_trino, // ?:
ast_ternary,
ast_binary,
ast_unary,
ast_for, // for keyword
ast_forindex, // forindex keyword
ast_foreach, // foreach keyword
ast_forei, // foreach or forindex loop
ast_while, // while
ast_iter, // iterator, used in forindex/foreach
ast_cond, // mark a sub-tree of conditional expression
ast_if, // if keyword
ast_elsif, // elsif keyword
ast_else, // else keyword
ast_multi_id, // multi identifiers sub-tree
ast_tuple, // tuple, only used in multiple assignment
ast_def, // definition
ast_multi_assign,// multi assignment sub-tree
ast_continue, // continue keyword, only used in loop
ast_break, // break keyword, only used in loop
ast_ret // return keyword, only used in function block
@ -79,7 +58,9 @@ enum class expr_type:u32 {
class ast_visitor;
class hash_pair;
class parameter;
class multi_define;
class code_block;
class if_expr;
class ast_abstract_node {
public:
@ -94,6 +75,12 @@ public:
expr(const span& location, expr_type node_type):
nd_loc(location), nd_type(node_type) {}
~expr() = default;
const span& get_location() const {return nd_loc;}
expr_type get_type() const {return nd_type;}
void update_location(const span& location) {
nd_loc.end_line = location.end_line;
nd_loc.end_column = location.end_column;
}
virtual void accept(ast_visitor*);
};
@ -231,86 +218,267 @@ public:
class parameter:public expr {
public:
enum class param_type {
normal_parameter,
default_parameter,
dynamic_parameter
};
private:
param_type type;
identifier* name;
expr* default_value;
public:
parameter(const span& location):
expr(location, expr_type::ast_param),
name(nullptr), default_value(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class ternary_operator:public expr {
private:
expr* condition;
expr* left;
expr* right;
public:
ternary_operator(const span& location):
expr(location, expr_type::ast_ternary),
condition(nullptr), left(nullptr), right(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class binary_operator:public expr {
public:
enum class binary_type {
add,
sub,
mult,
div,
concat,
cmpeq,
cmpneq,
less,
leq,
grt,
geq,
bitwise_or,
bitwise_xor,
bitwise_and
};
private:
binary_type type;
expr* left;
expr* right;
public:
binary_operator(const span& location):
expr(location, expr_type::ast_binary),
left(nullptr), right(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class unary_operator:public expr {
public:
virtual void accept(ast_visitor*) override;
};
enum class unary_type {
negative,
logical_not,
bitwise_not
};
private:
unary_type type;
expr* value;
class call_expr:public expr {
public:
unary_operator(const span& location):
expr(location, expr_type::ast_unary),
value(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class call_hash:public expr {
private:
std::string field;
public:
call_hash(const span& location):
expr(location, expr_type::ast_callh),
field("") {}
virtual void accept(ast_visitor*) override;
};
class call_vector:public expr {
private:
std::vector<expr*> calls;
public:
call_vector(const span& location):
expr(location, expr_type::ast_callv) {}
virtual void accept(ast_visitor*) override;
};
class call_function:public expr {
public:
call_function(const span& location):
expr(location, expr_type::ast_callf) {}
virtual void accept(ast_visitor*) override;
};
class slice_vector:public expr {
private:
expr* begin;
expr* end;
public:
slice_vector(const span& location):
expr(location, expr_type::ast_subvec),
begin(nullptr), end(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class definition:public expr {
private:
identifier* variable;
multi_define* variables;
expr* value;
public:
definition(const span& location):
expr(location, expr_type::ast_def),
variable(nullptr), variables(nullptr), value(nullptr) {}
virtual void accept(ast_visitor*) override;
};
class multi_define:public expr {
private:
std::vector<expr*> variables;
public:
multi_define(const span& location):
expr(location, expr_type::ast_multi_id) {}
virtual void accept(ast_visitor*) override;
};
class while_expr:public expr {
private:
expr* condition;
code_block* block;
public:
while_expr(const span& location):
expr(location, expr_type::ast_while),
condition(nullptr), block(nullptr) {}
void set_condition(expr* node) {condition = node;}
void set_code_block (code_block* node) {block = node;}
expr* get_condition() {return condition;}
code_block* get_code_block() {return block;}
virtual void accept(ast_visitor*) override;
};
class for_expr:public expr {
private:
expr* initializing;
expr* condition;
expr* step;
code_block* block;
public:
for_expr(const span& location):
expr(location, expr_type::ast_for),
initializing(nullptr), condition(nullptr),
step(nullptr), block(nullptr) {}
void set_initial(expr* node) {initializing = node;}
void set_condition(expr* node) {condition = node;}
void set_step(expr* node) {step = node;}
void set_code_block(code_block* node) {block = node;}
expr* get_initial() {return initializing;}
expr* get_condition() {return condition;}
expr* get_step() {return step;}
code_block* get_code_block() {return block;}
virtual void accept(ast_visitor*) override;
};
class foreach_expr:public expr {
class iter_expr:public expr {
private:
identifier* name;
expr* value;
public:
iter_expr(const span& location):
expr(location, expr_type::ast_iter),
name(nullptr), value(nullptr) {}
~iter_expr();
void set_name(identifier* node) {name = node;}
void set_value(expr* node) {value = node;}
identifier* get_name() {return name;}
expr* get_value() {return value;}
virtual void accept(ast_visitor*) override;
};
class forindex_expr:public expr {
class forei_expr:public expr {
public:
enum class forei_loop_type {
foreach,
forindex
};
private:
forei_loop_type type;
iter_expr* iterator;
expr* vector_node;
code_block* block;
public:
forei_expr(const span& location, forei_loop_type loop_type):
expr(location, expr_type::ast_forei),
type(loop_type), iterator(nullptr),
vector_node(nullptr), block(nullptr) {}
~forei_expr();
void set_iterator(iter_expr* node) {iterator = node;}
void set_value(expr* node) {vector_node = node;}
void set_code_block(code_block* node) {block = node;}
iter_expr* get_iterator() {return iterator;}
expr* get_value() {return vector_node;}
code_block* get_code_block() {return block;}
virtual void accept(ast_visitor*) override;
};
class condition_expr:public expr {
private:
if_expr* if_stmt;
std::vector<if_expr*> elsif_stmt;
if_expr* else_stmt;
public:
condition_expr(const span& location):
expr(location, expr_type::ast_cond),
if_stmt(nullptr), else_stmt(nullptr) {}
~condition_expr();
void set_if_statement(if_expr* node) {if_stmt = node;}
void add_elsif_statement(if_expr* node) {elsif_stmt.push_back(node);}
void set_else_statement(if_expr* node) {else_stmt = node;}
if_expr* get_if_statement() {return if_stmt;}
std::vector<if_expr*>& get_elsif_stataments() {return elsif_stmt;}
if_expr* get_else_statement() {return else_stmt;}
virtual void accept(ast_visitor*) override;
};
class if_expr:public expr {
private:
expr* condition;
code_block* block;
public:
if_expr(const span& location):
expr(location, expr_type::ast_if),
condition(nullptr), block(nullptr) {}
void set_condition(expr* node) {condition = node;}
void set_code_block(code_block* node) {block = node;}
expr* get_condition() {return condition;}
code_block* get_code_block() {return block;}
virtual void accept(ast_visitor*) override;
};
@ -331,6 +499,15 @@ public:
};
class return_expr:public expr {
private:
expr* value;
public:
return_expr(const span& location):
expr(location, expr_type::ast_ret),
value(nullptr) {}
~return_expr();
void set_value(expr* node) {value = node;}
expr* get_value() {return value;}
virtual void accept(ast_visitor*) override;
};

View File

@ -1,12 +1,14 @@
#include "nasal_new_ast.h"
#include "nasal_new_parse.h"
const error& parse::compile(const lexer& lexer) {
toks=lexer.result().data();
ptr=in_func=in_loop=0;
root={toks[0].loc, ast_root};
root = new code_block(toks[0].loc);
while(!lookahead(tok::eof)) {
root.add(expr());
root->add_expression(expr());
if (lookahead(tok::semi)) {
match(tok::semi);
} else if (need_semi_check(root.child().back()) && !lookahead(tok::eof)) {
@ -54,7 +56,7 @@ void parse::easter_egg() const {
<< " `\"` `\"` \n";
}
void parse::die(const span& loc, string info) {
void parse::die(const span& loc, std::string info) {
err.err("parse", loc, info);
}
@ -123,15 +125,13 @@ bool parse::check_func_end(const ast& node) {
return false;
}
if (node.child().empty() || (
type!=ast_def &&
type!=ast_equal &&
type!=ast_addeq &&
type!=ast_subeq &&
type!=ast_multeq &&
type!=ast_diveq &&
type!=ast_lnkeq
)
) {
type!=ast_def &&
type!=ast_equal &&
type!=ast_addeq &&
type!=ast_subeq &&
type!=ast_multeq &&
type!=ast_diveq &&
type!=ast_lnkeq)) {
return false;
} else {
return check_func_end(node.child().back());
@ -163,46 +163,50 @@ bool parse::check_special_call() {
return false;
}
bool parse::need_semi_check(const ast& node) {
u32 type=node.type();
if (type==ast_for || type==ast_foreach || type==ast_forindex || type==ast_while || type==ast_cond) {
bool parse::need_semi_check(expr* node) {
auto type=node->get_type();
if (type==expr_type::ast_for ||
type==expr_type::ast_forei ||
type==expr_type::ast_while ||
type==expr_type::ast_cond) {
return false;
}
return !check_func_end(node);
}
ast parse::null() {
return {toks[ptr].loc, ast_null};
void parse::update_location(expr* node) {
node->update_location(toks[ptr].loc);
}
ast parse::nil() {
return {toks[ptr].loc, ast_nil};
null_expr* parse::null() {
return new null_expr(toks[ptr].loc);
}
ast parse::num() {
ast node(toks[ptr].loc, ast_num);
node.set_num(str2num(toks[ptr].str.c_str()));
nil_expr* parse::nil() {
return new nil_expr(toks[ptr].loc);
}
number_literal* parse::num() {
auto node = new number_literal(toks[ptr].loc,
str2num(toks[ptr].str.c_str()));
match(tok::num);
return node;
}
ast parse::str() {
ast node(toks[ptr].loc, ast_str);
node.set_str(toks[ptr].str);
string_literal* parse::str() {
auto node = new string_literal(toks[ptr].loc, toks[ptr].str);
match(tok::str);
return node;
}
ast parse::id() {
ast node(toks[ptr].loc, ast_id);
node.set_str(toks[ptr].str);
identifier* parse::id() {
auto node = new identifier(toks[ptr].loc, toks[ptr].str);
match(tok::id);
return node;
}
ast parse::bools() {
ast node(toks[ptr].loc, ast_bool);
node.set_str(toks[ptr].str);
bool_literal* parse::bools() {
auto node = new bool_literal(toks[ptr].loc, toks[ptr].str=="true");
if (lookahead(tok::tktrue)) {
match(tok::tktrue);
} else {
@ -211,7 +215,7 @@ ast parse::bools() {
return node;
}
ast parse::vec() {
vector_expr* parse::vec() {
// panic set for this token is not ','
// this is the FIRST set of calculation
// array end with tok::null=0
@ -221,7 +225,7 @@ ast parse::vec() {
tok::func,tok::var,tok::lcurve,tok::floater,
tok::lbrace,tok::lbracket,tok::null
};
ast node(toks[ptr].loc, ast_vec);
auto node = new vector_expr(toks[ptr].loc);
match(tok::lbracket);
while(!lookahead(tok::rbracket)) {
node.add(calc());
@ -233,16 +237,16 @@ ast parse::vec() {
break;
}
}
node.update_span(thisspan);
update_location(node);
match(tok::rbracket, "expected ']' when generating vector");
return node;
}
ast parse::hash() {
ast node(toks[ptr].loc, ast_hash);
hash_expr* parse::hash() {
auto node = new hash_expr(toks[ptr].loc);
match(tok::lbrace);
while(!lookahead(tok::rbrace)) {
node.add(pair());
node->add_member(pair());
if (lookahead(tok::comma)) {
match(tok::comma);
} else if (lookahead(tok::id) || lookahead(tok::str)) { // first set of hashmember
@ -251,46 +255,45 @@ ast parse::hash() {
break;
}
}
node.update_span(thisspan);
update_location(node);
match(tok::rbrace, "expected '}' when generating hash");
return node;
}
ast parse::pair() {
ast node(toks[ptr].loc, ast_pair);
hash_pair* parse::pair() {
auto node = new hash_pair(toks[ptr].loc);
if (lookahead(tok::id)) {
node.add(id());
node->set_name(toks[ptr].str);
match(tok::id);
} else if (lookahead(tok::str)) {
node.add(str());
node->set_name(toks[ptr].str);
match(tok::str);
} else {
match(tok::id, "expected hashmap key");
}
match(tok::colon);
node.add(calc());
node.update_span();
node->set_element(calc());
update_location(node);
return node;
}
ast parse::func() {
function* parse::func() {
++in_func;
ast node(toks[ptr].loc, ast_func);
auto node = new function(toks[ptr].loc);
match(tok::func);
if (lookahead(tok::lcurve)) {
node.add(params());
} else {
node.add(null());
params(node);
}
node.add(exprs());
node->set_code_block(exprs());
--in_func;
node.update_span();
update_location(node);
return node;
}
ast parse::params() {
ast node(toks[ptr].loc, ast_params);
void parse::params(function* func_node) {
match(tok::lcurve);
while(!lookahead(tok::rcurve)) {
ast tmp=id();
auto tmp = id();
if (lookahead(tok::eq) || lookahead(tok::ellipsis)) {
ast special_arg(toks[ptr].loc, ast_null);
if (lookahead(tok::eq)) {
@ -315,18 +318,18 @@ ast parse::params() {
break;
}
}
node.update_span(thisspan);
update_location(node);
match(tok::rcurve, "expected ')' after parameter list");
return node;
}
ast parse::lcurve_expr() {
expr* parse::lcurve_expr() {
if (toks[ptr+1].type==tok::var)
return definition();
return check_tuple()?multi_assgin():calc();
}
ast parse::expr() {
expr* parse::expression() {
tok type=toks[ptr].type;
if ((type==tok::brk || type==tok::cont) && !in_loop) {
die(thisspan, "must use break/continue in loops");
@ -346,18 +349,18 @@ ast parse::expr() {
case tok::lbrace:
case tok::sub:
case tok::floater:
case tok::opnot: return calc(); break;
case tok::var: return definition(); break;
case tok::lcurve: return lcurve_expr(); break;
case tok::opnot: return calc();
case tok::var: return definition();
case tok::lcurve: return lcurve_expr();
case tok::rfor:
case tok::forindex:
case tok::foreach:
case tok::rwhile: return loop(); break;
case tok::rif: return cond(); break;
case tok::cont: return continue_expr(); break;
case tok::brk: return break_expr(); break;
case tok::ret: return ret_expr(); break;
case tok::semi: break;
case tok::rwhile: return loop();
case tok::rif: return cond();
case tok::cont: return continue_expression();
case tok::brk: return break_expression();
case tok::ret: return return_expression();
case tok::semi: break;
default:
die(thisspan, "incorrect token <"+toks[ptr].str+">");
next();
@ -365,19 +368,19 @@ ast parse::expr() {
}
// unreachable
return {toks[ptr].loc, ast_null};
return new null_expr(toks[ptr].loc);
}
ast parse::exprs() {
code_block* parse::expression_block() {
if (lookahead(tok::eof)) {
die(thisspan, "expected expression block");
return null();
return new code_block(toks[ptr].loc);
}
ast node(toks[ptr].loc, ast_block);
auto node = new code_block(toks[ptr].loc);
if (lookahead(tok::lbrace)) {
match(tok::lbrace);
while(!lookahead(tok::rbrace) && !lookahead(tok::eof)) {
node.add(expr());
node->add_expression(expression());
if (lookahead(tok::semi)) {
match(tok::semi);
} else if (need_semi_check(node.child().back()) && !lookahead(tok::rbrace)) {
@ -387,15 +390,16 @@ ast parse::exprs() {
}
match(tok::rbrace, "expected '}' when generating expressions");
} else {
node.add(expr());
if (lookahead(tok::semi))
node->add_expression(expression());
if (lookahead(tok::semi)) {
match(tok::semi);
}
}
node.update_span();
update_location(node);
return node;
}
ast parse::calc() {
expr* parse::calc() {
ast node=bitwise_or();
if (lookahead(tok::quesmark)) {
// trinocular calculation
@ -420,11 +424,11 @@ ast parse::calc() {
tmp.add(calc());
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::bitwise_or() {
expr* parse::bitwise_or() {
ast node=bitwise_xor();
while(lookahead(tok::btor)) {
ast tmp(toks[ptr].loc, ast_bitor);
@ -434,11 +438,11 @@ ast parse::bitwise_or() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::bitwise_xor() {
expr* parse::bitwise_xor() {
ast node=bitwise_and();
while(lookahead(tok::btxor)) {
ast tmp(toks[ptr].loc, ast_bitxor);
@ -448,11 +452,11 @@ ast parse::bitwise_xor() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::bitwise_and() {
expr* parse::bitwise_and() {
ast node=or_expr();
while(lookahead(tok::btand)) {
ast tmp(toks[ptr].loc, ast_bitand);
@ -462,11 +466,11 @@ ast parse::bitwise_and() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::or_expr() {
expr* parse::or_expr() {
ast node=and_expr();
while(lookahead(tok::opor)) {
ast tmp(toks[ptr].loc, ast_or);
@ -476,11 +480,11 @@ ast parse::or_expr() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::and_expr() {
expr* parse::and_expr() {
ast node=cmp_expr();
while(lookahead(tok::opand)) {
ast tmp(toks[ptr].loc, ast_and);
@ -490,11 +494,11 @@ ast parse::and_expr() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::cmp_expr() {
expr* parse::cmp_expr() {
ast node=additive_expr();
while(tok::cmpeq<=toks[ptr].type && toks[ptr].type<=tok::geq) {
// tok::cmpeq~tok::geq is 43~48,ast_cmpeq~ast_geq is 27~32
@ -505,11 +509,11 @@ ast parse::cmp_expr() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::additive_expr() {
expr* parse::additive_expr() {
ast node=multive_expr();
while(lookahead(tok::add) || lookahead(tok::sub) || lookahead(tok::floater)) {
ast tmp(toks[ptr].loc, ast_null);
@ -525,11 +529,11 @@ ast parse::additive_expr() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::multive_expr() {
expr* parse::multive_expr() {
ast node=(lookahead(tok::sub) || lookahead(tok::opnot) || lookahead(tok::floater))?unary():scalar();
while(lookahead(tok::mult) || lookahead(tok::div)) {
ast tmp(toks[ptr].loc, (u32)toks[ptr].type-(u32)tok::mult+ast_mult);
@ -539,11 +543,11 @@ ast parse::multive_expr() {
tmp.update_span();
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::unary() {
expr* parse::unary() {
ast node(toks[ptr].loc,ast_null);
switch(toks[ptr].type) {
case tok::sub: node.set_type(ast_neg);match(tok::sub);break;
@ -552,11 +556,11 @@ ast parse::unary() {
default: break;
}
node.add((lookahead(tok::sub) || lookahead(tok::opnot) || lookahead(tok::floater))?unary():scalar());
node.update_span();
update_location(node);
return node;
}
ast parse::scalar() {
expr* parse::scalar() {
ast node(toks[ptr].loc, ast_null);
if (lookahead(tok::tknil)) {
node=nil();
@ -580,7 +584,7 @@ ast parse::scalar() {
match(tok::lcurve);
node=calc();
node.set_begin(loc.begin_line, loc.begin_column);
node.update_span(thisspan);
update_location(node);
match(tok::rcurve);
} else if (lookahead(tok::var)) {
match(tok::var);
@ -601,11 +605,11 @@ ast parse::scalar() {
node.add(call_scalar());
}
}
node.update_span();
update_location(node);
return node;
}
ast parse::call_scalar() {
expr* parse::call_scalar() {
switch(toks[ptr].type) {
case tok::lcurve: return callf(); break;
case tok::lbracket: return callv(); break;
@ -616,7 +620,7 @@ ast parse::call_scalar() {
return {toks[ptr].loc, ast_nil};
}
ast parse::callh() {
expr* parse::callh() {
ast node(toks[ptr].loc, ast_callh);
match(tok::dot);
node.set_str(toks[ptr].str);
@ -625,7 +629,7 @@ ast parse::callh() {
return node;
}
ast parse::callv() {
expr* parse::callv() {
// panic set for this token is not ','
// this is the FIRST set of subvec
// array end with tok::null=0
@ -650,12 +654,12 @@ ast parse::callv() {
if (node.size()==0) {
die(thisspan, "expected index value");
}
node.update_span(thisspan);
update_location(node);
match(tok::rbracket, "expected ']' when calling vector");
return node;
}
ast parse::callf() {
expr* parse::callf() {
// panic set for this token is not ','
// this is the FIRST set of calculation/hashmember
// array end with tok::null=0
@ -677,12 +681,12 @@ ast parse::callf() {
else if (!lookahead(tok::rcurve) && !check_comma(panic))
break;
}
node.update_span(thisspan);
update_location(node);
match(tok::rcurve, "expected ')' when calling function");
return node;
}
ast parse::subvec() {
expr* parse::subvec() {
ast node=lookahead(tok::colon)?nil():calc();
if (lookahead(tok::colon)) {
ast tmp(toks[ptr].loc, ast_subvec);
@ -691,11 +695,11 @@ ast parse::subvec() {
tmp.add((lookahead(tok::comma) || lookahead(tok::rbracket))?nil():calc());
node=std::move(tmp);
}
node.update_span();
update_location(node);
return node;
}
ast parse::definition() {
expr* parse::definition() {
ast node(toks[ptr].loc, ast_def);
if (lookahead(tok::var)) {
match(tok::var);
@ -713,32 +717,32 @@ ast parse::definition() {
} else {
node.add(calc());
}
node.update_span();
update_location(node);
return node;
}
ast parse::incurve_def() {
expr* parse::incurve_def() {
const auto& loc=toks[ptr].loc;
match(tok::lcurve);
match(tok::var);
ast node=multi_id();
node.update_span(thisspan);
update_location(node);
match(tok::rcurve);
node.set_begin(loc.begin_line, loc.begin_column);
return node;
}
ast parse::outcurve_def() {
expr* parse::outcurve_def() {
const auto& loc=toks[ptr].loc;
match(tok::lcurve);
ast node=multi_id();
node.update_span(thisspan);
update_location(node);
match(tok::rcurve);
node.set_begin(loc.begin_line, loc.begin_column);
return node;
}
ast parse::multi_id() {
expr* parse::multi_id() {
ast node(toks[ptr].loc, ast_multi_id);
while(!lookahead(tok::eof)) {
// only identifier is allowed here
@ -752,11 +756,11 @@ ast parse::multi_id() {
break;
}
}
node.update_span();
update_location(node);
return node;
}
ast parse::multi_scalar() {
expr* parse::multi_scalar() {
// if check_call_memory is true,we will check if value called here can reach a memory space
const tok panic[]={
tok::id,tok::str,tok::num,tok::tktrue,
@ -776,12 +780,12 @@ ast parse::multi_scalar() {
break;
}
}
node.update_span(thisspan);
update_location(node);
match(tok::rcurve, "expected ')' after multi-scalar");
return node;
}
ast parse::multi_assgin() {
expr* parse::multi_assgin() {
ast node(toks[ptr].loc, ast_multi_assign);
node.add(multi_scalar());
match(tok::eq);
@ -794,11 +798,11 @@ ast parse::multi_assgin() {
} else {
node.add(calc());
}
node.update_span();
update_location(node);
return node;
}
ast parse::loop() {
expr* parse::loop() {
++in_loop;
ast node(toks[ptr].loc, ast_null);
switch(toks[ptr].type) {
@ -812,18 +816,18 @@ ast parse::loop() {
return node;
}
ast parse::while_loop() {
expr* parse::while_loop() {
ast node(toks[ptr].loc, ast_while);
match(tok::rwhile);
match(tok::lcurve);
node.add(calc());
match(tok::rcurve);
node.add(exprs());
node.update_span();
update_location(node);
return node;
}
ast parse::for_loop() {
expr* parse::for_loop() {
ast node(toks[ptr].loc, ast_for);
match(tok::rfor);
match(tok::lcurve);
@ -862,11 +866,11 @@ ast parse::for_loop() {
}
match(tok::rcurve);
node.add(exprs());
node.update_span();
update_location(node);
return node;
}
ast parse::forei_loop() {
expr* parse::forei_loop() {
ast node(toks[ptr].loc, ast_null);
switch(toks[ptr].type) {
case tok::forindex:node.set_type(ast_forindex);match(tok::forindex);break;
@ -887,11 +891,11 @@ ast parse::forei_loop() {
node.add(calc());
match(tok::rcurve);
node.add(exprs());
node.update_span();
update_location(node);
return node;
}
ast parse::iter_gen() {
expr* parse::iter_gen() {
ast node(toks[ptr].loc, ast_null);
if (lookahead(tok::var)) {
match(tok::var);
@ -904,11 +908,11 @@ ast parse::iter_gen() {
node.add(call_scalar());
}
}
node.update_span();
update_location(node);
return node;
}
ast parse::cond() {
expr* parse::cond() {
ast node(toks[ptr].loc, ast_cond);
// generate if
@ -941,24 +945,24 @@ ast parse::cond() {
elsenode.update_span();
node.add(std::move(elsenode));
}
node.update_span();
update_location(node);
return node;
}
ast parse::continue_expr() {
ast node(toks[ptr].loc, ast_continue);
continue_expr* parse::continue_expression() {
auto node = new continue_expr(toks[ptr].loc);
match(tok::cont);
return node;
}
ast parse::break_expr() {
ast node(toks[ptr].loc, ast_break);
break_expr* parse::break_expression() {
auto node = new break_expr(toks[ptr].loc);
match(tok::brk);
return node;
}
ast parse::ret_expr() {
ast node(toks[ptr].loc, ast_ret);
return_expr* parse::return_expression() {
auto node = new return_expr(toks[ptr].loc);
match(tok::ret);
tok type=toks[ptr].type;
if (type==tok::tknil || type==tok::num ||
@ -966,10 +970,11 @@ ast parse::ret_expr() {
type==tok::func || type==tok::sub ||
type==tok::opnot || type==tok::lcurve ||
type==tok::lbracket || type==tok::lbrace ||
type==tok::tktrue || type==tok::tkfalse
) {
node.add(calc());
type==tok::tktrue || type==tok::tkfalse) {
node->set_value(calc());
} else {
node->set_value(nil());
}
node.update_span();
update_location(node);
return node;
}

View File

@ -18,7 +18,7 @@ private:
u32 in_func; // count function block
u32 in_loop; // count loop block
const token* toks;
ast root;
code_block* root;
error& err;
private:
@ -77,7 +77,7 @@ private:
};
private:
void die(const span&,string);
void die(const span&,std::string);
void next() {++ptr;}
void match(tok type, const char* info=nullptr);
bool lookahead(tok);
@ -86,64 +86,64 @@ private:
bool check_tuple();
bool check_func_end(const ast&);
bool check_special_call();
bool need_semi_check(const ast&);
bool need_semi_check(expr*);
void update_location(expr*);
private:
ast null();
ast nil();
ast num();
ast str();
ast id();
ast bools();
ast vec();
ast hash();
ast pair();
ast func();
ast params();
ast lcurve_expr();
ast expr();
ast exprs();
ast calc();
ast bitwise_or();
ast bitwise_xor();
ast bitwise_and();
ast or_expr();
ast and_expr();
ast cmp_expr();
ast additive_expr();
ast multive_expr();
ast unary();
ast scalar();
ast call_scalar();
ast callh();
ast callv();
ast callf();
ast subvec();
ast definition();
ast incurve_def();
ast outcurve_def();
ast multi_id();
ast multi_scalar();
ast multi_assgin();
ast loop();
ast while_loop();
ast for_loop();
ast forei_loop();
ast iter_gen();
ast cond();
ast continue_expr();
ast break_expr();
ast ret_expr();
null_expr* null();
nil_expr* nil();
number_literal* num();
string_literal* str();
identifier* id();
bool_literal* bools();
vector_expr* vec();
hash_expr* hash();
hash_pair* pair();
function* func();
void params(function*);
expr* lcurve_expr();
expr* expression();
code_block* expression_block();
expr* calc();
expr* bitwise_or();
expr* bitwise_xor();
expr* bitwise_and();
expr* or_expr();
expr* and_expr();
expr* cmp_expr();
expr* additive_expr();
expr* multive_expr();
expr* unary();
expr* scalar();
expr* call_scalar();
expr* callh();
expr* callv();
expr* callf();
expr* subvec();
expr* definition();
expr* incurve_def();
expr* outcurve_def();
expr* multi_id();
expr* multi_scalar();
expr* multi_assgin();
expr* loop();
expr* while_loop();
expr* for_loop();
expr* forei_loop();
expr* iter_gen();
expr* cond();
continue_expr* continue_expression();
break_expr* break_expression();
return_expr* return_expression();
public:
inline ast& tree() {return root;}
inline const ast& tree() const {return root;}
inline code_block* tree() {return root;}
public:
parse(error& e):
ptr(0), in_func(0), in_loop(0),
toks(nullptr),
root({0, 0, 0, 0, ""}, ast_root),
root(nullptr),
err(e) {}
const error& compile(const lexer&);
void easter_egg() const;

View File

@ -31,7 +31,7 @@ nasal_new_ast.o: ast/nasal_new_ast.h ast/nasal_new_ast.cpp nasal.h
$(CXX) -std=$(STD) -c -O3 ast/nasal_new_ast.cpp -fno-exceptions -fPIC -o nasal_new_ast.o -I .
nasal_new_parse.o: ast/nasal_new_parse.h ast/nasal_new_parse.cpp nasal.h ast/nasal_new_ast.h
$(CXX) -std=$(STD) -c -O3 ast/nasal_new_ast.cpp -fno-exceptions -fPIC -o nasal_new_parse.o -I .
# $(CXX) -std=$(STD) -c -O3 ast/nasal_new_parse.cpp -fno-exceptions -fPIC -o nasal_new_parse.o -I .
ast_visitor.o: ast/nasal_new_ast.h ast/ast_visitor.h ast/ast_visitor.cpp
$(CXX) -std=$(STD) -c -O3 ast/ast_visitor.cpp -fno-exceptions -fPIC -o ast_visitor.o -I .