diff --git a/lint/peglint.cc b/lint/peglint.cc index 14423cc..1a2b30b 100644 --- a/lint/peglint.cc +++ b/lint/peglint.cc @@ -5,174 +5,151 @@ // MIT License // -#include #include +#include #include using namespace std; -inline bool read_file(const char* path, vector& buff) -{ - ifstream ifs(path, ios::in | ios::binary); - if (ifs.fail()) { - return false; - } +inline bool read_file(const char *path, vector &buff) { + ifstream ifs(path, ios::in | ios::binary); + if (ifs.fail()) { return false; } - buff.resize(static_cast(ifs.seekg(0, ios::end).tellg())); - if (!buff.empty()) { - ifs.seekg(0, ios::beg).read(&buff[0], static_cast(buff.size())); - } - return true; + buff.resize(static_cast(ifs.seekg(0, ios::end).tellg())); + if (!buff.empty()) { + ifs.seekg(0, ios::beg).read(&buff[0], static_cast(buff.size())); + } + return true; } -int main(int argc, const char** argv) -{ - auto opt_ast = false; - auto opt_optimize_ast_nodes = false; - auto opt_help = false; - auto opt_source = false; - vector source; - auto opt_trace = false; - vector path_list; - - auto argi = 1; - while (argi < argc) { - auto arg = argv[argi++]; - if (string("--help") == arg) { - opt_help = true; - } else if (string("--ast") == arg) { - opt_ast = true; - } else if (string("--optimize_ast_nodes") == arg || string("--opt") == arg) { - opt_optimize_ast_nodes = true; - } else if (string("--source") == arg) { - opt_source = true; - if (argi < argc) { - std::string text = argv[argi++]; - source.assign(text.begin(), text.end()); - } - } else if (string("--trace") == arg) { - opt_trace = true; - } else { - path_list.push_back(arg); - } - } - - if (path_list.empty() || opt_help) { - cerr << "usage: peglint [--ast] [--optimize_ast_nodes|--opt] [--source text] [--trace] [grammar file path] [source file path]" << endl; - return 1; - } - - // Check PEG grammar - auto syntax_path = path_list[0]; - - vector syntax; - if (!read_file(syntax_path, syntax)) { - cerr << "can't open the grammar file." << endl; - return -1; +int main(int argc, const char **argv) { + auto opt_ast = false; + auto opt_optimize_ast_nodes = false; + auto opt_help = false; + auto opt_source = false; + vector source; + auto opt_trace = false; + vector path_list; + + auto argi = 1; + while (argi < argc) { + auto arg = argv[argi++]; + if (string("--help") == arg) { + opt_help = true; + } else if (string("--ast") == arg) { + opt_ast = true; + } else if (string("--optimize_ast_nodes") == arg || + string("--opt") == arg) { + opt_optimize_ast_nodes = true; + } else if (string("--source") == arg) { + opt_source = true; + if (argi < argc) { + std::string text = argv[argi++]; + source.assign(text.begin(), text.end()); + } + } else if (string("--trace") == arg) { + opt_trace = true; + } else { + path_list.push_back(arg); } + } - peg::parser parser; + if (path_list.empty() || opt_help) { + cerr << "usage: peglint [--ast] [--optimize_ast_nodes|--opt] [--source " + "text] [--trace] [grammar file path] [source file path]" + << endl; + return 1; + } - parser.log = [&](size_t ln, size_t col, const string& msg) { - cerr << syntax_path << ":" << ln << ":" << col << ": " << msg << endl; - }; + // Check PEG grammar + auto syntax_path = path_list[0]; - if (!parser.load_grammar(syntax.data(), syntax.size())) { - return -1; - } + vector syntax; + if (!read_file(syntax_path, syntax)) { + cerr << "can't open the grammar file." << endl; + return -1; + } - if (path_list.size() < 2 && !opt_source) { - return 0; - } + peg::parser parser; - // Check source - std::string source_path = "[commandline]"; - if (path_list.size() >= 2) { - if (!read_file(path_list[1], source)) { - cerr << "can't open the code file." << endl; - return -1; - } - source_path = path_list[1]; - } + parser.log = [&](size_t ln, size_t col, const string &msg) { + cerr << syntax_path << ":" << ln << ":" << col << ": " << msg << endl; + }; - parser.log = [&](size_t ln, size_t col, const string& msg) { - cerr << source_path << ":" << ln << ":" << col << ": " << msg << endl; - }; - - if (opt_trace) { - size_t prev_pos = 0; - parser.enable_trace( - [&](const char* name, - const char* s, - size_t /*n*/, - const peg::SemanticValues& /*sv*/, - const peg::Context& c, - const peg::any& /*dt*/) { - auto pos = static_cast(s - c.s); - auto backtrack = (pos < prev_pos ? "*" : ""); - string indent; - auto level = c.trace_ids.size() - 1; - while (level--) { indent += "│"; } - std::cout - << "E " << pos << backtrack << "\t" - << indent << "┌" << name - << " #" << c.trace_ids.back() - << std::endl; - prev_pos = static_cast(pos); - }, - [&](const char* name, - const char* s, - size_t /*n*/, - const peg::SemanticValues& sv, - const peg::Context& c, - const peg::any& /*dt*/, - size_t len) { - auto pos = static_cast(s - c.s); - if (len != static_cast(-1)) { - pos += len; - } - string indent; - auto level = c.trace_ids.size() - 1; - while (level--) { indent += "│"; } - auto ret = len != static_cast(-1) ? "└o " : "└x "; - std::stringstream choice; - if (sv.choice_count() > 0) { - choice << " " << sv.choice() << "/" << sv.choice_count(); - } - std::string token; - if (!sv.tokens.empty()) { - const auto& tok = sv.tokens[0]; - token += " '" + std::string(tok.first, tok.second) + "'"; - } - std::cout - << "L " << pos << "\t" - << indent << ret << name - << " #" << c.trace_ids.back() - << choice.str() - << token - << std::endl; - } - ); - } + if (!parser.load_grammar(syntax.data(), syntax.size())) { return -1; } - if (opt_ast) { - parser.enable_ast(); + if (path_list.size() < 2 && !opt_source) { return 0; } - std::shared_ptr ast; - if (!parser.parse_n(source.data(), source.size(), ast)) { - return -1; - } - - ast = peg::AstOptimizer(opt_optimize_ast_nodes).optimize(ast); - std::cout << peg::ast_to_s(ast); - - } else { - if (!parser.parse_n(source.data(), source.size())) { - return -1; - } + // Check source + std::string source_path = "[commandline]"; + if (path_list.size() >= 2) { + if (!read_file(path_list[1], source)) { + cerr << "can't open the code file." << endl; + return -1; } - - return 0; + source_path = path_list[1]; + } + + parser.log = [&](size_t ln, size_t col, const string &msg) { + cerr << source_path << ":" << ln << ":" << col << ": " << msg << endl; + }; + + if (opt_trace) { + size_t prev_pos = 0; + parser.enable_trace( + [&](const char *name, const char *s, size_t /*n*/, + const peg::SemanticValues & /*sv*/, const peg::Context &c, + const peg::any & /*dt*/) { + auto pos = static_cast(s - c.s); + auto backtrack = (pos < prev_pos ? "*" : ""); + string indent; + auto level = c.trace_ids.size() - 1; + while (level--) { + indent += "│"; + } + std::cout << "E " << pos << backtrack << "\t" << indent << "┌" << name + << " #" << c.trace_ids.back() << std::endl; + prev_pos = static_cast(pos); + }, + [&](const char *name, const char *s, size_t /*n*/, + const peg::SemanticValues &sv, const peg::Context &c, + const peg::any & /*dt*/, size_t len) { + auto pos = static_cast(s - c.s); + if (len != static_cast(-1)) { pos += len; } + string indent; + auto level = c.trace_ids.size() - 1; + while (level--) { + indent += "│"; + } + auto ret = len != static_cast(-1) ? "└o " : "└x "; + std::stringstream choice; + if (sv.choice_count() > 0) { + choice << " " << sv.choice() << "/" << sv.choice_count(); + } + std::string token; + if (!sv.tokens.empty()) { + const auto &tok = sv.tokens[0]; + token += " '" + std::string(tok.first, tok.second) + "'"; + } + std::cout << "L " << pos << "\t" << indent << ret << name << " #" + << c.trace_ids.back() << choice.str() << token << std::endl; + }); + } + + if (opt_ast) { + parser.enable_ast(); + + std::shared_ptr ast; + if (!parser.parse_n(source.data(), source.size(), ast)) { return -1; } + + ast = peg::AstOptimizer(opt_optimize_ast_nodes).optimize(ast); + std::cout << peg::ast_to_s(ast); + + } else { + if (!parser.parse_n(source.data(), source.size())) { return -1; } + } + + return 0; } // vim: et ts=4 sw=4 cin cino={1s ff=unix