BareMetal.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. //===-- BareMetal.cpp - Bare Metal ToolChain --------------------*- 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. #include "BareMetal.h"
  9. #include "CommonArgs.h"
  10. #include "InputInfo.h"
  11. #include "Gnu.h"
  12. #include "clang/Driver/Compilation.h"
  13. #include "clang/Driver/Driver.h"
  14. #include "clang/Driver/DriverDiagnostic.h"
  15. #include "clang/Driver/Options.h"
  16. #include "llvm/Option/ArgList.h"
  17. #include "llvm/Support/Path.h"
  18. #include "llvm/Support/VirtualFileSystem.h"
  19. #include "llvm/Support/raw_ostream.h"
  20. using namespace llvm::opt;
  21. using namespace clang;
  22. using namespace clang::driver;
  23. using namespace clang::driver::tools;
  24. using namespace clang::driver::toolchains;
  25. BareMetal::BareMetal(const Driver &D, const llvm::Triple &Triple,
  26. const ArgList &Args)
  27. : ToolChain(D, Triple, Args) {
  28. getProgramPaths().push_back(getDriver().getInstalledDir());
  29. if (getDriver().getInstalledDir() != getDriver().Dir)
  30. getProgramPaths().push_back(getDriver().Dir);
  31. }
  32. BareMetal::~BareMetal() {}
  33. /// Is the triple {arm,thumb}-none-none-{eabi,eabihf} ?
  34. static bool isARMBareMetal(const llvm::Triple &Triple) {
  35. if (Triple.getArch() != llvm::Triple::arm &&
  36. Triple.getArch() != llvm::Triple::thumb)
  37. return false;
  38. if (Triple.getVendor() != llvm::Triple::UnknownVendor)
  39. return false;
  40. if (Triple.getOS() != llvm::Triple::UnknownOS)
  41. return false;
  42. if (Triple.getEnvironment() != llvm::Triple::EABI &&
  43. Triple.getEnvironment() != llvm::Triple::EABIHF)
  44. return false;
  45. return true;
  46. }
  47. bool BareMetal::handlesTarget(const llvm::Triple &Triple) {
  48. return isARMBareMetal(Triple);
  49. }
  50. Tool *BareMetal::buildLinker() const {
  51. return new tools::baremetal::Linker(*this);
  52. }
  53. std::string BareMetal::getRuntimesDir() const {
  54. SmallString<128> Dir(getDriver().ResourceDir);
  55. llvm::sys::path::append(Dir, "lib", "baremetal");
  56. return Dir.str();
  57. }
  58. void BareMetal::AddClangSystemIncludeArgs(const ArgList &DriverArgs,
  59. ArgStringList &CC1Args) const {
  60. if (DriverArgs.hasArg(options::OPT_nostdinc))
  61. return;
  62. if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) {
  63. SmallString<128> Dir(getDriver().ResourceDir);
  64. llvm::sys::path::append(Dir, "include");
  65. addSystemInclude(DriverArgs, CC1Args, Dir.str());
  66. }
  67. if (!DriverArgs.hasArg(options::OPT_nostdlibinc)) {
  68. SmallString<128> Dir(getDriver().SysRoot);
  69. llvm::sys::path::append(Dir, "include");
  70. addSystemInclude(DriverArgs, CC1Args, Dir.str());
  71. }
  72. }
  73. void BareMetal::addClangTargetOptions(const ArgList &DriverArgs,
  74. ArgStringList &CC1Args,
  75. Action::OffloadKind) const {
  76. CC1Args.push_back("-nostdsysteminc");
  77. }
  78. void BareMetal::AddClangCXXStdlibIncludeArgs(
  79. const ArgList &DriverArgs, ArgStringList &CC1Args) const {
  80. if (DriverArgs.hasArg(options::OPT_nostdinc) ||
  81. DriverArgs.hasArg(options::OPT_nostdlibinc) ||
  82. DriverArgs.hasArg(options::OPT_nostdincxx))
  83. return;
  84. StringRef SysRoot = getDriver().SysRoot;
  85. if (SysRoot.empty())
  86. return;
  87. switch (GetCXXStdlibType(DriverArgs)) {
  88. case ToolChain::CST_Libcxx: {
  89. SmallString<128> Dir(SysRoot);
  90. llvm::sys::path::append(Dir, "include", "c++", "v1");
  91. addSystemInclude(DriverArgs, CC1Args, Dir.str());
  92. break;
  93. }
  94. case ToolChain::CST_Libstdcxx: {
  95. SmallString<128> Dir(SysRoot);
  96. llvm::sys::path::append(Dir, "include", "c++");
  97. std::error_code EC;
  98. Generic_GCC::GCCVersion Version = {"", -1, -1, -1, "", "", ""};
  99. // Walk the subdirs, and find the one with the newest gcc version:
  100. for (llvm::vfs::directory_iterator
  101. LI = getDriver().getVFS().dir_begin(Dir.str(), EC),
  102. LE;
  103. !EC && LI != LE; LI = LI.increment(EC)) {
  104. StringRef VersionText = llvm::sys::path::filename(LI->path());
  105. auto CandidateVersion = Generic_GCC::GCCVersion::Parse(VersionText);
  106. if (CandidateVersion.Major == -1)
  107. continue;
  108. if (CandidateVersion <= Version)
  109. continue;
  110. Version = CandidateVersion;
  111. }
  112. if (Version.Major == -1)
  113. return;
  114. llvm::sys::path::append(Dir, Version.Text);
  115. addSystemInclude(DriverArgs, CC1Args, Dir.str());
  116. break;
  117. }
  118. }
  119. }
  120. void BareMetal::AddCXXStdlibLibArgs(const ArgList &Args,
  121. ArgStringList &CmdArgs) const {
  122. switch (GetCXXStdlibType(Args)) {
  123. case ToolChain::CST_Libcxx:
  124. CmdArgs.push_back("-lc++");
  125. CmdArgs.push_back("-lc++abi");
  126. break;
  127. case ToolChain::CST_Libstdcxx:
  128. CmdArgs.push_back("-lstdc++");
  129. CmdArgs.push_back("-lsupc++");
  130. break;
  131. }
  132. CmdArgs.push_back("-lunwind");
  133. }
  134. void BareMetal::AddLinkRuntimeLib(const ArgList &Args,
  135. ArgStringList &CmdArgs) const {
  136. CmdArgs.push_back(Args.MakeArgString("-lclang_rt.builtins-" +
  137. getTriple().getArchName() + ".a"));
  138. }
  139. void baremetal::Linker::ConstructJob(Compilation &C, const JobAction &JA,
  140. const InputInfo &Output,
  141. const InputInfoList &Inputs,
  142. const ArgList &Args,
  143. const char *LinkingOutput) const {
  144. ArgStringList CmdArgs;
  145. auto &TC = static_cast<const toolchains::BareMetal&>(getToolChain());
  146. AddLinkerInputs(TC, Inputs, Args, CmdArgs, JA);
  147. CmdArgs.push_back("-Bstatic");
  148. CmdArgs.push_back(Args.MakeArgString("-L" + TC.getRuntimesDir()));
  149. Args.AddAllArgs(CmdArgs, {options::OPT_L, options::OPT_T_Group,
  150. options::OPT_e, options::OPT_s, options::OPT_t,
  151. options::OPT_Z_Flag, options::OPT_r});
  152. if (TC.ShouldLinkCXXStdlib(Args))
  153. TC.AddCXXStdlibLibArgs(Args, CmdArgs);
  154. if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
  155. CmdArgs.push_back("-lc");
  156. CmdArgs.push_back("-lm");
  157. TC.AddLinkRuntimeLib(Args, CmdArgs);
  158. }
  159. CmdArgs.push_back("-o");
  160. CmdArgs.push_back(Output.getFilename());
  161. C.addCommand(std::make_unique<Command>(JA, *this,
  162. Args.MakeArgString(TC.GetLinkerPath()),
  163. CmdArgs, Inputs));
  164. }