From 5d9f0f36c5c1f10b0c495effe1b18a0ae755e84d Mon Sep 17 00:00:00 2001 From: Eugen Wissner Date: Tue, 18 Mar 2025 11:37:10 +0100 Subject: [PATCH] Skip parameter names in procedure type expressions --- boot/ast.cc | 28 +++++------- boot/driver.cc | 5 +-- boot/parser.yy | 69 +++++++++++++++++------------- boot/result.cc | 5 +-- boot/semantic.cc | 5 +-- boot/symbol.cc | 5 +-- gcc/elna-builtins.cc | 5 +-- gcc/elna-diagnostic.cc | 5 +-- gcc/elna-generic.cc | 23 +++++----- gcc/elna-tree.cc | 5 +-- include/elna/boot/ast.h | 26 +++++------ include/elna/boot/driver.h | 5 +-- include/elna/boot/result.h | 5 +-- include/elna/boot/semantic.h | 5 +-- include/elna/boot/symbol.h | 5 +-- include/elna/gcc/elna-builtins.h | 5 +-- include/elna/gcc/elna-diagnostic.h | 5 +-- include/elna/gcc/elna-generic.h | 5 +-- include/elna/gcc/elna-tree.h | 5 +-- source.elna | 6 +-- 20 files changed, 92 insertions(+), 135 deletions(-) diff --git a/boot/ast.cc b/boot/ast.cc index d3e0b9f..98a8820 100644 --- a/boot/ast.cc +++ b/boot/ast.cc @@ -17,9 +17,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/ast.h" -namespace elna -{ -namespace boot +namespace elna::boot { void empty_visitor::visit(variable_declaration *) { @@ -538,14 +536,19 @@ namespace boot delete m_body; } - procedure_type_expression::procedure_type_expression(const struct position position, - std::shared_ptr return_type) - : type_expression(position), return_type(return_type), no_return(false) + return_declaration::return_declaration(std::shared_ptr type) + : type(type) { } - procedure_type_expression::procedure_type_expression(const struct position position, no_return_t) - : type_expression(position), return_type(nullptr), no_return(true) + return_declaration::return_declaration(std::monostate) + : no_return(true) + { + } + + procedure_type_expression::procedure_type_expression(const struct position position, + return_declaration return_type) + : type_expression(position), return_type(return_type) { } @@ -554,14 +557,6 @@ namespace boot visitor->visit(this); } - procedure_type_expression::~procedure_type_expression() - { - for (auto parameter : this->parameters) - { - delete parameter; - } - } - procedure_definition::procedure_definition(const struct position position, const std::string& identifier, const bool exported, std::shared_ptr heading, block *body) : definition(position, identifier, exported), m_heading(heading), body(body) @@ -1212,4 +1207,3 @@ namespace boot __builtin_unreachable(); }; } -} diff --git a/boot/driver.cc b/boot/driver.cc index 0257d44..eb1ad1d 100644 --- a/boot/driver.cc +++ b/boot/driver.cc @@ -17,9 +17,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/driver.h" -namespace elna -{ -namespace boot +namespace elna::boot { position make_position(const yy::location& location) { @@ -79,4 +77,3 @@ namespace boot } } } -} diff --git a/boot/parser.yy b/boot/parser.yy index 2959d2e..cd11e89 100644 --- a/boot/parser.yy +++ b/boot/parser.yy @@ -122,10 +122,9 @@ along with GCC; see the file COPYING3. If not see %type literal; %type constant_definition; %type > constant_part constant_definitions; -%type > variable_declarations variable_part variable_declaration - formal_parameters formal_parameter_list; -%type formal_parameter +%type > variable_declarations variable_part variable_declaration; %type > type_expression; +%type >> type_expressions; %type traits_expression; %type expression operand; %type unary_expression; @@ -140,14 +139,15 @@ along with GCC; see the file COPYING3. If not see %type statement; %type > statements; %type procedure_definition; -%type > procedure_heading; +%type , std::shared_ptr>> procedure_heading; +%type return_declaration; %type > procedure_definitions procedure_part; %type type_definition; %type > type_definitions type_part; %type block; -%type field_declaration; +%type field_declaration formal_parameter; %type >>> - optional_fields required_fields; + optional_fields required_fields formal_parameters; %type > elsif_then_statements elsif_do_statements; %type cast_expression; %type defer_statement; @@ -191,31 +191,32 @@ identifier_definitions: $$.emplace($$.cbegin(), $1); } | identifier_definition { $$.emplace_back(std::move($1)); } +return_declaration: + /* proper procedure */ {} + | "->" "!" { $$ = elna::boot::return_declaration(std::monostate{}); } + | "->" type_expression { $$ = elna::boot::return_declaration($2); } procedure_heading: - formal_parameter_list + "(" formal_parameters ")" return_declaration { - $$ = std::make_shared(elna::boot::make_position(@1)); - std::swap($1, $$->parameters); - } - | formal_parameter_list "->" "!" - { - $$ = std::make_shared(elna::boot::make_position(@1), - elna::boot::no_return); - std::swap($1, $$->parameters); - } - | formal_parameter_list "->" type_expression - { - $$ = std::make_shared(elna::boot::make_position(@1), $3); - std::swap($1, $$->parameters); + $$.second = std::make_shared(elna::boot::make_position(@1), + std::move($4)); + for (auto& [name, type] : $2) + { + $$.first.emplace_back(std::move(name)); + $$.second->parameters.push_back(type); + } } procedure_definition: "proc" identifier_definition procedure_heading ";" block { - $$ = new elna::boot::procedure_definition(elna::boot::make_position(@1), $2.first, $2.second, $3, $5); + $$ = new elna::boot::procedure_definition(elna::boot::make_position(@1), + $2.first, $2.second, $3.second, $5); + std::swap($3.first, $$->parameter_names); } | "proc" identifier_definition procedure_heading ";" "extern" { - $$ = new elna::boot::procedure_definition(elna::boot::make_position(@1), $2.first, $2.second, $3); + $$ = new elna::boot::procedure_definition(elna::boot::make_position(@1), $2.first, $2.second, $3.second); + std::swap($3.first, $$->parameter_names); } procedure_definitions: procedure_definition procedure_definitions @@ -438,7 +439,14 @@ expressions: std::swap($$, $3); $$.emplace($$.cbegin(), $1); } - | expression { $$.emplace_back(std::move($1)); } + | expression { $$.push_back($1); } +type_expressions: + type_expression "," type_expressions + { + std::swap($$, $3); + $$.emplace($$.cbegin(), $1); + } + | type_expression { $$.push_back($1); } designator_expression: operand "[" expression "]" { @@ -500,9 +508,12 @@ type_expression: { $$ = std::make_shared(elna::boot::make_position(@1), std::move($2)); } - | "proc" procedure_heading + | "proc" "(" type_expressions ")" return_declaration { - $$ = $2; + auto result = std::make_shared(elna::boot::make_position(@1), + std::move($5)); + std::swap(result->parameters, $3); + $$ = result; } | IDENTIFIER { @@ -559,18 +570,16 @@ type_part: | "type" type_definitions { std::swap($$, $2); } formal_parameter: IDENTIFIER ":" type_expression { - $$ = new elna::boot::variable_declaration(elna::boot::make_position(@2), $1, $3); + $$ = std::make_pair($1, $3); } formal_parameters: - formal_parameter "," formal_parameters + /* no formal parameters */ {} + | formal_parameter "," formal_parameters { std::swap($$, $3); $$.emplace($$.cbegin(), $1); } | formal_parameter { $$.emplace_back(std::move($1)); } -formal_parameter_list: - "(" ")" {} - | "(" formal_parameters ")" { std::swap($$, $2); } actual_parameter_list: "(" ")" {} | "(" expressions ")" { std::swap($$, $2); } diff --git a/boot/result.cc b/boot/result.cc index 11d85ac..41a7d51 100644 --- a/boot/result.cc +++ b/boot/result.cc @@ -17,9 +17,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/result.h" -namespace elna -{ -namespace boot +namespace elna::boot { error::error(const char *path, const struct position position) : position(position), path(path) @@ -46,4 +44,3 @@ namespace boot return m_errors; } } -} diff --git a/boot/semantic.cc b/boot/semantic.cc index 408082a..17e8ad7 100644 --- a/boot/semantic.cc +++ b/boot/semantic.cc @@ -17,9 +17,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/semantic.h" -namespace elna -{ -namespace boot +namespace elna::boot { undeclared_error::undeclared_error(const std::string& identifier, const char *path, const struct position position) : error(path, position), identifier(identifier) @@ -122,4 +120,3 @@ namespace boot { } } -} diff --git a/boot/symbol.cc b/boot/symbol.cc index c3ba0f2..0e2de25 100644 --- a/boot/symbol.cc +++ b/boot/symbol.cc @@ -17,9 +17,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/symbol.h" -namespace elna -{ -namespace boot +namespace elna::boot { type::type() { @@ -301,4 +299,3 @@ namespace boot return result; } } -} diff --git a/gcc/elna-builtins.cc b/gcc/elna-builtins.cc index 262d567..576b61b 100644 --- a/gcc/elna-builtins.cc +++ b/gcc/elna-builtins.cc @@ -21,9 +21,7 @@ along with GCC; see the file COPYING3. If not see #include "stringpool.h" #include "elna/gcc/elna-tree.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { void init_ttree() { @@ -66,4 +64,3 @@ namespace gcc return symbol_table; } } -} diff --git a/gcc/elna-diagnostic.cc b/gcc/elna-diagnostic.cc index b988f61..3668048 100644 --- a/gcc/elna-diagnostic.cc +++ b/gcc/elna-diagnostic.cc @@ -19,9 +19,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/gcc/elna-tree.h" #include "elna/gcc/elna1.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { location_t get_location(const boot::position *position) { @@ -144,4 +142,3 @@ namespace gcc } } } -} diff --git a/gcc/elna-generic.cc b/gcc/elna-generic.cc index 0d4926d..9e556cd 100644 --- a/gcc/elna-generic.cc +++ b/gcc/elna-generic.cc @@ -34,9 +34,7 @@ along with GCC; see the file COPYING3. If not see #include "fold-const.h" #include "langhooks.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { tree get_inner_alias(const boot::type& type, std::shared_ptr symbols) { @@ -317,7 +315,7 @@ namespace gcc tree fndecl = build_fn_decl(definition->identifier.c_str(), declaration_type); this->symbols->enter(definition->identifier, fndecl); - if (definition->heading().no_return) + if (definition->heading().return_type.no_return) { TREE_THIS_VOLATILE(fndecl) = 1; } @@ -336,19 +334,22 @@ namespace gcc function_args_iterator parameter_type; function_args_iter_init(¶meter_type, declaration_type); - for (const boot::variable_declaration *parameter : definition->heading().parameters) + std::vector::const_iterator parameter_name = definition->parameter_names.cbegin(); + + for (std::shared_ptr parameter : definition->heading().parameters) { tree declaration_tree = build_decl(get_location(¶meter->position()), PARM_DECL, - get_identifier(parameter->identifier.c_str()), function_args_iter_cond(¶meter_type)); + get_identifier(parameter_name->c_str()), function_args_iter_cond(¶meter_type)); DECL_CONTEXT(declaration_tree) = fndecl; DECL_ARG_TYPE(declaration_tree) = function_args_iter_cond(¶meter_type); if (definition->body != nullptr) { - this->symbols->enter(parameter->identifier, declaration_tree); + this->symbols->enter(*parameter_name, declaration_tree); } argument_chain = chainon(argument_chain, declaration_tree); function_args_iter_next(¶meter_type); + ++parameter_name; } DECL_ARGUMENTS(fndecl) = argument_chain; TREE_PUBLIC(fndecl) = definition->exported; @@ -779,15 +780,14 @@ namespace gcc for (std::size_t i = 0; i < type.parameters.size(); ++i) { - boot::type_expression& parameter_type = type.parameters.at(i)->variable_type(); - parameter_type.accept(this); + type.parameters.at(i)->accept(this); parameter_types[i] = this->current_expression; } tree return_type = void_type_node; - if (type.return_type != nullptr) + if (type.return_type.type != nullptr) { - type.return_type->accept(this); + type.return_type.type->accept(this); return_type = this->current_expression; } this->current_expression = NULL_TREE; @@ -1226,4 +1226,3 @@ namespace gcc defer(leave_scope()); } } -} diff --git a/gcc/elna-tree.cc b/gcc/elna-tree.cc index c8262df..7b890b6 100644 --- a/gcc/elna-tree.cc +++ b/gcc/elna-tree.cc @@ -24,9 +24,7 @@ along with GCC; see the file COPYING3. If not see #include "fold-const.h" #include "diagnostic-core.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { bool is_pointer_type(tree type) { @@ -210,4 +208,3 @@ namespace gcc } } } -} diff --git a/include/elna/boot/ast.h b/include/elna/boot/ast.h index 6c32a54..d72205e 100644 --- a/include/elna/boot/ast.h +++ b/include/elna/boot/ast.h @@ -21,11 +21,10 @@ along with GCC; see the file COPYING3. If not see #include #include #include +#include #include "elna/boot/result.h" -namespace elna -{ -namespace boot +namespace elna::boot { enum class binary_operator { @@ -379,10 +378,15 @@ namespace boot /** * Tags a procedure type as never returning. */ - struct no_return_t + struct return_declaration { + return_declaration() = default; + explicit return_declaration(std::shared_ptr type); + explicit return_declaration(std::monostate); + + std::shared_ptr type{ nullptr }; + bool no_return{ false }; }; - constexpr no_return_t no_return{}; /** * Procedure type. @@ -390,18 +394,14 @@ namespace boot class procedure_type_expression : public type_expression { public: - const std::shared_ptr return_type; - const bool no_return; - std::vector parameters; + const return_declaration return_type; + std::vector> parameters; procedure_type_expression(const struct position position, - std::shared_ptr return_type = nullptr); - procedure_type_expression(const struct position position, no_return_t); + return_declaration return_type = return_declaration()); void accept(parser_visitor *visitor); std::shared_ptr is_procedure() override; - - virtual ~procedure_type_expression() override; }; /** @@ -413,6 +413,7 @@ namespace boot public: block *const body; + std::vector parameter_names; procedure_definition(const struct position position, const std::string& identifier, const bool exported, std::shared_ptr heading, block *body = nullptr); @@ -840,4 +841,3 @@ namespace boot const char *print_binary_operator(const binary_operator operation); } -} diff --git a/include/elna/boot/driver.h b/include/elna/boot/driver.h index f09da87..0f3a37f 100644 --- a/include/elna/boot/driver.h +++ b/include/elna/boot/driver.h @@ -21,9 +21,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/ast.h" #include "location.hh" -namespace elna -{ -namespace boot +namespace elna::boot { position make_position(const yy::location& location); @@ -50,4 +48,3 @@ namespace boot char escape_char(char escape); } -} diff --git a/include/elna/boot/result.h b/include/elna/boot/result.h index 65bfe59..f053ece 100644 --- a/include/elna/boot/result.h +++ b/include/elna/boot/result.h @@ -22,9 +22,7 @@ along with GCC; see the file COPYING3. If not see #include #include -namespace elna -{ -namespace boot +namespace elna::boot { /** * Position in the source text. @@ -82,4 +80,3 @@ namespace boot } }; } -} diff --git a/include/elna/boot/semantic.h b/include/elna/boot/semantic.h index 5517026..ac5993f 100644 --- a/include/elna/boot/semantic.h +++ b/include/elna/boot/semantic.h @@ -24,9 +24,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/result.h" #include "elna/boot/symbol.h" -namespace elna -{ -namespace boot +namespace elna::boot { class undeclared_error : public error { @@ -68,4 +66,3 @@ namespace boot void visit(procedure_type_expression *) override; }; } -} diff --git a/include/elna/boot/symbol.h b/include/elna/boot/symbol.h index ac783ab..1d6a779 100644 --- a/include/elna/boot/symbol.h +++ b/include/elna/boot/symbol.h @@ -23,9 +23,7 @@ along with GCC; see the file COPYING3. If not see #include #include -namespace elna -{ -namespace boot +namespace elna::boot { class alias_type; class primitive_type; @@ -257,4 +255,3 @@ namespace boot std::shared_ptr builtin_symbol_table(); } -} diff --git a/include/elna/gcc/elna-builtins.h b/include/elna/gcc/elna-builtins.h index ed5b749..251b894 100644 --- a/include/elna/gcc/elna-builtins.h +++ b/include/elna/gcc/elna-builtins.h @@ -25,11 +25,8 @@ along with GCC; see the file COPYING3. If not see #include "elna/gcc/elna-tree.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { void init_ttree(); std::shared_ptr builtin_symbol_table(); } -} diff --git a/include/elna/gcc/elna-diagnostic.h b/include/elna/gcc/elna-diagnostic.h index d225505..2f37ee9 100644 --- a/include/elna/gcc/elna-diagnostic.h +++ b/include/elna/gcc/elna-diagnostic.h @@ -29,12 +29,9 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/result.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { location_t get_location(const boot::position *position); std::string print_type(tree type); void report_errors(const std::deque>& errors); } -} diff --git a/include/elna/gcc/elna-generic.h b/include/elna/gcc/elna-generic.h index 5ae5f90..fd9f97a 100644 --- a/include/elna/gcc/elna-generic.h +++ b/include/elna/gcc/elna-generic.h @@ -30,9 +30,7 @@ along with GCC; see the file COPYING3. If not see #include -namespace elna -{ -namespace gcc +namespace elna::gcc { std::deque> do_semantic_analysis(const char *path, std::unique_ptr& ast, std::shared_ptr info_table, @@ -104,4 +102,3 @@ namespace gcc void visit(boot::defer_statement *statement) override; }; } -} diff --git a/include/elna/gcc/elna-tree.h b/include/elna/gcc/elna-tree.h index 636a817..6b4a249 100644 --- a/include/elna/gcc/elna-tree.h +++ b/include/elna/gcc/elna-tree.h @@ -29,9 +29,7 @@ along with GCC; see the file COPYING3. If not see #include "elna/boot/ast.h" #include "elna/boot/symbol.h" -namespace elna -{ -namespace gcc +namespace elna::gcc { using symbol_table = boot::symbol_map; @@ -82,4 +80,3 @@ namespace gcc tree_code operator_code, tree left, tree right); tree build_field(location_t location, tree record_type, const std::string name, tree type); } -} diff --git a/source.elna b/source.elna index f6e3400..99770f2 100644 --- a/source.elna +++ b/source.elna @@ -88,9 +88,9 @@ type position: Position input: ^Byte - empty: proc(data: ^Byte) -> Bool - advance: proc(data: ^Byte) - head: proc(data: ^Byte) -> Char + empty: proc(^Byte) -> Bool + advance: proc(^Byte) + head: proc(^Byte) -> Char end Token* = record kind: Int