//===-- driver.cpp - Flang Driver -----------------------------------------===// // // 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 is the entry point to the flang driver; it is a thin wrapper // for functionality in the Driver flang library. // //===----------------------------------------------------------------------===// #include "clang/Driver/Driver.h" #include "flang/Frontend/CompilerInvocation.h" #include "flang/Frontend/TextDiagnosticPrinter.h" #include "clang/Basic/Diagnostic.h" #include "clang/Basic/DiagnosticIDs.h" #include "clang/Basic/DiagnosticOptions.h" #include "clang/Driver/Compilation.h" #include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/IntrusiveRefCntPtr.h" #include "llvm/Option/ArgList.h" #include "llvm/Support/Host.h" #include "llvm/Support/InitLLVM.h" #include "llvm/Support/VirtualFileSystem.h" using llvm::StringRef; // main frontend method. Lives inside fc1_main.cpp extern int fc1_main(llvm::ArrayRef argv, const char *argv0); std::string GetExecutablePath(const char *argv0) { // This just needs to be some symbol in the binary void *p = (void *)(intptr_t)GetExecutablePath; return llvm::sys::fs::getMainExecutable(argv0, p); } // This lets us create the DiagnosticsEngine with a properly-filled-out // DiagnosticOptions instance static clang::DiagnosticOptions *CreateAndPopulateDiagOpts( llvm::ArrayRef argv) { auto *diagOpts = new clang::DiagnosticOptions; // Ignore missingArgCount and the return value of ParseDiagnosticArgs. // Any errors that would be diagnosed here will also be diagnosed later, // when the DiagnosticsEngine actually exists. unsigned missingArgIndex, missingArgCount; llvm::opt::InputArgList args = clang::driver::getDriverOptTable().ParseArgs( argv.slice(1), missingArgIndex, missingArgCount, /*FlagsToInclude=*/clang::driver::options::FlangOption); (void)Fortran::frontend::ParseDiagnosticArgs(*diagOpts, args); return diagOpts; } static int ExecuteFC1Tool(llvm::SmallVectorImpl &argV) { llvm::StringRef tool = argV[1]; if (tool == "-fc1") return fc1_main(makeArrayRef(argV).slice(2), argV[0]); // Reject unknown tools. // ATM it only supports fc1. Any fc1[*] is rejected. llvm::errs() << "error: unknown integrated tool '" << tool << "'. " << "Valid tools include '-fc1'.\n"; return 1; } int main(int argc_, const char **argv_) { // Initialize variables to call the driver llvm::InitLLVM x(argc_, argv_); llvm::SmallVector argv(argv_, argv_ + argc_); clang::driver::ParsedClangName targetandMode("flang", "--driver-mode=flang"); std::string driverPath = GetExecutablePath(argv[0]); // Check if flang-new is in the frontend mode auto firstArg = std::find_if( argv.begin() + 1, argv.end(), [](const char *a) { return a != nullptr; }); if (firstArg != argv.end()) { if (llvm::StringRef(argv[1]).startswith("-cc1")) { llvm::errs() << "error: unknown integrated tool '" << argv[1] << "'. " << "Valid tools include '-fc1'.\n"; return 1; } // Call flang-new frontend if (llvm::StringRef(argv[1]).startswith("-fc1")) { return ExecuteFC1Tool(argv); } } // Not in the frontend mode - continue in the compiler driver mode. // Create DiagnosticsEngine for the compiler driver llvm::IntrusiveRefCntPtr diagOpts = CreateAndPopulateDiagOpts(argv); llvm::IntrusiveRefCntPtr diagID( new clang::DiagnosticIDs()); Fortran::frontend::TextDiagnosticPrinter *diagClient = new Fortran::frontend::TextDiagnosticPrinter(llvm::errs(), &*diagOpts); diagClient->set_prefix( std::string(llvm::sys::path::stem(GetExecutablePath(argv[0])))); clang::DiagnosticsEngine diags(diagID, &*diagOpts, diagClient); // Prepare the driver clang::driver::Driver theDriver(driverPath, llvm::sys::getDefaultTargetTriple(), diags, "flang LLVM compiler"); theDriver.setTargetAndMode(targetandMode); std::unique_ptr c( theDriver.BuildCompilation(argv)); llvm::SmallVector, 4> failingCommands; // Run the driver int res = 1; bool isCrash = false; res = theDriver.ExecuteCompilation(*c, failingCommands); for (const auto &p : failingCommands) { int CommandRes = p.first; const clang::driver::Command *failingCommand = p.second; if (!res) res = CommandRes; // If result status is < 0 (e.g. when sys::ExecuteAndWait returns -1), // then the driver command signalled an error. On Windows, abort will // return an exit code of 3. In these cases, generate additional diagnostic // information if possible. isCrash = CommandRes < 0; #ifdef _WIN32 IsCrash |= CommandRes == 3; #endif if (isCrash) { theDriver.generateCompilationDiagnostics(*c, *failingCommand); break; } } diags.getClient()->finish(); // If we have multiple failing commands, we return the result of the first // failing command. return res; }