mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
Since a Job may be reached through multiple paths in the dependency graph, we need to include support for caching Jobs. (Currently, the cache maps Action/ToolChain pairs to Jobs.) As a result of these changes, Tool::constructJob() and Driver::buildJobsForAction() return a plain Job *, instead of a std::unique_ptr<Job>, since the Jobs returned by buildJobsForAction() no longer need to be unique. Swift SVN r12443
187 lines
6.5 KiB
C++
187 lines
6.5 KiB
C++
//===--- Tools.cpp - Tools Implementations --------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "Tools.h"
|
|
#include "ToolChains.h"
|
|
|
|
#include "swift/Basic/LLVM.h"
|
|
#include "swift/Basic/Range.h"
|
|
#include "swift/Driver/Driver.h"
|
|
#include "swift/Driver/Job.h"
|
|
#include "swift/Driver/Options.h"
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
#include "llvm/Option/Arg.h"
|
|
#include "llvm/Option/ArgList.h"
|
|
#include "llvm/Support/Path.h"
|
|
|
|
using namespace swift::driver;
|
|
using namespace swift::driver::tools;
|
|
using namespace llvm::opt;
|
|
|
|
/// Swift Tool
|
|
|
|
Job *Swift::constructJob(const JobAction &JA, std::unique_ptr<JobList> Inputs,
|
|
std::unique_ptr<CommandOutput> Output,
|
|
const ActionList &InputActions, const ArgList &Args,
|
|
StringRef LinkingOutput) const {
|
|
ArgStringList Arguments;
|
|
|
|
const char *Exec = getToolChain().getDriver().getSwiftProgramPath();
|
|
|
|
// Invoke ourselves in -frontend mode.
|
|
Arguments.push_back("-frontend");
|
|
|
|
Arguments.push_back("-target");
|
|
std::string TripleStr = getToolChain().getTripleString();
|
|
Arguments.push_back(Args.MakeArgString(TripleStr));
|
|
|
|
const char *OutputOption = nullptr;
|
|
switch (Output->getType()) {
|
|
case types::TY_Object:
|
|
OutputOption = "-c";
|
|
break;
|
|
case types::TY_RawSIL:
|
|
OutputOption = "-emit-silgen";
|
|
break;
|
|
case types::TY_SIL:
|
|
OutputOption = "-emit-sil";
|
|
break;
|
|
case types::TY_LLVM_IR:
|
|
OutputOption = "-emit-ir";
|
|
break;
|
|
case types::TY_LLVM_BC:
|
|
OutputOption = "-emit-bc";
|
|
break;
|
|
case types::TY_Nothing:
|
|
// We were told to output nothing, so get the last mode option and use that.
|
|
OutputOption =
|
|
Args.getLastArg(options::OPT_modes_Group)->getSpelling().data();
|
|
break;
|
|
default:
|
|
llvm_unreachable("Invalid output type");
|
|
}
|
|
|
|
assert(OutputOption != nullptr && "No output option specified!");
|
|
|
|
Arguments.push_back(OutputOption);
|
|
|
|
assert(Inputs->empty() &&
|
|
"The Swift frontend does not expect to be fed any input Jobs!");
|
|
assert(InputActions.size() == 1 &&
|
|
"The Swift frontend expects exactly one input (the primary file)!");
|
|
|
|
const InputAction *IA = dyn_cast<InputAction>(InputActions[0]);
|
|
assert(IA && "Only InputActions can be passed as inputs!");
|
|
const Arg &PrimaryInputArg = IA->getInputArg();
|
|
bool FoundPrimaryInput = false;
|
|
|
|
for (const Arg *A : make_range(Args.filtered_begin(options::OPT_INPUT),
|
|
Args.filtered_end())) {
|
|
Option Opt = A->getOption();
|
|
if (A->getOption().matches(options::OPT_INPUT)) {
|
|
// See if this input should be passed with -primary-file.
|
|
if (!FoundPrimaryInput && PrimaryInputArg.getIndex() == A->getIndex()) {
|
|
Arguments.push_back("-primary-file");
|
|
FoundPrimaryInput = true;
|
|
}
|
|
Arguments.push_back(A->getValue());
|
|
}
|
|
}
|
|
|
|
if (Args.hasArg(options::OPT_module_name)) {
|
|
Args.AddLastArg(Arguments, options::OPT_module_name);
|
|
} else if (const Arg *A = Args.getLastArgNoClaim(options::OPT_o)) {
|
|
Arguments.push_back("-module-name");
|
|
StringRef InferredModuleName = llvm::sys::path::stem(A->getValue());
|
|
Arguments.push_back(Args.MakeArgString(InferredModuleName));
|
|
}
|
|
|
|
Args.AddLastArg(Arguments, options::OPT_g);
|
|
|
|
// Pass the optimization level down to the frontend.
|
|
Args.AddLastArg(Arguments, options::OPT_O_Group);
|
|
|
|
// Set the SDK for the frontend.
|
|
Args.AddLastArg(Arguments, options::OPT_sdk);
|
|
|
|
// Pass through the values passed to -Xfrontend.
|
|
Args.AddAllArgValues(Arguments, options::OPT_Xfrontend);
|
|
|
|
Args.AddLastArg(Arguments, options::OPT_parse_as_library);
|
|
|
|
Args.AddLastArg(Arguments, options::OPT_parse_sil);
|
|
|
|
Args.AddLastArg(Arguments, options::OPT_parse_stdlib);
|
|
|
|
if (Args.hasArg(options::OPT_serialize_diagnostics)) {
|
|
Arguments.push_back("-serialize-diagnostics");
|
|
// TODO: pass -serialized-diagnostics-path with user-specified path,
|
|
// if present. (This requires an output file map.)
|
|
}
|
|
|
|
// Add the output file argument if necessary.
|
|
if (Output->getType() != types::TY_Nothing) {
|
|
Arguments.push_back("-o");
|
|
Arguments.push_back(Output->getFilename().data());
|
|
} else if (Args.getLastArg(options::OPT_modes_Group)->getOption().matches(
|
|
options::OPT_i)) {
|
|
Args.AddLastArg(Arguments, options::OPT__DASH_DASH);
|
|
}
|
|
|
|
return new Command(JA, *this, std::move(Inputs), std::move(Output), Exec,
|
|
Arguments);
|
|
}
|
|
|
|
/// Darwin Tools
|
|
|
|
llvm::Triple::ArchType darwin::getArchTypeForDarwinArchName(StringRef Arch) {
|
|
return llvm::StringSwitch<llvm::Triple::ArchType>(Arch)
|
|
.Cases("i386", "i486", "i486SX", "i586", "i686", llvm::Triple::x86)
|
|
.Cases("pentium", "pentpro", "pentIIm3", "pentIIm5", "pentium4",
|
|
llvm::Triple::x86)
|
|
|
|
.Case("x86_64", llvm::Triple::x86_64)
|
|
|
|
.Cases("arm", "armv4t", "armv5", "armv6", "armv6m", llvm::Triple::arm)
|
|
.Cases("armv7", "armv7em", "armv7f", "armv7k", "armv7m", llvm::Triple::arm)
|
|
.Cases("armv7s", "xscale", llvm::Triple::arm)
|
|
|
|
.Default(llvm::Triple::UnknownArch);
|
|
}
|
|
|
|
void darwin::DarwinTool::anchor() {}
|
|
|
|
void darwin::DarwinTool::AddDarwinArch(const ArgList &Args,
|
|
ArgStringList &CmdArgs) const {
|
|
StringRef ArchName = getDarwinToolChain().getDarwinArchName(Args);
|
|
|
|
CmdArgs.push_back("-arch");
|
|
CmdArgs.push_back(Args.MakeArgString(ArchName));
|
|
}
|
|
|
|
Job *darwin::Linker::constructJob(const JobAction &JA,
|
|
std::unique_ptr<JobList> Inputs,
|
|
std::unique_ptr<CommandOutput> Output,
|
|
const ActionList &InputActions,
|
|
const ArgList &Args,
|
|
StringRef LinkingOutput) const {
|
|
assert(Output->getType() == types::TY_Image && "Invalid linker output type.");
|
|
ArgStringList Arguments;
|
|
Arguments.push_back("-v");
|
|
|
|
std::string Exec = getToolChain().getDriver().getProgramPath("ld",
|
|
getToolChain());
|
|
|
|
return new Command(JA, *this, std::move(Inputs), std::move(Output),
|
|
Args.MakeArgString(Exec), Arguments);
|
|
}
|