13#include "llvm/ADT/StringExtras.h"
14#include "llvm/ADT/StringRef.h"
15#include "llvm/Object/Archive.h"
16#include "llvm/Object/ObjectFile.h"
17#include "llvm/Support/MD5.h"
18#include "llvm/Support/MemoryBuffer.h"
19#include "llvm/Support/Path.h"
20#include "llvm/Support/raw_ostream.h"
21#include "llvm/TargetParser/Triple.h"
31#if defined(_WIN32) || defined(_WIN64)
32#define NULL_FILE "nul"
34#define NULL_FILE "/dev/null"
38const unsigned HIPCodeObjectAlign = 4096;
44 return HasTargetID ? (
T.getArchName() +
"-" +
T.getVendorName() +
"-" +
45 T.getOSName() +
"-" +
T.getEnvironmentName())
47 :
T.normalize(llvm::Triple::CanonicalForm::FOUR_IDENT);
55 const llvm::opt::ArgList &Args_)
57 DiagID(C.getDriver().getDiags().getCustomDiagID(
59 "Error collecting HIP undefined fatbin symbols: %0")),
60 Quiet(C.getArgs().hasArg(
options::OPT__HASH_HASH_HASH)),
61 Verbose(C.getArgs().hasArg(
options::OPT_v)) {
65 for (
const auto &Name : FatBinSymbols)
66 llvm::errs() <<
"Found undefined HIP fatbin symbol: " << Name <<
"\n";
67 for (
const auto &Name : GPUBinHandleSymbols)
68 llvm::errs() <<
"Found undefined HIP gpubin handle symbol: " << Name
78 return GPUBinHandleSymbols;
86 llvm::Triple Triple(C.getDriver().getTargetTriple());
87 bool IsMSVC = Triple.isWindowsMSVCEnvironment();
88 llvm::StringRef Ext = IsMSVC ?
".lib" :
".a";
90 for (
const auto *Arg : Args.filtered(options::OPT_l)) {
91 llvm::StringRef
Value = Arg->getValue();
92 if (
Value.starts_with(
":"))
93 ExactLibNames.push_back(
Value.drop_front());
95 LibNames.push_back(
Value);
97 for (
const auto *Arg : Args.filtered(options::OPT_L)) {
98 auto Path = Arg->getValue();
99 LibPaths.push_back(Path);
101 llvm::errs() <<
"HIP fatbin symbol search uses library path: " << Path
105 auto ProcessLib = [&](llvm::StringRef LibName,
bool IsExact) {
107 IsExact ? Twine(LibName).str()
108 : IsMSVC ? (Twine(LibName) + Ext).str()
109 : (Twine(
"lib") + LibName + Ext).str());
112 for (
const auto Path : LibPaths) {
114 llvm::sys::path::append(FullPath, FullLibName);
116 if (llvm::sys::fs::exists(FullPath)) {
118 llvm::errs() <<
"HIP fatbin symbol search found library: "
120 auto BufferOrErr = llvm::MemoryBuffer::getFile(FullPath);
122 errorHandler(llvm::errorCodeToError(BufferOrErr.getError()));
125 processInput(BufferOrErr.get()->getMemBufferRef());
130 if (!
Found && Verbose)
131 llvm::errs() <<
"HIP fatbin symbol search could not find library: "
132 << FullLibName <<
"\n";
135 for (
const auto LibName : ExactLibNames)
136 ProcessLib(LibName,
true);
138 for (
const auto LibName : LibNames)
139 ProcessLib(LibName,
false);
144 const llvm::opt::ArgList &Args;
148 std::set<std::string> FatBinSymbols;
149 std::set<std::string> GPUBinHandleSymbols;
150 std::set<std::string, std::less<>> DefinedFatBinSymbols;
151 std::set<std::string, std::less<>> DefinedGPUBinHandleSymbols;
152 const std::string FatBinPrefix =
"__hip_fatbin";
153 const std::string GPUBinHandlePrefix =
"__hip_gpubin_handle";
155 void populateSymbols() {
156 std::deque<const Action *> WorkList;
157 std::set<const Action *> Visited;
159 for (
const auto &
Action :
C.getActions())
160 WorkList.push_back(
Action);
162 while (!WorkList.empty()) {
163 const Action *CurrentAction = WorkList.front();
164 WorkList.pop_front();
166 if (!CurrentAction || !Visited.insert(CurrentAction).second)
169 if (
const auto *IA = dyn_cast<InputAction>(CurrentAction)) {
170 std::string ID = IA->getId().str();
172 ID = llvm::utohexstr(llvm::MD5Hash(ID),
true);
173 FatBinSymbols.insert((FatBinPrefix + Twine(
'_') + ID).str());
174 GPUBinHandleSymbols.insert(
175 (GPUBinHandlePrefix + Twine(
'_') + ID).str());
178 if (IA->getInputArg().getNumValues() == 0)
180 const char *Filename = IA->getInputArg().getValue();
183 auto BufferOrErr = llvm::MemoryBuffer::getFile(Filename);
190 processInput(BufferOrErr.get()->getMemBufferRef());
192 llvm::append_range(WorkList, CurrentAction->
getInputs());
196 void processInput(
const llvm::MemoryBufferRef &Buffer) {
198 auto ObjFileOrErr = llvm::object::ObjectFile::createObjectFile(Buffer);
200 processSymbols(**ObjFileOrErr);
205 llvm::consumeError(ObjFileOrErr.takeError());
206 auto ArchiveOrErr = llvm::object::Archive::create(Buffer);
208 llvm::Error Err = llvm::Error::success();
209 llvm::object::Archive &Archive = *ArchiveOrErr.get();
210 for (
auto &Child : Archive.children(Err)) {
211 auto ChildBufOrErr = Child.getMemoryBufferRef();
213 processInput(*ChildBufOrErr);
215 errorHandler(ChildBufOrErr.takeError());
219 errorHandler(std::move(Err));
224 llvm::consumeError(ArchiveOrErr.takeError());
227 void processSymbols(
const llvm::object::ObjectFile &Obj) {
228 for (
const auto &Symbol : Obj.symbols()) {
229 auto FlagOrErr = Symbol.getFlags();
231 errorHandler(FlagOrErr.takeError());
235 auto NameOrErr = Symbol.getName();
237 errorHandler(NameOrErr.takeError());
240 llvm::StringRef Name = *NameOrErr;
243 FlagOrErr.get() & llvm::object::SymbolRef::SF_Undefined;
244 bool isHidden = FlagOrErr.get() & llvm::object::SymbolRef::SF_Hidden;
245 bool isFatBinSymbol = Name.starts_with(FatBinPrefix);
246 bool isGPUBinHandleSymbol = Name.starts_with(GPUBinHandlePrefix);
250 if (isFatBinSymbol &&
251 DefinedFatBinSymbols.find(Name) == DefinedFatBinSymbols.end())
252 FatBinSymbols.insert(Name.str());
253 else if (isGPUBinHandleSymbol &&
254 DefinedGPUBinHandleSymbols.find(Name) ==
255 DefinedGPUBinHandleSymbols.end())
256 GPUBinHandleSymbols.insert(Name.str());
265 if (isFatBinSymbol) {
266 DefinedFatBinSymbols.insert(Name.str());
267 FatBinSymbols.erase(Name.str());
268 }
else if (isGPUBinHandleSymbol) {
269 DefinedGPUBinHandleSymbols.insert(Name.str());
270 GPUBinHandleSymbols.erase(Name.str());
275 void errorHandler(llvm::Error Err) {
278 C.getDriver().Diag(DiagID) << llvm::toString(std::move(Err));
285 llvm::StringRef OutputFileName,
287 const llvm::opt::ArgList &Args,
291 ArgStringList BundlerArgs;
292 BundlerArgs.push_back(Args.MakeArgString(
"-type=o"));
293 BundlerArgs.push_back(
294 Args.MakeArgString(
"-bundle-align=" + Twine(HIPCodeObjectAlign)));
298 std::string BundlerTargetArg =
"-targets=host-x86_64-unknown-linux-gnu";
303 std::string OffloadKind =
"hip";
304 auto &TT =
T.getToolChain().getTriple();
306 OffloadKind = OffloadKind +
"v4";
307 for (
const auto &II : Inputs) {
308 const auto *A = II.getAction();
309 auto ArchStr = llvm::StringRef(A->getOffloadingArch());
310 BundlerTargetArg +=
',' + OffloadKind +
'-';
311 if (ArchStr ==
"amdgcnspirv")
316 if (!ArchStr.empty())
317 BundlerTargetArg +=
'-' + ArchStr.str();
319 BundlerArgs.push_back(Args.MakeArgString(BundlerTargetArg));
322 std::string BundlerInputArg =
"-input=" NULL_FILE;
323 BundlerArgs.push_back(Args.MakeArgString(BundlerInputArg));
324 for (
const auto &II : Inputs) {
325 BundlerInputArg = std::string(
"-input=") + II.getFilename();
326 BundlerArgs.push_back(Args.MakeArgString(BundlerInputArg));
329 std::string Output = std::string(OutputFileName);
330 auto *BundlerOutputArg =
331 Args.MakeArgString(std::string(
"-output=").append(Output));
332 BundlerArgs.push_back(BundlerOutputArg);
336 const char *Bundler = Args.MakeArgString(
337 T.getToolChain().GetProgramPath(
"clang-offload-bundler"));
338 C.addCommand(std::make_unique<Command>(
340 InputInfo(&JA, Args.MakeArgString(Output))));
350 const Driver &D =
C.getDriver();
351 std::string Name = std::string(llvm::sys::path::stem(Output.
getFilename()));
356 const char *ObjinFile;
357 const char *BundleFile;
359 ObjinFile =
C.getArgs().MakeArgString(Name +
".mcin");
360 BundleFile =
C.getArgs().MakeArgString(Name +
".hipfb");
363 ObjinFile =
C.addTempFile(
C.getArgs().MakeArgString(TmpNameMcin));
365 BundleFile =
C.addTempFile(
C.getArgs().MakeArgString(TmpNameFb));
370 std::string ObjBuffer;
371 llvm::raw_string_ostream ObjStream(ObjBuffer);
378 std::string PrimaryHipFatbinSymbol;
379 std::string PrimaryGpuBinHandleSymbol;
380 bool FoundPrimaryHipFatbinSymbol =
false;
381 bool FoundPrimaryGpuBinHandleSymbol =
false;
383 std::vector<std::string> AliasHipFatbinSymbols;
384 std::vector<std::string> AliasGpuBinHandleSymbols;
388 for (
const auto &Symbol : Symbols.getFatBinSymbols()) {
389 if (!FoundPrimaryHipFatbinSymbol) {
390 PrimaryHipFatbinSymbol = Symbol;
391 FoundPrimaryHipFatbinSymbol =
true;
393 AliasHipFatbinSymbols.push_back(Symbol);
396 for (
const auto &Symbol : Symbols.getGPUBinHandleSymbols()) {
397 if (!FoundPrimaryGpuBinHandleSymbol) {
398 PrimaryGpuBinHandleSymbol = Symbol;
399 FoundPrimaryGpuBinHandleSymbol =
true;
401 AliasGpuBinHandleSymbols.push_back(Symbol);
408 ObjStream <<
"# HIP Object Generator\n";
409 ObjStream <<
"# *** Automatically generated by Clang ***\n";
410 if (FoundPrimaryGpuBinHandleSymbol) {
412 if (HostTriple.isWindowsMSVCEnvironment())
413 ObjStream <<
" .section .hip_gpubin_handle,\"dw\"\n";
415 ObjStream <<
" .protected " << PrimaryGpuBinHandleSymbol <<
"\n";
416 ObjStream <<
" .type " << PrimaryGpuBinHandleSymbol <<
",@object\n";
417 ObjStream <<
" .section .hip_gpubin_handle,\"aw\"\n";
419 ObjStream <<
" .globl " << PrimaryGpuBinHandleSymbol <<
"\n";
420 ObjStream <<
" .p2align 3\n";
421 ObjStream << PrimaryGpuBinHandleSymbol <<
":\n";
422 ObjStream <<
" .zero 8\n";
425 for (
const auto &AliasSymbol : AliasGpuBinHandleSymbols) {
426 ObjStream <<
" .globl " << AliasSymbol <<
"\n";
427 ObjStream <<
" .set " << AliasSymbol <<
"," << PrimaryGpuBinHandleSymbol
431 if (FoundPrimaryHipFatbinSymbol) {
433 if (HostTriple.isWindowsMSVCEnvironment())
434 ObjStream <<
" .section .hip_fatbin,\"dw\"\n";
436 ObjStream <<
" .protected " << PrimaryHipFatbinSymbol <<
"\n";
437 ObjStream <<
" .type " << PrimaryHipFatbinSymbol <<
",@object\n";
438 ObjStream <<
" .section .hip_fatbin,\"a\",@progbits\n";
440 ObjStream <<
" .globl " << PrimaryHipFatbinSymbol <<
"\n";
441 ObjStream <<
" .p2align " << llvm::Log2(llvm::Align(HIPCodeObjectAlign))
444 for (
const auto &AliasSymbol : AliasHipFatbinSymbols) {
445 ObjStream <<
" .globl " << AliasSymbol <<
"\n";
446 ObjStream <<
" .set " << AliasSymbol <<
"," << PrimaryHipFatbinSymbol
449 ObjStream << PrimaryHipFatbinSymbol <<
":\n";
450 ObjStream <<
" .incbin ";
451 llvm::sys::printArg(ObjStream, BundleFile,
true);
454 if (HostTriple.isOSLinux() && HostTriple.isOSBinFormatELF())
455 ObjStream <<
" .section .note.GNU-stack, \"\", @progbits\n";
459 if (
C.getArgs().hasArg(options::OPT_fhip_dump_offload_linker_script))
460 llvm::errs() << ObjBuffer;
464 llvm::raw_fd_ostream Objf(ObjinFile, EC, llvm::sys::fs::OF_None);
467 D.
Diag(clang::diag::err_unable_to_make_temp) << EC.message();
473 ArgStringList ClangArgs{
"-target", Args.MakeArgString(HostTriple.normalize()),
485 StringRef Extension) {
486 if (
C.getDriver().isSaveTempsEnabled()) {
487 return C.getArgs().MakeArgString(Prefix +
"." + Extension);
489 auto TmpFile =
C.getDriver().GetTemporaryPath(Prefix, Extension);
490 return C.addTempFile(
C.getArgs().MakeArgString(TmpFile));
static LLVM_ATTRIBUTE_USED bool isHidden(const CheckerRegistryData &Registry, StringRef CheckerName)
static std::string normalizeForBundler(const llvm::Triple &T, bool HasTargetID)
const std::set< std::string > & getGPUBinHandleSymbols() const
void processStaticLibraries()
HIPUndefinedFatBinSymbols(const Compilation &C, const llvm::opt::ArgList &Args_)
const std::set< std::string > & getFatBinSymbols() const
Concrete class used by the front-end to report problems and issues.
Action - Represent an abstract compilation step to perform.
Compilation - A set of tasks to perform for a single driver invocation.
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
bool isSaveTempsEnabled() const
const char * getPrependArg() const
const char * getClangProgramPath() const
Get the path to the main clang executable.
DiagnosticBuilder Diag(unsigned DiagID) const
std::string GetTemporaryPath(StringRef Prefix, StringRef Suffix) const
GetTemporaryPath - Return the pathname of a temporary file to use as part of compilation; the file wi...
SmallVector< InputInfo, 4 > InputInfoList
The JSON file list parser is used to communicate input to InstallAPI.
const FunctionProtoType * T
static constexpr ResponseFileSupport None()
Returns a ResponseFileSupport indicating that response files are not supported.