1e5dd7070Spatrick //===--- DragonFly.cpp - DragonFly 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 "DragonFly.h"
10e5dd7070Spatrick #include "CommonArgs.h"
11e5dd7070Spatrick #include "clang/Driver/Compilation.h"
12e5dd7070Spatrick #include "clang/Driver/Driver.h"
13e5dd7070Spatrick #include "clang/Driver/Options.h"
14e5dd7070Spatrick #include "llvm/Option/ArgList.h"
15e5dd7070Spatrick
16e5dd7070Spatrick using namespace clang::driver;
17e5dd7070Spatrick using namespace clang::driver::tools;
18e5dd7070Spatrick using namespace clang::driver::toolchains;
19e5dd7070Spatrick using namespace clang;
20e5dd7070Spatrick using namespace llvm::opt;
21e5dd7070Spatrick
22e5dd7070Spatrick /// DragonFly Tools
23e5dd7070Spatrick
24e5dd7070Spatrick // For now, DragonFly Assemble does just about the same as for
25e5dd7070Spatrick // FreeBSD, but this may change soon.
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const26e5dd7070Spatrick void dragonfly::Assembler::ConstructJob(Compilation &C, const JobAction &JA,
27e5dd7070Spatrick const InputInfo &Output,
28e5dd7070Spatrick const InputInfoList &Inputs,
29e5dd7070Spatrick const ArgList &Args,
30e5dd7070Spatrick const char *LinkingOutput) const {
31e5dd7070Spatrick claimNoWarnArgs(Args);
32e5dd7070Spatrick ArgStringList CmdArgs;
33e5dd7070Spatrick
34e5dd7070Spatrick // When building 32-bit code on DragonFly/pc64, we have to explicitly
35e5dd7070Spatrick // instruct as in the base system to assemble 32-bit code.
36e5dd7070Spatrick if (getToolChain().getArch() == llvm::Triple::x86)
37e5dd7070Spatrick CmdArgs.push_back("--32");
38e5dd7070Spatrick
39e5dd7070Spatrick Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA, options::OPT_Xassembler);
40e5dd7070Spatrick
41e5dd7070Spatrick CmdArgs.push_back("-o");
42e5dd7070Spatrick CmdArgs.push_back(Output.getFilename());
43e5dd7070Spatrick
44e5dd7070Spatrick for (const auto &II : Inputs)
45e5dd7070Spatrick CmdArgs.push_back(II.getFilename());
46e5dd7070Spatrick
47e5dd7070Spatrick const char *Exec = Args.MakeArgString(getToolChain().GetProgramPath("as"));
48a9ac8606Spatrick C.addCommand(std::make_unique<Command>(JA, *this,
49a9ac8606Spatrick ResponseFileSupport::AtFileCurCP(),
50a9ac8606Spatrick Exec, CmdArgs, Inputs, Output));
51e5dd7070Spatrick }
52e5dd7070Spatrick
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const53e5dd7070Spatrick void dragonfly::Linker::ConstructJob(Compilation &C, const JobAction &JA,
54e5dd7070Spatrick const InputInfo &Output,
55e5dd7070Spatrick const InputInfoList &Inputs,
56e5dd7070Spatrick const ArgList &Args,
57e5dd7070Spatrick const char *LinkingOutput) const {
58e5dd7070Spatrick const Driver &D = getToolChain().getDriver();
59e5dd7070Spatrick ArgStringList CmdArgs;
60e5dd7070Spatrick
61e5dd7070Spatrick if (!D.SysRoot.empty())
62e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString("--sysroot=" + D.SysRoot));
63e5dd7070Spatrick
64e5dd7070Spatrick CmdArgs.push_back("--eh-frame-hdr");
65e5dd7070Spatrick if (Args.hasArg(options::OPT_static)) {
66e5dd7070Spatrick CmdArgs.push_back("-Bstatic");
67e5dd7070Spatrick } else {
68e5dd7070Spatrick if (Args.hasArg(options::OPT_rdynamic))
69e5dd7070Spatrick CmdArgs.push_back("-export-dynamic");
70e5dd7070Spatrick if (Args.hasArg(options::OPT_shared))
71e5dd7070Spatrick CmdArgs.push_back("-Bshareable");
72*12c85518Srobert else if (!Args.hasArg(options::OPT_r)) {
73e5dd7070Spatrick CmdArgs.push_back("-dynamic-linker");
74e5dd7070Spatrick CmdArgs.push_back("/usr/libexec/ld-elf.so.2");
75e5dd7070Spatrick }
76e5dd7070Spatrick CmdArgs.push_back("--hash-style=gnu");
77e5dd7070Spatrick CmdArgs.push_back("--enable-new-dtags");
78e5dd7070Spatrick }
79e5dd7070Spatrick
80e5dd7070Spatrick // When building 32-bit code on DragonFly/pc64, we have to explicitly
81e5dd7070Spatrick // instruct ld in the base system to link 32-bit code.
82e5dd7070Spatrick if (getToolChain().getArch() == llvm::Triple::x86) {
83e5dd7070Spatrick CmdArgs.push_back("-m");
84e5dd7070Spatrick CmdArgs.push_back("elf_i386");
85e5dd7070Spatrick }
86e5dd7070Spatrick
87e5dd7070Spatrick if (Output.isFilename()) {
88e5dd7070Spatrick CmdArgs.push_back("-o");
89e5dd7070Spatrick CmdArgs.push_back(Output.getFilename());
90e5dd7070Spatrick } else {
91e5dd7070Spatrick assert(Output.isNothing() && "Invalid output.");
92e5dd7070Spatrick }
93e5dd7070Spatrick
94*12c85518Srobert if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles,
95*12c85518Srobert options::OPT_r)) {
96e5dd7070Spatrick if (!Args.hasArg(options::OPT_shared)) {
97e5dd7070Spatrick if (Args.hasArg(options::OPT_pg))
98e5dd7070Spatrick CmdArgs.push_back(
99e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("gcrt1.o")));
100e5dd7070Spatrick else {
101e5dd7070Spatrick if (Args.hasArg(options::OPT_pie))
102e5dd7070Spatrick CmdArgs.push_back(
103e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("Scrt1.o")));
104e5dd7070Spatrick else
105e5dd7070Spatrick CmdArgs.push_back(
106e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("crt1.o")));
107e5dd7070Spatrick }
108e5dd7070Spatrick }
109e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath("crti.o")));
110e5dd7070Spatrick if (Args.hasArg(options::OPT_shared) || Args.hasArg(options::OPT_pie))
111e5dd7070Spatrick CmdArgs.push_back(
112e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("crtbeginS.o")));
113e5dd7070Spatrick else
114e5dd7070Spatrick CmdArgs.push_back(
115e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("crtbegin.o")));
116e5dd7070Spatrick }
117e5dd7070Spatrick
118e5dd7070Spatrick Args.AddAllArgs(CmdArgs,
119e5dd7070Spatrick {options::OPT_L, options::OPT_T_Group, options::OPT_e});
120e5dd7070Spatrick
121e5dd7070Spatrick AddLinkerInputs(getToolChain(), Inputs, Args, CmdArgs, JA);
122e5dd7070Spatrick
123*12c85518Srobert if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs,
124*12c85518Srobert options::OPT_r)) {
125a9ac8606Spatrick CmdArgs.push_back("-L/usr/lib/gcc80");
126e5dd7070Spatrick
127e5dd7070Spatrick if (!Args.hasArg(options::OPT_static)) {
128e5dd7070Spatrick CmdArgs.push_back("-rpath");
129a9ac8606Spatrick CmdArgs.push_back("/usr/lib/gcc80");
130e5dd7070Spatrick }
131e5dd7070Spatrick
132e5dd7070Spatrick if (D.CCCIsCXX()) {
133e5dd7070Spatrick if (getToolChain().ShouldLinkCXXStdlib(Args))
134e5dd7070Spatrick getToolChain().AddCXXStdlibLibArgs(Args, CmdArgs);
135e5dd7070Spatrick CmdArgs.push_back("-lm");
136e5dd7070Spatrick }
137e5dd7070Spatrick
138e5dd7070Spatrick if (Args.hasArg(options::OPT_pthread))
139e5dd7070Spatrick CmdArgs.push_back("-lpthread");
140e5dd7070Spatrick
141e5dd7070Spatrick if (!Args.hasArg(options::OPT_nolibc)) {
142e5dd7070Spatrick CmdArgs.push_back("-lc");
143e5dd7070Spatrick }
144e5dd7070Spatrick
145e5dd7070Spatrick if (Args.hasArg(options::OPT_static) ||
146e5dd7070Spatrick Args.hasArg(options::OPT_static_libgcc)) {
147e5dd7070Spatrick CmdArgs.push_back("-lgcc");
148e5dd7070Spatrick CmdArgs.push_back("-lgcc_eh");
149e5dd7070Spatrick } else {
150e5dd7070Spatrick if (Args.hasArg(options::OPT_shared_libgcc)) {
151e5dd7070Spatrick CmdArgs.push_back("-lgcc_pic");
152e5dd7070Spatrick if (!Args.hasArg(options::OPT_shared))
153e5dd7070Spatrick CmdArgs.push_back("-lgcc");
154e5dd7070Spatrick } else {
155e5dd7070Spatrick CmdArgs.push_back("-lgcc");
156e5dd7070Spatrick CmdArgs.push_back("--as-needed");
157e5dd7070Spatrick CmdArgs.push_back("-lgcc_pic");
158e5dd7070Spatrick CmdArgs.push_back("--no-as-needed");
159e5dd7070Spatrick }
160e5dd7070Spatrick }
161e5dd7070Spatrick }
162e5dd7070Spatrick
163*12c85518Srobert if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles,
164*12c85518Srobert options::OPT_r)) {
165e5dd7070Spatrick if (Args.hasArg(options::OPT_shared) || Args.hasArg(options::OPT_pie))
166e5dd7070Spatrick CmdArgs.push_back(
167e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("crtendS.o")));
168e5dd7070Spatrick else
169e5dd7070Spatrick CmdArgs.push_back(
170e5dd7070Spatrick Args.MakeArgString(getToolChain().GetFilePath("crtend.o")));
171e5dd7070Spatrick CmdArgs.push_back(Args.MakeArgString(getToolChain().GetFilePath("crtn.o")));
172e5dd7070Spatrick }
173e5dd7070Spatrick
174e5dd7070Spatrick getToolChain().addProfileRTLibs(Args, CmdArgs);
175e5dd7070Spatrick
176e5dd7070Spatrick const char *Exec = Args.MakeArgString(getToolChain().GetLinkerPath());
177a9ac8606Spatrick C.addCommand(std::make_unique<Command>(JA, *this,
178a9ac8606Spatrick ResponseFileSupport::AtFileCurCP(),
179a9ac8606Spatrick Exec, CmdArgs, Inputs, Output));
180e5dd7070Spatrick }
181e5dd7070Spatrick
182e5dd7070Spatrick /// DragonFly - DragonFly tool chain which can call as(1) and ld(1) directly.
183e5dd7070Spatrick
DragonFly(const Driver & D,const llvm::Triple & Triple,const ArgList & Args)184e5dd7070Spatrick DragonFly::DragonFly(const Driver &D, const llvm::Triple &Triple,
185e5dd7070Spatrick const ArgList &Args)
186e5dd7070Spatrick : Generic_ELF(D, Triple, Args) {
187e5dd7070Spatrick
188e5dd7070Spatrick // Path mangling to find libexec
189e5dd7070Spatrick getProgramPaths().push_back(getDriver().getInstalledDir());
190e5dd7070Spatrick if (getDriver().getInstalledDir() != getDriver().Dir)
191e5dd7070Spatrick getProgramPaths().push_back(getDriver().Dir);
192e5dd7070Spatrick
193e5dd7070Spatrick getFilePaths().push_back(getDriver().Dir + "/../lib");
194e5dd7070Spatrick getFilePaths().push_back("/usr/lib");
195a9ac8606Spatrick getFilePaths().push_back("/usr/lib/gcc80");
196e5dd7070Spatrick }
197e5dd7070Spatrick
buildAssembler() const198e5dd7070Spatrick Tool *DragonFly::buildAssembler() const {
199e5dd7070Spatrick return new tools::dragonfly::Assembler(*this);
200e5dd7070Spatrick }
201e5dd7070Spatrick
buildLinker() const202e5dd7070Spatrick Tool *DragonFly::buildLinker() const {
203e5dd7070Spatrick return new tools::dragonfly::Linker(*this);
204e5dd7070Spatrick }
205