mirror of
https://github.com/yhirose/cpp-peglib.git
synced 2025-01-22 21:35:29 +00:00
202 lines
5.8 KiB
C++
202 lines
5.8 KiB
C++
//
|
|
// peglint.cc
|
|
//
|
|
// Copyright (c) 2021 Yuji Hirose. All rights reserved.
|
|
// MIT License
|
|
//
|
|
|
|
#include <fstream>
|
|
#include <peglib.h>
|
|
#include <sstream>
|
|
|
|
using namespace std;
|
|
|
|
inline bool read_file(const char *path, vector<char> &buff) {
|
|
ifstream ifs(path, ios::in | ios::binary);
|
|
if (ifs.fail()) { return false; }
|
|
|
|
buff.resize(static_cast<unsigned int>(ifs.seekg(0, ios::end).tellg()));
|
|
if (!buff.empty()) {
|
|
ifs.seekg(0, ios::beg).read(&buff[0], static_cast<streamsize>(buff.size()));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
inline vector<string> split(const string &s, char delim) {
|
|
vector<string> elems;
|
|
stringstream ss(s);
|
|
string elem;
|
|
while (getline(ss, elem, delim)) {
|
|
elems.push_back(elem);
|
|
}
|
|
return elems;
|
|
}
|
|
|
|
int main(int argc, const char **argv) {
|
|
auto opt_packrat = false;
|
|
auto opt_ast = false;
|
|
auto opt_optimize = false;
|
|
auto opt_mode = true;
|
|
auto opt_help = false;
|
|
auto opt_source = false;
|
|
vector<char> source;
|
|
auto opt_trace = false;
|
|
vector<const char *> path_list;
|
|
|
|
auto argi = 1;
|
|
while (argi < argc) {
|
|
auto arg = argv[argi++];
|
|
if (string("--help") == arg) {
|
|
opt_help = true;
|
|
} else if (string("--packrat") == arg) {
|
|
opt_packrat = true;
|
|
} else if (string("--ast") == arg) {
|
|
opt_ast = true;
|
|
} else if (string("--opt") == arg || string("--opt-all") == arg) {
|
|
opt_optimize = true;
|
|
opt_mode = true;
|
|
} else if (string("--opt-only") == arg) {
|
|
opt_optimize = true;
|
|
opt_mode = false;
|
|
} 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 << R"(usage: grammar_file_path [source_file_path]
|
|
|
|
options:
|
|
--source: source text
|
|
--packrat: enable packrat memoise
|
|
--ast: show AST tree
|
|
--opt, --opt-all: optimize all AST nodes except nodes selected with `no_ast_opt` instruction
|
|
--opt-only: optimize only AST nodes selected with `no_ast_opt` instruction
|
|
--trace: show trace messages
|
|
)";
|
|
|
|
return 1;
|
|
}
|
|
|
|
// Check PEG grammar
|
|
auto syntax_path = path_list[0];
|
|
|
|
vector<char> syntax;
|
|
if (!read_file(syntax_path, syntax)) {
|
|
cerr << "can't open the grammar file." << endl;
|
|
return -1;
|
|
}
|
|
|
|
peg::parser parser;
|
|
|
|
parser.log = [&](size_t ln, size_t col, const string &msg) {
|
|
cerr << syntax_path << ":" << ln << ":" << col << ": " << msg << endl;
|
|
};
|
|
|
|
if (!parser.load_grammar(syntax.data(), syntax.size())) { return -1; }
|
|
|
|
if (path_list.size() < 2 && !opt_source) { return 0; }
|
|
|
|
// 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 << source_path << ":" << ln << ":" << col << ": " << msg << endl;
|
|
};
|
|
|
|
if (opt_packrat) {
|
|
parser.enable_packrat_parsing();
|
|
}
|
|
|
|
if (opt_trace) {
|
|
size_t prev_pos = 0;
|
|
parser.enable_trace(
|
|
[&](const peg::Ope &ope, const char *s, size_t /*n*/,
|
|
const peg::SemanticValues & /*sv*/, const peg::Context &c,
|
|
const std::any & /*dt*/) {
|
|
auto pos = static_cast<size_t>(s - c.s);
|
|
auto backtrack = (pos < prev_pos ? "*" : "");
|
|
string indent;
|
|
auto level = c.trace_ids.size() - 1;
|
|
while (level--) {
|
|
indent += "│";
|
|
}
|
|
std::string name;
|
|
{
|
|
name = peg::TraceOpeName::get(const_cast<peg::Ope &>(ope));
|
|
|
|
auto lit = dynamic_cast<const peg::LiteralString *>(&ope);
|
|
if (lit) { name += " '" + peg::escape_characters(lit->lit_) + "'"; }
|
|
}
|
|
std::cout << "E " << pos << backtrack << "\t" << indent << "┌" << name
|
|
<< " #" << c.trace_ids.back() << std::endl;
|
|
prev_pos = static_cast<size_t>(pos);
|
|
},
|
|
[&](const peg::Ope &ope, const char *s, size_t /*n*/,
|
|
const peg::SemanticValues &sv, const peg::Context &c,
|
|
const std::any & /*dt*/, size_t len) {
|
|
auto pos = static_cast<size_t>(s - c.s);
|
|
if (len != static_cast<size_t>(-1)) { pos += len; }
|
|
string indent;
|
|
auto level = c.trace_ids.size() - 1;
|
|
while (level--) {
|
|
indent += "│";
|
|
}
|
|
auto ret = len != static_cast<size_t>(-1) ? "└o " : "└x ";
|
|
auto name = peg::TraceOpeName::get(const_cast<peg::Ope &>(ope));
|
|
std::stringstream choice;
|
|
if (sv.choice_count() > 0) {
|
|
choice << " " << sv.choice() << "/" << sv.choice_count();
|
|
}
|
|
std::string token;
|
|
if (!sv.tokens.empty()) {
|
|
token += ", token '";
|
|
token += sv.tokens[0];
|
|
token += "'";
|
|
}
|
|
std::string matched;
|
|
if (peg::success(len) &&
|
|
peg::TokenChecker::is_token(const_cast<peg::Ope &>(ope))) {
|
|
matched = ", match '" + peg::escape_characters(s, len) + "'";
|
|
}
|
|
std::cout << "L " << pos << "\t" << indent << ret << name << " #"
|
|
<< c.trace_ids.back() << choice.str() << token << matched << std::endl;
|
|
});
|
|
}
|
|
|
|
if (opt_ast) {
|
|
parser.enable_ast();
|
|
|
|
std::shared_ptr<peg::Ast> ast;
|
|
auto ret = parser.parse_n(source.data(), source.size(), ast);
|
|
|
|
if (ast) {
|
|
if (opt_optimize) {
|
|
ast = parser.optimize_ast(ast, opt_mode);
|
|
}
|
|
std::cout << peg::ast_to_s(ast);
|
|
}
|
|
|
|
if (!ret) { return -1; }
|
|
} else {
|
|
if (!parser.parse_n(source.data(), source.size())) { return -1; }
|
|
}
|
|
|
|
return 0;
|
|
}
|