Index: llvm/test/tools/llvm-elfabi/binary-read-neededlibs.test =================================================================== --- /dev/null +++ llvm/test/tools/llvm-elfabi/binary-read-neededlibs.test @@ -0,0 +1,55 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-elfabi %t --emit-tbe=- | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_DYN + Machine: EM_X86_64 +Sections: + - Name: .dynamic + Type: SHT_DYNAMIC + Flags: [ SHF_ALLOC ] + Address: 0x1000 + Content: "010000000000000001000000000000000e00000000000000150000000000000001000000000000000b000000000000000a0000000000000024000000000000000500000000000000d00200000000000000000000000000000000000000000000" + # DT_NEEDED 1 (0x01) + # DT_SONAME 21 (0x15) + # DT_NEEDED 11 (0x0b) + # DT_STRSZ 36 (0x24) + # DT_STRTAB 0x2d0 + # DT_NULL 0x0 + Size: 96 + - Name: .dynsym + Type: SHT_DYNSYM + Flags: [ SHF_ALLOC ] + Address: 0x2000 + EntSize: 24 + Size: 0 + Link: .dynstr + - Name: .dynstr + Type: SHT_STRTAB + Flags: [ SHF_ALLOC ] + Address: 0x3000 + # "\0libfoo.so\0libbar.so\0somelib.so\0foo\0" + Content: "006c6962666f6f2e736f006c69626261722e736f00736f6d656c69622e736f00666f6f00" +ProgramHeaders: + - Type: PT_LOAD + VAddr: 0x0000 + PAddr: 0x0000 + - Type: PT_DYNAMIC + Flags: [ PF_X, PF_R ] + VAddr: 0x1000 + PAddr: 0x1000 + Sections: + - Section: .dynamic + +# CHECK: --- !tapi-tbe +# CHECK-NEXT: TbeVersion: {{[1-9]\d*\.(0|([1-9]\d*))}} +# CHECK-NEXT: SoName: somelib.so +# CHECK-NEXT: Arch: x86_64 +# CHECK-NEXT: NeededLibs: +# CHECK-NEXT: - libfoo.so +# CHECK-NEXT: - libbar.so +# CHECK-NEXT: Symbols: {} +# CHECK-NEXT: ... Index: llvm/test/tools/llvm-elfabi/binary-read-soname-no-null.test =================================================================== --- llvm/test/tools/llvm-elfabi/binary-read-soname-no-null.test +++ llvm/test/tools/llvm-elfabi/binary-read-soname-no-null.test @@ -42,4 +42,4 @@ Sections: - Section: .dynamic -# CHECK: String overran bounds of string table (no null terminator) when reading SONAME +# CHECK: String overran bounds of string table (no null terminator) when reading DT_SONAME Index: llvm/tools/llvm-elfabi/ELFObjHandler.cpp =================================================================== --- llvm/tools/llvm-elfabi/ELFObjHandler.cpp +++ llvm/tools/llvm-elfabi/ELFObjHandler.cpp @@ -31,6 +31,7 @@ uint64_t StrTabOffset = 0; uint64_t StrSize = 0; Optional SONameOffset; + std::vector NeededLibNames; }; /// This function behaves similarly to StringRef::substr(), but attempts to @@ -99,6 +100,9 @@ if (Dyn.StrSize == 0) return createError("DT_STRSZ must be greater than zero"); break; + case DT_NEEDED: + Dyn.NeededLibNames.push_back(Entry.d_un.d_val); + break; } } @@ -114,6 +118,12 @@ return createError( "DT_SONAME string offset outside of dynamic string table"); } + for (uint64_t Offset : Dyn.NeededLibNames) { + if (Offset >= Dyn.StrSize){ + return createError( + "DT_NEEDED string offset outside of dynamic string table"); + } + } return Error::success(); } @@ -149,12 +159,21 @@ Expected NameOrErr = terminatedSubstr(DynStr, *DynEnt.SONameOffset); if (!NameOrErr) { - return appendToError(NameOrErr.takeError(), "when reading SONAME"); + return appendToError(NameOrErr.takeError(), "when reading DT_SONAME"); } DestStub->SoName = *NameOrErr; } - // TODO: Populate NeededLibs from .dynamic entries and linked string table. + // Populate NeededLibs from .dynamic entries and dynamic string table. + for (uint64_t NeededStrOffset : DynEnt.NeededLibNames) { + Expected LibNameOrErr = + terminatedSubstr(DynStr, NeededStrOffset); + if (!LibNameOrErr) { + return appendToError(LibNameOrErr.takeError(), "when reading DT_NEEDED"); + } + DestStub->NeededLibs.push_back(*LibNameOrErr); + } + // TODO: Populate Symbols from .dynsym table and linked string table. return std::move(DestStub);