Fixes for the c++ backend
This commit is contained in:
parent
f2d4dc31e7
commit
eaf2786738
|
@ -51,11 +51,11 @@ namespace backends {
|
||||||
topLevel["num_symbols"] = templ::make_string(std::to_string(config.grammar.terminals.size() + 1 + config.grammar.variables.size())); // + 1 for EOF
|
topLevel["num_symbols"] = templ::make_string(std::to_string(config.grammar.terminals.size() + 1 + config.grammar.variables.size())); // + 1 for EOF
|
||||||
|
|
||||||
std::vector<templ::TemplateContext> symbols;
|
std::vector<templ::TemplateContext> symbols;
|
||||||
symbols.push_back(templ::make_map({{"symbol", "TEOF"}}));
|
symbols.push_back(templ::make_map({{"symbol", templ::make_string("T_EOF")}}));
|
||||||
for (auto& s : config.grammar.terminals)
|
for (auto& s : config.grammar.terminals)
|
||||||
symbols.push_back(templ::make_map({{"symbol", s}}));
|
symbols.push_back(templ::make_map({{"symbol", templ::make_string("T_" + s)}}));
|
||||||
for (auto& s : config.grammar.variables)
|
for (auto& s : config.grammar.variables)
|
||||||
symbols.push_back(templ::make_map({{"symbol", s}}));
|
symbols.push_back(templ::make_map({{"symbol", templ::make_string("V_" + s)}}));
|
||||||
topLevel["symbols"] = std::move(symbols);
|
topLevel["symbols"] = std::move(symbols);
|
||||||
|
|
||||||
std::vector<templ::TemplateContext> rules;
|
std::vector<templ::TemplateContext> rules;
|
||||||
|
@ -64,7 +64,7 @@ namespace backends {
|
||||||
r["index"] = templ::make_string(std::to_string(i));
|
r["index"] = templ::make_string(std::to_string(i));
|
||||||
r["rhs_length"] = templ::make_string(std::to_string(config.grammar.rules[i]->tail.size()));
|
r["rhs_length"] = templ::make_string(std::to_string(config.grammar.rules[i]->tail.size()));
|
||||||
if (false /* the rule has a name */)
|
if (false /* the rule has a name */)
|
||||||
r["name"] = templ::make_string(""); //The name
|
r["rname"] = templ::make_string(""); //The name
|
||||||
rules.push_back(templ::make_map(std::move(r)));
|
rules.push_back(templ::make_map(std::move(r)));
|
||||||
}
|
}
|
||||||
topLevel["rules"] = templ::make_array(std::move(rules));
|
topLevel["rules"] = templ::make_array(std::move(rules));
|
||||||
|
@ -76,8 +76,8 @@ namespace backends {
|
||||||
std::map<const std::string, templ::TemplateContext> st;
|
std::map<const std::string, templ::TemplateContext> st;
|
||||||
|
|
||||||
std::vector<templ::TemplateContext> actions;
|
std::vector<templ::TemplateContext> actions;
|
||||||
for (const auto& term : terminals) {
|
for (std::string term : terminals) {
|
||||||
std::string a = "";
|
std::string a = "ERROR";
|
||||||
std::string data = "0";
|
std::string data = "0";
|
||||||
if (table.act[i].find(term) != table.act[i].end()) {
|
if (table.act[i].find(term) != table.act[i].end()) {
|
||||||
const auto& tmp = table.act[i][term];
|
const auto& tmp = table.act[i][term];
|
||||||
|
@ -88,7 +88,7 @@ namespace backends {
|
||||||
break;
|
break;
|
||||||
case lr::Action::REDUCE:
|
case lr::Action::REDUCE:
|
||||||
a = "REDUCE";
|
a = "REDUCE";
|
||||||
data = term + " << 31 | " + std::to_string(tmp.second);
|
data = "static_cast<std::uint64_t>(" + parserName + "_Symbol::V_" + config.grammar.rules[tmp.second]->head + ") << 31 | " + std::to_string(tmp.second);
|
||||||
break;
|
break;
|
||||||
case lr::Action::ACCEPT:
|
case lr::Action::ACCEPT:
|
||||||
a = "ACCEPT";
|
a = "ACCEPT";
|
||||||
|
@ -101,22 +101,23 @@ namespace backends {
|
||||||
actions.push_back(templ::make_map({{"action", templ::make_string(a)},
|
actions.push_back(templ::make_map({{"action", templ::make_string(a)},
|
||||||
{"data", templ::make_string(data)}}));
|
{"data", templ::make_string(data)}}));
|
||||||
|
|
||||||
std::vector<templ::TemplateContext> gotos;
|
|
||||||
for (const auto& nonterm : config.grammar.variables) {
|
|
||||||
gotos.push_back(templ::make_string(std::to_string(table.goto_[i][nonterm])));
|
|
||||||
}
|
|
||||||
|
|
||||||
st["actions"] = templ::make_array(std::move(actions));
|
|
||||||
st["gotos"] = templ::make_array(std::move(gotos));
|
|
||||||
states[i] = templ::make_map(std::move(st));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<templ::TemplateContext> gotos;
|
||||||
|
for (const auto& nonterm : config.grammar.variables) {
|
||||||
|
gotos.push_back(templ::make_string(std::to_string(table.goto_[i][nonterm])));
|
||||||
|
}
|
||||||
|
|
||||||
|
st["actions"] = templ::make_array(std::move(actions));
|
||||||
|
st["gotos"] = templ::make_array(std::move(gotos));
|
||||||
|
states[i] = templ::make_map(std::move(st));
|
||||||
}
|
}
|
||||||
topLevel["states"] = templ::make_array(std::move(states));
|
topLevel["states"] = templ::make_array(std::move(states));
|
||||||
|
|
||||||
std::unique_ptr<std::ostream> headerOut = getOstreamForFileName(parserName + ".h");
|
std::unique_ptr<std::ostream> headerOut = getOstreamForFileName(parserName + ".h");
|
||||||
std::unique_ptr<std::ostream> implOut = getOstreamForFileName(parserName + ".cpp");
|
std::unique_ptr<std::ostream> implOut = getOstreamForFileName(parserName + ".cpp");
|
||||||
this->doTemplate(*headerOut, "c++/lr.h", topLevel);
|
this->doTemplate(*headerOut, "c++/lr.h", topLevel);
|
||||||
this->doTemplate(*implOut, "c++/lr.h", topLevel);
|
this->doTemplate(*implOut, "c++/lr.cpp", topLevel);
|
||||||
}
|
}
|
||||||
private:
|
private:
|
||||||
util::ParserType m_parserType;
|
util::ParserType m_parserType;
|
||||||
|
|
|
@ -5,9 +5,9 @@
|
||||||
|
|
||||||
const std::uint64_t TABLE[{{num_states}}][{{num_symbols}}] = {
|
const std::uint64_t TABLE[{{num_states}}][{{num_symbols}}] = {
|
||||||
{{#states}}
|
{{#states}}
|
||||||
{ {{#actions}}({{name}}::{{action}} | {{data}} << 2),{{/actions}}
|
{ {{#actions}}({{action}} | ({{data}}) << 2),{{/actions}}
|
||||||
{{#gotos}}({{data}} << 2),{{/gotos}} },
|
{{#gotos}}(({{.}}) << 2),{{/gotos}} },
|
||||||
{{/states}}
|
{{/states}}
|
||||||
};
|
};
|
||||||
|
|
||||||
const unsigned char REDUCE_COUNT[{{num_rules}}] = { {{#rules}}{{rhs_length}}{{/rules}} };
|
const unsigned char REDUCE_COUNT[{{num_rules}}] = { {{#rules}}{{rhs_length}},{{/rules}} };
|
||||||
|
|
|
@ -3,9 +3,19 @@
|
||||||
#define PARSODUS_PARSER_{{name}}_H
|
#define PARSODUS_PARSER_{{name}}_H
|
||||||
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
#include <cstdint>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
#include <stack>
|
#include <stack>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the type of the symbol (both terminals and nonterminals)
|
||||||
|
*/
|
||||||
|
enum class {{name}}_Symbol : std::uint64_t {
|
||||||
|
{{#symbols}}
|
||||||
|
{{symbol}},
|
||||||
|
{{/symbols}}
|
||||||
|
};
|
||||||
|
|
||||||
template <typename Value>
|
template <typename Value>
|
||||||
class {{name}} {
|
class {{name}} {
|
||||||
public:
|
public:
|
||||||
|
@ -19,21 +29,12 @@ class {{name}} {
|
||||||
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/**
|
|
||||||
* Represents the type of the symbol (both terminals and nonterminals)
|
|
||||||
*/
|
|
||||||
enum Symbol {
|
|
||||||
{{#symbols}}
|
|
||||||
{{symbol}},
|
|
||||||
{{/symbols}}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A token, consisting of a Symbol type (should be a terminal) and a Value
|
* A token, consisting of a Symbol type (should be a terminal) and a Value
|
||||||
*/
|
*/
|
||||||
struct Token {
|
struct Token {
|
||||||
Symbol symbol,
|
{{name}}_Symbol symbol;
|
||||||
Value value
|
Value value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,8 +51,8 @@ class {{name}} {
|
||||||
* Apply a reduction (a grammar rule in reverse)
|
* Apply a reduction (a grammar rule in reverse)
|
||||||
*/
|
*/
|
||||||
{{#rules}}
|
{{#rules}}
|
||||||
{{#name}}virtual Value reduce_{{name}}(std::deque<Token> subparts) = 0;{{/name}}
|
{{#rname}}virtual Value reduce_{{rname}}(std::deque<Token> subparts) = 0;{{/rname}}
|
||||||
{{^name}}virtual Value reduce_{{index}}(std::deque<Token> subparts) = 0;{{/name}}
|
{{^rname}}virtual Value reduce_{{index}}(std::deque<Token> subparts) = 0;{{/rname}}
|
||||||
{{/rules}}
|
{{/rules}}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -75,7 +76,7 @@ enum Action {
|
||||||
* Parser implementation *
|
* Parser implementation *
|
||||||
***************************/
|
***************************/
|
||||||
template <typename Value>
|
template <typename Value>
|
||||||
Value {{name}}::parse() {
|
Value {{name}}<Value>::parse() {
|
||||||
std::stack<Token> valueStack;
|
std::stack<Token> valueStack;
|
||||||
std::stack<std::uint64_t> stateStack;
|
std::stack<std::uint64_t> stateStack;
|
||||||
|
|
||||||
|
@ -98,11 +99,11 @@ Value {{name}}::parse() {
|
||||||
case REDUCE:
|
case REDUCE:
|
||||||
{
|
{
|
||||||
std::uint64_t tmp = act >> 2;
|
std::uint64_t tmp = act >> 2;
|
||||||
Symbol symbol = tmp >> 31;
|
{{name}}_Symbol symbol = static_cast<{{name}}_Symbol>(tmp >> 31);
|
||||||
std::uint32_t rule = tmp & ((1ull << 32) - 1);
|
std::uint32_t rule = tmp & ((1ull << 32) - 1);
|
||||||
|
|
||||||
std::deque<Token> dq;
|
std::deque<Token> dq;
|
||||||
for (decltype(RECUDE_COUNT[rule]) i = 0; i < REDUCE_COUNT[rule]; i++) {
|
for (unsigned char i = 0; i < REDUCE_COUNT[rule]; i++) {
|
||||||
dq.emplace_front(std::move(valueStack.top()));
|
dq.emplace_front(std::move(valueStack.top()));
|
||||||
valueStack.pop();
|
valueStack.pop();
|
||||||
stateStack.pop();
|
stateStack.pop();
|
||||||
|
@ -111,8 +112,8 @@ Value {{name}}::parse() {
|
||||||
switch (rule) {
|
switch (rule) {
|
||||||
{{#rules}}
|
{{#rules}}
|
||||||
case {{index}}:
|
case {{index}}:
|
||||||
{{#name}}valueStack.emplace({symbol, reduce_{{name}}(std::move(dq))}){{/name}}
|
{{#rname}}valueStack.emplace({symbol, reduce_{{rname}}(std::move(dq))});{{/rname}}
|
||||||
{{^name}}valueStack.emplace({symbol, reduce_{{index}}(std::move(dq))}){{/name}}
|
{{^rname}}valueStack.emplace({symbol, reduce_{{index}}(std::move(dq))});{{/rname}}
|
||||||
break;
|
break;
|
||||||
{{/rules}}
|
{{/rules}}
|
||||||
default:
|
default:
|
||||||
|
|
Loading…
Reference in New Issue