2021-08-18 19:43:30 +08:00
|
|
|
//===-- clang-nvlink-wrapper/ClangNvlinkWrapper.cpp - wrapper over nvlink-===//
|
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===---------------------------------------------------------------------===//
|
|
|
|
///
|
|
|
|
/// \file
|
|
|
|
/// This tool works as a wrapper over nvlink program. It transparently passes
|
|
|
|
/// every input option and objects to nvlink except archive files. It reads
|
|
|
|
/// each input archive file to extract archived cubin files as temporary files.
|
|
|
|
/// These temp (*.cubin) files are passed to nvlink, because nvlink does not
|
|
|
|
/// support linking of archive files implicitly.
|
|
|
|
///
|
2021-09-03 19:14:22 +08:00
|
|
|
/// During linking of heterogeneous device archive libraries, the
|
2021-08-18 19:43:30 +08:00
|
|
|
/// clang-offload-bundler creates a device specific archive of cubin files.
|
|
|
|
/// Such an archive is then passed to this tool to extract cubin files before
|
|
|
|
/// passing to nvlink.
|
|
|
|
///
|
|
|
|
/// Example:
|
|
|
|
/// clang-nvlink-wrapper -o a.out-openmp-nvptx64 /tmp/libTest-nvptx-sm_50.a
|
|
|
|
///
|
|
|
|
/// 1. Extract (libTest-nvptx-sm_50.a) => /tmp/a.cubin /tmp/b.cubin
|
|
|
|
/// 2. nvlink -o a.out-openmp-nvptx64 /tmp/a.cubin /tmp/b.cubin
|
|
|
|
//===---------------------------------------------------------------------===//
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
#include "clang/Basic/Version.h"
|
2021-08-18 19:43:30 +08:00
|
|
|
#include "llvm/Object/Archive.h"
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Errc.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include "llvm/Support/Path.h"
|
|
|
|
#include "llvm/Support/Program.h"
|
|
|
|
#include "llvm/Support/Signals.h"
|
|
|
|
#include "llvm/Support/StringSaver.h"
|
|
|
|
#include "llvm/Support/WithColor.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
static cl::opt<bool> Help("h", cl::desc("Alias for -help"), cl::Hidden);
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
// Mark all our options with this category, everything else (except for -help)
|
|
|
|
// will be hidden.
|
|
|
|
static cl::OptionCategory
|
|
|
|
ClangNvlinkWrapperCategory("clang-nvlink-wrapper options");
|
|
|
|
|
|
|
|
static cl::opt<std::string> NvlinkUserPath("nvlink-path",
|
|
|
|
cl::desc("Path of nvlink binary"),
|
|
|
|
cl::cat(ClangNvlinkWrapperCategory));
|
|
|
|
|
|
|
|
// Do not parse nvlink options
|
|
|
|
static cl::list<std::string>
|
|
|
|
NVArgs(cl::Sink, cl::desc("<options to be passed to nvlink>..."));
|
|
|
|
|
2022-01-20 21:29:16 +08:00
|
|
|
static bool isEmptyFile(StringRef Filename) {
|
|
|
|
ErrorOr<std::unique_ptr<MemoryBuffer>> BufOrErr =
|
|
|
|
MemoryBuffer::getFileOrSTDIN(Filename, false, false);
|
|
|
|
if (std::error_code EC = BufOrErr.getError())
|
|
|
|
return false;
|
|
|
|
return (*BufOrErr)->getBuffer().empty();
|
|
|
|
}
|
|
|
|
|
2021-08-18 19:43:30 +08:00
|
|
|
static Error runNVLink(std::string NVLinkPath,
|
|
|
|
SmallVectorImpl<std::string> &Args) {
|
|
|
|
std::vector<StringRef> NVLArgs;
|
|
|
|
NVLArgs.push_back(NVLinkPath);
|
2022-01-20 21:29:16 +08:00
|
|
|
StringRef Output = *(llvm::find(Args, "-o") + 1);
|
2021-08-18 19:43:30 +08:00
|
|
|
for (auto &Arg : Args) {
|
2022-01-20 21:29:16 +08:00
|
|
|
if (!(sys::fs::exists(Arg) && Arg != Output && isEmptyFile(Arg)))
|
|
|
|
NVLArgs.push_back(Arg);
|
2021-08-18 19:43:30 +08:00
|
|
|
}
|
|
|
|
|
2021-12-27 05:31:40 +08:00
|
|
|
if (sys::ExecuteAndWait(NVLinkPath, NVLArgs))
|
2021-08-18 19:43:30 +08:00
|
|
|
return createStringError(inconvertibleErrorCode(), "'nvlink' failed");
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Error extractArchiveFiles(StringRef Filename,
|
|
|
|
SmallVectorImpl<std::string> &Args,
|
|
|
|
SmallVectorImpl<std::string> &TmpFiles) {
|
|
|
|
std::vector<std::unique_ptr<MemoryBuffer>> ArchiveBuffers;
|
|
|
|
|
|
|
|
ErrorOr<std::unique_ptr<MemoryBuffer>> BufOrErr =
|
2021-09-03 19:14:22 +08:00
|
|
|
MemoryBuffer::getFileOrSTDIN(Filename, false, false);
|
2021-08-18 19:43:30 +08:00
|
|
|
if (std::error_code EC = BufOrErr.getError())
|
|
|
|
return createFileError(Filename, EC);
|
|
|
|
|
|
|
|
ArchiveBuffers.push_back(std::move(*BufOrErr));
|
|
|
|
Expected<std::unique_ptr<llvm::object::Archive>> LibOrErr =
|
|
|
|
object::Archive::create(ArchiveBuffers.back()->getMemBufferRef());
|
|
|
|
if (!LibOrErr)
|
|
|
|
return LibOrErr.takeError();
|
|
|
|
|
|
|
|
auto Archive = std::move(*LibOrErr);
|
|
|
|
|
|
|
|
Error Err = Error::success();
|
|
|
|
auto ChildEnd = Archive->child_end();
|
|
|
|
for (auto ChildIter = Archive->child_begin(Err); ChildIter != ChildEnd;
|
|
|
|
++ChildIter) {
|
|
|
|
if (Err)
|
|
|
|
return Err;
|
|
|
|
auto ChildNameOrErr = (*ChildIter).getName();
|
|
|
|
if (!ChildNameOrErr)
|
|
|
|
return ChildNameOrErr.takeError();
|
|
|
|
|
|
|
|
StringRef ChildName = sys::path::filename(ChildNameOrErr.get());
|
|
|
|
|
|
|
|
auto ChildBufferRefOrErr = (*ChildIter).getMemoryBufferRef();
|
|
|
|
if (!ChildBufferRefOrErr)
|
|
|
|
return ChildBufferRefOrErr.takeError();
|
|
|
|
|
|
|
|
auto ChildBuffer =
|
|
|
|
MemoryBuffer::getMemBuffer(ChildBufferRefOrErr.get(), false);
|
|
|
|
auto ChildNameSplit = ChildName.split('.');
|
|
|
|
|
|
|
|
SmallString<16> Path;
|
|
|
|
int FileDesc;
|
|
|
|
if (std::error_code EC = sys::fs::createTemporaryFile(
|
|
|
|
(ChildNameSplit.first), (ChildNameSplit.second), FileDesc, Path))
|
|
|
|
return createFileError(ChildName, EC);
|
|
|
|
|
|
|
|
std::string TmpFileName(Path.str());
|
|
|
|
Args.push_back(TmpFileName);
|
|
|
|
TmpFiles.push_back(TmpFileName);
|
|
|
|
std::error_code EC;
|
|
|
|
raw_fd_ostream OS(Path.c_str(), EC, sys::fs::OF_None);
|
|
|
|
if (EC)
|
|
|
|
return createFileError(TmpFileName, errc::io_error);
|
|
|
|
OS << ChildBuffer->getBuffer();
|
|
|
|
OS.close();
|
|
|
|
}
|
|
|
|
return Err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Error cleanupTmpFiles(SmallVectorImpl<std::string> &TmpFiles) {
|
|
|
|
for (auto &TmpFile : TmpFiles) {
|
|
|
|
if (std::error_code EC = sys::fs::remove(TmpFile))
|
|
|
|
return createFileError(TmpFile, errc::no_such_file_or_directory);
|
|
|
|
}
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
static void PrintVersion(raw_ostream &OS) {
|
|
|
|
OS << clang::getClangToolFullVersion("clang-nvlink-wrapper") << '\n';
|
|
|
|
}
|
|
|
|
|
2021-08-18 19:43:30 +08:00
|
|
|
int main(int argc, const char **argv) {
|
|
|
|
sys::PrintStackTraceOnErrorSignal(argv[0]);
|
2021-10-09 20:06:11 +08:00
|
|
|
cl::SetVersionPrinter(PrintVersion);
|
|
|
|
cl::HideUnrelatedOptions(ClangNvlinkWrapperCategory);
|
|
|
|
cl::ParseCommandLineOptions(
|
|
|
|
argc, argv,
|
|
|
|
"A wrapper tool over nvlink program. It transparently passes every \n"
|
|
|
|
"input option and objects to nvlink except archive files and path of \n"
|
|
|
|
"nvlink binary. It reads each input archive file to extract archived \n"
|
|
|
|
"cubin files as temporary files.\n");
|
2021-08-18 19:43:30 +08:00
|
|
|
|
|
|
|
if (Help) {
|
|
|
|
cl::PrintHelpMessage();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto reportError = [argv](Error E) {
|
|
|
|
logAllUnhandledErrors(std::move(E), WithColor::error(errs(), argv[0]));
|
|
|
|
exit(1);
|
|
|
|
};
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
std::string NvlinkPath;
|
2021-08-18 19:43:30 +08:00
|
|
|
SmallVector<const char *, 0> Argv(argv, argv + argc);
|
|
|
|
SmallVector<std::string, 0> ArgvSubst;
|
|
|
|
SmallVector<std::string, 0> TmpFiles;
|
|
|
|
BumpPtrAllocator Alloc;
|
|
|
|
StringSaver Saver(Alloc);
|
|
|
|
cl::ExpandResponseFiles(Saver, cl::TokenizeGNUCommandLine, Argv);
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
for (const std::string &Arg : NVArgs) {
|
2021-08-18 19:43:30 +08:00
|
|
|
if (sys::path::extension(Arg) == ".a") {
|
|
|
|
if (Error Err = extractArchiveFiles(Arg, ArgvSubst, TmpFiles))
|
|
|
|
reportError(std::move(Err));
|
|
|
|
} else {
|
|
|
|
ArgvSubst.push_back(Arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-09 20:06:11 +08:00
|
|
|
NvlinkPath = NvlinkUserPath;
|
|
|
|
|
|
|
|
// If user hasn't specified nvlink binary then search it in PATH
|
|
|
|
if (NvlinkPath.empty()) {
|
|
|
|
ErrorOr<std::string> NvlinkPathErr = sys::findProgramByName("nvlink");
|
|
|
|
if (!NvlinkPathErr) {
|
|
|
|
reportError(createStringError(NvlinkPathErr.getError(),
|
|
|
|
"unable to find 'nvlink' in path"));
|
|
|
|
}
|
|
|
|
NvlinkPath = NvlinkPathErr.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Error Err = runNVLink(NvlinkPath, ArgvSubst))
|
2021-08-18 19:43:30 +08:00
|
|
|
reportError(std::move(Err));
|
|
|
|
if (Error Err = cleanupTmpFiles(TmpFiles))
|
|
|
|
reportError(std::move(Err));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|