LR support for explicit rule precedences

This commit is contained in:
Robin Jadoul 2017-01-20 20:05:41 +01:00
parent 8358813204
commit 982d385ea3
3 changed files with 14 additions and 22 deletions

2
TODO
View File

@ -5,7 +5,7 @@
T> Parsodus regex parser in Lexesis T> Parsodus regex parser in Lexesis
-> Vrijgeven in libraryformaat: mogelijkheid verschillende tokens opvragen -> Vrijgeven in libraryformaat: mogelijkheid verschillende tokens opvragen
K> Parsodus Parsodus parser K> Parsodus Parsodus parser
-> Precedence -> Precedence + fill precedence in rules
-> rule naming (problem: multiple rules same name -> change in backend) -> rule naming (problem: multiple rules same name -> change in backend)
-> grammar struct change to contain optional name per rule -> grammar struct change to contain optional name per rule
-> Volgorde belangrijk? -> Volgorde belangrijk?

View File

@ -11,6 +11,12 @@
#include <vector> #include <vector>
namespace pds { namespace pds {
enum class PrecedenceType {
LEFT,
RIGHT,
NONASSOC
};
/** /**
* Represents a grammar rule * Represents a grammar rule
* head -> tail[0] tail[1] ... tail[tail.size() - 1] * head -> tail[0] tail[1] ... tail[tail.size() - 1]
@ -19,6 +25,7 @@ namespace pds {
std::string head; ///< The replaced variable std::string head; ///< The replaced variable
std::vector<std::string> tail; ///< The replacement rule std::vector<std::string> tail; ///< The replacement rule
const std::string name; ///< An optional name for this rule, if it's empty, there's no name const std::string name; ///< An optional name for this rule, if it's empty, there's no name
std::pair<bool, std::pair<int, PrecedenceType>> precedence; ///< precedence for this rule, the bool indicates whether it's valid (invalid if there is no rightmost terminal and no explicit precedence)
bool operator<(const Rule& other) const { bool operator<(const Rule& other) const {
if(head != other.head){ if(head != other.head){
@ -33,12 +40,6 @@ namespace pds {
Rule(const std::string& h, const std::vector<std::string>& t, const std::string& name) : head(h), tail(t), name(name) {} Rule(const std::string& h, const std::vector<std::string>& t, const std::string& name) : head(h), tail(t), name(name) {}
}; };
enum class PrecedenceType {
LEFT,
RIGHT,
NONASSOC
};
/** /**
* A context free grammar * A context free grammar
* Keeps track of variables, terminals and replacement rules * Keeps track of variables, terminals and replacement rules

View File

@ -132,32 +132,23 @@ LRTable Generator<Itemset>::generate() {
} else if (table.act[curIdx].count(term)) { } else if (table.act[curIdx].count(term)) {
if (table.act[curIdx][term].first == Action::SHIFT) { if (table.act[curIdx][term].first == Action::SHIFT) {
//Shift-Reduce conflict, rapport and resolve it (TODO) //Shift-Reduce conflict, rapport and resolve it (TODO)
std::string leftToken = "";
for (std::size_t i = m_gram.rules[rule_applied]->tail.size(); i > 0; i--) {
std::string leftSym = m_gram.rules[rule_applied]->tail[i - 1];
if (m_gram.terminals.count(leftSym)) {
leftToken = leftSym;
break;
}
}
bool handled = false; bool handled = false;
auto leftTokenIt = m_gram.precedence.find(leftToken);
auto rightTokenIt = m_gram.precedence.find(term); auto rightTokenIt = m_gram.precedence.find(term);
if (leftTokenIt != m_gram.precedence.end() && rightTokenIt != m_gram.precedence.end()) { if (m_gram.rules[rule_applied]->precedence.first && rightTokenIt != m_gram.precedence.end()) {
if (leftTokenIt->second.first > rightTokenIt->second.first) { auto& leftPrec = m_gram.rules[rule_applied]->precedence.second;
if (leftPrec.first > rightTokenIt->second.first) {
// Keep the shift // Keep the shift
handled = true; handled = true;
} else if (leftTokenIt->second.first > rightTokenIt->second.first) { } else if (leftPrec.first > rightTokenIt->second.first) {
// replace with a reduce // replace with a reduce
table.act[curIdx][term] = {Action::REDUCE, rule_applied}; table.act[curIdx][term] = {Action::REDUCE, rule_applied};
handled = true; handled = true;
} else if (leftTokenIt->second.second == PrecedenceType::LEFT) { } else if (leftPrec.second == PrecedenceType::LEFT) {
// replace with a reduce // replace with a reduce
table.act[curIdx][term] = {Action::REDUCE, rule_applied}; table.act[curIdx][term] = {Action::REDUCE, rule_applied};
handled = true; handled = true;
} else if (leftTokenIt->second.second == PrecedenceType::RIGHT) { } else if (leftPrec.second == PrecedenceType::RIGHT) {
// Keep the shift // Keep the shift
handled = true; handled = true;
} //Nonassoc falls through } //Nonassoc falls through