102d32aecSArgyrios Kyrtzidis //===--- CreateInvocationFromCommandLine.cpp - CompilerInvocation from Args ==// 202d32aecSArgyrios Kyrtzidis // 302d32aecSArgyrios Kyrtzidis // The LLVM Compiler Infrastructure 402d32aecSArgyrios Kyrtzidis // 502d32aecSArgyrios Kyrtzidis // This file is distributed under the University of Illinois Open Source 602d32aecSArgyrios Kyrtzidis // License. See LICENSE.TXT for details. 702d32aecSArgyrios Kyrtzidis // 802d32aecSArgyrios Kyrtzidis //===----------------------------------------------------------------------===// 902d32aecSArgyrios Kyrtzidis // 1002d32aecSArgyrios Kyrtzidis // Construct a compiler invocation object for command line driver arguments 1102d32aecSArgyrios Kyrtzidis // 1202d32aecSArgyrios Kyrtzidis //===----------------------------------------------------------------------===// 1302d32aecSArgyrios Kyrtzidis 1402d32aecSArgyrios Kyrtzidis #include "clang/Frontend/Utils.h" 153a02247dSChandler Carruth #include "clang/Basic/DiagnosticOptions.h" 1602d32aecSArgyrios Kyrtzidis #include "clang/Driver/Compilation.h" 1702d32aecSArgyrios Kyrtzidis #include "clang/Driver/Driver.h" 18*0ff05cd1SArtem Belevich #include "clang/Driver/Action.h" 1902d32aecSArgyrios Kyrtzidis #include "clang/Driver/Options.h" 2002d32aecSArgyrios Kyrtzidis #include "clang/Driver/Tool.h" 213a02247dSChandler Carruth #include "clang/Frontend/CompilerInstance.h" 223a02247dSChandler Carruth #include "clang/Frontend/FrontendDiagnostic.h" 23898229abSReid Kleckner #include "llvm/Option/ArgList.h" 2402d32aecSArgyrios Kyrtzidis #include "llvm/Support/Host.h" 2502d32aecSArgyrios Kyrtzidis using namespace clang; 26898229abSReid Kleckner using namespace llvm::opt; 2702d32aecSArgyrios Kyrtzidis 2802d32aecSArgyrios Kyrtzidis /// createInvocationFromCommandLine - Construct a compiler invocation object for 2902d32aecSArgyrios Kyrtzidis /// a command line argument vector. 3002d32aecSArgyrios Kyrtzidis /// 3102d32aecSArgyrios Kyrtzidis /// \return A CompilerInvocation, or 0 if none was built for the given 3202d32aecSArgyrios Kyrtzidis /// argument vector. 3302d32aecSArgyrios Kyrtzidis CompilerInvocation * 3454b1677dSChris Lattner clang::createInvocationFromCommandLine(ArrayRef<const char *> ArgList, 35c95d8192SDylan Noblesmith IntrusiveRefCntPtr<DiagnosticsEngine> Diags) { 36f994cef8SAlp Toker if (!Diags.get()) { 3702d32aecSArgyrios Kyrtzidis // No diagnostics engine was provided, so create our own diagnostics object 3802d32aecSArgyrios Kyrtzidis // with the default options. 39f1b49e23SSean Silva Diags = CompilerInstance::createDiagnostics(new DiagnosticOptions); 4002d32aecSArgyrios Kyrtzidis } 4102d32aecSArgyrios Kyrtzidis 420e62c1ccSChris Lattner SmallVector<const char *, 16> Args; 4302d32aecSArgyrios Kyrtzidis Args.push_back("<clang>"); // FIXME: Remove dummy argument. 4402d32aecSArgyrios Kyrtzidis Args.insert(Args.end(), ArgList.begin(), ArgList.end()); 4502d32aecSArgyrios Kyrtzidis 463169e806SArgyrios Kyrtzidis // FIXME: Find a cleaner way to force the driver into restricted modes. 4702d32aecSArgyrios Kyrtzidis Args.push_back("-fsyntax-only"); 4802d32aecSArgyrios Kyrtzidis 4902d32aecSArgyrios Kyrtzidis // FIXME: We shouldn't have to pass in the path info. 508188c8a1SSebastian Pop driver::Driver TheDriver("clang", llvm::sys::getDefaultTargetTriple(), 511761f118SAlp Toker *Diags); 5202d32aecSArgyrios Kyrtzidis 5302d32aecSArgyrios Kyrtzidis // Don't check that inputs exist, they may have been remapped. 5402d32aecSArgyrios Kyrtzidis TheDriver.setCheckInputsExist(false); 5502d32aecSArgyrios Kyrtzidis 56b8984329SAhmed Charles std::unique_ptr<driver::Compilation> C(TheDriver.BuildCompilation(Args)); 5702d32aecSArgyrios Kyrtzidis 5802d32aecSArgyrios Kyrtzidis // Just print the cc1 options if -### was present. 5902d32aecSArgyrios Kyrtzidis if (C->getArgs().hasArg(driver::options::OPT__HASH_HASH_HASH)) { 60b212b34fSHans Wennborg C->getJobs().Print(llvm::errs(), "\n", true); 6149a2790fSCraig Topper return nullptr; 6202d32aecSArgyrios Kyrtzidis } 6302d32aecSArgyrios Kyrtzidis 6402d32aecSArgyrios Kyrtzidis // We expect to get back exactly one command job, if we didn't something 65*0ff05cd1SArtem Belevich // failed. CUDA compilation is an exception as it creates multiple jobs. If 66*0ff05cd1SArtem Belevich // that's the case, we proceed with the first job. If caller needs particular 67*0ff05cd1SArtem Belevich // CUDA job, it should be controlled via --cuda-{host|device}-only option 68*0ff05cd1SArtem Belevich // passed to the driver. 6902d32aecSArgyrios Kyrtzidis const driver::JobList &Jobs = C->getJobs(); 70*0ff05cd1SArtem Belevich bool CudaCompilation = false; 71*0ff05cd1SArtem Belevich if (Jobs.size() > 1) { 72*0ff05cd1SArtem Belevich for (auto &A : C->getActions()){ 73*0ff05cd1SArtem Belevich // On MacOSX real actions may end up being wrapped in BindArchAction 74*0ff05cd1SArtem Belevich if (isa<driver::BindArchAction>(A)) 75*0ff05cd1SArtem Belevich A = *A->begin(); 76*0ff05cd1SArtem Belevich if (isa<driver::CudaDeviceAction>(A)) { 77*0ff05cd1SArtem Belevich CudaCompilation = true; 78*0ff05cd1SArtem Belevich break; 79*0ff05cd1SArtem Belevich } 80*0ff05cd1SArtem Belevich } 81*0ff05cd1SArtem Belevich } 82*0ff05cd1SArtem Belevich if (Jobs.size() == 0 || !isa<driver::Command>(*Jobs.begin()) || 83*0ff05cd1SArtem Belevich (Jobs.size() > 1 && !CudaCompilation)) { 842c1dd271SDylan Noblesmith SmallString<256> Msg; 8502d32aecSArgyrios Kyrtzidis llvm::raw_svector_ostream OS(Msg); 86b212b34fSHans Wennborg Jobs.Print(OS, "; ", true); 8702d32aecSArgyrios Kyrtzidis Diags->Report(diag::err_fe_expected_compiler_job) << OS.str(); 8849a2790fSCraig Topper return nullptr; 8902d32aecSArgyrios Kyrtzidis } 9002d32aecSArgyrios Kyrtzidis 91aab9792bSJustin Bogner const driver::Command &Cmd = cast<driver::Command>(*Jobs.begin()); 92c11bf802SDavid Blaikie if (StringRef(Cmd.getCreator().getName()) != "clang") { 9302d32aecSArgyrios Kyrtzidis Diags->Report(diag::err_fe_expected_clang_command); 9449a2790fSCraig Topper return nullptr; 9502d32aecSArgyrios Kyrtzidis } 9602d32aecSArgyrios Kyrtzidis 97c11bf802SDavid Blaikie const ArgStringList &CCArgs = Cmd.getArguments(); 98b8984329SAhmed Charles std::unique_ptr<CompilerInvocation> CI(new CompilerInvocation()); 99e99b27f3SDylan Noblesmith if (!CompilerInvocation::CreateFromArgs(*CI, 10002d32aecSArgyrios Kyrtzidis const_cast<const char **>(CCArgs.data()), 10102d32aecSArgyrios Kyrtzidis const_cast<const char **>(CCArgs.data()) + 10202d32aecSArgyrios Kyrtzidis CCArgs.size(), 103e99b27f3SDylan Noblesmith *Diags)) 10449a2790fSCraig Topper return nullptr; 1059a16beb8SAhmed Charles return CI.release(); 10602d32aecSArgyrios Kyrtzidis } 107