| //===--- ExecuteCompilerInvocation.cpp ------------------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file holds ExecuteCompilerInvocation(). It is split into its own file to |
| // minimize the impact of pulling in essentially everything else in Flang. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "flang/Frontend/CompilerInstance.h" |
| #include "flang/Frontend/FrontendActions.h" |
| #include "flang/Frontend/FrontendPluginRegistry.h" |
| #include "clang/Driver/Options.h" |
| #include "llvm/Option/OptTable.h" |
| #include "llvm/Option/Option.h" |
| #include "llvm/Support/BuryPointer.h" |
| #include "llvm/Support/CommandLine.h" |
| |
| namespace Fortran::frontend { |
| |
| static std::unique_ptr<FrontendAction> CreateFrontendBaseAction( |
| CompilerInstance &ci) { |
| |
| ActionKind ak = ci.frontendOpts().programAction; |
| switch (ak) { |
| case InputOutputTest: |
| return std::make_unique<InputOutputTestAction>(); |
| case PrintPreprocessedInput: |
| return std::make_unique<PrintPreprocessedAction>(); |
| case ParseSyntaxOnly: |
| return std::make_unique<ParseSyntaxOnlyAction>(); |
| case EmitObj: |
| return std::make_unique<EmitObjAction>(); |
| case DebugUnparse: |
| return std::make_unique<DebugUnparseAction>(); |
| case DebugUnparseNoSema: |
| return std::make_unique<DebugUnparseNoSemaAction>(); |
| case DebugUnparseWithSymbols: |
| return std::make_unique<DebugUnparseWithSymbolsAction>(); |
| case DebugDumpSymbols: |
| return std::make_unique<DebugDumpSymbolsAction>(); |
| case DebugDumpParseTree: |
| return std::make_unique<DebugDumpParseTreeAction>(); |
| case DebugDumpParseTreeNoSema: |
| return std::make_unique<DebugDumpParseTreeNoSemaAction>(); |
| case DebugDumpAll: |
| return std::make_unique<DebugDumpAllAction>(); |
| case DebugDumpProvenance: |
| return std::make_unique<DebugDumpProvenanceAction>(); |
| case DebugDumpParsingLog: |
| return std::make_unique<DebugDumpParsingLogAction>(); |
| case DebugMeasureParseTree: |
| return std::make_unique<DebugMeasureParseTreeAction>(); |
| case DebugPreFIRTree: |
| return std::make_unique<DebugPreFIRTreeAction>(); |
| case GetDefinition: |
| return std::make_unique<GetDefinitionAction>(); |
| case GetSymbolsSources: |
| return std::make_unique<GetSymbolsSourcesAction>(); |
| case InitOnly: |
| return std::make_unique<InitOnlyAction>(); |
| case PluginAction: { |
| for (const FrontendPluginRegistry::entry &plugin : |
| FrontendPluginRegistry::entries()) { |
| if (plugin.getName() == ci.frontendOpts().ActionName) { |
| std::unique_ptr<PluginParseTreeAction> p(plugin.instantiate()); |
| return std::move(p); |
| } |
| } |
| unsigned diagID = ci.diagnostics().getCustomDiagID( |
| clang::DiagnosticsEngine::Error, "unable to find plugin '%0'"); |
| ci.diagnostics().Report(diagID) << ci.frontendOpts().ActionName; |
| return nullptr; |
| } |
| default: |
| break; |
| // TODO: |
| // case ParserSyntaxOnly: |
| // case EmitLLVM: |
| // case EmitLLVMOnly: |
| // case EmitCodeGenOnly: |
| // (...) |
| } |
| return 0; |
| } |
| |
| std::unique_ptr<FrontendAction> CreateFrontendAction(CompilerInstance &ci) { |
| // Create the underlying action. |
| std::unique_ptr<FrontendAction> act = CreateFrontendBaseAction(ci); |
| if (!act) |
| return nullptr; |
| |
| return act; |
| } |
| |
| bool ExecuteCompilerInvocation(CompilerInstance *flang) { |
| // Honor -help. |
| if (flang->frontendOpts().showHelp) { |
| clang::driver::getDriverOptTable().printHelp(llvm::outs(), |
| "flang-new -fc1 [options] file...", "LLVM 'Flang' Compiler", |
| /*Include=*/clang::driver::options::FC1Option, |
| /*Exclude=*/llvm::opt::DriverFlag::HelpHidden, |
| /*ShowAllAliases=*/false); |
| return true; |
| } |
| |
| // Honor -version. |
| if (flang->frontendOpts().showVersion) { |
| llvm::cl::PrintVersionMessage(); |
| return true; |
| } |
| |
| // Load any requested plugins. |
| for (const std::string &Path : flang->frontendOpts().plugins) { |
| std::string Error; |
| if (llvm::sys::DynamicLibrary::LoadLibraryPermanently( |
| Path.c_str(), &Error)) { |
| unsigned diagID = flang->diagnostics().getCustomDiagID( |
| clang::DiagnosticsEngine::Error, "unable to load plugin '%0': '%1'"); |
| flang->diagnostics().Report(diagID) << Path << Error; |
| } |
| } |
| |
| // If there were errors in processing arguments, don't do anything else. |
| if (flang->diagnostics().hasErrorOccurred()) { |
| return false; |
| } |
| |
| // Create and execute the frontend action. |
| std::unique_ptr<FrontendAction> act(CreateFrontendAction(*flang)); |
| if (!act) |
| return false; |
| |
| bool success = flang->ExecuteAction(*act); |
| return success; |
| } |
| |
| } // namespace Fortran::frontend |