2015-03-31 04:09:03 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) 2015, Matthias Schiffer <mschiffer@universe-factory.net>
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer in the documentation
|
|
|
|
and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
|
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "generator.hpp"
|
|
|
|
|
2015-03-31 18:38:59 +02:00
|
|
|
#include <queue>
|
|
|
|
|
2015-03-31 04:09:03 +02:00
|
|
|
|
|
|
|
namespace solar {
|
|
|
|
|
2015-03-31 18:38:59 +02:00
|
|
|
std::set<item_t> generator_t::get_set(const std::string &nonterm) {
|
|
|
|
std::set<item_t> set;
|
|
|
|
|
|
|
|
auto entries = rules.equal_range(nonterm);
|
|
|
|
for (auto entry = entries.first; entry != entries.second; ++entry)
|
|
|
|
set.insert(entry->second);
|
|
|
|
|
|
|
|
return set;
|
|
|
|
}
|
|
|
|
|
|
|
|
void generator_t::close_set(std::set<item_t> *set) {
|
|
|
|
std::queue<item_t> queue;
|
|
|
|
for (const item_t &item : *set)
|
|
|
|
queue.emplace(item);
|
|
|
|
|
|
|
|
for (; !queue.empty(); queue.pop()) {
|
|
|
|
const item_t &cur = queue.front();
|
|
|
|
if (!cur.has_next())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const symbol_t &sym = cur.get_next_symbol();
|
|
|
|
if (sym.get_type() != SYMBOL_TYPE_NONTERM)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (const item_t &item : get_set(sym.get_value())) {
|
|
|
|
if (set->insert(item).second)
|
|
|
|
queue.push(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void generator_t::print_item(const item_t &item) {
|
|
|
|
printf("%s |=", item.get_lhs().c_str());
|
|
|
|
|
|
|
|
for (size_t i = 0; i <= item.get_rhs().size(); i++) {
|
|
|
|
if (i == item.get_point())
|
|
|
|
printf(" .");
|
|
|
|
|
|
|
|
if (i == item.get_rhs().size())
|
|
|
|
break;
|
|
|
|
|
|
|
|
switch (item.get_rhs()[i].get_type()) {
|
|
|
|
case SYMBOL_TYPE_CHAR:
|
|
|
|
printf(" '%c'", item.get_rhs()[i].get_value()[0]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf(" %s", item.get_rhs()[i].get_value().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void generator_t::print_set(const std::set<item_t> &set) {
|
|
|
|
for (const item_t &item : set)
|
|
|
|
print_item(item);
|
|
|
|
}
|
|
|
|
|
2015-03-31 04:09:03 +02:00
|
|
|
generator_t::generator_t(const std::set<item_t> &rules0) {
|
|
|
|
for (item_t rule : rules0) {
|
|
|
|
rules.emplace(rule.get_lhs(), rule);
|
|
|
|
|
|
|
|
while (rule.has_next()) {
|
|
|
|
items.emplace(rule.get_next_symbol(), rule);
|
|
|
|
rule.shift();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-31 18:38:59 +02:00
|
|
|
std::set<item_t> set0 = get_set("");
|
|
|
|
close_set(&set0);
|
|
|
|
|
|
|
|
print_set(set0);
|
2015-03-31 04:09:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|