1//===--- OpenBSD.cpp - OpenBSD 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 "OpenBSD.h"
10#include "Arch/ARM.h"
11#include "Arch/Mips.h"
12#include "Arch/Sparc.h"
13#include "clang/Config/config.h"
14#include "clang/Driver/CommonArgs.h"
15#include "clang/Driver/Compilation.h"
16#include "clang/Driver/SanitizerArgs.h"
17#include "clang/Options/Options.h"
18#include "llvm/Option/ArgList.h"
19#include "llvm/Support/Path.h"
20#include "llvm/Support/VirtualFileSystem.h"
21
22using namespace clang::driver;
23using namespace clang::driver::tools;
24using namespace clang::driver::toolchains;
25using namespace clang;
26using namespace llvm::opt;
27
28void openbsd::Assembler::ConstructJob(Compilation &C, const JobAction &JA,
29 const InputInfo &Output,
30 const InputInfoList &Inputs,
31 const ArgList &Args,
32 const char *LinkingOutput) const {
33 const auto &ToolChain = static_cast<const OpenBSD &>(getToolChain());
34 const Driver &D = ToolChain.getDriver();
35 const llvm::Triple &Triple = ToolChain.getTriple();
36 ArgStringList CmdArgs;
37
38 claimNoWarnArgs(Args);
39
40 switch (ToolChain.getArch()) {
41 case llvm::Triple::x86:
42 // When building 32-bit code on OpenBSD/amd64, we have to explicitly
43 // instruct as in the base system to assemble 32-bit code.
44 CmdArgs.push_back(Elt: "--32");
45 break;
46
47 case llvm::Triple::arm: {
48 StringRef MArch, MCPU;
49 arm::getARMArchCPUFromArgs(Args, Arch&: MArch, CPU&: MCPU, /*FromAs*/ true);
50 std::string Arch = arm::getARMTargetCPU(CPU: MCPU, Arch: MArch, Triple);
51 CmdArgs.push_back(Elt: Args.MakeArgString(Str: "-mcpu=" + Arch));
52 break;
53 }
54
55 case llvm::Triple::ppc:
56 CmdArgs.push_back(Elt: "-mppc");
57 CmdArgs.push_back(Elt: "-many");
58 break;
59
60 case llvm::Triple::sparcv9: {
61 CmdArgs.push_back(Elt: "-64");
62 std::string CPU = getCPUName(D, Args, T: Triple);
63 CmdArgs.push_back(Elt: sparc::getSparcAsmModeForCPU(Name: CPU, Triple));
64 AddAssemblerKPIC(ToolChain, Args, CmdArgs);
65 break;
66 }
67
68 case llvm::Triple::mips64:
69 case llvm::Triple::mips64el: {
70 StringRef CPUName;
71 StringRef ABIName;
72 mips::getMipsCPUAndABI(Args, Triple, CPUName, ABIName);
73
74 CmdArgs.push_back(Elt: "-march");
75 CmdArgs.push_back(Elt: CPUName.data());
76
77 CmdArgs.push_back(Elt: "-mabi");
78 CmdArgs.push_back(Elt: mips::getGnuCompatibleMipsABIName(ABI: ABIName).data());
79
80 if (Triple.isLittleEndian())
81 CmdArgs.push_back(Elt: "-EL");
82 else
83 CmdArgs.push_back(Elt: "-EB");
84
85 AddAssemblerKPIC(ToolChain, Args, CmdArgs);
86 break;
87 }
88
89 default:
90 break;
91 }
92
93 Args.AddAllArgValues(Output&: CmdArgs, Id0: options::OPT_Wa_COMMA, Id1: options::OPT_Xassembler);
94
95 CmdArgs.push_back(Elt: "-o");
96 CmdArgs.push_back(Elt: Output.getFilename());
97
98 for (const auto &II : Inputs)
99 CmdArgs.push_back(Elt: II.getFilename());
100
101 const char *Exec = Args.MakeArgString(Str: ToolChain.GetProgramPath(Name: "as"));
102 C.addCommand(Cmd: std::make_unique<Command>(args: JA, args: *this,
103 args: ResponseFileSupport::AtFileCurCP(),
104 args&: Exec, args&: CmdArgs, args: Inputs, args: Output));
105}
106
107void openbsd::Linker::ConstructJob(Compilation &C, const JobAction &JA,
108 const InputInfo &Output,
109 const InputInfoList &Inputs,
110 const ArgList &Args,
111 const char *LinkingOutput) const {
112 const auto &ToolChain = static_cast<const OpenBSD &>(getToolChain());
113 const Driver &D = ToolChain.getDriver();
114 const llvm::Triple &Triple = ToolChain.getTriple();
115 const llvm::Triple::ArchType Arch = ToolChain.getArch();
116 const bool Static = Args.hasArg(Ids: options::OPT_static);
117 const bool Shared = Args.hasArg(Ids: options::OPT_shared);
118 const bool Profiling = Args.hasArg(Ids: options::OPT_pg);
119 const bool Pie = Args.hasArg(Ids: options::OPT_pie);
120 const bool Nopie = Args.hasArg(Ids: options::OPT_no_pie, Ids: options::OPT_nopie);
121 const bool Relocatable = Args.hasArg(Ids: options::OPT_r);
122 ArgStringList CmdArgs;
123
124 // Silence warning for "clang -g foo.o -o foo"
125 Args.ClaimAllArgs(Id0: options::OPT_g_Group);
126 // and "clang -emit-llvm foo.o -o foo"
127 Args.ClaimAllArgs(Id0: options::OPT_emit_llvm);
128 // and for "clang -w foo.o -o foo". Other warning options are already
129 // handled somewhere else.
130 Args.ClaimAllArgs(Id0: options::OPT_w);
131
132 if (!D.SysRoot.empty())
133 CmdArgs.push_back(Elt: Args.MakeArgString(Str: "--sysroot=" + D.SysRoot));
134
135 if (Arch == llvm::Triple::mips64)
136 CmdArgs.push_back(Elt: "-EB");
137 else if (Arch == llvm::Triple::mips64el)
138 CmdArgs.push_back(Elt: "-EL");
139
140 if (!Args.hasArg(Ids: options::OPT_nostdlib) && !Shared && !Relocatable) {
141 CmdArgs.push_back(Elt: "-e");
142 CmdArgs.push_back(Elt: "__start");
143 }
144
145 CmdArgs.push_back(Elt: "--eh-frame-hdr");
146 if (Static) {
147 CmdArgs.push_back(Elt: "-Bstatic");
148 } else {
149 if (Args.hasArg(Ids: options::OPT_rdynamic))
150 CmdArgs.push_back(Elt: "-export-dynamic");
151 if (Shared) {
152 CmdArgs.push_back(Elt: "-shared");
153 } else if (!Relocatable) {
154 CmdArgs.push_back(Elt: "-dynamic-linker");
155 CmdArgs.push_back(Elt: "/usr/libexec/ld.so");
156 }
157 }
158
159 if (Pie)
160 CmdArgs.push_back(Elt: "-pie");
161 if (Nopie || Profiling)
162 CmdArgs.push_back(Elt: "-nopie");
163
164 if (Triple.isLoongArch64() || Triple.isRISCV64()) {
165 CmdArgs.push_back(Elt: "-X");
166 if (Args.hasArg(Ids: options::OPT_mno_relax))
167 CmdArgs.push_back(Elt: "--no-relax");
168 }
169
170 assert((Output.isFilename() || Output.isNothing()) && "Invalid output.");
171 if (Output.isFilename()) {
172 CmdArgs.push_back(Elt: "-o");
173 CmdArgs.push_back(Elt: Output.getFilename());
174 }
175
176 if (!Args.hasArg(Ids: options::OPT_nostdlib, Ids: options::OPT_nostartfiles,
177 Ids: options::OPT_r)) {
178 const char *crt0 = nullptr;
179 const char *crtbegin = nullptr;
180 if (!Shared) {
181 if (Profiling)
182 crt0 = "gcrt0.o";
183 else if (Static && !Nopie)
184 crt0 = "rcrt0.o";
185 else
186 crt0 = "crt0.o";
187 crtbegin = "crtbegin.o";
188 } else {
189 crtbegin = "crtbeginS.o";
190 }
191
192 if (crt0)
193 CmdArgs.push_back(Elt: Args.MakeArgString(Str: ToolChain.GetFilePath(Name: crt0)));
194 CmdArgs.push_back(Elt: Args.MakeArgString(Str: ToolChain.GetFilePath(Name: crtbegin)));
195 }
196
197 Args.AddAllArgs(Output&: CmdArgs, Id0: options::OPT_L);
198 ToolChain.AddFilePathLibArgs(Args, CmdArgs);
199 Args.addAllArgs(Output&: CmdArgs,
200 Ids: {options::OPT_T_Group, options::OPT_s, options::OPT_t});
201
202 if (auto LTO = ToolChain.getLTOMode(Args); LTO != LTOK_None)
203 addLTOOptions(ToolChain, Args, CmdArgs, Output, Inputs, IsThinLTO: LTO == LTOK_Thin);
204
205 bool NeedsSanitizerDeps = addSanitizerRuntimes(TC: ToolChain, Args, CmdArgs);
206 bool NeedsXRayDeps = addXRayRuntime(TC: ToolChain, Args, CmdArgs);
207 AddLinkerInputs(TC: ToolChain, Inputs, Args, CmdArgs, JA);
208
209 if (!Args.hasArg(Ids: options::OPT_nostdlib, Ids: options::OPT_nodefaultlibs,
210 Ids: options::OPT_r)) {
211 // Use the static OpenMP runtime with -static-openmp
212 bool StaticOpenMP = Args.hasArg(Ids: options::OPT_static_openmp) && !Static;
213 addOpenMPRuntime(C, CmdArgs, TC: ToolChain, Args, ForceStaticHostRuntime: StaticOpenMP);
214
215 if (D.CCCIsCXX()) {
216 if (ToolChain.ShouldLinkCXXStdlib(Args))
217 ToolChain.AddCXXStdlibLibArgs(Args, CmdArgs);
218 if (Profiling)
219 CmdArgs.push_back(Elt: "-lm_p");
220 else
221 CmdArgs.push_back(Elt: "-lm");
222 }
223
224 // Silence warnings when linking C code with a C++ '-stdlib' argument.
225 Args.ClaimAllArgs(Id0: options::OPT_stdlib_EQ);
226
227 // Additional linker set-up and flags for Fortran. This is required in order
228 // to generate executables. As Fortran runtime depends on the C runtime,
229 // these dependencies need to be listed before the C runtime below (i.e.
230 // AddRunTimeLibs).
231 if (D.IsFlangMode() &&
232 !Args.hasArg(Ids: options::OPT_nostdlib, Ids: options::OPT_nodefaultlibs)) {
233 ToolChain.addFortranRuntimeLibraryPath(Args, CmdArgs);
234 ToolChain.addFortranRuntimeLibs(Args, CmdArgs);
235 if (Profiling)
236 CmdArgs.push_back(Elt: "-lm_p");
237 else
238 CmdArgs.push_back(Elt: "-lm");
239 }
240
241 if (NeedsSanitizerDeps) {
242 CmdArgs.push_back(Elt: ToolChain.getCompilerRTArgString(Args, Component: "builtins"));
243 linkSanitizerRuntimeDeps(TC: ToolChain, Args, CmdArgs);
244 }
245 if (NeedsXRayDeps) {
246 CmdArgs.push_back(Elt: ToolChain.getCompilerRTArgString(Args, Component: "builtins"));
247 linkXRayRuntimeDeps(TC: ToolChain, Args, CmdArgs);
248 }
249 // FIXME: For some reason GCC passes -lgcc before adding
250 // the default system libraries. Just mimic this for now.
251 CmdArgs.push_back(Elt: "-lcompiler_rt");
252
253 if (Args.hasArg(Ids: options::OPT_pthread)) {
254 if (!Shared && Profiling)
255 CmdArgs.push_back(Elt: "-lpthread_p");
256 else
257 CmdArgs.push_back(Elt: "-lpthread");
258 }
259
260 if (!Shared) {
261 if (Profiling)
262 CmdArgs.push_back(Elt: "-lc_p");
263 else
264 CmdArgs.push_back(Elt: "-lc");
265 }
266
267 CmdArgs.push_back(Elt: "-lcompiler_rt");
268 }
269
270 if (!Args.hasArg(Ids: options::OPT_nostdlib, Ids: options::OPT_nostartfiles,
271 Ids: options::OPT_r)) {
272 const char *crtend = nullptr;
273 if (!Shared)
274 crtend = "crtend.o";
275 else
276 crtend = "crtendS.o";
277
278 CmdArgs.push_back(Elt: Args.MakeArgString(Str: ToolChain.GetFilePath(Name: crtend)));
279 }
280
281 ToolChain.addProfileRTLibs(Args, CmdArgs);
282
283 const char *Exec = Args.MakeArgString(Str: ToolChain.GetLinkerPath());
284 C.addCommand(Cmd: std::make_unique<Command>(args: JA, args: *this,
285 args: ResponseFileSupport::AtFileCurCP(),
286 args&: Exec, args&: CmdArgs, args: Inputs, args: Output));
287}
288
289SanitizerMask
290OpenBSD::getSupportedSanitizers(BoundArch BA,
291 Action::OffloadKind DeviceOffloadKind) const {
292 const bool IsX86 = getTriple().getArch() == llvm::Triple::x86;
293 const bool IsX86_64 = getTriple().getArch() == llvm::Triple::x86_64;
294 SanitizerMask Res = ToolChain::getSupportedSanitizers(BA, DeviceOffloadKind);
295 if (IsX86 || IsX86_64) {
296 Res |= SanitizerKind::Vptr;
297 Res |= SanitizerKind::Fuzzer;
298 Res |= SanitizerKind::FuzzerNoLink;
299 }
300 if (IsX86_64) {
301 Res |= SanitizerKind::KernelAddress;
302 }
303 return Res;
304}
305
306/// OpenBSD - OpenBSD tool chain which can call as(1) and ld(1) directly.
307
308OpenBSD::OpenBSD(const Driver &D, const llvm::Triple &Triple,
309 const ArgList &Args)
310 : Generic_ELF(D, Triple, Args) {
311 getFilePaths().push_back(Elt: concat(Path: getDriver().SysRoot, A: "/usr/lib"));
312}
313
314void OpenBSD::AddClangSystemIncludeArgs(
315 const llvm::opt::ArgList &DriverArgs,
316 llvm::opt::ArgStringList &CC1Args) const {
317 const Driver &D = getDriver();
318
319 if (DriverArgs.hasArg(Ids: options::OPT_nostdinc))
320 return;
321
322 if (!DriverArgs.hasArg(Ids: options::OPT_nobuiltininc)) {
323 SmallString<128> Dir(D.ResourceDir);
324 llvm::sys::path::append(path&: Dir, a: "include");
325 addSystemInclude(DriverArgs, CC1Args, Path: Dir.str());
326 }
327
328 if (DriverArgs.hasArg(Ids: options::OPT_nostdlibinc))
329 return;
330
331 // Check for configure-time C include directories.
332 StringRef CIncludeDirs(C_INCLUDE_DIRS);
333 if (CIncludeDirs != "") {
334 SmallVector<StringRef, 5> dirs;
335 CIncludeDirs.split(A&: dirs, Separator: ":");
336 for (StringRef dir : dirs) {
337 StringRef Prefix =
338 llvm::sys::path::is_absolute(path: dir) ? StringRef(D.SysRoot) : "";
339 addExternCSystemInclude(DriverArgs, CC1Args, Path: Prefix + dir);
340 }
341 return;
342 }
343
344 addExternCSystemInclude(DriverArgs, CC1Args,
345 Path: concat(Path: D.SysRoot, A: "/usr/include"));
346}
347
348void OpenBSD::addLibCxxIncludePaths(const llvm::opt::ArgList &DriverArgs,
349 llvm::opt::ArgStringList &CC1Args) const {
350 addSystemInclude(DriverArgs, CC1Args,
351 Path: concat(Path: getDriver().SysRoot, A: "/usr/include/c++/v1"));
352}
353
354void OpenBSD::AddCXXStdlibLibArgs(const ArgList &Args,
355 ArgStringList &CmdArgs) const {
356 bool Profiling = Args.hasArg(Ids: options::OPT_pg);
357
358 CmdArgs.push_back(Elt: Profiling ? "-lc++_p" : "-lc++");
359 if (Args.hasArg(Ids: options::OPT_fexperimental_library))
360 CmdArgs.push_back(Elt: "-lc++experimental");
361 CmdArgs.push_back(Elt: Profiling ? "-lc++abi_p" : "-lc++abi");
362 CmdArgs.push_back(Elt: Profiling ? "-lpthread_p" : "-lpthread");
363}
364
365std::string OpenBSD::getCompilerRT(const ArgList &Args, StringRef Component,
366 FileType Type, bool IsFortran) const {
367 if (Component == "builtins") {
368 SmallString<128> Path(getDriver().SysRoot);
369 llvm::sys::path::append(path&: Path, a: "/usr/lib/libcompiler_rt.a");
370 if (getVFS().exists(Path))
371 return std::string(Path);
372 }
373 SmallString<128> P(getDriver().ResourceDir);
374 std::string CRTBasename = buildCompilerRTBasename(
375 Args, Component, Type, /*AddArch=*/false, IsFortran);
376 llvm::sys::path::append(path&: P, a: "lib", b: CRTBasename);
377 // Checks if this is the base system case which uses a different location.
378 if (getVFS().exists(Path: P))
379 return std::string(P);
380 return ToolChain::getCompilerRT(Args, Component, Type, IsFortran);
381}
382
383Tool *OpenBSD::buildAssembler() const {
384 return new tools::openbsd::Assembler(*this);
385}
386
387Tool *OpenBSD::buildLinker() const { return new tools::openbsd::Linker(*this); }
388
389bool OpenBSD::HasNativeLLVMSupport() const { return true; }
390
391ToolChain::UnwindTableLevel
392OpenBSD::getDefaultUnwindTableLevel(const ArgList &Args) const {
393 switch (getArch()) {
394 case llvm::Triple::arm:
395 return UnwindTableLevel::None;
396 default:
397 return UnwindTableLevel::Asynchronous;
398 }
399}
400