made setup for expression parsing
This commit is contained in:
@@ -35,7 +35,7 @@ namespace parser {
|
||||
case token::type::RETURN:
|
||||
return parse_return();
|
||||
default:
|
||||
return nullptr;
|
||||
return parse_expression_stmt();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -102,4 +102,12 @@ namespace parser {
|
||||
for (const auto& e : errors)
|
||||
delete e;
|
||||
}
|
||||
|
||||
void parser::register_prefix(token::type type, prefix_parse_fn fn) {
|
||||
prefix_parse_fns[type] = fn;
|
||||
};
|
||||
|
||||
void parser::register_infix(token::type type, infix_parse_fn fn) {
|
||||
infix_parse_fns[type] = fn;
|
||||
};
|
||||
} // namespace parser
|
||||
|
@@ -9,9 +9,14 @@
|
||||
#include "lexer/lexer.hpp"
|
||||
#include "token/token.hpp"
|
||||
|
||||
#include <functional>
|
||||
#include <vector>
|
||||
|
||||
namespace parser {
|
||||
|
||||
using prefix_parse_fn = std::function<ast::expression*()>;
|
||||
using infix_parse_fn = std::function<ast::expression*(ast::expression*)>;
|
||||
|
||||
struct parser {
|
||||
parser(lexer::lexer& lexer);
|
||||
~parser();
|
||||
@@ -23,6 +28,9 @@ namespace parser {
|
||||
lexer::lexer& lexer;
|
||||
token::token current, next;
|
||||
|
||||
std::unordered_map<token::type, prefix_parse_fn> prefix_parse_fns;
|
||||
std::unordered_map<token::type, infix_parse_fn> infix_parse_fns;
|
||||
|
||||
void next_token();
|
||||
ast::statement* parse_statement();
|
||||
ast::expression* parse_expression();
|
||||
@@ -31,5 +39,8 @@ namespace parser {
|
||||
ast::expression_stmt* parse_expression_stmt();
|
||||
bool expect_next(token::type);
|
||||
void next_error(token::type);
|
||||
|
||||
void register_prefix(token::type, prefix_parse_fn);
|
||||
void register_infix(token::type, infix_parse_fn);
|
||||
};
|
||||
} // namespace parser
|
||||
|
32
test/parser/expression.cpp
Normal file
32
test/parser/expression.cpp
Normal file
@@ -0,0 +1,32 @@
|
||||
#include "ast/expressions/identifier.hpp"
|
||||
#include "utils.hpp"
|
||||
|
||||
#include <doctest.h>
|
||||
|
||||
TEST_SUITE("Parser: expression") {
|
||||
TEST_CASE_FIXTURE(ParserFixture, "Simple expression statement") {
|
||||
setup("foobar;");
|
||||
REQUIRE(program->statements.size() == 1);
|
||||
ast::expression_stmt* expression_stmt;
|
||||
REQUIRE_NOTHROW(
|
||||
expression_stmt =
|
||||
dynamic_cast<ast::expression_stmt*>(program->statements[0])
|
||||
);
|
||||
REQUIRE_MESSAGE(
|
||||
expression_stmt != nullptr,
|
||||
"Couldn't cast statement to an expression statement"
|
||||
);
|
||||
|
||||
ast::identifier* ident;
|
||||
REQUIRE_NOTHROW(
|
||||
ident = dynamic_cast<ast::identifier*>(expression_stmt->expression)
|
||||
);
|
||||
REQUIRE_MESSAGE(
|
||||
ident != nullptr,
|
||||
"Couldn't cast expression to an identifier"
|
||||
);
|
||||
|
||||
REQUIRE(ident->value == "foobar");
|
||||
REQUIRE(ident->token_literal() == "foobar");
|
||||
};
|
||||
}
|
Reference in New Issue
Block a user