Skip to content

Commit 64f0681

Browse files
authored
[Libomptarget] Rework image checking further (#76120)
Summary: In the future, we may have more checks for different kinds of inputs, e.g. SPIR-V. This patch simply reworks the handling to be more generic and do the magic detection up-front. The checks inside the routines are now asserts so we don't spend time checking this stuff over and over again. This patch also tweaked the bitcode check. I used a different function to get the Lazy-IR module now, as it returns the raw expected value rather than the SM diganostic. No functionality change intended.
1 parent 1da9d8a commit 64f0681

File tree

5 files changed

+42
-41
lines changed

5 files changed

+42
-41
lines changed

openmp/libomptarget/plugins-nextgen/common/include/JIT.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ struct JITEngine {
5757

5858
/// Return true if \p Image is a bitcode image that can be JITed for the given
5959
/// architecture.
60-
bool checkBitcodeImage(const __tgt_device_image &Image);
60+
Expected<bool> checkBitcodeImage(StringRef Buffer) const;
6161

6262
private:
6363
/// Compile the bitcode image \p Image and generate the binary image that can

openmp/libomptarget/plugins-nextgen/common/include/PluginInterface.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -1067,7 +1067,7 @@ struct GenericPluginTy {
10671067

10681068
/// Top level interface to verify if a given ELF image can be executed on a
10691069
/// given target. Returns true if the \p Image is compatible with the plugin.
1070-
Expected<bool> checkELFImage(__tgt_device_image &Image) const;
1070+
Expected<bool> checkELFImage(StringRef Image) const;
10711071

10721072
/// Indicate if an image is compatible with the plugin devices. Notice that
10731073
/// this function may be called before actually initializing the devices. So

openmp/libomptarget/plugins-nextgen/common/src/JIT.cpp

+9-15
Original file line numberDiff line numberDiff line change
@@ -330,24 +330,18 @@ JITEngine::process(const __tgt_device_image &Image,
330330
return &Image;
331331
}
332332

333-
bool JITEngine::checkBitcodeImage(const __tgt_device_image &Image) {
333+
Expected<bool> JITEngine::checkBitcodeImage(StringRef Buffer) const {
334334
TimeTraceScope TimeScope("Check bitcode image");
335335

336-
if (!isImageBitcode(Image))
337-
return false;
338-
339-
StringRef Data(reinterpret_cast<const char *>(Image.ImageStart),
340-
target::getPtrDiff(Image.ImageEnd, Image.ImageStart));
341-
auto MB = MemoryBuffer::getMemBuffer(Data, /*BufferName=*/"",
342-
/*RequiresNullTerminator=*/false);
343-
if (!MB)
344-
return false;
336+
assert(identify_magic(Buffer) == file_magic::bitcode &&
337+
"Input is not bitcode");
345338

346339
LLVMContext Context;
347-
SMDiagnostic Diagnostic;
348-
std::unique_ptr<Module> M =
349-
llvm::getLazyIRModule(std::move(MB), Diagnostic, Context,
350-
/*ShouldLazyLoadMetadata=*/true);
340+
auto ModuleOrErr = getLazyBitcodeModule(MemoryBufferRef(Buffer, ""), Context,
341+
/*ShouldLazyLoadMetadata=*/true);
342+
if (!ModuleOrErr)
343+
return ModuleOrErr.takeError();
344+
Module &M = **ModuleOrErr;
351345

352-
return M && Triple(M->getTargetTriple()).getArch() == TT.getArch();
346+
return Triple(M.getTargetTriple()).getArch() == TT.getArch();
353347
}

openmp/libomptarget/plugins-nextgen/common/src/PluginInterface.cpp

+30-22
Original file line numberDiff line numberDiff line change
@@ -1632,24 +1632,21 @@ Error GenericPluginTy::deinitDevice(int32_t DeviceId) {
16321632
return Plugin::success();
16331633
}
16341634

1635-
Expected<bool> GenericPluginTy::checkELFImage(__tgt_device_image &Image) const {
1636-
StringRef Buffer(reinterpret_cast<const char *>(Image.ImageStart),
1637-
target::getPtrDiff(Image.ImageEnd, Image.ImageStart));
1638-
1635+
Expected<bool> GenericPluginTy::checkELFImage(StringRef Image) const {
16391636
// First check if this image is a regular ELF file.
1640-
if (!utils::elf::isELF(Buffer))
1637+
if (!utils::elf::isELF(Image))
16411638
return false;
16421639

16431640
// Check if this image is an ELF with a matching machine value.
1644-
auto MachineOrErr = utils::elf::checkMachine(Buffer, getMagicElfBits());
1641+
auto MachineOrErr = utils::elf::checkMachine(Image, getMagicElfBits());
16451642
if (!MachineOrErr)
16461643
return MachineOrErr.takeError();
16471644

16481645
if (!*MachineOrErr)
16491646
return false;
16501647

16511648
// Perform plugin-dependent checks for the specific architecture if needed.
1652-
return isELFCompatible(Buffer);
1649+
return isELFCompatible(Image);
16531650
}
16541651

16551652
const bool llvm::omp::target::plugin::libomptargetSupportsRPC() {
@@ -1678,27 +1675,38 @@ int32_t __tgt_rtl_init_plugin() {
16781675
return OFFLOAD_SUCCESS;
16791676
}
16801677

1681-
int32_t __tgt_rtl_is_valid_binary(__tgt_device_image *TgtImage) {
1682-
// TODO: We should be able to perform a trivial ELF machine check without
1683-
// initializing the plugin first to save time if the plugin is not needed.
1678+
int32_t __tgt_rtl_is_valid_binary(__tgt_device_image *Image) {
16841679
if (!Plugin::isActive())
16851680
return false;
16861681

1687-
// Check if this is a valid ELF with a matching machine and processor.
1688-
auto MatchOrErr = Plugin::get().checkELFImage(*TgtImage);
1689-
if (Error Err = MatchOrErr.takeError()) {
1682+
StringRef Buffer(reinterpret_cast<const char *>(Image->ImageStart),
1683+
target::getPtrDiff(Image->ImageEnd, Image->ImageStart));
1684+
1685+
auto HandleError = [&](Error Err) -> bool {
16901686
[[maybe_unused]] std::string ErrStr = toString(std::move(Err));
1691-
DP("Failure to check validity of image %p: %s", TgtImage, ErrStr.c_str());
1687+
DP("Failure to check validity of image %p: %s", Image, ErrStr.c_str());
1688+
return false;
1689+
};
1690+
switch (identify_magic(Buffer)) {
1691+
case file_magic::elf:
1692+
case file_magic::elf_relocatable:
1693+
case file_magic::elf_executable:
1694+
case file_magic::elf_shared_object:
1695+
case file_magic::elf_core: {
1696+
auto MatchOrErr = Plugin::get().checkELFImage(Buffer);
1697+
if (Error Err = MatchOrErr.takeError())
1698+
return HandleError(std::move(Err));
1699+
return *MatchOrErr;
1700+
}
1701+
case file_magic::bitcode: {
1702+
auto MatchOrErr = Plugin::get().getJIT().checkBitcodeImage(Buffer);
1703+
if (Error Err = MatchOrErr.takeError())
1704+
return HandleError(std::move(Err));
1705+
return *MatchOrErr;
1706+
}
1707+
default:
16921708
return false;
1693-
} else if (*MatchOrErr) {
1694-
return true;
16951709
}
1696-
1697-
// Check if this is a valid LLVM-IR file with matching triple.
1698-
if (Plugin::get().getJIT().checkBitcodeImage(*TgtImage))
1699-
return true;
1700-
1701-
return false;
17021710
}
17031711

17041712
int32_t __tgt_rtl_supports_empty_images() {

openmp/libomptarget/plugins-nextgen/common/src/Utils/ELF.cpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -37,8 +37,7 @@ bool utils::elf::isELF(StringRef Buffer) {
3737
}
3838

3939
Expected<bool> utils::elf::checkMachine(StringRef Object, uint16_t EMachine) {
40-
if (!isELF(Object))
41-
return createError("Input is not an ELF.");
40+
assert(isELF(Object) && "Input is not an ELF!");
4241

4342
Expected<ELF64LEObjectFile> ElfOrErr =
4443
ELF64LEObjectFile::create(MemoryBufferRef(Object, /*Identifier=*/""),

0 commit comments

Comments
 (0)