1efddda3dSPavel Labath //===-- TestObjectFileELF.cpp -----------------------------------*- C++ -*-===// 2efddda3dSPavel Labath // 3efddda3dSPavel Labath // 42946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 52946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 62946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 7efddda3dSPavel Labath // 8efddda3dSPavel Labath //===----------------------------------------------------------------------===// 9efddda3dSPavel Labath 10efddda3dSPavel Labath #include "Plugins/ObjectFile/ELF/ObjectFileELF.h" 117ed3e22fSPavel Labath #include "Plugins/SymbolVendor/ELF/SymbolVendorELF.h" 12e2867bc4SPavel Labath #include "TestingSupport/TestUtilities.h" 13efddda3dSPavel Labath #include "lldb/Core/Module.h" 14efddda3dSPavel Labath #include "lldb/Core/ModuleSpec.h" 15efddda3dSPavel Labath #include "lldb/Core/Section.h" 1646376966SJonas Devlieghere #include "lldb/Host/FileSystem.h" 17efddda3dSPavel Labath #include "lldb/Host/HostInfo.h" 186868d2ddSNathan Lanza #include "lldb/Utility/DataBufferHeap.h" 19e7d85688SVedant Kumar #include "llvm/ADT/Optional.h" 20e2867bc4SPavel Labath #include "llvm/Support/Compression.h" 21efddda3dSPavel Labath #include "llvm/Support/FileUtilities.h" 22efddda3dSPavel Labath #include "llvm/Support/Path.h" 23efddda3dSPavel Labath #include "llvm/Support/Program.h" 24efddda3dSPavel Labath #include "llvm/Support/raw_ostream.h" 25*8a777920SGreg Clayton #include "llvm/Testing/Support/Error.h" 26efddda3dSPavel Labath #include "gtest/gtest.h" 27efddda3dSPavel Labath 28efddda3dSPavel Labath using namespace lldb_private; 29efddda3dSPavel Labath using namespace lldb; 30efddda3dSPavel Labath 31efddda3dSPavel Labath class ObjectFileELFTest : public testing::Test { 32efddda3dSPavel Labath public: 33efddda3dSPavel Labath void SetUp() override { 3446376966SJonas Devlieghere FileSystem::Initialize(); 35efddda3dSPavel Labath HostInfo::Initialize(); 36efddda3dSPavel Labath ObjectFileELF::Initialize(); 37efddda3dSPavel Labath SymbolVendorELF::Initialize(); 38efddda3dSPavel Labath } 39efddda3dSPavel Labath 40efddda3dSPavel Labath void TearDown() override { 41efddda3dSPavel Labath SymbolVendorELF::Terminate(); 42efddda3dSPavel Labath ObjectFileELF::Terminate(); 43efddda3dSPavel Labath HostInfo::Terminate(); 4446376966SJonas Devlieghere FileSystem::Terminate(); 45efddda3dSPavel Labath } 46efddda3dSPavel Labath 47efddda3dSPavel Labath protected: 48efddda3dSPavel Labath }; 49efddda3dSPavel Labath 50efddda3dSPavel Labath TEST_F(ObjectFileELFTest, SectionsResolveConsistently) { 517ed3e22fSPavel Labath llvm::SmallString<128> obj; 52efddda3dSPavel Labath ASSERT_NO_ERROR(llvm::sys::fs::createTemporaryFile( 53efddda3dSPavel Labath "sections-resolve-consistently-%%%%%%", "obj", obj)); 54efddda3dSPavel Labath llvm::FileRemover remover(obj); 55*8a777920SGreg Clayton ASSERT_THAT_ERROR( 56*8a777920SGreg Clayton ReadYAMLObjectFile("sections-resolve-consistently.yaml", obj), 57*8a777920SGreg Clayton llvm::Succeeded()); 58efddda3dSPavel Labath 598f3be7a3SJonas Devlieghere ModuleSpec spec{FileSpec(obj)}; 608f3be7a3SJonas Devlieghere spec.GetSymbolFileSpec().SetFile(obj, FileSpec::Style::native); 61efddda3dSPavel Labath auto module_sp = std::make_shared<Module>(spec); 62efddda3dSPavel Labath SectionList *list = module_sp->GetSectionList(); 63efddda3dSPavel Labath ASSERT_NE(nullptr, list); 64efddda3dSPavel Labath 65efddda3dSPavel Labath auto bss_sp = list->FindSectionByName(ConstString(".bss")); 66efddda3dSPavel Labath ASSERT_NE(nullptr, bss_sp); 67efddda3dSPavel Labath auto data_sp = list->FindSectionByName(ConstString(".data")); 68efddda3dSPavel Labath ASSERT_NE(nullptr, data_sp); 69efddda3dSPavel Labath auto text_sp = list->FindSectionByName(ConstString(".text")); 70efddda3dSPavel Labath ASSERT_NE(nullptr, text_sp); 71efddda3dSPavel Labath 72efddda3dSPavel Labath const Symbol *X = module_sp->FindFirstSymbolWithNameAndType(ConstString("X"), 73efddda3dSPavel Labath eSymbolTypeAny); 74efddda3dSPavel Labath ASSERT_NE(nullptr, X); 75efddda3dSPavel Labath EXPECT_EQ(bss_sp, X->GetAddress().GetSection()); 76efddda3dSPavel Labath 77efddda3dSPavel Labath const Symbol *Y = module_sp->FindFirstSymbolWithNameAndType(ConstString("Y"), 78efddda3dSPavel Labath eSymbolTypeAny); 79efddda3dSPavel Labath ASSERT_NE(nullptr, Y); 80efddda3dSPavel Labath EXPECT_EQ(data_sp, Y->GetAddress().GetSection()); 81efddda3dSPavel Labath 82efddda3dSPavel Labath const Symbol *start = module_sp->FindFirstSymbolWithNameAndType( 83efddda3dSPavel Labath ConstString("_start"), eSymbolTypeAny); 84efddda3dSPavel Labath ASSERT_NE(nullptr, start); 85efddda3dSPavel Labath EXPECT_EQ(text_sp, start->GetAddress().GetSection()); 86efddda3dSPavel Labath } 874f033125SPavel Labath 884f033125SPavel Labath // Test that GetModuleSpecifications works on an "atypical" object file which 894f033125SPavel Labath // has section headers right after the ELF header (instead of the more common 904f033125SPavel Labath // layout where the section headers are at the very end of the object file). 91bca95f8aSPavel Labath // 92bca95f8aSPavel Labath // Test file generated with yaml2obj (@svn rev 324254) from the following input: 93bca95f8aSPavel Labath /* 94bca95f8aSPavel Labath --- !ELF 95bca95f8aSPavel Labath FileHeader: 96bca95f8aSPavel Labath Class: ELFCLASS64 97bca95f8aSPavel Labath Data: ELFDATA2LSB 98bca95f8aSPavel Labath Type: ET_EXEC 99bca95f8aSPavel Labath Machine: EM_X86_64 100bca95f8aSPavel Labath Entry: 0x00000000004003D0 101bca95f8aSPavel Labath Sections: 102bca95f8aSPavel Labath - Name: .note.gnu.build-id 103bca95f8aSPavel Labath Type: SHT_NOTE 104bca95f8aSPavel Labath Flags: [ SHF_ALLOC ] 105bca95f8aSPavel Labath Address: 0x0000000000400274 106bca95f8aSPavel Labath AddressAlign: 0x0000000000000004 107bca95f8aSPavel Labath Content: 040000001400000003000000474E55001B8A73AC238390E32A7FF4AC8EBE4D6A41ECF5C9 108bca95f8aSPavel Labath - Name: .text 109bca95f8aSPavel Labath Type: SHT_PROGBITS 110bca95f8aSPavel Labath Flags: [ SHF_ALLOC, SHF_EXECINSTR ] 111bca95f8aSPavel Labath Address: 0x00000000004003D0 112bca95f8aSPavel Labath AddressAlign: 0x0000000000000010 113bca95f8aSPavel Labath Content: DEADBEEFBAADF00D 114bca95f8aSPavel Labath ... 115bca95f8aSPavel Labath */ 1164f033125SPavel Labath TEST_F(ObjectFileELFTest, GetModuleSpecifications_EarlySectionHeaders) { 1174f033125SPavel Labath std::string SO = GetInputFilePath("early-section-headers.so"); 1184f033125SPavel Labath ModuleSpecList Specs; 1198f3be7a3SJonas Devlieghere ASSERT_EQ(1u, ObjectFile::GetModuleSpecifications(FileSpec(SO), 0, 0, Specs)); 1204f033125SPavel Labath ModuleSpec Spec; 1214f033125SPavel Labath ASSERT_TRUE(Specs.GetModuleSpecAtIndex(0, Spec)) ; 1224f033125SPavel Labath UUID Uuid; 1234f033125SPavel Labath Uuid.SetFromStringRef("1b8a73ac238390e32a7ff4ac8ebe4d6a41ecf5c9", 20); 1244f033125SPavel Labath EXPECT_EQ(Spec.GetUUID(), Uuid); 1254f033125SPavel Labath } 1266868d2ddSNathan Lanza 127b37f1ec8SDavide Italiano static void CHECK_ABS32(uint8_t *bytes, uint32_t offset, uint32_t addend) { 128b37f1ec8SDavide Italiano uint32_t res; 12905620d8eSDavide Italiano memcpy(&res, reinterpret_cast<uint32_t *>(bytes + offset), sizeof(uint32_t)); 130b37f1ec8SDavide Italiano ASSERT_EQ(addend, res); 131b37f1ec8SDavide Italiano } 132b37f1ec8SDavide Italiano 133b37f1ec8SDavide Italiano static void CHECK_ABS64(uint8_t *bytes, uint64_t offset, uint64_t addend) { 134b37f1ec8SDavide Italiano uint64_t res; 13505620d8eSDavide Italiano memcpy(&res, reinterpret_cast<uint64_t *>(bytes + offset), sizeof(uint64_t)); 136b37f1ec8SDavide Italiano ASSERT_EQ(addend, res); 137b37f1ec8SDavide Italiano } 1386868d2ddSNathan Lanza 1396868d2ddSNathan Lanza TEST_F(ObjectFileELFTest, TestAARCH64Relocations) { 1406868d2ddSNathan Lanza llvm::SmallString<128> obj; 1416868d2ddSNathan Lanza ASSERT_NO_ERROR(llvm::sys::fs::createTemporaryFile( 1426868d2ddSNathan Lanza "debug-info-relocations-%%%%%%", "obj", obj)); 1436868d2ddSNathan Lanza llvm::FileRemover remover(obj); 144*8a777920SGreg Clayton ASSERT_THAT_ERROR(ReadYAMLObjectFile("debug-info-relocations.pcm.yaml", obj), 145*8a777920SGreg Clayton llvm::Succeeded()); 1466868d2ddSNathan Lanza 1476868d2ddSNathan Lanza ModuleSpec spec{FileSpec(obj)}; 1486868d2ddSNathan Lanza spec.GetSymbolFileSpec().SetFile(obj, FileSpec::Style::native); 1496868d2ddSNathan Lanza auto module_sp = std::make_shared<Module>(spec); 1506868d2ddSNathan Lanza 1516868d2ddSNathan Lanza auto objfile = static_cast<ObjectFileELF *>(module_sp->GetObjectFile()); 1526868d2ddSNathan Lanza SectionList *section_list = objfile->GetSectionList(); 1536868d2ddSNathan Lanza ASSERT_NE(nullptr, section_list); 1546868d2ddSNathan Lanza 1556868d2ddSNathan Lanza auto debug_info_sp = 1566868d2ddSNathan Lanza section_list->FindSectionByName(ConstString(".debug_info")); 1576868d2ddSNathan Lanza ASSERT_NE(nullptr, debug_info_sp); 1586868d2ddSNathan Lanza objfile->RelocateSection(debug_info_sp.get()); 1596868d2ddSNathan Lanza 1606868d2ddSNathan Lanza DataExtractor data; 1616868d2ddSNathan Lanza // length of 0x10 is not needed but length 0x0 crashes 1626868d2ddSNathan Lanza objfile->GetData(0x00, 0x10, data); 1636868d2ddSNathan Lanza DataBufferSP &data_buffer_sp = data.GetSharedDataBuffer(); 1646868d2ddSNathan Lanza uint8_t *bytes = data_buffer_sp->GetBytes(); 1656868d2ddSNathan Lanza 1666868d2ddSNathan Lanza addr_t debug_info_offset = debug_info_sp->GetFileOffset(); 1676868d2ddSNathan Lanza bytes += debug_info_offset; 1686868d2ddSNathan Lanza 1696868d2ddSNathan Lanza // Sanity check - The first byte from the yaml file is 0x47 1706868d2ddSNathan Lanza ASSERT_EQ(0x47, *bytes); 1716868d2ddSNathan Lanza 1726868d2ddSNathan Lanza // .rela.debug_info contains 9 relocations: 1736868d2ddSNathan Lanza // 7 R_AARCH64_ABS32 - 2 R_AARCH64_ABS64 1746868d2ddSNathan Lanza // None have a value. Four have addends. 175b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x6, 0); 176b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0xC, 0); 177b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x12, 45); 178b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x16, 0); 179b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x1A, 55); 180b37f1ec8SDavide Italiano CHECK_ABS64(bytes, 0x1E, 0); 181b37f1ec8SDavide Italiano CHECK_ABS64(bytes, 0x2B, 0); 182b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x39, 73); 183b37f1ec8SDavide Italiano CHECK_ABS32(bytes, 0x44, 75); 1846868d2ddSNathan Lanza } 185