diff --git a/boot/semantic.cc b/boot/semantic.cc index d1b73c2..d29771a 100644 --- a/boot/semantic.cc +++ b/boot/semantic.cc @@ -15,6 +15,8 @@ You should have received a copy of the GNU General Public License along with GCC; see the file COPYING3. If not see . */ +#include + #include "elna/boot/semantic.h" namespace elna @@ -52,7 +54,8 @@ namespace boot { for (type_definition *const type : program->types) { - if (!this->unresolved.insert({ type->identifier, std::make_shared() }).second) + if (!this->unresolved.insert({ type->identifier, std::make_shared(type->identifier) }).second + || this->symbols->contains(type->identifier)) { add_error(type->identifier, this->input_file, type->position()); } @@ -61,6 +64,11 @@ namespace boot { type->accept(this); } + for (auto& unresolved : this->unresolved) + { + auto info = std::make_shared(type_info(type(unresolved.second))); + this->symbols->enter(std::move(unresolved.first), info); + } } void declaration_visitor::visit(type_definition *definition) @@ -79,7 +87,7 @@ namespace boot { this->current_type = type(unresolved_alias->second); } - else if (auto from_symbol_table = this->symbols.lookup(type_expression->name)) + else if (auto from_symbol_table = this->symbols->lookup(type_expression->name)) { this->current_type = from_symbol_table->is_type()->symbol; } @@ -102,32 +110,54 @@ namespace boot this->current_type = type(std::make_shared(this->current_type, type_expression->size)); } + bool declaration_visitor::build_composite_type(const std::vector& declarations, + std::vector& fields) + { + std::set field_names; + + for (auto& field : declarations) + { + if (field_names.find(field.first) != field_names.cend()) + { + add_error(field.first, this->input_file, field.second->position()); + this->current_type = type(); + return false; + } + field.second->accept(this); + if (!this->current_type.empty()) + { + fields.push_back({ field.first, type(this->current_type) }); + } + } + return true; + } + void declaration_visitor::visit(record_type_expression *type_expression) { auto type_definition = std::make_shared(); - for (auto& field : type_expression->fields) + if (build_composite_type(type_expression->fields, type_definition->fields)) { - field.second->accept(this); - if (!this->current_type.empty()) - { - type_definition->fields.push_back({ field.first, type(this->current_type) }); - } + this->current_type = type(type_definition); + } + else + { + this->current_type = type(); } - this->current_type = type(type_definition); } void declaration_visitor::visit(union_type_expression *type_expression) { auto type_definition = std::make_shared(); - for (auto& field : type_expression->fields) + if (build_composite_type(type_expression->fields, type_definition->fields)) { - field.second->accept(this); - - type_definition->fields.push_back({ field.first, type(this->current_type) }); + this->current_type = type(type_definition); + } + else + { + this->current_type = type(); } - this->current_type = type(type_definition); } void declaration_visitor::visit(procedure_type_expression *type_expression) diff --git a/boot/symbol.cc b/boot/symbol.cc index c1de6e6..0391579 100644 --- a/boot/symbol.cc +++ b/boot/symbol.cc @@ -247,6 +247,11 @@ namespace boot return tag == type_tag::empty; } + alias_type::alias_type(const std::string& name) + : name(name), reference() + { + } + pointer_type::pointer_type(type base) : base(base) { diff --git a/gcc/Make-lang.in b/gcc/Make-lang.in index a9fed82..3159853 100644 --- a/gcc/Make-lang.in +++ b/gcc/Make-lang.in @@ -136,7 +136,7 @@ elna.stagefeedback: stagefeedback-start -mv elna/*$(objext) stagefeedback/elna ELNA_INCLUDES = -I $(srcdir)/elna/include -I elna/generated -ELNA_CXXFLAGS = -std=c++14 +ELNA_CXXFLAGS = -std=c++17 elna/%.o: elna/boot/%.cc elna/generated/parser.hh elna/generated/location.hh $(COMPILE) $(ELNA_CXXFLAGS) $(ELNA_INCLUDES) $< diff --git a/gcc/elna-generic.cc b/gcc/elna-generic.cc index db39118..6906815 100644 --- a/gcc/elna-generic.cc +++ b/gcc/elna-generic.cc @@ -38,11 +38,15 @@ namespace elna { namespace gcc { - static tree get_inner_alias(std::shared_ptr symbol_table, const boot::type& type) + tree handle_symbol(const std::string& symbol_name, const boot::type& type, + std::shared_ptr from, std::shared_ptr to); + + tree get_inner_alias(const boot::type& type, + std::shared_ptr from, std::shared_ptr to) { if (auto reference = type.get()) { - return symbol_table->lookup(reference->identifier); + return to->lookup(reference->identifier); } else if (auto reference = type.get()) { @@ -54,15 +58,36 @@ namespace gcc } else if (auto reference = type.get()) { - return build_pointer_type_for_mode(get_inner_alias(symbol_table, reference->base), VOIDmode, true); + return build_pointer_type_for_mode(get_inner_alias(reference->base, from, to), VOIDmode, true); + } + else if (auto reference = type.get()) + { + tree lower_bound = build_int_cst_type(integer_type_node, 0); + tree upper_bound = build_int_cst_type(integer_type_node, reference->size); + tree range_type = build_range_type(integer_type_node, lower_bound, upper_bound); + + return build_array_type(get_inner_alias(reference->base, from, to), range_type); } else if (auto reference = type.get()) { - return get_inner_alias(symbol_table, reference->reference); + return handle_symbol(reference->name, reference->reference, from, to); } return error_mark_node; } + tree handle_symbol(const std::string& symbol_name, const boot::type& type, + std::shared_ptr from, std::shared_ptr to) + { + auto looked_up = to->lookup(symbol_name); + + if (looked_up == NULL_TREE) + { + looked_up = get_inner_alias(type, from, to); + to->enter(symbol_name, looked_up); + } + return looked_up; + } + std::deque> do_semantic_analysis(const char *path, std::unique_ptr& ast, std::shared_ptr symbols) { @@ -73,10 +98,13 @@ namespace gcc if (declaration_visitor.errors().empty()) { - for (auto unresolved : declaration_visitor.unresolved) + for (auto& [symbol_name, symbol_info] : *info_table) { - auto inner_alias = get_inner_alias(symbols, boot::type(unresolved.second)); - symbols->enter(unresolved.first, inner_alias); + handle_symbol(symbol_name, symbol_info->is_type()->symbol, info_table, symbols); + } + for (auto& [symbol_name, symbol_info] : *info_table) + { + // printf("%s\n", symbol_name.c_str()); } } return std::move(declaration_visitor.errors()); @@ -1154,7 +1182,7 @@ namespace gcc void generic_visitor::visit(boot::record_type_expression *type) { std::set field_names; - tree record_type_node = this->current_expression == NULL_TREE + tree composite_type_node = this->current_expression == NULL_TREE ? make_node(RECORD_TYPE) : this->current_expression; @@ -1174,19 +1202,19 @@ namespace gcc return; } tree field_declaration = build_field(get_location(&field.second->position()), - record_type_node, field.first, this->current_expression); - TYPE_FIELDS(record_type_node) = chainon(TYPE_FIELDS(record_type_node), field_declaration); + composite_type_node, field.first, this->current_expression); + TYPE_FIELDS(composite_type_node) = chainon(TYPE_FIELDS(composite_type_node), field_declaration); this->current_expression = NULL_TREE; } - layout_type(record_type_node); + layout_type(composite_type_node); - this->current_expression = record_type_node; + this->current_expression = composite_type_node; } void generic_visitor::visit(boot::union_type_expression *type) { std::set field_names; - tree union_type_node = this->current_expression == NULL_TREE + tree composite_type_node = this->current_expression == NULL_TREE ? make_node(UNION_TYPE) : this->current_expression; @@ -1206,13 +1234,13 @@ namespace gcc return; } tree field_declaration = build_field(get_location(&field.second->position()), - union_type_node, field.first, this->current_expression); - TYPE_FIELDS(union_type_node) = chainon(TYPE_FIELDS(union_type_node), field_declaration); + composite_type_node, field.first, this->current_expression); + TYPE_FIELDS(composite_type_node) = chainon(TYPE_FIELDS(composite_type_node), field_declaration); this->current_expression = NULL_TREE; } - layout_type(union_type_node); + layout_type(composite_type_node); - this->current_expression = union_type_node; + this->current_expression = composite_type_node; } void generic_visitor::visit(boot::procedure_type_expression *type) diff --git a/include/elna/boot/semantic.h b/include/elna/boot/semantic.h index afbf353..c17287e 100644 --- a/include/elna/boot/semantic.h +++ b/include/elna/boot/semantic.h @@ -51,11 +51,13 @@ namespace boot class declaration_visitor final : public empty_visitor, public error_container { type current_type; - symbol_table symbols; - - public: + std::shared_ptr symbols; std::unordered_map> unresolved; + bool build_composite_type(const std::vector& declarations, + std::vector& fields); + + public: explicit declaration_visitor(const char *path, std::shared_ptr symbols); void visit(primitive_type_expression *type_expression) override; diff --git a/include/elna/boot/symbol.h b/include/elna/boot/symbol.h index 7e45f2f..404cf30 100644 --- a/include/elna/boot/symbol.h +++ b/include/elna/boot/symbol.h @@ -85,7 +85,10 @@ namespace boot struct alias_type { + const std::string name; type reference; + + explicit alias_type(const std::string& name); }; struct pointer_type @@ -173,22 +176,30 @@ namespace boot iterator begin() { - return entries.begin(); + return this->entries.begin(); } iterator end() { - return entries.end(); + return this->entries.end(); } const_iterator cbegin() const { - return entries.cbegin(); + return this->entries.cbegin(); } const_iterator cend() const { - return entries.cend(); + return this->entries.cend(); + } + + /** + * \return Symbol count in the current scope. + */ + std::size_t size() const + { + return this->entries.size(); } /** @@ -213,6 +224,15 @@ namespace boot return nothing; } + /** + * \param name Symbol name. + * \return Whether the table contains a symbol with the given name. + */ + bool contains(const std::string& name) + { + return lookup(name) != nothing; + } + /** * Registers new symbol. *