1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
#include "cru/parse/Grammar.hpp"
#include "cru/parse/Symbol.hpp"
#include <algorithm>
#include <iterator>
#include <unordered_map>
namespace cru::parse {
Grammar::Grammar() {}
Grammar::~Grammar() {
for (auto symbol : symbols_) {
delete symbol;
}
for (auto production : productions_) {
delete production;
}
}
Terminal* Grammar::CreateTerminal(String name) {
auto terminal = new Terminal(this, std::move(name));
terminals_.push_back(terminal);
symbols_.push_back(terminal);
return terminal;
}
Nonterminal* Grammar::CreateNonterminal(String name) {
auto nonterminal = new Nonterminal(this, std::move(name));
nonterminals_.push_back(nonterminal);
symbols_.push_back(nonterminal);
return nonterminal;
}
Production* Grammar::CreateProduction(String name, Nonterminal* left,
std::vector<Symbol*> right) {
Expects(left->GetGrammar() == this);
Expects(std::all_of(right.cbegin(), right.cend(), [this](Symbol* symbol) {
return symbol->GetGrammar() == this;
}));
auto production =
new Production(this, std::move(name), left, std::move(right));
productions_.push_back(production);
return production;
}
bool Grammar::RemoveSymbol(Symbol* symbol) {
for (auto production : productions_) {
if (production->GetLeft() == symbol) return false;
for (auto s : production->GetRight()) {
if (s == symbol) return false;
}
}
auto i = std::find(symbols_.begin(), symbols_.end(), symbol);
if (i == symbols_.cend()) return false;
symbols_.erase(i);
if (auto t = dynamic_cast<Terminal*>(symbol)) {
terminals_.erase(std::find(terminals_.begin(), terminals_.end(), t));
} else if (auto n = dynamic_cast<Nonterminal*>(symbol)) {
nonterminals_.erase(
std::find(nonterminals_.begin(), nonterminals_.end(), n));
}
delete symbol;
return true;
}
bool Grammar::RemoveProduction(Production* production) {
auto i = std::find(productions_.begin(), productions_.end(), production);
if (i == productions_.cend()) return false;
productions_.erase(i);
delete production;
return true;
}
Grammar* Grammar::Clone() const {
Grammar* g = new Grammar();
std::unordered_map<Symbol*, Symbol*> symbol_map;
for (auto old_terminal : terminals_) {
auto new_terminal = g->CreateTerminal(old_terminal->GetName());
symbol_map.emplace(old_terminal, new_terminal);
}
for (auto old_nonterminal : nonterminals_) {
auto new_nonterminal = g->CreateNonterminal(old_nonterminal->GetName());
symbol_map.emplace(old_nonterminal, new_nonterminal);
}
g->SetStartSymbol(static_cast<Nonterminal*>(symbol_map[start_symbol_]));
for (auto old_production : productions_) {
std::vector<Symbol*> new_right;
std::transform(old_production->GetRight().cbegin(),
old_production->GetRight().cend(),
std::back_inserter(new_right),
[&symbol_map](Symbol* old) { return symbol_map[old]; });
g->CreateProduction(
old_production->GetName(),
static_cast<Nonterminal*>(symbol_map[old_production->GetLeft()]),
std::move(new_right));
}
return g;
}
} // namespace cru::parse
|