Debug output from the table generator
This commit is contained in:
parent
dd895152f7
commit
10798dea7a
16
TODO
16
TODO
|
@ -18,28 +18,28 @@ K> Parsodus Parsodus parser
|
|||
#R 1 enkel LR(1)
|
||||
#K 1 LR(1) + LALR(1)
|
||||
#T 1 LR(0)
|
||||
|
||||
-> Table generator (independent of specific tables)???
|
||||
-> JSON/example tests
|
||||
|
||||
-> presentation
|
||||
|
||||
-> Error reporting
|
||||
-> publication / LICENSE
|
||||
|
||||
-> --- logging
|
||||
-> write configuration sets, table
|
||||
-> Generator: logging
|
||||
-> driver/main: debug flag
|
||||
|
||||
-----------------------------------------------------------------------
|
||||
- MOSTLY DONE -
|
||||
-----------------------------------------------------------------------
|
||||
R> README
|
||||
R> README: bash completion + precedence
|
||||
R> JSON port -> rewrite once named rules exist
|
||||
|
||||
|
||||
-----------------------------------------------------------------------
|
||||
- DONE -
|
||||
-----------------------------------------------------------------------
|
||||
-> logging
|
||||
-> write configuration sets, table
|
||||
-> Generator: logging
|
||||
-> driver/main: debug flag
|
||||
-> Error reporting
|
||||
-> bool specialization in backend?
|
||||
T> Parsodus regex parser in Lexesis
|
||||
-> Vrijgeven in libraryformaat: mogelijkheid verschillende tokens opvragen
|
||||
|
|
|
@ -4,6 +4,9 @@
|
|||
|
||||
#include "Parsodus/lrtables/LR0Item.h"
|
||||
|
||||
#include <ostream>
|
||||
#include <set>
|
||||
|
||||
namespace pds {
|
||||
namespace lr {
|
||||
|
||||
|
@ -24,6 +27,21 @@ public:
|
|||
bool merge(const Itemset& rhs);
|
||||
bool empty() const;
|
||||
|
||||
friend std::ostream& operator<<(std::ostream& os, const LR0ItemsetBase<Itemset>& its) {
|
||||
for (const auto& it : its.m_items) {
|
||||
os << it.rule->head << " ->";
|
||||
for (std::size_t i = 0; i < it.rule->tail.size(); i++) {
|
||||
if (i == it.dotIdx)
|
||||
os << " ·";
|
||||
os << " " << it.rule->tail[i];
|
||||
}
|
||||
if (it.dotIdx == it.rule->tail.size())
|
||||
os << " ·";
|
||||
os << std::endl;
|
||||
}
|
||||
return os;
|
||||
}
|
||||
|
||||
protected:
|
||||
std::set<LR0Item> m_items;
|
||||
};
|
||||
|
|
|
@ -6,6 +6,8 @@
|
|||
#include "Parsodus/util/symbols.h"
|
||||
|
||||
#include <algorithm>
|
||||
#include <ostream>
|
||||
#include <set>
|
||||
|
||||
namespace pds {
|
||||
namespace lr {
|
||||
|
@ -24,6 +26,28 @@ public:
|
|||
bool empty() const;
|
||||
std::set<std::size_t> getReduces(const Grammar& g, std::string lookahead) const;
|
||||
|
||||
friend std::ostream& operator<<(std::ostream& os, const LR1ItemsetBase<Itemset>& its) {
|
||||
for (const auto& it : its.m_items) {
|
||||
os << it.rule->head << " ->";
|
||||
for (std::size_t i = 0; i < it.rule->tail.size(); i++) {
|
||||
if (i == it.dotIdx)
|
||||
os << " ·";
|
||||
os << " " << it.rule->tail[i];
|
||||
}
|
||||
if (it.dotIdx == it.rule->tail.size())
|
||||
os << " ·";
|
||||
|
||||
os << " [";
|
||||
for (auto la = it.lookaheads.cbegin(); la != it.lookaheads.cend(); la++) {
|
||||
if (la != it.lookaheads.cbegin())
|
||||
os << ", ";
|
||||
os << *la;
|
||||
}
|
||||
os << "]" << std::endl;
|
||||
}
|
||||
return os;
|
||||
}
|
||||
|
||||
protected:
|
||||
std::vector<LR1Item> m_items;
|
||||
};
|
||||
|
|
|
@ -6,6 +6,8 @@
|
|||
#include "Parsodus/util/symbols.h"
|
||||
#include "Parsodus/lrtables/table.h"
|
||||
|
||||
#include "g3log/g3log.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
#include <cassert>
|
||||
#include <memory>
|
||||
|
@ -80,6 +82,9 @@ LRTable Generator<Itemset>::generate() {
|
|||
m_gram.variables.begin(), m_gram.variables.end(),
|
||||
std::inserter(symbols, symbols.end()));
|
||||
|
||||
std::map<std::size_t, std::map<std::string, std::pair<Action, std::size_t>>> errors;
|
||||
bool reduceReduce = false;
|
||||
|
||||
std::queue<std::pair<std::size_t, Itemset>> q;
|
||||
q.emplace(0, itemsets[0]);
|
||||
while (!q.empty()) {
|
||||
|
@ -131,7 +136,6 @@ LRTable Generator<Itemset>::generate() {
|
|||
table.act[curIdx][term] = std::make_pair(Action::ACCEPT, 0);
|
||||
} else if (table.act[curIdx].count(term)) {
|
||||
if (table.act[curIdx][term].first == Action::SHIFT) {
|
||||
//Shift-Reduce conflict, rapport and resolve it (TODO)
|
||||
bool handled = false;
|
||||
|
||||
auto rightTokenIt = m_gram.precedence.find(term);
|
||||
|
@ -151,15 +155,21 @@ LRTable Generator<Itemset>::generate() {
|
|||
} else if (leftPrec.second == PrecedenceType::RIGHT) {
|
||||
// Keep the shift
|
||||
handled = true;
|
||||
} //Nonassoc falls through
|
||||
} else {
|
||||
//make it an error
|
||||
table.act[curIdx][term] = {Action::ERROR, 0};
|
||||
handled = true;
|
||||
}
|
||||
}
|
||||
if (!handled) {
|
||||
LOG(WARNING) << "Shift/reduce conflict in state " << curIdx << ", favouring shift" << std::endl;
|
||||
errors[curIdx][term] = {Action::REDUCE, rule_applied};
|
||||
}
|
||||
|
||||
if (!handled)
|
||||
throw std::runtime_error("shift-reduce");
|
||||
} else if (table.act[curIdx][term].first == Action::REDUCE
|
||||
&& table.act[curIdx][term].second != rule_applied) {
|
||||
//Reduce-Reduce conflict, rapport it (TODO)
|
||||
throw std::runtime_error("reduce-reduce");
|
||||
LOG(WARNING) << "Reduce/reduce conflict in state " << curIdx << std::endl;
|
||||
errors[curIdx][term] = {Action::REDUCE, rule_applied};
|
||||
reduceReduce = true;
|
||||
} else {
|
||||
//Reduce using the same rule, no problem, NO-OP
|
||||
}
|
||||
|
@ -171,10 +181,59 @@ LRTable Generator<Itemset>::generate() {
|
|||
}
|
||||
}
|
||||
|
||||
auto reporter = [&table](std::size_t idx, std::string sym, const std::pair<Action, std::size_t>& p) -> std::string {
|
||||
switch (p.first) {
|
||||
case Action::ERROR:
|
||||
return "on " + sym + " fail\n";
|
||||
case Action::SHIFT:
|
||||
return "on " + sym + " shift and go to " + std::to_string(table.act[idx][sym].second) + "\n";
|
||||
case Action::REDUCE:
|
||||
return "on " + sym + " reduce with rule " + std::to_string(table.act[idx][sym].second) + "\n";
|
||||
case Action::ACCEPT:
|
||||
return "on " + sym + " accept\n";
|
||||
default:
|
||||
return "";
|
||||
}
|
||||
};
|
||||
|
||||
for (std::size_t idx = 0; idx < itemsets.size(); idx++) {
|
||||
LOG(INFO) << "State " << idx << std::endl
|
||||
<< itemsets[idx] << std::endl;
|
||||
for (std::string terminal : m_gram.terminals) {
|
||||
auto pIt = table.act[idx].find(terminal);
|
||||
if (pIt != table.act[idx].end()) {
|
||||
LOG(INFO) << reporter(idx, terminal, pIt->second);
|
||||
} else {
|
||||
LOG(INFO) << reporter(idx, terminal, {Action::ERROR, 0});
|
||||
}
|
||||
}
|
||||
LOG(INFO) << std::endl;
|
||||
bool doneGoto = false, doneError = false;
|
||||
for (std::string var : m_gram.variables) {
|
||||
if (table.goto_[idx].count(var)) {
|
||||
LOG(INFO) << "after " << var << " go to state " << table.goto_[idx][var] << std::endl;
|
||||
doneGoto = true;
|
||||
}
|
||||
}
|
||||
if (doneGoto)
|
||||
LOG(INFO) << std::endl;
|
||||
for (auto& p : errors[idx]) {
|
||||
doneError = true;
|
||||
LOG(INFO) << "! " << reporter(idx, p.first, p.second);
|
||||
}
|
||||
if (doneError)
|
||||
LOG(INFO) << std::endl;
|
||||
LOG(INFO) << std::string(50, '=') << std::endl << std::endl;
|
||||
}
|
||||
|
||||
if (reduceReduce) {
|
||||
throw std::runtime_error("Stopped generating because of errors in the file");
|
||||
}
|
||||
|
||||
return table;
|
||||
}
|
||||
|
||||
} /* lr */
|
||||
} /* pdf */
|
||||
} /* pds */
|
||||
|
||||
#endif /* PARSODUS_LRTABLES_GENERATOR_H_YW3GIUNH */
|
||||
|
|
|
@ -119,9 +119,6 @@ inline std::string to_string({{name}}_Symbol s) {
|
|||
case {{name}}_Symbol::{{symbol}}:
|
||||
return "{{symbol}}";
|
||||
{{/symbols}}
|
||||
default:
|
||||
//should not happen
|
||||
return "?";
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -45,15 +45,6 @@ TEST(lr0, test0) {
|
|||
grammar.rules.push_back(std::make_shared<pds::Rule>(*rule));
|
||||
|
||||
|
||||
{
|
||||
pds::lr::Generator<pds::lr::LR0Itemset> g(grammar);
|
||||
ASSERT_THROW(g.generate(), std::runtime_error);
|
||||
try {
|
||||
g.generate();
|
||||
} catch (std::runtime_error& e) {
|
||||
ASSERT_EQ(std::string("shift-reduce"), e.what());
|
||||
}
|
||||
}
|
||||
{
|
||||
pds::lr::Generator<pds::lr::SLR1Itemset> g(grammar);
|
||||
ASSERT_NO_THROW(g.generate());
|
||||
|
|
|
@ -86,11 +86,6 @@ TEST(lr1, only) {
|
|||
{
|
||||
pds::lr::Generator<pds::lr::LALR1Itemset> g(grammar);
|
||||
ASSERT_THROW(g.generate(), std::runtime_error);
|
||||
try {
|
||||
g.generate();
|
||||
} catch (std::runtime_error& e) {
|
||||
ASSERT_EQ(std::string("reduce-reduce"), e.what());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue