180814287SRaphael Isemann //===-- TestObjectFileELF.cpp ---------------------------------------------===//
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"
125dca0596SRaphael Isemann #include "TestingSupport/SubsystemRAII.h"
13e2867bc4SPavel Labath #include "TestingSupport/TestUtilities.h"
14efddda3dSPavel Labath #include "lldb/Core/Module.h"
15efddda3dSPavel Labath #include "lldb/Core/ModuleSpec.h"
16efddda3dSPavel Labath #include "lldb/Core/Section.h"
1746376966SJonas Devlieghere #include "lldb/Host/FileSystem.h"
18efddda3dSPavel Labath #include "lldb/Host/HostInfo.h"
196868d2ddSNathan Lanza #include "lldb/Utility/DataBufferHeap.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 {
325dca0596SRaphael Isemann SubsystemRAII<FileSystem, HostInfo, ObjectFileELF, SymbolFileSymtab>
335dca0596SRaphael Isemann subsystems;
34efddda3dSPavel Labath };
35efddda3dSPavel Labath
TEST_F(ObjectFileELFTest,SectionsResolveConsistently)36efddda3dSPavel Labath TEST_F(ObjectFileELFTest, SectionsResolveConsistently) {
37a2e270faSPavel Labath auto ExpectedFile = TestFile::fromYaml(R"(
38a2e270faSPavel Labath --- !ELF
39a2e270faSPavel Labath FileHeader:
40a2e270faSPavel Labath Class: ELFCLASS64
41a2e270faSPavel Labath Data: ELFDATA2LSB
42a2e270faSPavel Labath Type: ET_EXEC
43a2e270faSPavel Labath Machine: EM_X86_64
44a2e270faSPavel Labath Entry: 0x0000000000400180
45a2e270faSPavel Labath Sections:
46a2e270faSPavel Labath - Name: .note.gnu.build-id
47a2e270faSPavel Labath Type: SHT_NOTE
48a2e270faSPavel Labath Flags: [ SHF_ALLOC ]
49a2e270faSPavel Labath Address: 0x0000000000400158
50a2e270faSPavel Labath AddressAlign: 0x0000000000000004
51a2e270faSPavel Labath Content: 040000001400000003000000474E55003F3EC29E3FD83E49D18C4D49CD8A730CC13117B6
52a2e270faSPavel Labath - Name: .text
53a2e270faSPavel Labath Type: SHT_PROGBITS
54a2e270faSPavel Labath Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
55a2e270faSPavel Labath Address: 0x0000000000400180
56a2e270faSPavel Labath AddressAlign: 0x0000000000000010
57a2e270faSPavel Labath Content: 554889E58B042500106000890425041060005DC3
58a2e270faSPavel Labath - Name: .data
59a2e270faSPavel Labath Type: SHT_PROGBITS
60a2e270faSPavel Labath Flags: [ SHF_WRITE, SHF_ALLOC ]
61a2e270faSPavel Labath Address: 0x0000000000601000
62a2e270faSPavel Labath AddressAlign: 0x0000000000000004
63a2e270faSPavel Labath Content: 2F000000
64a2e270faSPavel Labath - Name: .bss
65a2e270faSPavel Labath Type: SHT_NOBITS
66a2e270faSPavel Labath Flags: [ SHF_WRITE, SHF_ALLOC ]
67a2e270faSPavel Labath Address: 0x0000000000601004
68a2e270faSPavel Labath AddressAlign: 0x0000000000000004
69a2e270faSPavel Labath Size: 0x0000000000000004
70a2e270faSPavel Labath Symbols:
71a2e270faSPavel Labath - Name: Y
72a2e270faSPavel Labath Type: STT_OBJECT
73a2e270faSPavel Labath Section: .data
74a2e270faSPavel Labath Value: 0x0000000000601000
75a2e270faSPavel Labath Size: 0x0000000000000004
76a2e270faSPavel Labath Binding: STB_GLOBAL
77a2e270faSPavel Labath - Name: _start
78a2e270faSPavel Labath Type: STT_FUNC
79a2e270faSPavel Labath Section: .text
80a2e270faSPavel Labath Value: 0x0000000000400180
81a2e270faSPavel Labath Size: 0x0000000000000014
82a2e270faSPavel Labath Binding: STB_GLOBAL
83a2e270faSPavel Labath - Name: X
84a2e270faSPavel Labath Type: STT_OBJECT
85a2e270faSPavel Labath Section: .bss
86a2e270faSPavel Labath Value: 0x0000000000601004
87a2e270faSPavel Labath Size: 0x0000000000000004
88a2e270faSPavel Labath Binding: STB_GLOBAL
89a2e270faSPavel Labath ...
90a2e270faSPavel Labath )");
91a2e270faSPavel Labath ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded());
92efddda3dSPavel Labath
93a4a00cedSFred Riss auto module_sp = std::make_shared<Module>(ExpectedFile->moduleSpec());
94efddda3dSPavel Labath SectionList *list = module_sp->GetSectionList();
95efddda3dSPavel Labath ASSERT_NE(nullptr, list);
96efddda3dSPavel Labath
97efddda3dSPavel Labath auto bss_sp = list->FindSectionByName(ConstString(".bss"));
98efddda3dSPavel Labath ASSERT_NE(nullptr, bss_sp);
99efddda3dSPavel Labath auto data_sp = list->FindSectionByName(ConstString(".data"));
100efddda3dSPavel Labath ASSERT_NE(nullptr, data_sp);
101efddda3dSPavel Labath auto text_sp = list->FindSectionByName(ConstString(".text"));
102efddda3dSPavel Labath ASSERT_NE(nullptr, text_sp);
103efddda3dSPavel Labath
104efddda3dSPavel Labath const Symbol *X = module_sp->FindFirstSymbolWithNameAndType(ConstString("X"),
105efddda3dSPavel Labath eSymbolTypeAny);
106efddda3dSPavel Labath ASSERT_NE(nullptr, X);
107efddda3dSPavel Labath EXPECT_EQ(bss_sp, X->GetAddress().GetSection());
108efddda3dSPavel Labath
109efddda3dSPavel Labath const Symbol *Y = module_sp->FindFirstSymbolWithNameAndType(ConstString("Y"),
110efddda3dSPavel Labath eSymbolTypeAny);
111efddda3dSPavel Labath ASSERT_NE(nullptr, Y);
112efddda3dSPavel Labath EXPECT_EQ(data_sp, Y->GetAddress().GetSection());
113efddda3dSPavel Labath
114efddda3dSPavel Labath const Symbol *start = module_sp->FindFirstSymbolWithNameAndType(
115efddda3dSPavel Labath ConstString("_start"), eSymbolTypeAny);
116efddda3dSPavel Labath ASSERT_NE(nullptr, start);
117efddda3dSPavel Labath EXPECT_EQ(text_sp, start->GetAddress().GetSection());
118efddda3dSPavel Labath }
1194f033125SPavel Labath
1204f033125SPavel Labath // Test that GetModuleSpecifications works on an "atypical" object file which
1214f033125SPavel Labath // has section headers right after the ELF header (instead of the more common
1224f033125SPavel Labath // layout where the section headers are at the very end of the object file).
123bca95f8aSPavel Labath //
124bca95f8aSPavel Labath // Test file generated with yaml2obj (@svn rev 324254) from the following input:
125bca95f8aSPavel Labath /*
126bca95f8aSPavel Labath --- !ELF
127bca95f8aSPavel Labath FileHeader:
128bca95f8aSPavel Labath Class: ELFCLASS64
129bca95f8aSPavel Labath Data: ELFDATA2LSB
130bca95f8aSPavel Labath Type: ET_EXEC
131bca95f8aSPavel Labath Machine: EM_X86_64
132bca95f8aSPavel Labath Entry: 0x00000000004003D0
133bca95f8aSPavel Labath Sections:
134bca95f8aSPavel Labath - Name: .note.gnu.build-id
135bca95f8aSPavel Labath Type: SHT_NOTE
136bca95f8aSPavel Labath Flags: [ SHF_ALLOC ]
137bca95f8aSPavel Labath Address: 0x0000000000400274
138bca95f8aSPavel Labath AddressAlign: 0x0000000000000004
139bca95f8aSPavel Labath Content: 040000001400000003000000474E55001B8A73AC238390E32A7FF4AC8EBE4D6A41ECF5C9
140bca95f8aSPavel Labath - Name: .text
141bca95f8aSPavel Labath Type: SHT_PROGBITS
142bca95f8aSPavel Labath Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
143bca95f8aSPavel Labath Address: 0x00000000004003D0
144bca95f8aSPavel Labath AddressAlign: 0x0000000000000010
145bca95f8aSPavel Labath Content: DEADBEEFBAADF00D
146bca95f8aSPavel Labath ...
147bca95f8aSPavel Labath */
TEST_F(ObjectFileELFTest,GetModuleSpecifications_EarlySectionHeaders)1484f033125SPavel Labath TEST_F(ObjectFileELFTest, GetModuleSpecifications_EarlySectionHeaders) {
1494f033125SPavel Labath std::string SO = GetInputFilePath("early-section-headers.so");
1504f033125SPavel Labath ModuleSpecList Specs;
1518f3be7a3SJonas Devlieghere ASSERT_EQ(1u, ObjectFile::GetModuleSpecifications(FileSpec(SO), 0, 0, Specs));
1524f033125SPavel Labath ModuleSpec Spec;
1534f033125SPavel Labath ASSERT_TRUE(Specs.GetModuleSpecAtIndex(0, Spec)) ;
1544f033125SPavel Labath UUID Uuid;
1551beffc18SJaroslav Sevcik Uuid.SetFromStringRef("1b8a73ac238390e32a7ff4ac8ebe4d6a41ecf5c9");
1564f033125SPavel Labath EXPECT_EQ(Spec.GetUUID(), Uuid);
1574f033125SPavel Labath }
158ad6690afSAntonio Afonso
TEST_F(ObjectFileELFTest,GetModuleSpecifications_OffsetSizeWithNormalFile)159*b0b9605aSKazuki Sakamoto TEST_F(ObjectFileELFTest, GetModuleSpecifications_OffsetSizeWithNormalFile) {
160*b0b9605aSKazuki Sakamoto std::string SO = GetInputFilePath("liboffset-test.so");
161*b0b9605aSKazuki Sakamoto ModuleSpecList Specs;
162*b0b9605aSKazuki Sakamoto ASSERT_EQ(1u, ObjectFile::GetModuleSpecifications(FileSpec(SO), 0, 0, Specs));
163*b0b9605aSKazuki Sakamoto ModuleSpec Spec;
164*b0b9605aSKazuki Sakamoto ASSERT_TRUE(Specs.GetModuleSpecAtIndex(0, Spec)) ;
165*b0b9605aSKazuki Sakamoto UUID Uuid;
166*b0b9605aSKazuki Sakamoto Uuid.SetFromStringRef("7D6E4738");
167*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetUUID(), Uuid);
168*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetObjectOffset(), 0UL);
169*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetObjectSize(), 3600UL);
170*b0b9605aSKazuki Sakamoto EXPECT_EQ(FileSystem::Instance().GetByteSize(FileSpec(SO)), 3600UL);
171*b0b9605aSKazuki Sakamoto }
172*b0b9605aSKazuki Sakamoto
TEST_F(ObjectFileELFTest,GetModuleSpecifications_OffsetSizeWithOffsetFile)173*b0b9605aSKazuki Sakamoto TEST_F(ObjectFileELFTest, GetModuleSpecifications_OffsetSizeWithOffsetFile) {
174*b0b9605aSKazuki Sakamoto // The contents of offset-test.bin are
175*b0b9605aSKazuki Sakamoto // - 0-1023: \0
176*b0b9605aSKazuki Sakamoto // - 1024-4623: liboffset-test.so (offset: 1024, size: 3600, CRC32: 7D6E4738)
177*b0b9605aSKazuki Sakamoto // - 4624-4639: \0
178*b0b9605aSKazuki Sakamoto std::string SO = GetInputFilePath("offset-test.bin");
179*b0b9605aSKazuki Sakamoto ModuleSpecList Specs;
180*b0b9605aSKazuki Sakamoto ASSERT_EQ(
181*b0b9605aSKazuki Sakamoto 1u, ObjectFile::GetModuleSpecifications(FileSpec(SO), 1024, 3600, Specs));
182*b0b9605aSKazuki Sakamoto ModuleSpec Spec;
183*b0b9605aSKazuki Sakamoto ASSERT_TRUE(Specs.GetModuleSpecAtIndex(0, Spec)) ;
184*b0b9605aSKazuki Sakamoto UUID Uuid;
185*b0b9605aSKazuki Sakamoto Uuid.SetFromStringRef("7D6E4738");
186*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetUUID(), Uuid);
187*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetObjectOffset(), 1024UL);
188*b0b9605aSKazuki Sakamoto EXPECT_EQ(Spec.GetObjectSize(), 3600UL);
189*b0b9605aSKazuki Sakamoto EXPECT_EQ(FileSystem::Instance().GetByteSize(FileSpec(SO)), 4640UL);
190*b0b9605aSKazuki Sakamoto }
191*b0b9605aSKazuki Sakamoto
TEST_F(ObjectFileELFTest,GetSymtab_NoSymEntryPointArmThumbAddressClass)192ad6690afSAntonio Afonso TEST_F(ObjectFileELFTest, GetSymtab_NoSymEntryPointArmThumbAddressClass) {
193ad6690afSAntonio Afonso /*
194ad6690afSAntonio Afonso // nosym-entrypoint-arm-thumb.s
195ad6690afSAntonio Afonso .thumb_func
196ad6690afSAntonio Afonso _start:
197ad6690afSAntonio Afonso mov r0, #42
198ad6690afSAntonio Afonso mov r7, #1
199ad6690afSAntonio Afonso svc #0
200ad6690afSAntonio Afonso // arm-linux-androideabi-as nosym-entrypoint-arm-thumb.s
201ad6690afSAntonio Afonso // -o nosym-entrypoint-arm-thumb.o
202ad6690afSAntonio Afonso // arm-linux-androideabi-ld nosym-entrypoint-arm-thumb.o
203ad6690afSAntonio Afonso // -o nosym-entrypoint-arm-thumb -e 0x8075 -s
204ad6690afSAntonio Afonso */
205ad6690afSAntonio Afonso auto ExpectedFile = TestFile::fromYaml(R"(
206ad6690afSAntonio Afonso --- !ELF
207ad6690afSAntonio Afonso FileHeader:
208ad6690afSAntonio Afonso Class: ELFCLASS32
209ad6690afSAntonio Afonso Data: ELFDATA2LSB
210ad6690afSAntonio Afonso Type: ET_EXEC
211ad6690afSAntonio Afonso Machine: EM_ARM
212ad6690afSAntonio Afonso Flags: [ EF_ARM_SOFT_FLOAT, EF_ARM_EABI_VER5 ]
213ad6690afSAntonio Afonso Entry: 0x0000000000008075
214ad6690afSAntonio Afonso Sections:
215ad6690afSAntonio Afonso - Name: .text
216ad6690afSAntonio Afonso Type: SHT_PROGBITS
217ad6690afSAntonio Afonso Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
218ad6690afSAntonio Afonso Address: 0x0000000000008074
219ad6690afSAntonio Afonso AddressAlign: 0x0000000000000002
220ad6690afSAntonio Afonso Content: 2A20012700DF
221ad6690afSAntonio Afonso - Name: .data
222ad6690afSAntonio Afonso Type: SHT_PROGBITS
223ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ]
224ad6690afSAntonio Afonso Address: 0x0000000000009000
225ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
226ad6690afSAntonio Afonso Content: ''
227ad6690afSAntonio Afonso - Name: .bss
228ad6690afSAntonio Afonso Type: SHT_NOBITS
229ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ]
230ad6690afSAntonio Afonso Address: 0x0000000000009000
231ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
232ad6690afSAntonio Afonso - Name: .note.gnu.gold-version
233ad6690afSAntonio Afonso Type: SHT_NOTE
234ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004
235ad6690afSAntonio Afonso Content: 040000000900000004000000474E5500676F6C6420312E3131000000
236ad6690afSAntonio Afonso - Name: .ARM.attributes
237ad6690afSAntonio Afonso Type: SHT_ARM_ATTRIBUTES
238ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
239ad6690afSAntonio Afonso Content: '4113000000616561626900010900000006020901'
240ad6690afSAntonio Afonso ...
241ad6690afSAntonio Afonso )");
242ad6690afSAntonio Afonso ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded());
243ad6690afSAntonio Afonso
244a4a00cedSFred Riss auto module_sp = std::make_shared<Module>(ExpectedFile->moduleSpec());
245ad6690afSAntonio Afonso
246ad6690afSAntonio Afonso auto entry_point_addr = module_sp->GetObjectFile()->GetEntryPointAddress();
247ad6690afSAntonio Afonso ASSERT_TRUE(entry_point_addr.GetOffset() & 1);
248ad6690afSAntonio Afonso // Decrease the offsite by 1 to make it into a breakable address since this
249ad6690afSAntonio Afonso // is Thumb.
250ad6690afSAntonio Afonso entry_point_addr.SetOffset(entry_point_addr.GetOffset() - 1);
251ad6690afSAntonio Afonso ASSERT_EQ(entry_point_addr.GetAddressClass(),
252ad6690afSAntonio Afonso AddressClass::eCodeAlternateISA);
253ad6690afSAntonio Afonso }
254ad6690afSAntonio Afonso
TEST_F(ObjectFileELFTest,GetSymtab_NoSymEntryPointArmAddressClass)255ad6690afSAntonio Afonso TEST_F(ObjectFileELFTest, GetSymtab_NoSymEntryPointArmAddressClass) {
256ad6690afSAntonio Afonso /*
257ad6690afSAntonio Afonso // nosym-entrypoint-arm.s
258ad6690afSAntonio Afonso _start:
259ad6690afSAntonio Afonso movs r0, #42
260ad6690afSAntonio Afonso movs r7, #1
261ad6690afSAntonio Afonso svc #0
262ad6690afSAntonio Afonso // arm-linux-androideabi-as nosym-entrypoint-arm.s
263ad6690afSAntonio Afonso // -o nosym-entrypoint-arm.o
264ad6690afSAntonio Afonso // arm-linux-androideabi-ld nosym-entrypoint-arm.o
265ad6690afSAntonio Afonso // -o nosym-entrypoint-arm -e 0x8074 -s
266ad6690afSAntonio Afonso */
267ad6690afSAntonio Afonso auto ExpectedFile = TestFile::fromYaml(R"(
268ad6690afSAntonio Afonso --- !ELF
269ad6690afSAntonio Afonso FileHeader:
270ad6690afSAntonio Afonso Class: ELFCLASS32
271ad6690afSAntonio Afonso Data: ELFDATA2LSB
272ad6690afSAntonio Afonso Type: ET_EXEC
273ad6690afSAntonio Afonso Machine: EM_ARM
274ad6690afSAntonio Afonso Flags: [ EF_ARM_SOFT_FLOAT, EF_ARM_EABI_VER5 ]
275ad6690afSAntonio Afonso Entry: 0x0000000000008074
276ad6690afSAntonio Afonso Sections:
277ad6690afSAntonio Afonso - Name: .text
278ad6690afSAntonio Afonso Type: SHT_PROGBITS
279ad6690afSAntonio Afonso Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
280ad6690afSAntonio Afonso Address: 0x0000000000008074
281ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004
282ad6690afSAntonio Afonso Content: 2A00A0E30170A0E3000000EF
283ad6690afSAntonio Afonso - Name: .data
284ad6690afSAntonio Afonso Type: SHT_PROGBITS
285ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ]
286ad6690afSAntonio Afonso Address: 0x0000000000009000
287ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
288ad6690afSAntonio Afonso Content: ''
289ad6690afSAntonio Afonso - Name: .bss
290ad6690afSAntonio Afonso Type: SHT_NOBITS
291ad6690afSAntonio Afonso Flags: [ SHF_WRITE, SHF_ALLOC ]
292ad6690afSAntonio Afonso Address: 0x0000000000009000
293ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
294ad6690afSAntonio Afonso - Name: .note.gnu.gold-version
295ad6690afSAntonio Afonso Type: SHT_NOTE
296ad6690afSAntonio Afonso AddressAlign: 0x0000000000000004
297ad6690afSAntonio Afonso Content: 040000000900000004000000474E5500676F6C6420312E3131000000
298ad6690afSAntonio Afonso - Name: .ARM.attributes
299ad6690afSAntonio Afonso Type: SHT_ARM_ATTRIBUTES
300ad6690afSAntonio Afonso AddressAlign: 0x0000000000000001
301ad6690afSAntonio Afonso Content: '4113000000616561626900010900000006010801'
302ad6690afSAntonio Afonso ...
303ad6690afSAntonio Afonso )");
304ad6690afSAntonio Afonso ASSERT_THAT_EXPECTED(ExpectedFile, llvm::Succeeded());
305ad6690afSAntonio Afonso
306a4a00cedSFred Riss auto module_sp = std::make_shared<Module>(ExpectedFile->moduleSpec());
307ad6690afSAntonio Afonso
308ad6690afSAntonio Afonso auto entry_point_addr = module_sp->GetObjectFile()->GetEntryPointAddress();
309ad6690afSAntonio Afonso ASSERT_EQ(entry_point_addr.GetAddressClass(), AddressClass::eCode);
310ad6690afSAntonio Afonso }
311