cpp-peglib/example/calc3.cc

69 lines
1.6 KiB
C++
Raw Permalink Normal View History

2015-02-22 04:19:54 +00:00
//
// calc3.cc
//
// Copyright (c) 2015 Yuji Hirose. All rights reserved.
// MIT License
//
#include <cstdlib>
2020-10-02 01:26:04 +00:00
#include <iostream>
#include <peglib.h>
2015-02-22 04:19:54 +00:00
2015-08-10 20:37:56 +00:00
using namespace peg;
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
int main(int argc, const char **argv) {
if (argc < 2 || std::string("--help") == argv[1]) {
std::cout << "usage: calc3 [formula]" << std::endl;
return 1;
}
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
std::function<long(const Ast &)> eval = [&](const Ast &ast) {
if (ast.name == "NUMBER") {
return ast.token_to_number<long>();
} else {
const auto &nodes = ast.nodes;
auto result = eval(*nodes[0]);
for (auto i = 1u; i < nodes.size(); i += 2) {
auto num = eval(*nodes[i + 1]);
auto ope = nodes[i]->token[0];
switch (ope) {
case '+': result += num; break;
case '-': result -= num; break;
case '*': result *= num; break;
case '/': result /= num; break;
2015-06-14 12:06:09 +00:00
}
2020-10-02 01:26:04 +00:00
}
return result;
}
};
2015-06-04 15:41:14 +00:00
2020-10-02 01:26:04 +00:00
parser parser(R"(
EXPRESSION <- TERM (TERM_OPERATOR TERM)*
2015-08-07 03:09:37 +00:00
TERM <- FACTOR (FACTOR_OPERATOR FACTOR)*
FACTOR <- NUMBER / '(' EXPRESSION ')'
TERM_OPERATOR <- < [-+] >
FACTOR_OPERATOR <- < [/*] >
NUMBER <- < [0-9]+ >
%whitespace <- [ \t\r\n]*
2015-08-07 03:09:37 +00:00
)");
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
parser.enable_ast();
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
auto expr = argv[1];
std::shared_ptr<Ast> ast;
if (parser.parse(expr, ast)) {
2021-01-22 01:56:05 +00:00
ast = parser.optimize_ast(ast);
2020-10-02 01:26:04 +00:00
std::cout << ast_to_s(ast);
std::cout << expr << " = " << eval(*ast) << std::endl;
return 0;
}
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
std::cout << "syntax error..." << std::endl;
2015-02-22 04:19:54 +00:00
2020-10-02 01:26:04 +00:00
return -1;
2015-02-22 04:19:54 +00:00
}
// vim: et ts=4 sw=4 cin cino={1s ff=unix