1e5dd7070Spatrick //===--- CloudABI.cpp - CloudABI ToolChain Implementations ------*- C++ -*-===//
2e5dd7070Spatrick //
3e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e5dd7070Spatrick //
7e5dd7070Spatrick //===----------------------------------------------------------------------===//
8e5dd7070Spatrick
9e5dd7070Spatrick #include "CloudABI.h"
10e5dd7070Spatrick #include "CommonArgs.h"
11e5dd7070Spatrick #include "clang/Driver/Compilation.h"
12e5dd7070Spatrick #include "clang/Driver/Driver.h"
13a9ac8606Spatrick #include "clang/Driver/InputInfo.h"
14e5dd7070Spatrick #include "clang/Driver/Options.h"
15e5dd7070Spatrick #include "llvm/ADT/SmallString.h"
16e5dd7070Spatrick #include "llvm/Option/ArgList.h"
17e5dd7070Spatrick #include "llvm/Support/Path.h"
18e5dd7070Spatrick
19e5dd7070Spatrick using namespace clang::driver;
20e5dd7070Spatrick using namespace clang::driver::tools;
21e5dd7070Spatrick using namespace clang::driver::toolchains;
22e5dd7070Spatrick using namespace clang;
23e5dd7070Spatrick using namespace llvm::opt;
24e5dd7070Spatrick
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const25e5dd7070Spatrick void cloudabi::Linker::ConstructJob(Compilation &C, const JobAction &JA,
26e5dd7070Spatrick const InputInfo &Output,
27e5dd7070Spatrick const InputInfoList &Inputs,
28e5dd7070Spatrick const ArgList &Args,
29e5dd7070Spatrick const char *LinkingOutput) const {
30e5dd7070Spatrick const ToolChain &ToolChain = getToolChain();
31e5dd7070Spatrick const Driver &D = ToolChain.getDriver();
32e5dd7070Spatrick ArgStringList CmdArgs;
33e5dd7070Spatrick
34e5dd7070Spatrick // Silence warning for "clang -g foo.o -o foo"
35e5dd7070Spatrick Args.ClaimAllArgs(options::OPT_g_Group);
36e5dd7070Spatrick // and "clang -emit-llvm foo.o -o foo"
37e5dd7070Spatrick Args.ClaimAllArgs(options::OPT_emit_llvm);
38e5dd7070Spatrick // and for "clang -w foo.o -o foo". Other warning options are already
39e5dd7070Spatrick // handled somewhere else.
40e5dd7070Spatrick Args.ClaimAllArgs(options::OPT_w);
41e5dd7070Spatrick
42e5dd7070Spatrick if (!D.SysRoot.empty())
43e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString("--sysroot=" + D.SysRoot));
44e5dd7070Spatrick
45e5dd7070Spatrick // CloudABI only supports static linkage.
46e5dd7070Spatrick CmdArgs.push_back("-Bstatic");
47e5dd7070Spatrick CmdArgs.push_back("--no-dynamic-linker");
48e5dd7070Spatrick
49e5dd7070Spatrick // Provide PIE linker flags in case PIE is default for the architecture.
50*12c85518Srobert if (ToolChain.isPIEDefault(Args)) {
51e5dd7070Spatrick CmdArgs.push_back("-pie");
52e5dd7070Spatrick CmdArgs.push_back("-zrelro");
53e5dd7070Spatrick }
54e5dd7070Spatrick
55e5dd7070Spatrick CmdArgs.push_back("--eh-frame-hdr");
56e5dd7070Spatrick CmdArgs.push_back("--gc-sections");
57e5dd7070Spatrick
58e5dd7070Spatrick if (Output.isFilename()) {
59e5dd7070Spatrick CmdArgs.push_back("-o");
60e5dd7070Spatrick CmdArgs.push_back(Output.getFilename());
61e5dd7070Spatrick } else {
62e5dd7070Spatrick assert(Output.isNothing() && "Invalid output.");
63e5dd7070Spatrick }
64e5dd7070Spatrick
65e5dd7070Spatrick if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles)) {
66e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString(ToolChain.GetFilePath("crt0.o")));
67e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString(ToolChain.GetFilePath("crtbegin.o")));
68e5dd7070Spatrick }
69e5dd7070Spatrick
70e5dd7070Spatrick Args.AddAllArgs(CmdArgs, options::OPT_L);
71e5dd7070Spatrick ToolChain.AddFilePathLibArgs(Args, CmdArgs);
72e5dd7070Spatrick Args.AddAllArgs(CmdArgs,
73e5dd7070Spatrick {options::OPT_T_Group, options::OPT_e, options::OPT_s,
74e5dd7070Spatrick options::OPT_t, options::OPT_Z_Flag, options::OPT_r});
75e5dd7070Spatrick
76e5dd7070Spatrick if (D.isUsingLTO()) {
77e5dd7070Spatrick assert(!Inputs.empty() && "Must have at least one input.");
78ec727ea7Spatrick addLTOOptions(ToolChain, Args, CmdArgs, Output, Inputs[0],
79e5dd7070Spatrick D.getLTOMode() == LTOK_Thin);
80e5dd7070Spatrick }
81e5dd7070Spatrick
82e5dd7070Spatrick AddLinkerInputs(ToolChain, Inputs, Args, CmdArgs, JA);
83e5dd7070Spatrick
84e5dd7070Spatrick if (ToolChain.ShouldLinkCXXStdlib(Args))
85e5dd7070Spatrick ToolChain.AddCXXStdlibLibArgs(Args, CmdArgs);
86e5dd7070Spatrick if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
87e5dd7070Spatrick CmdArgs.push_back("-lc");
88e5dd7070Spatrick CmdArgs.push_back("-lcompiler_rt");
89e5dd7070Spatrick }
90e5dd7070Spatrick
91e5dd7070Spatrick if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles))
92e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString(ToolChain.GetFilePath("crtend.o")));
93e5dd7070Spatrick
94e5dd7070Spatrick const char *Exec = Args.MakeArgString(ToolChain.GetLinkerPath());
95a9ac8606Spatrick C.addCommand(std::make_unique<Command>(JA, *this,
96a9ac8606Spatrick ResponseFileSupport::AtFileCurCP(),
97a9ac8606Spatrick Exec, CmdArgs, Inputs, Output));
98e5dd7070Spatrick }
99e5dd7070Spatrick
100e5dd7070Spatrick // CloudABI - CloudABI tool chain which can call ld(1) directly.
101e5dd7070Spatrick
CloudABI(const Driver & D,const llvm::Triple & Triple,const ArgList & Args)102e5dd7070Spatrick CloudABI::CloudABI(const Driver &D, const llvm::Triple &Triple,
103e5dd7070Spatrick const ArgList &Args)
104e5dd7070Spatrick : Generic_ELF(D, Triple, Args) {
105e5dd7070Spatrick SmallString<128> P(getDriver().Dir);
106e5dd7070Spatrick llvm::sys::path::append(P, "..", getTriple().str(), "lib");
107ec727ea7Spatrick getFilePaths().push_back(std::string(P.str()));
108e5dd7070Spatrick }
109e5dd7070Spatrick
addLibCxxIncludePaths(const llvm::opt::ArgList & DriverArgs,llvm::opt::ArgStringList & CC1Args) const110e5dd7070Spatrick void CloudABI::addLibCxxIncludePaths(const llvm::opt::ArgList &DriverArgs,
111e5dd7070Spatrick llvm::opt::ArgStringList &CC1Args) const {
112e5dd7070Spatrick SmallString<128> P(getDriver().Dir);
113e5dd7070Spatrick llvm::sys::path::append(P, "..", getTriple().str(), "include/c++/v1");
114e5dd7070Spatrick addSystemInclude(DriverArgs, CC1Args, P.str());
115e5dd7070Spatrick }
116e5dd7070Spatrick
AddCXXStdlibLibArgs(const ArgList & Args,ArgStringList & CmdArgs) const117e5dd7070Spatrick void CloudABI::AddCXXStdlibLibArgs(const ArgList &Args,
118e5dd7070Spatrick ArgStringList &CmdArgs) const {
119e5dd7070Spatrick CmdArgs.push_back("-lc++");
120*12c85518Srobert if (Args.hasArg(options::OPT_fexperimental_library))
121*12c85518Srobert CmdArgs.push_back("-lc++experimental");
122e5dd7070Spatrick CmdArgs.push_back("-lc++abi");
123e5dd7070Spatrick CmdArgs.push_back("-lunwind");
124e5dd7070Spatrick }
125e5dd7070Spatrick
buildLinker() const126e5dd7070Spatrick Tool *CloudABI::buildLinker() const {
127e5dd7070Spatrick return new tools::cloudabi::Linker(*this);
128e5dd7070Spatrick }
129e5dd7070Spatrick
isPIEDefault(const llvm::opt::ArgList & Args) const130*12c85518Srobert bool CloudABI::isPIEDefault(const llvm::opt::ArgList &Args) const {
131e5dd7070Spatrick // Only enable PIE on architectures that support PC-relative
132e5dd7070Spatrick // addressing. PC-relative addressing is required, as the process
133e5dd7070Spatrick // startup code must be able to relocate itself.
134e5dd7070Spatrick switch (getTriple().getArch()) {
135e5dd7070Spatrick case llvm::Triple::aarch64:
136e5dd7070Spatrick case llvm::Triple::x86_64:
137e5dd7070Spatrick return true;
138e5dd7070Spatrick default:
139e5dd7070Spatrick return false;
140e5dd7070Spatrick }
141e5dd7070Spatrick }
142e5dd7070Spatrick
getSupportedSanitizers() const143e5dd7070Spatrick SanitizerMask CloudABI::getSupportedSanitizers() const {
144e5dd7070Spatrick SanitizerMask Res = ToolChain::getSupportedSanitizers();
145e5dd7070Spatrick Res |= SanitizerKind::SafeStack;
146e5dd7070Spatrick return Res;
147e5dd7070Spatrick }
148e5dd7070Spatrick
getDefaultSanitizers() const149e5dd7070Spatrick SanitizerMask CloudABI::getDefaultSanitizers() const {
150e5dd7070Spatrick return SanitizerKind::SafeStack;
151e5dd7070Spatrick }
152