// // peglint.cc // // Copyright (c) 2015 Yuji Hirose. All rights reserved. // MIT License // #include #include using namespace std; int run_server(int port, const vector& syntax, const vector& source); 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; } int main(int argc, const char** argv) { auto opt_ast = false; auto opt_optimize_ast_nodes = false; auto opt_help = false; auto opt_server = false; int port = 1234; 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("--server") == arg) { opt_server = true; if (argi < argc) { port = std::stoi(argv[argi++]); } } else { path_list.push_back(arg); } } if ((path_list.empty() && !opt_server) || opt_help) { cerr << "usage: peglint [--ast] [--optimize_ast_nodes|--opt] [--server [PORT=1234]] [grammar file path] [source file path]" << endl; return 1; } // Sever mode if (opt_server) { vector syntax; vector source; if (path_list.size() >= 1 && !read_file(path_list[0], syntax)) { cerr << "can't open the grammar file." << endl; return -1; } if (path_list.size() >= 2 && !read_file(path_list[1], source)) { cerr << "can't open the code file." << endl; return -1; } return run_server(port, syntax, source); } // 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; } peg::parser parser; parser.log = [&](auto ln, auto col, const auto& msg) { cerr << syntax_path << ":" << ln << ":" << col << ": " << msg << endl; }; if (!parser.load_grammar(syntax.data(), syntax.size())) { return -1; } if (path_list.size() < 2) { return 0; } // Check source auto source_path = path_list[1]; vector source; if (!read_file(source_path, source)) { auto beg = source_path; auto end = source_path + strlen(source_path); source.assign(beg, end); source_path = "[commendline]"; } parser.log = [&](auto ln, auto col, const auto& msg) { cerr << source_path << ":" << ln << ":" << col << ": " << msg << 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); peg::print_ast(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