mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 01:19:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			173 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2023, Dan Klishch <danilklishch@gmail.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/Format.h>
 | |
| #include <LibCore/ArgsParser.h>
 | |
| #include <LibMain/Main.h>
 | |
| 
 | |
| #include "Compiler/Passes/CFGBuildingPass.h"
 | |
| #include "Compiler/Passes/CFGSimplificationPass.h"
 | |
| #include "Compiler/Passes/DeadCodeEliminationPass.h"
 | |
| #include "Compiler/Passes/IfBranchMergingPass.h"
 | |
| #include "Compiler/Passes/ReferenceResolvingPass.h"
 | |
| #include "Compiler/Passes/SSABuildingPass.h"
 | |
| #include "Function.h"
 | |
| #include "Parser/CppASTConverter.h"
 | |
| #include "Parser/SpecificationParsing.h"
 | |
| 
 | |
| using namespace JSSpecCompiler;
 | |
| 
 | |
| struct CompilationStepWithDumpOptions {
 | |
|     OwnPtr<CompilationStep> step;
 | |
|     bool dump_ast = false;
 | |
|     bool dump_cfg = false;
 | |
| };
 | |
| 
 | |
| class CompilationPipeline {
 | |
| public:
 | |
|     template<typename T>
 | |
|     void add_compilation_pass()
 | |
|     {
 | |
|         auto func = +[](TranslationUnitRef translation_unit) {
 | |
|             T { translation_unit }.run();
 | |
|         };
 | |
|         add_step(adopt_own_if_nonnull(new NonOwningCompilationStep(T::name, func)));
 | |
|     }
 | |
| 
 | |
|     template<typename T>
 | |
|     void for_each_step_in(StringView pass_list, T&& func)
 | |
|     {
 | |
|         HashTable<StringView> selected_steps;
 | |
|         for (auto pass : pass_list.split_view(',')) {
 | |
|             if (pass == "all") {
 | |
|                 for (auto const& step : m_pipeline)
 | |
|                     selected_steps.set(step.step->name());
 | |
|             } else if (pass == "last") {
 | |
|                 selected_steps.set(m_pipeline.last().step->name());
 | |
|             } else if (pass.starts_with('-')) {
 | |
|                 VERIFY(selected_steps.remove(pass.substring_view(1)));
 | |
|             } else {
 | |
|                 selected_steps.set(pass);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         for (auto& step : m_pipeline)
 | |
|             if (selected_steps.contains(step.step->name()))
 | |
|                 func(step);
 | |
|     }
 | |
| 
 | |
|     void add_step(OwnPtr<CompilationStep>&& step)
 | |
|     {
 | |
|         m_pipeline.append({ move(step) });
 | |
|     }
 | |
| 
 | |
|     auto const& pipeline() const { return m_pipeline; }
 | |
| 
 | |
| private:
 | |
|     Vector<CompilationStepWithDumpOptions> m_pipeline;
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct AK::Formatter<ReadonlySpan<FunctionArgument>> : AK::Formatter<StringView> {
 | |
|     ErrorOr<void> format(FormatBuilder& builder, ReadonlySpan<FunctionArgument> const& arguments)
 | |
|     {
 | |
|         size_t previous_optional_group = 0;
 | |
|         for (size_t i = 0; i < arguments.size(); ++i) {
 | |
|             if (previous_optional_group != arguments[i].optional_arguments_group) {
 | |
|                 previous_optional_group = arguments[i].optional_arguments_group;
 | |
|                 TRY(builder.put_string("["sv));
 | |
|             }
 | |
|             TRY(builder.put_string(arguments[i].name));
 | |
|             if (i + 1 != arguments.size())
 | |
|                 TRY(builder.put_literal(", "sv));
 | |
|         }
 | |
|         TRY(builder.put_string(TRY(String::repeated(']', previous_optional_group))));
 | |
|         return {};
 | |
|     }
 | |
| };
 | |
| 
 | |
| ErrorOr<int> serenity_main(Main::Arguments arguments)
 | |
| {
 | |
|     Core::ArgsParser args_parser;
 | |
| 
 | |
|     StringView filename;
 | |
|     args_parser.add_positional_argument(filename, "File to compile", "file");
 | |
| 
 | |
|     constexpr StringView language_spec = "spec"sv;
 | |
|     constexpr StringView language_cpp = "c++"sv;
 | |
|     StringView language = language_spec;
 | |
|     args_parser.add_option(Core::ArgsParser::Option {
 | |
|         .argument_mode = Core::ArgsParser::OptionArgumentMode::Optional,
 | |
|         .help_string = "Specify the language of the input file.",
 | |
|         .short_name = 'x',
 | |
|         .value_name = "{c++|spec}",
 | |
|         .accept_value = [&](StringView value) {
 | |
|             language = value;
 | |
|             return language.is_one_of(language_spec, language_cpp);
 | |
|         },
 | |
|     });
 | |
| 
 | |
|     StringView passes_to_dump_ast;
 | |
|     args_parser.add_option(passes_to_dump_ast, "Dump AST after specified passes.", "dump-ast", 0, "{all|last|<pass-name>|-<pass-name>[,...]}");
 | |
| 
 | |
|     StringView passes_to_dump_cfg;
 | |
|     args_parser.add_option(passes_to_dump_cfg, "Dump CFG after specified passes.", "dump-cfg", 0, "{all|last|<pass-name>|-<pass-name>[,...]}");
 | |
| 
 | |
|     bool silence_diagnostics = false;
 | |
|     args_parser.add_option(silence_diagnostics, "Silence all diagnostics.", "silence-diagnostics");
 | |
| 
 | |
|     args_parser.parse(arguments);
 | |
| 
 | |
|     CompilationPipeline pipeline;
 | |
|     if (language == language_cpp)
 | |
|         pipeline.add_step(adopt_own_if_nonnull(new CppParsingStep()));
 | |
|     else
 | |
|         pipeline.add_step(adopt_own_if_nonnull(new SpecificationParsingStep()));
 | |
|     pipeline.add_compilation_pass<IfBranchMergingPass>();
 | |
|     pipeline.add_compilation_pass<ReferenceResolvingPass>();
 | |
|     pipeline.add_compilation_pass<CFGBuildingPass>();
 | |
|     pipeline.add_compilation_pass<CFGSimplificationPass>();
 | |
|     pipeline.add_compilation_pass<SSABuildingPass>();
 | |
|     pipeline.add_compilation_pass<DeadCodeEliminationPass>();
 | |
| 
 | |
|     pipeline.for_each_step_in(passes_to_dump_ast, [](CompilationStepWithDumpOptions& step) {
 | |
|         step.dump_ast = true;
 | |
|     });
 | |
|     pipeline.for_each_step_in(passes_to_dump_cfg, [](CompilationStepWithDumpOptions& step) {
 | |
|         step.dump_cfg = true;
 | |
|     });
 | |
| 
 | |
|     TranslationUnit translation_unit(filename);
 | |
| 
 | |
|     for (auto const& step : pipeline.pipeline()) {
 | |
|         step.step->run(&translation_unit);
 | |
| 
 | |
|         if (translation_unit.diag().has_fatal_errors()) {
 | |
|             translation_unit.diag().print_diagnostics();
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         if (step.dump_ast) {
 | |
|             outln(stderr, "===== AST after {} =====", step.step->name());
 | |
|             for (auto const& function : translation_unit.functions_to_compile()) {
 | |
|                 outln(stderr, "{}({}):", function->name(), function->arguments());
 | |
|                 outln(stderr, "{}", function->m_ast);
 | |
|             }
 | |
|         }
 | |
|         if (step.dump_cfg && translation_unit.functions_to_compile().size() && translation_unit.functions_to_compile()[0]->m_cfg != nullptr) {
 | |
|             outln(stderr, "===== CFG after {} =====", step.step->name());
 | |
|             for (auto const& function : translation_unit.functions_to_compile()) {
 | |
|                 outln(stderr, "{}({}):", function->name(), function->arguments());
 | |
|                 outln(stderr, "{}", *function->m_cfg);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!silence_diagnostics)
 | |
|         translation_unit.diag().print_diagnostics();
 | |
| 
 | |
|     return 0;
 | |
| }
 |