git-subtree-dir: xo-reader git-subtree-mainline:dacdeb2cd7git-subtree-split:c46c0f1cc4
279 lines
11 KiB
C++
279 lines
11 KiB
C++
/* file parser.test.cpp
|
|
*
|
|
* author: Roland Conybeare
|
|
*/
|
|
|
|
#include "xo/reader/parser.hpp"
|
|
#include "xo/reader/define_xs.hpp"
|
|
#include <catch2/catch.hpp>
|
|
|
|
namespace xo {
|
|
using parser_type = xo::scm::parser;
|
|
using token_type = parser_type::token_type;
|
|
using xo::scm::exprstatetype;
|
|
using xo::scm::define_xs;
|
|
using xo::scm::defexprstatetype;
|
|
using std::cerr;
|
|
using std::endl;
|
|
|
|
//using xo::ast::Expression;
|
|
|
|
namespace ut {
|
|
TEST_CASE("parser", "[parser]") {
|
|
for (std::size_t i_tc = 0; i_tc < 2; ++i_tc) {
|
|
parser_type parser;
|
|
|
|
constexpr bool c_debug_flag = true;
|
|
scope log(XO_DEBUG(c_debug_flag), xtag("i_tc", i_tc));
|
|
|
|
parser.begin_translation_unit();
|
|
|
|
REQUIRE(parser.stack_size() == 1);
|
|
REQUIRE(parser.i_exstype(0)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
|
|
/* input:
|
|
* def
|
|
*/
|
|
{
|
|
auto r1 = parser.include_token(token_type::def());
|
|
REQUIRE(r1.get() == nullptr);
|
|
|
|
/* stack should be:
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
* def_1
|
|
* expect_symbol
|
|
*/
|
|
CHECK(parser.stack_size() == 3);
|
|
if (parser.stack_size() > 0)
|
|
CHECK(parser.i_exstype(0) == exprstatetype::expect_symbol);
|
|
if (parser.stack_size() > 1) {
|
|
CHECK(parser.i_exstype(1) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(1)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(1))->defxs_type() == defexprstatetype::def_1);
|
|
}
|
|
if (parser.stack_size() > 2)
|
|
CHECK(parser.i_exstype(2)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
}
|
|
|
|
/* input:
|
|
* def foo
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r2 = parser.include_token(token_type::symbol_token("foo"));
|
|
|
|
cerr << "parser state after [def foo]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r2.get() == nullptr);
|
|
|
|
/* stack should be:
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
* def_1
|
|
*/
|
|
CHECK(parser.stack_size() == 2);
|
|
if (parser.stack_size() > 0) {
|
|
CHECK(parser.i_exstype(0) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(0)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(0))->defxs_type() == defexprstatetype::def_2);
|
|
}
|
|
if (parser.stack_size() > 1)
|
|
CHECK(parser.i_exstype(1)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
|
|
}
|
|
|
|
if (i_tc == 0) {
|
|
;
|
|
} else if (i_tc == 1) {
|
|
/* input:
|
|
* def foo :
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r3 = parser.include_token(token_type::colon());
|
|
|
|
cerr << "parser state after [def foo :]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r3.get() == nullptr);
|
|
|
|
/* stack should be:
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
* def_3
|
|
* expect_symbol
|
|
*/
|
|
CHECK(parser.stack_size() == 3);
|
|
if (parser.stack_size() > 0)
|
|
CHECK(parser.i_exstype(0) == exprstatetype::expect_type);
|
|
if (parser.stack_size() > 1) {
|
|
CHECK(parser.i_exstype(1) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(1)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(1))->defxs_type() == defexprstatetype::def_3);
|
|
}
|
|
if (parser.stack_size() > 2)
|
|
CHECK(parser.i_exstype(2)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
}
|
|
|
|
/* input:
|
|
* def foo : f64
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r4 = parser.include_token(token_type::symbol_token("f64"));
|
|
|
|
cerr << "parser state after [def foo : f64]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r4.get() == nullptr);
|
|
|
|
CHECK(parser.stack_size() == 2);
|
|
|
|
/* stack should be:
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
* def_4
|
|
*/
|
|
CHECK(parser.stack_size() == 2);
|
|
if (parser.stack_size() > 0) {
|
|
CHECK(parser.i_exstype(0) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(0)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(0))->defxs_type() == defexprstatetype::def_4);
|
|
}
|
|
if (parser.stack_size() > 1)
|
|
CHECK(parser.i_exstype(1)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
|
|
/* expecting either:
|
|
* = rhs-expression
|
|
* new-expression
|
|
*/
|
|
}
|
|
}
|
|
|
|
/* input:
|
|
*
|
|
* i_tc==0:
|
|
* def foo =
|
|
* ^ ^
|
|
* 0 1
|
|
*
|
|
* i_tc==1:
|
|
* def foo : f64 =
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r5 = parser.include_token(token_type::singleassign());
|
|
|
|
cerr << "parser state after [def foo : f64 =]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r5.get() == nullptr);
|
|
|
|
CHECK(parser.stack_size() == 3);
|
|
|
|
/* stack should be
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
* def_5
|
|
* expect_expression
|
|
*/
|
|
CHECK(parser.stack_size() == 3);
|
|
if (parser.stack_size() > 0)
|
|
CHECK(parser.i_exstype(0) == exprstatetype::expect_rhs_expression);
|
|
if (parser.stack_size() > 1) {
|
|
CHECK(parser.i_exstype(1) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(1)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(1))->defxs_type() == defexprstatetype::def_5);
|
|
}
|
|
if (parser.stack_size() > 2)
|
|
CHECK(parser.i_exstype(2)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
}
|
|
|
|
/* input:
|
|
*
|
|
* i_tc==0:
|
|
* def foo = 3.14159265
|
|
* ^ ^
|
|
* 0 1
|
|
*
|
|
* i_tc==1:
|
|
* def foo : f64 = 3.14159265
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r6 = parser.include_token(token_type::f64_token("3.14159265"));
|
|
|
|
cerr << "parser state after [def foo : f64 = 3.14159265]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r6.get() == nullptr);
|
|
|
|
/* stack should be
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
*/
|
|
CHECK(parser.stack_size() == 4);
|
|
if (parser.stack_size() > 0)
|
|
CHECK(parser.i_exstype(0) == exprstatetype::expr_progress);
|
|
if (parser.stack_size() > 1)
|
|
CHECK(parser.i_exstype(1) == exprstatetype::expect_rhs_expression);
|
|
if (parser.stack_size() > 2) {
|
|
CHECK(parser.i_exstype(2) == exprstatetype::defexpr);
|
|
REQUIRE(define_xs::from(parser.i_exstate(2)) != nullptr);
|
|
CHECK(define_xs::from(parser.i_exstate(2))->defxs_type() == defexprstatetype::def_5);
|
|
}
|
|
if (parser.stack_size() > 3)
|
|
CHECK(parser.i_exstype(3)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
}
|
|
|
|
/* input:
|
|
*
|
|
* i_tc==0:
|
|
* def foo = 3.14159265 ;
|
|
* ^ ^
|
|
* 0 1
|
|
*
|
|
* i_tc==1:
|
|
* def foo : f64 = 3.14159265 ;
|
|
* ^ ^
|
|
* 0 1
|
|
*/
|
|
{
|
|
auto r7 = parser.include_token(token_type::semicolon());
|
|
|
|
cerr << "parser state after [def foo : f64 = 3.14159265;]" << endl;
|
|
cerr << parser << endl;
|
|
|
|
REQUIRE(r7.get() != nullptr);
|
|
|
|
CHECK(parser.stack_size() == 1);
|
|
|
|
/* stack should be
|
|
*
|
|
* expect_toplevel_expression_sequence
|
|
*/
|
|
CHECK(parser.stack_size() == 1);
|
|
if (parser.stack_size() > 0)
|
|
CHECK(parser.i_exstype(0)
|
|
== exprstatetype::expect_toplevel_expression_sequence);
|
|
}
|
|
}
|
|
} /*TEST_CASE(parser)*/
|
|
} /*namespace ut*/
|
|
} /*namespace xo*/
|
|
|
|
/* end parser.test.cpp */
|