//===--- 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 CreateFrontendBaseAction( CompilerInstance &ci) { ActionKind ak = ci.frontendOpts().programAction; switch (ak) { case InputOutputTest: return std::make_unique(); case PrintPreprocessedInput: return std::make_unique(); case ParseSyntaxOnly: return std::make_unique(); case EmitObj: return std::make_unique(); case DebugUnparse: return std::make_unique(); case DebugUnparseNoSema: return std::make_unique(); case DebugUnparseWithSymbols: return std::make_unique(); case DebugDumpSymbols: return std::make_unique(); case DebugDumpParseTree: return std::make_unique(); case DebugDumpParseTreeNoSema: return std::make_unique(); case DebugDumpAll: return std::make_unique(); case DebugDumpProvenance: return std::make_unique(); case DebugDumpParsingLog: return std::make_unique(); case DebugMeasureParseTree: return std::make_unique(); case DebugPreFIRTree: return std::make_unique(); case GetDefinition: return std::make_unique(); case GetSymbolsSources: return std::make_unique(); case InitOnly: return std::make_unique(); case PluginAction: { for (const FrontendPluginRegistry::entry &plugin : FrontendPluginRegistry::entries()) { if (plugin.getName() == ci.frontendOpts().ActionName) { std::unique_ptr 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 CreateFrontendAction(CompilerInstance &ci) { // Create the underlying action. std::unique_ptr 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 act(CreateFrontendAction(*flang)); if (!act) return false; bool success = flang->ExecuteAction(*act); return success; } } // namespace Fortran::frontend