blob: 31d16922cc43eea96933ef430535eba19f00170c [file] [log] [blame]
Jonathan Roelofs901c7762017-05-25 15:42:13 +00001//===--- BaremMetal.cpp - Bare Metal ToolChain ------------------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "BareMetal.h"
11
12#include "CommonArgs.h"
13#include "InputInfo.h"
14#include "Gnu.h"
15
Jonathan Roelofs901c7762017-05-25 15:42:13 +000016#include "clang/Driver/Compilation.h"
17#include "clang/Driver/Driver.h"
18#include "clang/Driver/DriverDiagnostic.h"
19#include "clang/Driver/Options.h"
20#include "llvm/Option/ArgList.h"
21#include "llvm/Support/Path.h"
Jonas Devliegherefc514902018-10-10 13:27:25 +000022#include "llvm/Support/VirtualFileSystem.h"
Jonathan Roelofs901c7762017-05-25 15:42:13 +000023#include "llvm/Support/raw_ostream.h"
24
25using namespace llvm::opt;
26using namespace clang;
27using namespace clang::driver;
28using namespace clang::driver::tools;
29using namespace clang::driver::toolchains;
30
31BareMetal::BareMetal(const Driver &D, const llvm::Triple &Triple,
32 const ArgList &Args)
33 : ToolChain(D, Triple, Args) {
34 getProgramPaths().push_back(getDriver().getInstalledDir());
35 if (getDriver().getInstalledDir() != getDriver().Dir)
36 getProgramPaths().push_back(getDriver().Dir);
37}
38
39BareMetal::~BareMetal() {}
40
41/// Is the triple {arm,thumb}-none-none-{eabi,eabihf} ?
42static bool isARMBareMetal(const llvm::Triple &Triple) {
43 if (Triple.getArch() != llvm::Triple::arm &&
44 Triple.getArch() != llvm::Triple::thumb)
45 return false;
46
47 if (Triple.getVendor() != llvm::Triple::UnknownVendor)
48 return false;
49
50 if (Triple.getOS() != llvm::Triple::UnknownOS)
51 return false;
52
53 if (Triple.getEnvironment() != llvm::Triple::EABI &&
54 Triple.getEnvironment() != llvm::Triple::EABIHF)
55 return false;
56
57 return true;
58}
59
60bool BareMetal::handlesTarget(const llvm::Triple &Triple) {
61 return isARMBareMetal(Triple);
62}
63
64Tool *BareMetal::buildLinker() const {
65 return new tools::baremetal::Linker(*this);
66}
67
Jonathan Roelofs901c7762017-05-25 15:42:13 +000068std::string BareMetal::getRuntimesDir() const {
69 SmallString<128> Dir(getDriver().ResourceDir);
70 llvm::sys::path::append(Dir, "lib", "baremetal");
71 return Dir.str();
72}
73
74void BareMetal::AddClangSystemIncludeArgs(const ArgList &DriverArgs,
75 ArgStringList &CC1Args) const {
76 if (DriverArgs.hasArg(options::OPT_nostdinc))
77 return;
78
79 if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
80 SmallString<128> Dir(getDriver().ResourceDir);
81 llvm::sys::path::append(Dir, "include");
82 addSystemInclude(DriverArgs, CC1Args, Dir.str());
83 }
84
85 if (!DriverArgs.hasArg(options::OPT_nostdlibinc)) {
86 SmallString<128> Dir(getDriver().SysRoot);
87 llvm::sys::path::append(Dir, "include");
88 addSystemInclude(DriverArgs, CC1Args, Dir.str());
89 }
90}
91
92void BareMetal::addClangTargetOptions(const ArgList &DriverArgs,
Gheorghe-Teodor Berceaf0f29602017-07-06 16:22:21 +000093 ArgStringList &CC1Args,
94 Action::OffloadKind) const {
Jonathan Roelofs901c7762017-05-25 15:42:13 +000095 CC1Args.push_back("-nostdsysteminc");
96}
97
Petr Hosek8d612142018-04-10 19:55:55 +000098void BareMetal::AddClangCXXStdlibIncludeArgs(
99 const ArgList &DriverArgs, ArgStringList &CC1Args) const {
100 if (DriverArgs.hasArg(options::OPT_nostdinc) ||
101 DriverArgs.hasArg(options::OPT_nostdlibinc) ||
102 DriverArgs.hasArg(options::OPT_nostdincxx))
103 return;
104
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000105 StringRef SysRoot = getDriver().SysRoot;
106 if (SysRoot.empty())
Petr Hosek8d612142018-04-10 19:55:55 +0000107 return;
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000108
Petr Hosek8d612142018-04-10 19:55:55 +0000109 switch (GetCXXStdlibType(DriverArgs)) {
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000110 case ToolChain::CST_Libcxx: {
111 SmallString<128> Dir(SysRoot);
112 llvm::sys::path::append(Dir, "include", "c++", "v1");
Petr Hosek8d612142018-04-10 19:55:55 +0000113 addSystemInclude(DriverArgs, CC1Args, Dir.str());
114 break;
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000115 }
116 case ToolChain::CST_Libstdcxx: {
117 SmallString<128> Dir(SysRoot);
118 llvm::sys::path::append(Dir, "include", "c++");
119 std::error_code EC;
120 Generic_GCC::GCCVersion Version = {"", -1, -1, -1, "", "", ""};
121 // Walk the subdirs, and find the one with the newest gcc version:
Jonas Devliegherefc514902018-10-10 13:27:25 +0000122 for (llvm::vfs::directory_iterator
123 LI = getDriver().getVFS().dir_begin(Dir.str(), EC),
124 LE;
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000125 !EC && LI != LE; LI = LI.increment(EC)) {
Sam McCall0ae00562018-09-14 12:47:38 +0000126 StringRef VersionText = llvm::sys::path::filename(LI->path());
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000127 auto CandidateVersion = Generic_GCC::GCCVersion::Parse(VersionText);
128 if (CandidateVersion.Major == -1)
129 continue;
130 if (CandidateVersion <= Version)
131 continue;
132 Version = CandidateVersion;
133 }
134 if (Version.Major == -1)
Petr Hosek8d612142018-04-10 19:55:55 +0000135 return;
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000136 llvm::sys::path::append(Dir, Version.Text);
Petr Hosek8d612142018-04-10 19:55:55 +0000137 addSystemInclude(DriverArgs, CC1Args, Dir.str());
138 break;
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000139 }
140 }
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000141}
142
143void BareMetal::AddCXXStdlibLibArgs(const ArgList &Args,
144 ArgStringList &CmdArgs) const {
145 switch (GetCXXStdlibType(Args)) {
146 case ToolChain::CST_Libcxx:
147 CmdArgs.push_back("-lc++");
148 CmdArgs.push_back("-lc++abi");
149 break;
150 case ToolChain::CST_Libstdcxx:
151 CmdArgs.push_back("-lstdc++");
152 CmdArgs.push_back("-lsupc++");
153 break;
154 }
155 CmdArgs.push_back("-lunwind");
156}
157
158void BareMetal::AddLinkRuntimeLib(const ArgList &Args,
159 ArgStringList &CmdArgs) const {
160 CmdArgs.push_back(Args.MakeArgString("-lclang_rt.builtins-" +
161 getTriple().getArchName() + ".a"));
162}
163
164void baremetal::Linker::ConstructJob(Compilation &C, const JobAction &JA,
165 const InputInfo &Output,
166 const InputInfoList &Inputs,
167 const ArgList &Args,
168 const char *LinkingOutput) const {
169 ArgStringList CmdArgs;
170
171 auto &TC = static_cast<const toolchains::BareMetal&>(getToolChain());
172
173 AddLinkerInputs(TC, Inputs, Args, CmdArgs, JA);
174
175 CmdArgs.push_back("-Bstatic");
176
177 CmdArgs.push_back(Args.MakeArgString("-L" + TC.getRuntimesDir()));
178
179 Args.AddAllArgs(CmdArgs, {options::OPT_L, options::OPT_T_Group,
180 options::OPT_e, options::OPT_s, options::OPT_t,
181 options::OPT_Z_Flag, options::OPT_r});
182
Nico Weber0ee47d92017-07-25 18:02:57 +0000183 if (TC.ShouldLinkCXXStdlib(Args))
184 TC.AddCXXStdlibLibArgs(Args, CmdArgs);
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000185 if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
Jonathan Roelofs901c7762017-05-25 15:42:13 +0000186 CmdArgs.push_back("-lc");
187 CmdArgs.push_back("-lm");
188
189 TC.AddLinkRuntimeLib(Args, CmdArgs);
190 }
191
192 CmdArgs.push_back("-o");
193 CmdArgs.push_back(Output.getFilename());
194
195 C.addCommand(llvm::make_unique<Command>(JA, *this,
196 Args.MakeArgString(TC.GetLinkerPath()),
197 CmdArgs, Inputs));
198}