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" 1184a68569SPavel Labath #include "Plugins/SymbolFile/Symtab/SymbolFileSymtab.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" 258a777920SGreg 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(); 3784a68569SPavel Labath SymbolFileSymtab::Initialize(); 38efddda3dSPavel Labath } 39efddda3dSPavel Labath 40efddda3dSPavel Labath void TearDown() override { 4184a68569SPavel Labath SymbolFileSymtab::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) { 51a2e270faSPavel Labath auto ExpectedFile = TestFile::fromYaml(R"( 52a2e270faSPavel Labath --- !ELF 53a2e270faSPavel Labath FileHeader: 54a2e270faSPavel Labath Class: ELFCLASS64 55a2e270faSPavel Labath Data: ELFDATA2LSB 56a2e270faSPavel Labath Type: ET_EXEC 57a2e270faSPavel Labath Machine: EM_X86_64 58a2e270faSPavel Labath Entry: 0x0000000000400180 59a2e270faSPavel Labath Sections: 60a2e270faSPavel Labath - Name: .note.gnu.build-id 61a2e270faSPavel Labath Type: SHT_NOTE 62a2e270faSPavel Labath Flags: [ SHF_ALLOC ] 63a2e270faSPavel Labath Address: 0x0000000000400158 64a2e270faSPavel Labath AddressAlign: 0x0000000000000004 65a2e270faSPavel Labath Content: 040000001400000003000000474E55003F3EC29E3FD83E49D18C4D49CD8A730CC13117B6 66a2e270faSPavel Labath - Name: .text 67a2e270faSPavel Labath Type: SHT_PROGBITS 68a2e270faSPavel Labath Flags: [ SHF_ALLOC, SHF_EXECINSTR ] 69a2e270faSPavel Labath Address: 0x0000000000400180 70a2e270faSPavel Labath AddressAlign: 0x0000000000000010 71a2e270faSPavel Labath Content: 554889E58B042500106000890425041060005DC3 72a2e270faSPavel Labath - Name: .data 73a2e270faSPavel Labath Type: SHT_PROGBITS 74a2e270faSPavel Labath Flags: [ SHF_WRITE, SHF_ALLOC ] 75a2e270faSPavel Labath Address: 0x0000000000601000 76a2e270faSPavel Labath AddressAlign: 0x0000000000000004 77a2e270faSPavel Labath Content: 2F000000 78a2e270faSPavel Labath - Name: .bss 79a2e270faSPavel Labath Type: SHT_NOBITS 80a2e270faSPavel Labath Flags: [ SHF_WRITE, SHF_ALLOC ] 81a2e270faSPavel Labath Address: 0x0000000000601004 82a2e270faSPavel Labath AddressAlign: 0x0000000000000004 83a2e270faSPavel Labath Size: 0x0000000000000004 84a2e270faSPavel Labath Symbols: 85a2e270faSPavel Labath - Name: Y 86a2e270faSPavel Labath Type: STT_OBJECT 87a2e270faSPavel Labath Section: .data 88a2e270faSPavel Labath Value: 0x0000000000601000 89a2e270faSPavel Labath Size: 0x0000000000000004 90a2e270faSPavel Labath Binding: STB_GLOBAL 91a2e270faSPavel Labath - Name: _start 92a2e270faSPavel Labath Type: STT_FUNC 93a2e270faSPavel Labath Section: .text 94a2e270faSPavel Labath Value: 0x0000000000400180 95a2e270faSPavel Labath Size: 0x0000000000000014 96a2e270faSPavel Labath Binding: STB_GLOBAL 97a2e270faSPavel Labath - Name: X 98a2e270faSPavel Labath Type: STT_OBJECT 99a2e270faSPavel Labath Section: .bss 100a2e270faSPavel Labath Value: 0x0000000000601004 101a2e270faSPavel Labath Size: 0x0000000000000004 102a2e270faSPavel Labath Binding: STB_GLOBAL 103a2e270faSPavel Labath ... 104a2e270faSPavel Labath )"); 105a2e270faSPavel Labath ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded()); 106efddda3dSPavel Labath 107a2e270faSPavel Labath ModuleSpec spec{FileSpec(ExpectedFile->name())}; 108a2e270faSPavel Labath spec.GetSymbolFileSpec().SetFile(ExpectedFile->name(), 109a2e270faSPavel Labath FileSpec::Style::native); 110efddda3dSPavel Labath auto module_sp = std::make_shared<Module>(spec); 111efddda3dSPavel Labath SectionList *list = module_sp->GetSectionList(); 112efddda3dSPavel Labath ASSERT_NE(nullptr, list); 113efddda3dSPavel Labath 114efddda3dSPavel Labath auto bss_sp = list->FindSectionByName(ConstString(".bss")); 115efddda3dSPavel Labath ASSERT_NE(nullptr, bss_sp); 116efddda3dSPavel Labath auto data_sp = list->FindSectionByName(ConstString(".data")); 117efddda3dSPavel Labath ASSERT_NE(nullptr, data_sp); 118efddda3dSPavel Labath auto text_sp = list->FindSectionByName(ConstString(".text")); 119efddda3dSPavel Labath ASSERT_NE(nullptr, text_sp); 120efddda3dSPavel Labath 121efddda3dSPavel Labath const Symbol *X = module_sp->FindFirstSymbolWithNameAndType(ConstString("X"), 122efddda3dSPavel Labath eSymbolTypeAny); 123efddda3dSPavel Labath ASSERT_NE(nullptr, X); 124efddda3dSPavel Labath EXPECT_EQ(bss_sp, X->GetAddress().GetSection()); 125efddda3dSPavel Labath 126efddda3dSPavel Labath const Symbol *Y = module_sp->FindFirstSymbolWithNameAndType(ConstString("Y"), 127efddda3dSPavel Labath eSymbolTypeAny); 128efddda3dSPavel Labath ASSERT_NE(nullptr, Y); 129efddda3dSPavel Labath EXPECT_EQ(data_sp, Y->GetAddress().GetSection()); 130efddda3dSPavel Labath 131efddda3dSPavel Labath const Symbol *start = module_sp->FindFirstSymbolWithNameAndType( 132efddda3dSPavel Labath ConstString("_start"), eSymbolTypeAny); 133efddda3dSPavel Labath ASSERT_NE(nullptr, start); 134efddda3dSPavel Labath EXPECT_EQ(text_sp, start->GetAddress().GetSection()); 135efddda3dSPavel Labath } 1364f033125SPavel Labath 1374f033125SPavel Labath // Test that GetModuleSpecifications works on an "atypical" object file which 1384f033125SPavel Labath // has section headers right after the ELF header (instead of the more common 1394f033125SPavel Labath // layout where the section headers are at the very end of the object file). 140bca95f8aSPavel Labath // 141bca95f8aSPavel Labath // Test file generated with yaml2obj (@svn rev 324254) from the following input: 142bca95f8aSPavel Labath /* 143bca95f8aSPavel Labath --- !ELF 144bca95f8aSPavel Labath FileHeader: 145bca95f8aSPavel Labath Class: ELFCLASS64 146bca95f8aSPavel Labath Data: ELFDATA2LSB 147bca95f8aSPavel Labath Type: ET_EXEC 148bca95f8aSPavel Labath Machine: EM_X86_64 149bca95f8aSPavel Labath Entry: 0x00000000004003D0 150bca95f8aSPavel Labath Sections: 151bca95f8aSPavel Labath - Name: .note.gnu.build-id 152bca95f8aSPavel Labath Type: SHT_NOTE 153bca95f8aSPavel Labath Flags: [ SHF_ALLOC ] 154bca95f8aSPavel Labath Address: 0x0000000000400274 155bca95f8aSPavel Labath AddressAlign: 0x0000000000000004 156bca95f8aSPavel Labath Content: 040000001400000003000000474E55001B8A73AC238390E32A7FF4AC8EBE4D6A41ECF5C9 157bca95f8aSPavel Labath - Name: .text 158bca95f8aSPavel Labath Type: SHT_PROGBITS 159bca95f8aSPavel Labath Flags: [ SHF_ALLOC, SHF_EXECINSTR ] 160bca95f8aSPavel Labath Address: 0x00000000004003D0 161bca95f8aSPavel Labath AddressAlign: 0x0000000000000010 162bca95f8aSPavel Labath Content: DEADBEEFBAADF00D 163bca95f8aSPavel Labath ... 164bca95f8aSPavel Labath */ 1654f033125SPavel Labath TEST_F(ObjectFileELFTest, GetModuleSpecifications_EarlySectionHeaders) { 1664f033125SPavel Labath std::string SO = GetInputFilePath("early-section-headers.so"); 1674f033125SPavel Labath ModuleSpecList Specs; 1688f3be7a3SJonas Devlieghere ASSERT_EQ(1u, ObjectFile::GetModuleSpecifications(FileSpec(SO), 0, 0, Specs)); 1694f033125SPavel Labath ModuleSpec Spec; 1704f033125SPavel Labath ASSERT_TRUE(Specs.GetModuleSpecAtIndex(0, Spec)) ; 1714f033125SPavel Labath UUID Uuid; 1724f033125SPavel Labath Uuid.SetFromStringRef("1b8a73ac238390e32a7ff4ac8ebe4d6a41ecf5c9", 20); 1734f033125SPavel Labath EXPECT_EQ(Spec.GetUUID(), Uuid); 1744f033125SPavel Labath } 175*ad6690afSAntonio Afonso 176*ad6690afSAntonio Afonso TEST_F(ObjectFileELFTest, GetSymtab_NoSymEntryPointArmThumbAddressClass) { 177*ad6690afSAntonio Afonso /* 178*ad6690afSAntonio Afonso // nosym-entrypoint-arm-thumb.s 179*ad6690afSAntonio Afonso .thumb_func 180*ad6690afSAntonio Afonso _start: 181*ad6690afSAntonio Afonso mov r0, #42 182*ad6690afSAntonio Afonso mov r7, #1 183*ad6690afSAntonio Afonso svc #0 184*ad6690afSAntonio Afonso // arm-linux-androideabi-as nosym-entrypoint-arm-thumb.s 185*ad6690afSAntonio Afonso // -o nosym-entrypoint-arm-thumb.o 186*ad6690afSAntonio Afonso // arm-linux-androideabi-ld nosym-entrypoint-arm-thumb.o 187*ad6690afSAntonio Afonso // -o nosym-entrypoint-arm-thumb -e 0x8075 -s 188*ad6690afSAntonio Afonso */ 189*ad6690afSAntonio Afonso auto ExpectedFile = TestFile::fromYaml(R"( 190*ad6690afSAntonio Afonso --- !ELF 191*ad6690afSAntonio Afonso FileHeader: 192*ad6690afSAntonio Afonso Class: ELFCLASS32 193*ad6690afSAntonio Afonso Data: ELFDATA2LSB 194*ad6690afSAntonio Afonso Type: ET_EXEC 195*ad6690afSAntonio Afonso Machine: EM_ARM 196*ad6690afSAntonio Afonso Flags: [ EF_ARM_SOFT_FLOAT, EF_ARM_EABI_VER5 ] 197*ad6690afSAntonio Afonso Entry: 0x0000000000008075 198*ad6690afSAntonio Afonso Sections: 199*ad6690afSAntonio Afonso - Name: .text 200*ad6690afSAntonio Afonso Type: SHT_PROGBITS 201*ad6690afSAntonio Afonso Flags: [ SHF_ALLOC, SHF_EXECINSTR ] 202*ad6690afSAntonio Afonso Address: 0x0000000000008074 203*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000002 204*ad6690afSAntonio Afonso Content: 2A20012700DF 205*ad6690afSAntonio Afonso - Name: .data 206*ad6690afSAntonio Afonso Type: SHT_PROGBITS 207*ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ] 208*ad6690afSAntonio Afonso Address: 0x0000000000009000 209*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 210*ad6690afSAntonio Afonso Content: '' 211*ad6690afSAntonio Afonso - Name: .bss 212*ad6690afSAntonio Afonso Type: SHT_NOBITS 213*ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ] 214*ad6690afSAntonio Afonso Address: 0x0000000000009000 215*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 216*ad6690afSAntonio Afonso - Name: .note.gnu.gold-version 217*ad6690afSAntonio Afonso Type: SHT_NOTE 218*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004 219*ad6690afSAntonio Afonso Content: 040000000900000004000000474E5500676F6C6420312E3131000000 220*ad6690afSAntonio Afonso - Name: .ARM.attributes 221*ad6690afSAntonio Afonso Type: SHT_ARM_ATTRIBUTES 222*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 223*ad6690afSAntonio Afonso Content: '4113000000616561626900010900000006020901' 224*ad6690afSAntonio Afonso ... 225*ad6690afSAntonio Afonso )"); 226*ad6690afSAntonio Afonso ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded()); 227*ad6690afSAntonio Afonso 228*ad6690afSAntonio Afonso ModuleSpec spec{FileSpec(ExpectedFile->name())}; 229*ad6690afSAntonio Afonso spec.GetSymbolFileSpec().SetFile(ExpectedFile->name(), 230*ad6690afSAntonio Afonso FileSpec::Style::native); 231*ad6690afSAntonio Afonso auto module_sp = std::make_shared<Module>(spec); 232*ad6690afSAntonio Afonso 233*ad6690afSAntonio Afonso auto entry_point_addr = module_sp->GetObjectFile()->GetEntryPointAddress(); 234*ad6690afSAntonio Afonso ASSERT_TRUE(entry_point_addr.GetOffset() & 1); 235*ad6690afSAntonio Afonso // Decrease the offsite by 1 to make it into a breakable address since this 236*ad6690afSAntonio Afonso // is Thumb. 237*ad6690afSAntonio Afonso entry_point_addr.SetOffset(entry_point_addr.GetOffset() - 1); 238*ad6690afSAntonio Afonso ASSERT_EQ(entry_point_addr.GetAddressClass(), 239*ad6690afSAntonio Afonso AddressClass::eCodeAlternateISA); 240*ad6690afSAntonio Afonso } 241*ad6690afSAntonio Afonso 242*ad6690afSAntonio Afonso TEST_F(ObjectFileELFTest, GetSymtab_NoSymEntryPointArmAddressClass) { 243*ad6690afSAntonio Afonso /* 244*ad6690afSAntonio Afonso // nosym-entrypoint-arm.s 245*ad6690afSAntonio Afonso _start: 246*ad6690afSAntonio Afonso movs r0, #42 247*ad6690afSAntonio Afonso movs r7, #1 248*ad6690afSAntonio Afonso svc #0 249*ad6690afSAntonio Afonso // arm-linux-androideabi-as nosym-entrypoint-arm.s 250*ad6690afSAntonio Afonso // -o nosym-entrypoint-arm.o 251*ad6690afSAntonio Afonso // arm-linux-androideabi-ld nosym-entrypoint-arm.o 252*ad6690afSAntonio Afonso // -o nosym-entrypoint-arm -e 0x8074 -s 253*ad6690afSAntonio Afonso */ 254*ad6690afSAntonio Afonso auto ExpectedFile = TestFile::fromYaml(R"( 255*ad6690afSAntonio Afonso --- !ELF 256*ad6690afSAntonio Afonso FileHeader: 257*ad6690afSAntonio Afonso Class: ELFCLASS32 258*ad6690afSAntonio Afonso Data: ELFDATA2LSB 259*ad6690afSAntonio Afonso Type: ET_EXEC 260*ad6690afSAntonio Afonso Machine: EM_ARM 261*ad6690afSAntonio Afonso Flags: [ EF_ARM_SOFT_FLOAT, EF_ARM_EABI_VER5 ] 262*ad6690afSAntonio Afonso Entry: 0x0000000000008074 263*ad6690afSAntonio Afonso Sections: 264*ad6690afSAntonio Afonso - Name: .text 265*ad6690afSAntonio Afonso Type: SHT_PROGBITS 266*ad6690afSAntonio Afonso Flags: [ SHF_ALLOC, SHF_EXECINSTR ] 267*ad6690afSAntonio Afonso Address: 0x0000000000008074 268*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004 269*ad6690afSAntonio Afonso Content: 2A00A0E30170A0E3000000EF 270*ad6690afSAntonio Afonso - Name: .data 271*ad6690afSAntonio Afonso Type: SHT_PROGBITS 272*ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ] 273*ad6690afSAntonio Afonso Address: 0x0000000000009000 274*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 275*ad6690afSAntonio Afonso Content: '' 276*ad6690afSAntonio Afonso - Name: .bss 277*ad6690afSAntonio Afonso Type: SHT_NOBITS 278*ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ] 279*ad6690afSAntonio Afonso Address: 0x0000000000009000 280*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 281*ad6690afSAntonio Afonso - Name: .note.gnu.gold-version 282*ad6690afSAntonio Afonso Type: SHT_NOTE 283*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004 284*ad6690afSAntonio Afonso Content: 040000000900000004000000474E5500676F6C6420312E3131000000 285*ad6690afSAntonio Afonso - Name: .ARM.attributes 286*ad6690afSAntonio Afonso Type: SHT_ARM_ATTRIBUTES 287*ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001 288*ad6690afSAntonio Afonso Content: '4113000000616561626900010900000006010801' 289*ad6690afSAntonio Afonso ... 290*ad6690afSAntonio Afonso )"); 291*ad6690afSAntonio Afonso ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded()); 292*ad6690afSAntonio Afonso 293*ad6690afSAntonio Afonso ModuleSpec spec{FileSpec(ExpectedFile->name())}; 294*ad6690afSAntonio Afonso spec.GetSymbolFileSpec().SetFile(ExpectedFile->name(), 295*ad6690afSAntonio Afonso FileSpec::Style::native); 296*ad6690afSAntonio Afonso auto module_sp = std::make_shared<Module>(spec); 297*ad6690afSAntonio Afonso 298*ad6690afSAntonio Afonso auto entry_point_addr = module_sp->GetObjectFile()->GetEntryPointAddress(); 299*ad6690afSAntonio Afonso ASSERT_EQ(entry_point_addr.GetAddressClass(), AddressClass::eCode); 300*ad6690afSAntonio Afonso }