LR support for explicit rule precedences
This commit is contained in:
parent
8358813204
commit
982d385ea3
2
TODO
2
TODO
|
@ -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?
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue