Parse parameter list
This commit is contained in:
		| @@ -157,6 +157,8 @@ namespace elna::riscv | ||||
|  | ||||
|     void visitor::prologue() | ||||
|     { | ||||
|         this->variable_counter = 1; | ||||
|  | ||||
|         this->instructions.push_back(instruction(base_opcode::opImm)); | ||||
|         this->instructions.push_back(instruction(base_opcode::store)); | ||||
|         this->instructions.push_back(instruction(base_opcode::store)); | ||||
| @@ -195,7 +197,7 @@ namespace elna::riscv | ||||
|         this->instructions.push_back(instruction(base_opcode::op) | ||||
|             .r(x_register::a0, funct3_t::_and, x_register::zero, x_register::zero)); | ||||
|  | ||||
|         epilogue(static_cast<std::uint32_t>(variable_counter * 4 + 8 + main_symbol->stack_size())); | ||||
|         epilogue(static_cast<std::uint32_t>(this->variable_counter * 4 + 8 + main_symbol->stack_size())); | ||||
|         this->writer->sink(definition->identifier(), | ||||
|                 reinterpret_cast<const std::byte *>(this->instructions.data()), | ||||
|                 this->instructions.size() * sizeof(instruction)); | ||||
| @@ -224,7 +226,7 @@ namespace elna::riscv | ||||
|         this->instructions.push_back(instruction(base_opcode::op) | ||||
|             .r(x_register::a0, funct3_t::_and, x_register::zero, x_register::zero)); | ||||
|  | ||||
|         epilogue(static_cast<std::uint32_t>(variable_counter * 4 + 8 + main_symbol->stack_size())); | ||||
|         epilogue(static_cast<std::uint32_t>(this->variable_counter * 4 + 8 + main_symbol->stack_size())); | ||||
|         this->writer->sink("main", reinterpret_cast<const std::byte *>(this->instructions.data()), | ||||
|                 this->instructions.size() * sizeof(instruction)); | ||||
|     } | ||||
|   | ||||
| @@ -38,7 +38,7 @@ namespace elna::cli | ||||
|         { | ||||
|             for (const auto& compile_error : lex_result.errors()) | ||||
|             { | ||||
|                 std::cout << compile_error->line() << ':' << compile_error->column() | ||||
|                 std::cerr << compile_error->line() << ':' << compile_error->column() | ||||
|                     << ": " << compile_error->what() << std::endl; | ||||
|             } | ||||
|             return 1; | ||||
| @@ -49,7 +49,7 @@ namespace elna::cli | ||||
|         { | ||||
|             for (const auto& compile_error : parser.errors()) | ||||
|             { | ||||
|                 std::cout << compile_error->line() << ':' << compile_error->column() | ||||
|                 std::cerr << compile_error->line() << ':' << compile_error->column() | ||||
|                     << ": " << compile_error->what() << std::endl; | ||||
|             } | ||||
|             return 2; | ||||
|   | ||||
| @@ -44,8 +44,10 @@ namespace elna | ||||
|         void add_exit_code(const test_status result) noexcept; | ||||
|     }; | ||||
|  | ||||
|     test_result run_for_output(boost::asio::io_context& context, const std::filesystem::path& binary, | ||||
|             std::initializer_list<boost::string_view> arguments); | ||||
|     boost::process::v2::process_stdio get_output_streams(const std::uint8_t stream_number, | ||||
|             boost::asio::readable_pipe& read_pipe); | ||||
|     test_result run_for_output(boost::asio::io_context& context, const std::uint8_t stream_number, | ||||
|             const std::filesystem::path& binary, std::initializer_list<boost::string_view> arguments); | ||||
|     test_result build_test(boost::asio::io_context& context, const std::filesystem::directory_entry& test_entry); | ||||
|     test_result run_test(boost::asio::io_context& context, const std::filesystem::directory_entry& test_entry); | ||||
|     void print_result(const std::filesystem::directory_entry& test_entry, const test_result& result); | ||||
|   | ||||
| @@ -516,18 +516,43 @@ namespace elna::source | ||||
|         } | ||||
|         auto definition_identifier = iterator.advance(token::type::identifier); | ||||
|  | ||||
|         if (!definition_identifier.has_value() || !iterator.skip(token::type::semicolon)) | ||||
|         if (!definition_identifier.has_value() || !iterator.skip(token::type::left_paren)) | ||||
|         { | ||||
|             return nullptr; | ||||
|         } | ||||
|         std::vector<std::unique_ptr<declaration>> declarations; | ||||
|         while (!iterator.current(token::type::right_paren)) | ||||
|         { | ||||
|             std::unique_ptr<declaration> parsed_declaration = parse_declaration(); | ||||
|             if (parsed_declaration == nullptr) | ||||
|             { | ||||
|                 return nullptr; | ||||
|             } | ||||
|             declarations.push_back(std::move(parsed_declaration)); | ||||
|  | ||||
|             if (iterator->of() == token::type::comma) | ||||
|             { | ||||
|                 ++iterator; | ||||
|                 continue; | ||||
|             } | ||||
|             else if (iterator->of() != token::type::right_paren) | ||||
|             { | ||||
|                 iterator.add_error(*iterator); | ||||
|                 return nullptr; | ||||
|             } | ||||
|         } | ||||
|         iterator.skip(token::type::right_paren); | ||||
|         auto definition_body = parse_block(); | ||||
|  | ||||
|         if (definition_body == nullptr || !iterator.skip(token::type::semicolon)) | ||||
|         { | ||||
|             return nullptr; | ||||
|         } | ||||
|         return std::make_unique<procedure_definition>(definition_identifier->get().identifier(), | ||||
|         auto procedure = std::make_unique<procedure_definition>(definition_identifier->get().identifier(), | ||||
|                 std::move(definition_body)); | ||||
|         procedure->parameters() = std::move(declarations); | ||||
|  | ||||
|         return procedure; | ||||
|     } | ||||
|  | ||||
|     std::unique_ptr<declaration> parser::parse_declaration() | ||||
|   | ||||
							
								
								
									
										2
									
								
								tests/expectations/procedure_2_statements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								tests/expectations/procedure_2_statements.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 5 | ||||
| 3 | ||||
							
								
								
									
										1
									
								
								tests/expectations/procedure_print_argument.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/expectations/procedure_print_argument.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| 5 | ||||
							
								
								
									
										9
									
								
								tests/procedure_2_statements.eln
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								tests/procedure_2_statements.eln
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| proc g() | ||||
| begin | ||||
|   writei(5); | ||||
|   writei(3) | ||||
| end; | ||||
|  | ||||
| begin | ||||
|   g() | ||||
| end. | ||||
| @@ -1,4 +1,4 @@ | ||||
| proc f; writei(5); | ||||
| proc f() writei(5); | ||||
|  | ||||
| begin | ||||
|   f() | ||||
|   | ||||
							
								
								
									
										5
									
								
								tests/procedure_print_argument.eln
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								tests/procedure_print_argument.eln
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| proc f(a: Int) writei(a); | ||||
|  | ||||
| begin | ||||
|   f(5) | ||||
| end. | ||||
| @@ -43,17 +43,33 @@ namespace elna | ||||
|         return in_build_directory() / path; | ||||
|     } | ||||
|  | ||||
|     test_result run_for_output(boost::asio::io_context& context, const std::filesystem::path& binary, | ||||
|             std::initializer_list<boost::string_view> arguments) | ||||
|     boost::process::v2::process_stdio get_output_streams(const std::uint8_t stream_number, | ||||
|             boost::asio::readable_pipe& read_pipe) | ||||
|     { | ||||
|         if (stream_number == 1) | ||||
|         { | ||||
|             return boost::process::v2::process_stdio{ nullptr, read_pipe }; | ||||
|         } | ||||
|         if (stream_number == 2) | ||||
|         { | ||||
|             return boost::process::v2::process_stdio{ nullptr, {}, read_pipe }; | ||||
|         } | ||||
|         return boost::process::v2::process_stdio{ nullptr, read_pipe, read_pipe }; | ||||
|     } | ||||
|  | ||||
|     test_result run_for_output(boost::asio::io_context& context, const std::uint8_t stream_number, | ||||
|             const std::filesystem::path& binary, std::initializer_list<boost::string_view> arguments) | ||||
|     { | ||||
|         boost::asio::readable_pipe read_pipe{ context }; | ||||
|         test_result result{}; | ||||
|         std::string output; | ||||
|         boost::asio::dynamic_string_buffer buffer = boost::asio::dynamic_buffer(output); | ||||
|         boost::system::error_code ec; | ||||
|  | ||||
|         boost::process::v2::process_stdio process_stdio; | ||||
|         boost::process::v2::process elna_child(context, | ||||
|                 binary, arguments, | ||||
|                 boost::process::v2::process_stdio{ nullptr, read_pipe, read_pipe }); | ||||
|                 get_output_streams(stream_number, read_pipe)); | ||||
|         do | ||||
|         { | ||||
|             std::size_t transferred = read_pipe.read_some(buffer.prepare(512), ec); | ||||
| @@ -84,7 +100,7 @@ namespace elna | ||||
|         std::filesystem::remove(test_binary); | ||||
|         std::filesystem::remove(test_object); | ||||
|  | ||||
|         test_result result = run_for_output(context, "./build/bin/elna", | ||||
|         test_result result = run_for_output(context, 2, "./build/bin/elna", | ||||
|                 { "-o", test_object.string(), test_entry.path().string() }); | ||||
|  | ||||
|         if (result.code != 0) | ||||
| @@ -157,7 +173,7 @@ namespace elna | ||||
|         std::filesystem::path test_binary = in_build_directory(test_filename); | ||||
|         test_binary.replace_extension(); | ||||
|  | ||||
|         return run_for_output(context, | ||||
|         return run_for_output(context, 1, | ||||
|                 boost::process::search_path("qemu-riscv32"), | ||||
|                 { test_binary.string() }); | ||||
|     } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user