Fix LALR infinite loop

This commit is contained in:
Thomas Avé 2017-01-11 18:43:05 +01:00
parent cf237b04eb
commit 029dbf6bcb
7 changed files with 28 additions and 14 deletions

View File

@ -13,7 +13,7 @@ public:
LALR1Itemset(std::shared_ptr<Rule> start); LALR1Itemset(std::shared_ptr<Rule> start);
bool canMerge(const LALR1Itemset& rhs) const; bool canMerge(const LALR1Itemset& rhs) const;
void merge(const LALR1Itemset& rhs); bool merge(const LALR1Itemset& rhs);
private: private:
}; };

View File

@ -21,7 +21,7 @@ public:
Itemset succ(std::string sym) const; Itemset succ(std::string sym) const;
bool operator==(const Itemset& rhs) const; bool operator==(const Itemset& rhs) const;
bool canMerge(const Itemset& rhs) const; bool canMerge(const Itemset& rhs) const;
void merge(const Itemset& rhs); bool merge(const Itemset& rhs);
bool empty() const; bool empty() const;
protected: protected:
@ -89,8 +89,9 @@ bool LR0ItemsetBase<Itemset>::canMerge(const Itemset&) const {
} }
template <typename Itemset> template <typename Itemset>
void LR0ItemsetBase<Itemset>::merge(const Itemset&) { bool LR0ItemsetBase<Itemset>::merge(const Itemset&) {
//NO-OP //NO-OP
return false;
} }
template <typename Itemset> template <typename Itemset>

View File

@ -13,7 +13,7 @@ public:
LR1Itemset(std::shared_ptr<Rule> start); LR1Itemset(std::shared_ptr<Rule> start);
bool canMerge(const LR1Itemset& rhs) const; bool canMerge(const LR1Itemset& rhs) const;
void merge(const LR1Itemset& rhs); bool merge(const LR1Itemset& rhs);
private: private:
}; };

View File

@ -24,7 +24,7 @@ namespace lr {
* - Itemset succ(std::string) const; compute the successor of this set, over the given symbol * - Itemset succ(std::string) const; compute the successor of this set, over the given symbol
* - bool operator==(const Itemset&); are these two Itemsets equal * - bool operator==(const Itemset&); are these two Itemsets equal
* - bool canMerge(const Itemset&) const; Can the given Itemset be merged into this one * - bool canMerge(const Itemset&) const; Can the given Itemset be merged into this one
* - void merge(const Itemset&); Merge the given Itemset into this one * - bool merge(const Itemset&); Merge the given Itemset into this one, return whether there was a change
* - bool empty() const; is this Itemset empty (== not useful) * - bool empty() const; is this Itemset empty (== not useful)
* - std::set<std::size_t> getReduces(const Grammar&, std::string) const; get all Rule indices where a reduce should happen with given lookahead (not necessarily a set, but iterable) * - std::set<std::size_t> getReduces(const Grammar&, std::string) const; get all Rule indices where a reduce should happen with given lookahead (not necessarily a set, but iterable)
* - static bool needsFollow() const; does this type of Itemset need Follow sets to work, if so the first and follow unique_ptr's of the grammar passed will be initialized * - static bool needsFollow() const; does this type of Itemset need Follow sets to work, if so the first and follow unique_ptr's of the grammar passed will be initialized
@ -99,8 +99,9 @@ LRTable Generator<Itemset>::generate() {
if (itemsets[idx] == s) { if (itemsets[idx] == s) {
break; break;
} else if (itemsets[idx].canMerge(s)) { } else if (itemsets[idx].canMerge(s)) {
itemsets[idx].merge(s); if (itemsets[idx].merge(s)) {
q.emplace(idx, std::move(s)); q.emplace(idx, std::move(s));
}
break; break;
} }
} }

View File

@ -1,8 +1,12 @@
parser: lalr(1) parser: SLR(1)
lexesis: lexer.lxs lexesis: lexer.lxs
terminals: terminals:
A B TERMINAL
start: s start: s
grammar: grammar:
s -> x x; s -> a s
x -> A x | B; | b
;
a → TERMINAL;
b -> a;

View File

@ -31,18 +31,25 @@ bool LALR1Itemset::canMerge(const LALR1Itemset& rhs) const {
return true; return true;
} }
void LALR1Itemset::merge(const LALR1Itemset& rhs) { bool LALR1Itemset::merge(const LALR1Itemset& rhs) {
bool change = false;
for (auto& toMerge : rhs.m_items) { for (auto& toMerge : rhs.m_items) {
bool found = false; bool found = false;
for (auto& it : m_items) { for (auto& it : m_items) {
if (it.rule == toMerge.rule && it.dotIdx == toMerge.dotIdx) { if (it.rule == toMerge.rule && it.dotIdx == toMerge.dotIdx) {
found = true; found = true;
it.lookaheads.insert(toMerge.lookaheads.begin(), toMerge.lookaheads.end()); for (auto& la : toMerge.lookaheads) {
if (!it.lookaheads.count(la)) {
change = true;
it.lookaheads.insert(la);
}
}
break; break;
} }
} }
assert(found); assert(found);
} }
return change;
} }
} /* lr */ } /* lr */

View File

@ -13,8 +13,9 @@ bool LR1Itemset::canMerge(const LR1Itemset&) const {
return false; return false;
} }
void LR1Itemset::merge(const LR1Itemset&) { bool LR1Itemset::merge(const LR1Itemset&) {
//NO-OP //NO-OP
return false;
} }
} /* lr */ } /* lr */