From 5d485f850583c67d0649aecb48c80f7bb1e68156 Mon Sep 17 00:00:00 2001 From: Eugen Wissner Date: Fri, 14 Mar 2025 11:22:11 +0100 Subject: [PATCH] Forbid redefenition of builtin types --- boot/semantic.cc | 10 ++++++++-- boot/symbol.cc | 5 +++++ gcc/Make-lang.in | 2 +- include/elna/boot/semantic.h | 2 +- include/elna/boot/symbol.h | 28 ++++++++++++++++++++++++---- 5 files changed, 39 insertions(+), 8 deletions(-) diff --git a/boot/semantic.cc b/boot/semantic.cc index d1b73c2..5752672 100644 --- a/boot/semantic.cc +++ b/boot/semantic.cc @@ -52,7 +52,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 +62,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 +85,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; } 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/include/elna/boot/semantic.h b/include/elna/boot/semantic.h index afbf353..d675708 100644 --- a/include/elna/boot/semantic.h +++ b/include/elna/boot/semantic.h @@ -51,7 +51,7 @@ namespace boot class declaration_visitor final : public empty_visitor, public error_container { type current_type; - symbol_table symbols; + std::shared_ptr symbols; public: std::unordered_map> unresolved; 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. *