From 013bf91fbd6eec6abae84457079201ccf6bd0e6a Mon Sep 17 00:00:00 2001 From: Eugen Wissner Date: Mon, 31 Mar 2025 12:48:30 +0200 Subject: [PATCH] Save procedure info in the symbol table --- boot/semantic.cc | 63 +++++++++++++------------ boot/symbol.cc | 89 +++++++++++++++++------------------- include/elna/boot/semantic.h | 2 + include/elna/boot/symbol.h | 19 +++++++- 4 files changed, 96 insertions(+), 77 deletions(-) diff --git a/boot/semantic.cc b/boot/semantic.cc index 8c480f5..72fea6b 100644 --- a/boot/semantic.cc +++ b/boot/semantic.cc @@ -46,6 +46,33 @@ namespace elna::boot { } + procedure_type declaration_visitor::build_procedure(procedure_type_expression& type_expression) + { + procedure_type::return_t result_return; + + if (type_expression.return_type.no_return) + { + result_return = procedure_type::return_t(std::monostate{}); + } + else if (type_expression.return_type.proper_type != nullptr) + { + type_expression.return_type.proper_type->accept(this); + result_return = procedure_type::return_t(this->current_type); + } + else + { + result_return = procedure_type::return_t(); + } + procedure_type result_type = procedure_type(result_return); + + for (std::shared_ptr parameter : type_expression.parameters) + { + parameter->accept(this); + result_type.parameters.push_back(this->current_type); + } + return result_type; + } + void declaration_visitor::visit(program *program) { for (type_definition *const type : program->types) @@ -134,7 +161,7 @@ namespace elna::boot { auto result_type = std::make_shared(); - for (auto& field : type_expression->fields) + for (const field_declaration& field : type_expression->fields) { field.second->accept(this); result_type->fields.push_back(std::make_pair(field.first, this->current_type)); @@ -144,26 +171,10 @@ namespace elna::boot void declaration_visitor::visit(procedure_type_expression *type_expression) { - std::shared_ptr result_type; + std::shared_ptr result_type = + std::make_shared(std::move(build_procedure(*type_expression))); - if (type_expression->return_type.no_return) - { - result_type = std::make_shared(procedure_type::return_t(std::monostate{})); - } - else if (type_expression->return_type.proper_type != nullptr) - { - type_expression->return_type.proper_type->accept(this); - result_type = std::make_shared(procedure_type::return_t(this->current_type)); - } - else - { - result_type = std::make_shared(procedure_type::return_t()); - } - for (auto& parameter : type_expression->parameters) - { - parameter->accept(this); - result_type->parameters.push_back(this->current_type); - } + this->current_type = type(result_type); } void declaration_visitor::visit(variable_declaration *declaration) @@ -177,14 +188,10 @@ namespace elna::boot void declaration_visitor::visit(procedure_definition *definition) { - for (auto heading_parameter : definition->heading().parameters) - { - heading_parameter->accept(this); - } - if (definition->heading().return_type.proper_type != nullptr) - { - definition->heading().return_type.proper_type->accept(this); - } + std::shared_ptr info = std::make_shared( + build_procedure(definition->heading()), definition->parameter_names); + + this->symbols->enter(definition->identifier, info); if (definition->body != nullptr) { definition->body->accept(this); diff --git a/boot/symbol.cc b/boot/symbol.cc index 80a9d77..643ea4e 100644 --- a/boot/symbol.cc +++ b/boot/symbol.cc @@ -53,6 +53,11 @@ namespace elna::boot { } + type::type(std::shared_ptr procedure) + : tag(type_tag::procedure), procedure(procedure) + { + } + void type::copy(const type& other) { switch (other.tag) @@ -77,6 +82,9 @@ namespace elna::boot case type_tag::array: new (&array) std::shared_ptr(other.array); break; + case type_tag::procedure: + new (&procedure) std::shared_ptr(other.procedure); + break; } } @@ -110,6 +118,9 @@ namespace elna::boot case type_tag::array: new (&array) std::shared_ptr(std::move(other.array)); break; + case type_tag::procedure: + new (&procedure) std::shared_ptr(std::move(other.procedure)); + break; } } @@ -164,85 +175,52 @@ namespace elna::boot case type_tag::array: this->array.~shared_ptr(); break; + case type_tag::procedure: + this->procedure.~shared_ptr(); + break; } } template<> std::shared_ptr type::get() const { - if (tag == type_tag::alias) - { - return this->alias.lock(); - } - else - { - return nullptr; - } + return tag == type_tag::alias ? this->alias.lock() : nullptr; } template<> std::shared_ptr type::get() const { - if (tag == type_tag::primitive) - { - return this->primitive; - } - else - { - return nullptr; - } + return tag == type_tag::primitive ? this->primitive : nullptr; } template<> std::shared_ptr type::get() const { - if (tag == type_tag::record) - { - return this->record; - } - else - { - return nullptr; - } + return tag == type_tag::record ? this->record : nullptr; } template<> std::shared_ptr type::get() const { - if (tag == type_tag::_union) - { - return this->_union; - } - else - { - return nullptr; - } + return tag == type_tag::_union ? this->_union : nullptr; } template<> std::shared_ptr type::get() const { - if (tag == type_tag::pointer) - { - return this->pointer; - } - else - { - return nullptr; - } + return tag == type_tag::pointer ? this->pointer : nullptr; } template<> std::shared_ptr type::get() const { - if (tag == type_tag::array) - { - return this->array; - } - else - { - return nullptr; - } + return tag == type_tag::array ? this->array : nullptr; + } + + template<> + std::shared_ptr type::get() const + { + return tag == type_tag::procedure ? this->procedure : nullptr; } bool type::empty() const @@ -284,6 +262,11 @@ namespace elna::boot return nullptr; } + std::shared_ptr info::is_procedure() + { + return nullptr; + } + type_info::type_info(const type symbol) : symbol(symbol) { @@ -294,6 +277,16 @@ namespace elna::boot return std::static_pointer_cast(shared_from_this()); } + procedure_info::procedure_info(const procedure_type symbol, const std::vector names) + : symbol(symbol), names(names) + { + } + + std::shared_ptr procedure_info::is_procedure() + { + return std::static_pointer_cast(shared_from_this()); + } + std::shared_ptr builtin_symbol_table() { auto result = std::make_shared(); diff --git a/include/elna/boot/semantic.h b/include/elna/boot/semantic.h index 295ec2c..76de3dd 100644 --- a/include/elna/boot/semantic.h +++ b/include/elna/boot/semantic.h @@ -51,6 +51,8 @@ namespace elna::boot type current_type; std::shared_ptr symbols; + procedure_type build_procedure(procedure_type_expression& type_expression); + public: std::unordered_map> unresolved; diff --git a/include/elna/boot/symbol.h b/include/elna/boot/symbol.h index 20439b0..45629c8 100644 --- a/include/elna/boot/symbol.h +++ b/include/elna/boot/symbol.h @@ -33,6 +33,7 @@ namespace elna::boot class union_type; class pointer_type; class array_type; + class procedure_type; class type { @@ -44,7 +45,8 @@ namespace elna::boot record, _union, pointer, - array + array, + procedure }; type_tag tag{ type_tag::empty }; union @@ -55,6 +57,7 @@ namespace elna::boot std::shared_ptr _union; std::shared_ptr pointer; std::shared_ptr array; + std::shared_ptr procedure; }; void copy(const type& other); @@ -68,6 +71,7 @@ namespace elna::boot explicit type(std::shared_ptr _union); explicit type(std::shared_ptr pointer); explicit type(std::shared_ptr array); + explicit type(std::shared_ptr procedure); type(const type& other); type& operator=(const type& other); @@ -138,6 +142,7 @@ namespace elna::boot }; class type_info; + class procedure_info; class info : public std::enable_shared_from_this { @@ -145,6 +150,7 @@ namespace elna::boot virtual ~info() = 0; virtual std::shared_ptr is_type(); + virtual std::shared_ptr is_procedure(); }; class type_info : public info @@ -157,6 +163,17 @@ namespace elna::boot std::shared_ptr is_type() override; }; + class procedure_info : public info + { + public: + const procedure_type symbol; + const std::vector names; + + procedure_info(const procedure_type symbol, const std::vector names); + + std::shared_ptr is_procedure() override; + }; + /** * Symbol table. */