clang 18.0.0git
DragonFly.cpp
Go to the documentation of this file.
1//===--- DragonFly.cpp - DragonFly ToolChain Implementations ----*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include "DragonFly.h"
10#include "CommonArgs.h"
12#include "clang/Driver/Driver.h"
14#include "llvm/Option/ArgList.h"
15#include "llvm/Support/Path.h"
16
17using namespace clang::driver;
18using namespace clang::driver::tools;
19using namespace clang::driver::toolchains;
20using namespace clang;
21using namespace llvm::opt;
22
23/// DragonFly Tools
24
25// For now, DragonFly Assemble does just about the same as for
26// FreeBSD, but this may change soon.
28 const InputInfo &Output,
29 const InputInfoList &Inputs,
30 const ArgList &Args,
31 const char *LinkingOutput) const {
32 claimNoWarnArgs(Args);
33 ArgStringList CmdArgs;
34
35 // When building 32-bit code on DragonFly/pc64, we have to explicitly
36 // instruct as in the base system to assemble 32-bit code.
37 if (getToolChain().getArch() == llvm::Triple::x86)
38 CmdArgs.push_back("--32");
39
40 Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA, options::OPT_Xassembler);
41
42 CmdArgs.push_back("-o");
43 CmdArgs.push_back(Output.getFilename());
44
45 for (const auto &II : Inputs)
46 CmdArgs.push_back(II.getFilename());
47
48 const char *Exec = Args.MakeArgString(getToolChain().GetProgramPath("as"));
49 C.addCommand(std::make_unique<Command>(JA, *this,
51 Exec, CmdArgs, Inputs, Output));
52}
53
55 const InputInfo &Output,
56 const InputInfoList &Inputs,
57 const ArgList &Args,
58 const char *LinkingOutput) const {
59 const Driver &D = getToolChain().getDriver();
60 ArgStringList CmdArgs;
61
62 if (!D.SysRoot.empty())
63 CmdArgs.push_back(Args.MakeArgString("--sysroot=" + D.SysRoot));
64
65 CmdArgs.push_back("--eh-frame-hdr");
66 if (Args.hasArg(options::OPT_static)) {
67 CmdArgs.push_back("-Bstatic");
68 } else {
69 if (Args.hasArg(options::OPT_rdynamic))
70 CmdArgs.push_back("-export-dynamic");
71 if (Args.hasArg(options::OPT_shared))
72 CmdArgs.push_back("-shared");
73 else if (!Args.hasArg(options::OPT_r)) {
74 CmdArgs.push_back("-dynamic-linker");
75 CmdArgs.push_back("/usr/libexec/ld-elf.so.2");
76 }
77 CmdArgs.push_back("--hash-style=gnu");
78 CmdArgs.push_back("--enable-new-dtags");
79 }
80
81 // When building 32-bit code on DragonFly/pc64, we have to explicitly
82 // instruct ld in the base system to link 32-bit code.
83 if (getToolChain().getArch() == llvm::Triple::x86) {
84 CmdArgs.push_back("-m");
85 CmdArgs.push_back("elf_i386");
86 }
87
88 assert((Output.isFilename() || Output.isNothing()) && "Invalid output.");
89 if (Output.isFilename()) {
90 CmdArgs.push_back("-o");
91 CmdArgs.push_back(Output.getFilename());
92 }
93
94 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles,
95 options::OPT_r)) {
96 if (!Args.hasArg(options::OPT_shared)) {
97 if (Args.hasArg(options::OPT_pg))
98 CmdArgs.push_back(
99 Args.MakeArgString(getToolChain().GetFilePath("gcrt1.o")));
100 else {
101 if (Args.hasArg(options::OPT_pie))
102 CmdArgs.push_back(
103 Args.MakeArgString(getToolChain().GetFilePath("Scrt1.o")));
104 else
105 CmdArgs.push_back(
106 Args.MakeArgString(getToolChain().GetFilePath("crt1.o")));
107 }
108 }
109 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath("crti.o")));
110 if (Args.hasArg(options::OPT_shared) || Args.hasArg(options::OPT_pie))
111 CmdArgs.push_back(
112 Args.MakeArgString(getToolChain().GetFilePath("crtbeginS.o")));
113 else
114 CmdArgs.push_back(
115 Args.MakeArgString(getToolChain().GetFilePath("crtbegin.o")));
116 }
117
118 Args.AddAllArgs(CmdArgs,
119 {options::OPT_L, options::OPT_T_Group});
120
121 AddLinkerInputs(getToolChain(), Inputs, Args, CmdArgs, JA);
122
123 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs,
124 options::OPT_r)) {
126 llvm::sys::path::append(Dir, "/usr/lib/gcc80");
127 CmdArgs.push_back(Args.MakeArgString("-L" + Dir));
128
129 if (!Args.hasArg(options::OPT_static)) {
130 CmdArgs.push_back("-rpath");
131 CmdArgs.push_back("/usr/lib/gcc80");
132 }
133
134 if (D.CCCIsCXX()) {
135 if (getToolChain().ShouldLinkCXXStdlib(Args))
136 getToolChain().AddCXXStdlibLibArgs(Args, CmdArgs);
137 CmdArgs.push_back("-lm");
138 }
139
140 if (Args.hasArg(options::OPT_pthread))
141 CmdArgs.push_back("-lpthread");
142
143 if (!Args.hasArg(options::OPT_nolibc)) {
144 CmdArgs.push_back("-lc");
145 }
146
147 if (Args.hasArg(options::OPT_static) ||
148 Args.hasArg(options::OPT_static_libgcc)) {
149 CmdArgs.push_back("-lgcc");
150 CmdArgs.push_back("-lgcc_eh");
151 } else {
152 if (Args.hasArg(options::OPT_shared_libgcc)) {
153 CmdArgs.push_back("-lgcc_pic");
154 if (!Args.hasArg(options::OPT_shared))
155 CmdArgs.push_back("-lgcc");
156 } else {
157 CmdArgs.push_back("-lgcc");
158 CmdArgs.push_back("--as-needed");
159 CmdArgs.push_back("-lgcc_pic");
160 CmdArgs.push_back("--no-as-needed");
161 }
162 }
163 }
164
165 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles,
166 options::OPT_r)) {
167 if (Args.hasArg(options::OPT_shared) || Args.hasArg(options::OPT_pie))
168 CmdArgs.push_back(
169 Args.MakeArgString(getToolChain().GetFilePath("crtendS.o")));
170 else
171 CmdArgs.push_back(
172 Args.MakeArgString(getToolChain().GetFilePath("crtend.o")));
173 CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath("crtn.o")));
174 }
175
176 getToolChain().addProfileRTLibs(Args, CmdArgs);
177
178 const char *Exec = Args.MakeArgString(getToolChain().GetLinkerPath());
179 C.addCommand(std::make_unique<Command>(JA, *this,
181 Exec, CmdArgs, Inputs, Output));
182}
183
184/// DragonFly - DragonFly tool chain which can call as(1) and ld(1) directly.
185
186DragonFly::DragonFly(const Driver &D, const llvm::Triple &Triple,
187 const ArgList &Args)
188 : Generic_ELF(D, Triple, Args) {
189
190 // Path mangling to find libexec
191 getProgramPaths().push_back(getDriver().getInstalledDir());
192 if (getDriver().getInstalledDir() != getDriver().Dir)
193 getProgramPaths().push_back(getDriver().Dir);
194
195 getFilePaths().push_back(getDriver().Dir + "/../lib");
196 getFilePaths().push_back(concat(getDriver().SysRoot, "/usr/lib"));
197 getFilePaths().push_back(concat(getDriver().SysRoot, "/usr/lib/gcc80"));
198}
199
201 const llvm::opt::ArgList &DriverArgs,
202 llvm::opt::ArgStringList &CC1Args) const {
203 const Driver &D = getDriver();
204
205 if (DriverArgs.hasArg(clang::driver::options::OPT_nostdinc))
206 return;
207
208 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
210 llvm::sys::path::append(Dir, "include");
211 addSystemInclude(DriverArgs, CC1Args, Dir.str());
212 }
213
214 if (DriverArgs.hasArg(options::OPT_nostdlibinc))
215 return;
216
217 addExternCSystemInclude(DriverArgs, CC1Args,
218 concat(D.SysRoot, "/usr/include"));
219}
220
221void DragonFly::addLibStdCxxIncludePaths(const llvm::opt::ArgList &DriverArgs,
222 llvm::opt::ArgStringList &CC1Args) const {
223 addLibStdCXXIncludePaths(concat(getDriver().SysRoot, "/usr/include/c++/8.0"), "", "",
224 DriverArgs, CC1Args);
225}
226
228 return new tools::dragonfly::Assembler(*this);
229}
230
232 return new tools::dragonfly::Linker(*this);
233}
Compilation - A set of tasks to perform for a single driver invocation.
Definition: Compilation.h:45
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
Definition: Driver.h:77
std::string SysRoot
sysroot, if present
Definition: Driver.h:183
std::string ResourceDir
The path to the compiler resource directory.
Definition: Driver.h:167
bool CCCIsCXX() const
Whether the driver should follow g++ like behavior.
Definition: Driver.h:216
InputInfo - Wrapper for information about an input source.
Definition: InputInfo.h:22
const char * getFilename() const
Definition: InputInfo.h:83
bool isNothing() const
Definition: InputInfo.h:74
bool isFilename() const
Definition: InputInfo.h:75
static void addSystemInclude(const llvm::opt::ArgList &DriverArgs, llvm::opt::ArgStringList &CC1Args, const Twine &Path)
Utility function to add a system include directory to CC1 arguments.
Definition: ToolChain.cpp:1133
static void addExternCSystemInclude(const llvm::opt::ArgList &DriverArgs, llvm::opt::ArgStringList &CC1Args, const Twine &Path)
Utility function to add a system include directory with extern "C" semantics to CC1 arguments.
Definition: ToolChain.cpp:1148
path_list & getFilePaths()
Definition: ToolChain.h:287
const Driver & getDriver() const
Definition: ToolChain.h:245
static std::string concat(StringRef Path, const Twine &A, const Twine &B="", const Twine &C="", const Twine &D="")
Definition: ToolChain.cpp:1172
path_list & getProgramPaths()
Definition: ToolChain.h:290
Tool - Information on a specific compilation tool.
Definition: Tool.h:32
const ToolChain & getToolChain() const
Definition: Tool.h:52
Tool * buildLinker() const override
Definition: DragonFly.cpp:231
void addLibStdCxxIncludePaths(const llvm::opt::ArgList &DriverArgs, llvm::opt::ArgStringList &CC1Args) const override
Definition: DragonFly.cpp:221
void AddClangSystemIncludeArgs(const llvm::opt::ArgList &DriverArgs, llvm::opt::ArgStringList &CC1Args) const override
Add the clang cc1 arguments for system include paths.
Definition: DragonFly.cpp:200
DragonFly(const Driver &D, const llvm::Triple &Triple, const llvm::opt::ArgList &Args)
DragonFly - DragonFly tool chain which can call as(1) and ld(1) directly.
Definition: DragonFly.cpp:186
Tool * buildAssembler() const override
Definition: DragonFly.cpp:227
bool addLibStdCXXIncludePaths(Twine IncludeDir, StringRef Triple, Twine IncludeSuffix, const llvm::opt::ArgList &DriverArgs, llvm::opt::ArgStringList &CC1Args, bool DetectDebian=false) const
Definition: Gnu.cpp:3159
void ConstructJob(Compilation &C, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &TCArgs, const char *LinkingOutput) const override
DragonFly Tools.
Definition: DragonFly.cpp:27
void ConstructJob(Compilation &C, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &TCArgs, const char *LinkingOutput) const override
ConstructJob - Construct jobs to perform the action JA, writing to Output and with Inputs,...
Definition: DragonFly.cpp:54
void claimNoWarnArgs(const llvm::opt::ArgList &Args)
void AddLinkerInputs(const ToolChain &TC, const InputInfoList &Inputs, const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CmdArgs, const JobAction &JA)
@ C
Languages that the frontend can parse and compile.
static constexpr ResponseFileSupport AtFileCurCP()
Definition: Job.h:92