Fixes for the c++ backend

This commit is contained in:
Robin Jadoul 2017-01-02 17:02:43 +01:00
parent f2d4dc31e7
commit eaf2786738
3 changed files with 39 additions and 37 deletions

View File

@ -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;

View File

@ -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}} };

View File

@ -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: