1 //===- GOFFObjectFileTest.cpp - Tests for GOFFObjectFile ------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "llvm/Object/GOFFObjectFile.h"
10 #include "llvm/Support/MemoryBuffer.h"
11 #include "llvm/Testing/Support/Error.h"
12 #include "gtest/gtest.h"
15 using namespace llvm::object
;
16 using namespace llvm::GOFF
;
19 char GOFFData
[GOFF::RecordLength
* 3] = {0x00};
21 void constructValidGOFF(size_t Size
) {
22 StringRef
ValidSize(GOFFData
, Size
);
23 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
24 object::ObjectFile::createGOFFObjectFile(
25 MemoryBufferRef(ValidSize
, "dummyGOFF"));
27 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
30 void constructInvalidGOFF(size_t Size
) {
31 // Construct GOFFObject with record of length != multiple of 80.
32 StringRef
InvalidData(GOFFData
, Size
);
33 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
34 object::ObjectFile::createGOFFObjectFile(
35 MemoryBufferRef(InvalidData
, "dummyGOFF"));
39 FailedWithMessage("object file is not the right size. Must be a multiple "
40 "of 80 bytes, but is " +
41 std::to_string(Size
) + " bytes"));
45 TEST(GOFFObjectFileTest
, ConstructGOFFObjectValidSize
) {
46 GOFFData
[0] = (char)0x03;
47 GOFFData
[1] = (char)0xF0;
48 GOFFData
[80] = (char)0x03;
49 GOFFData
[81] = (char)0x40;
50 constructValidGOFF(160);
51 constructValidGOFF(0);
54 TEST(GOFFObjectFileTest
, ConstructGOFFObjectInvalidSize
) {
55 constructInvalidGOFF(70);
56 constructInvalidGOFF(79);
57 constructInvalidGOFF(81);
60 TEST(GOFFObjectFileTest
, MissingHDR
) {
61 char GOFFData
[GOFF::RecordLength
* 2] = {0x00};
64 GOFFData
[0] = (char)0x03;
67 GOFFData
[GOFF::RecordLength
] = (char)0x03;
68 GOFFData
[GOFF::RecordLength
+ 1] = (char)0x40;
70 StringRef
Data(GOFFData
, GOFF::RecordLength
* 2);
72 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
73 object::ObjectFile::createGOFFObjectFile(
74 MemoryBufferRef(Data
, "dummyGOFF"));
78 FailedWithMessage("object file must start with HDR record"));
81 TEST(GOFFObjectFileTest
, MissingEND
) {
82 char GOFFData
[GOFF::RecordLength
* 2] = {0x00};
85 GOFFData
[0] = (char)0x03;
86 GOFFData
[1] = (char)0xF0;
89 GOFFData
[GOFF::RecordLength
] = (char)0x03;
91 StringRef
Data(GOFFData
, GOFF::RecordLength
* 2);
93 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
94 object::ObjectFile::createGOFFObjectFile(
95 MemoryBufferRef(Data
, "dummyGOFF"));
98 GOFFObjOrErr
, FailedWithMessage("object file must end with END record"));
101 TEST(GOFFObjectFileTest
, GetSymbolName
) {
102 char GOFFData
[GOFF::RecordLength
* 3] = {0x00};
105 GOFFData
[0] = (char)0x03;
106 GOFFData
[1] = (char)0xF0;
109 GOFFData
[GOFF::RecordLength
] = (char)0x03;
110 GOFFData
[GOFF::RecordLength
+ 3] = (char)0x02;
111 GOFFData
[GOFF::RecordLength
+ 7] = (char)0x01;
112 GOFFData
[GOFF::RecordLength
+ 11] = (char)0x01;
113 GOFFData
[GOFF::RecordLength
+ 71] = (char)0x05; // Size of symbol name.
114 GOFFData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name is Hello.
115 GOFFData
[GOFF::RecordLength
+ 73] = (char)0x85;
116 GOFFData
[GOFF::RecordLength
+ 74] = (char)0x93;
117 GOFFData
[GOFF::RecordLength
+ 75] = (char)0x93;
118 GOFFData
[GOFF::RecordLength
+ 76] = (char)0x96;
121 GOFFData
[GOFF::RecordLength
* 2] = 0x03;
122 GOFFData
[GOFF::RecordLength
* 2 + 1] = 0x40;
124 StringRef
Data(GOFFData
, GOFF::RecordLength
* 3);
126 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
127 object::ObjectFile::createGOFFObjectFile(
128 MemoryBufferRef(Data
, "dummyGOFF"));
130 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
132 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
134 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
135 Expected
<StringRef
> SymbolNameOrErr
= GOFFObj
->getSymbolName(Symbol
);
136 ASSERT_THAT_EXPECTED(SymbolNameOrErr
, Succeeded());
137 StringRef SymbolName
= SymbolNameOrErr
.get();
139 EXPECT_EQ(SymbolName
, "Hello");
143 TEST(GOFFObjectFileTest
, ConcatenatedGOFFFile
) {
144 char GOFFData
[GOFF::RecordLength
* 6] = {0x00};
147 GOFFData
[0] = (char)0x03;
148 GOFFData
[1] = (char)0xF0;
150 GOFFData
[GOFF::RecordLength
] = (char)0x03;
152 GOFFData
[GOFF::RecordLength
* 2] = (char)0x03;
153 GOFFData
[GOFF::RecordLength
* 2 + 1] = (char)0x40;
155 GOFFData
[GOFF::RecordLength
* 3] = (char)0x03;
156 GOFFData
[GOFF::RecordLength
* 3 + 1] = (char)0xF0;
158 GOFFData
[GOFF::RecordLength
* 4] = (char)0x03;
160 GOFFData
[GOFF::RecordLength
* 5] = (char)0x03;
161 GOFFData
[GOFF::RecordLength
* 5 + 1] = (char)0x40;
163 StringRef
Data(GOFFData
, GOFF::RecordLength
* 6);
165 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
166 object::ObjectFile::createGOFFObjectFile(
167 MemoryBufferRef(Data
, "dummyGOFF"));
169 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
172 TEST(GOFFObjectFileTest
, ContinuationGetSymbolName
) {
173 char GOFFContData
[GOFF::RecordLength
* 4] = {0x00};
176 GOFFContData
[0] = (char)0x03;
177 GOFFContData
[1] = (char)0xF0;
180 GOFFContData
[GOFF::RecordLength
] = (char)0x03;
181 GOFFContData
[GOFF::RecordLength
+ 1] = (char)0x01;
182 GOFFContData
[GOFF::RecordLength
+ 3] = (char)0x02;
183 GOFFContData
[GOFF::RecordLength
+ 7] = (char)0x01;
184 GOFFContData
[GOFF::RecordLength
+ 11] = (char)0x01;
185 GOFFContData
[GOFF::RecordLength
+ 71] = (char)0x0A; // Size of symbol name.
186 GOFFContData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name is HelloWorld.
187 GOFFContData
[GOFF::RecordLength
+ 73] = (char)0x85;
188 GOFFContData
[GOFF::RecordLength
+ 74] = (char)0x93;
189 GOFFContData
[GOFF::RecordLength
+ 75] = (char)0x93;
190 GOFFContData
[GOFF::RecordLength
+ 76] = (char)0x96;
191 GOFFContData
[GOFF::RecordLength
+ 77] = (char)0xA6;
192 GOFFContData
[GOFF::RecordLength
+ 78] = (char)0x96;
193 GOFFContData
[GOFF::RecordLength
+ 79] = (char)0x99;
195 // ESD continuation record.
196 GOFFContData
[GOFF::RecordLength
* 2] = (char)0x03;
197 GOFFContData
[GOFF::RecordLength
* 2 + 1] = (char)0x02; // No further continuations.
198 GOFFContData
[GOFF::RecordLength
* 2 + 3] = (char)0x93;
199 GOFFContData
[GOFF::RecordLength
* 2 + 4] = (char)0x84;
202 GOFFContData
[GOFF::RecordLength
* 3] = (char)0x03;
203 GOFFContData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
205 StringRef
Data(GOFFContData
, GOFF::RecordLength
* 4);
207 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
208 object::ObjectFile::createGOFFObjectFile(
209 MemoryBufferRef(Data
, "dummyGOFF"));
211 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
213 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
215 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
216 Expected
<StringRef
> SymbolNameOrErr
= GOFFObj
->getSymbolName(Symbol
);
217 ASSERT_THAT_EXPECTED(SymbolNameOrErr
, Succeeded());
218 StringRef SymbolName
= SymbolNameOrErr
.get();
219 EXPECT_EQ(SymbolName
, "Helloworld");
223 TEST(GOFFObjectFileTest
, ContinuationBitNotSet
) {
224 char GOFFContData
[GOFF::RecordLength
* 4] = {0x00};
227 GOFFContData
[0] = (char)0x03;
228 GOFFContData
[1] = (char)0xF0;
231 GOFFContData
[GOFF::RecordLength
] = (char)0x03;
232 GOFFContData
[GOFF::RecordLength
+ 1] = (char)0x01;
233 GOFFContData
[GOFF::RecordLength
+ 3] = (char)0x02;
234 GOFFContData
[GOFF::RecordLength
+ 7] = (char)0x01;
235 GOFFContData
[GOFF::RecordLength
+ 11] = (char)0x01;
236 GOFFContData
[GOFF::RecordLength
+ 71] = (char)0x0A; // Size of symbol name.
237 GOFFContData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name is HelloWorld.
238 GOFFContData
[GOFF::RecordLength
+ 73] = (char)0x85;
239 GOFFContData
[GOFF::RecordLength
+ 74] = (char)0x93;
240 GOFFContData
[GOFF::RecordLength
+ 75] = (char)0x93;
241 GOFFContData
[GOFF::RecordLength
+ 76] = (char)0x96;
242 GOFFContData
[GOFF::RecordLength
+ 77] = (char)0xA6;
243 GOFFContData
[GOFF::RecordLength
+ 78] = (char)0x96;
244 GOFFContData
[GOFF::RecordLength
+ 79] = (char)0x99;
246 // ESD continuation record.
247 GOFFContData
[GOFF::RecordLength
* 2] = (char)0x03;
248 GOFFContData
[GOFF::RecordLength
* 2 + 1] = (char)0x00;
249 GOFFContData
[GOFF::RecordLength
* 2 + 3] = (char)0x93;
250 GOFFContData
[GOFF::RecordLength
* 2 + 4] = (char)0x84;
253 GOFFContData
[GOFF::RecordLength
* 3] = (char)0x03;
254 GOFFContData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
256 StringRef
Data(GOFFContData
, GOFF::RecordLength
* 4);
258 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
259 object::ObjectFile::createGOFFObjectFile(
260 MemoryBufferRef(Data
, "dummyGOFF"));
261 EXPECT_THAT_EXPECTED(
263 FailedWithMessage("record 2 is not a continuation record but the "
264 "preceding record is continued"));
267 TEST(GOFFObjectFileTest
, ContinuationRecordNotTerminated
) {
268 char GOFFContData
[GOFF::RecordLength
* 4] = {0x00};
271 GOFFContData
[0] = (char)0x03;
272 GOFFContData
[1] = (char)0xF0;
275 GOFFContData
[GOFF::RecordLength
] = (char)0x03;
276 GOFFContData
[GOFF::RecordLength
+ 1] = (char)0x01;
277 GOFFContData
[GOFF::RecordLength
+ 3] = (char)0x02;
278 GOFFContData
[GOFF::RecordLength
+ 7] = (char)0x01;
279 GOFFContData
[GOFF::RecordLength
+ 11] = (char)0x01;
280 GOFFContData
[GOFF::RecordLength
+ 71] = (char)0x0A; // Size of symbol name.
281 GOFFContData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name is HelloWorld.
282 GOFFContData
[GOFF::RecordLength
+ 73] = (char)0x85;
283 GOFFContData
[GOFF::RecordLength
+ 74] = (char)0x93;
284 GOFFContData
[GOFF::RecordLength
+ 75] = (char)0x93;
285 GOFFContData
[GOFF::RecordLength
+ 76] = (char)0x96;
286 GOFFContData
[GOFF::RecordLength
+ 77] = (char)0xA6;
287 GOFFContData
[GOFF::RecordLength
+ 78] = (char)0x96;
288 GOFFContData
[GOFF::RecordLength
+ 79] = (char)0x99;
290 // ESD continuation record.
291 GOFFContData
[GOFF::RecordLength
* 2] = (char)0x03;
292 GOFFContData
[GOFF::RecordLength
* 2 + 1] = (char)0x03; // Continued bit set.
293 GOFFContData
[GOFF::RecordLength
* 2 + 3] = (char)0x93;
294 GOFFContData
[GOFF::RecordLength
* 2 + 4] = (char)0x84;
297 GOFFContData
[GOFF::RecordLength
* 3] = (char)0x03;
298 GOFFContData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
300 StringRef
Data(GOFFContData
, GOFF::RecordLength
* 4);
302 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
303 object::ObjectFile::createGOFFObjectFile(
304 MemoryBufferRef(Data
, "dummyGOFF"));
305 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
307 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
309 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
310 Expected
<StringRef
> SymbolNameOrErr
= GOFFObj
->getSymbolName(Symbol
);
311 EXPECT_THAT_EXPECTED(SymbolNameOrErr
,
312 FailedWithMessage("continued bit should not be set"));
316 TEST(GOFFObjectFileTest
, PrevNotContinued
) {
317 char GOFFContData
[GOFF::RecordLength
* 4] = {0x00};
320 GOFFContData
[0] = (char)0x03;
321 GOFFContData
[1] = (char)0xF0;
323 // ESD record, with continued bit not set.
324 GOFFContData
[GOFF::RecordLength
] = (char)0x03;
326 // ESD continuation record.
327 GOFFContData
[GOFF::RecordLength
* 2] = (char)0x03;
328 GOFFContData
[GOFF::RecordLength
* 2 + 1] = (char)0x02;
331 GOFFContData
[GOFF::RecordLength
* 3] = (char)0x03;
332 GOFFContData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
334 StringRef
Data(GOFFContData
, GOFF::RecordLength
* 4);
336 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
337 object::ObjectFile::createGOFFObjectFile(
338 MemoryBufferRef(Data
, "dummyGOFF"));
340 ASSERT_THAT_EXPECTED(
342 FailedWithMessage("record 2 is a continuation record that is not "
343 "preceded by a continued record"));
346 TEST(GOFFObjectFileTest
, ContinuationTypeMismatch
) {
347 char GOFFContData
[GOFF::RecordLength
* 4] = {0x00};
350 GOFFContData
[0] = (char)0x03;
351 GOFFContData
[1] = (char)0xF0;
354 GOFFContData
[GOFF::RecordLength
] = (char)0x03;
355 GOFFContData
[GOFF::RecordLength
+ 1] = (char)0x01; // Continued to next record.
357 // END continuation record.
358 GOFFContData
[GOFF::RecordLength
* 2] = (char)0x03;
359 GOFFContData
[GOFF::RecordLength
* 2 + 1] = (char)0x42;
362 GOFFContData
[GOFF::RecordLength
* 3] = (char)0x03;
363 GOFFContData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
365 StringRef
Data(GOFFContData
, GOFF::RecordLength
* 4);
367 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
368 object::ObjectFile::createGOFFObjectFile(
369 MemoryBufferRef(Data
, "dummyGOFF"));
371 ASSERT_THAT_EXPECTED(
373 FailedWithMessage("record 2 is a continuation record that does not match "
374 "the type of the previous record"));
377 TEST(GOFFObjectFileTest
, TwoSymbols
) {
378 char GOFFData
[GOFF::RecordLength
* 4] = {0x00};
381 GOFFData
[0] = (char)0x03;
382 GOFFData
[1] = (char)0xF0;
385 GOFFData
[GOFF::RecordLength
] = (char)0x03;
386 GOFFData
[GOFF::RecordLength
+ 3] = (char)0x00;
387 GOFFData
[GOFF::RecordLength
+ 7] = (char)0x01; // ESDID.
388 GOFFData
[GOFF::RecordLength
+ 71] = (char)0x01; // Size of symbol name.
389 GOFFData
[GOFF::RecordLength
+ 72] = (char)0xa7; // Symbol name is x.
392 GOFFData
[GOFF::RecordLength
* 2] = (char)0x03;
393 GOFFData
[GOFF::RecordLength
* 2 + 3] = (char)0x03;
394 GOFFData
[GOFF::RecordLength
* 2 + 7] = (char)0x02; // ESDID.
395 GOFFData
[GOFF::RecordLength
* 2 + 11] = (char)0x01; // Parent ESDID.
396 GOFFData
[GOFF::RecordLength
* 2 + 71] = (char)0x05; // Size of symbol name.
397 GOFFData
[GOFF::RecordLength
* 2 + 72] = (char)0xC8; // Symbol name is Hello.
398 GOFFData
[GOFF::RecordLength
* 2 + 73] = (char)0x85;
399 GOFFData
[GOFF::RecordLength
* 2 + 74] = (char)0x93;
400 GOFFData
[GOFF::RecordLength
* 2 + 75] = (char)0x93;
401 GOFFData
[GOFF::RecordLength
* 2 + 76] = (char)0x96;
404 GOFFData
[GOFF::RecordLength
* 3] = (char)0x03;
405 GOFFData
[GOFF::RecordLength
* 3 + 1] = (char)0x40;
407 StringRef
Data(GOFFData
, GOFF::RecordLength
* 4);
409 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
410 object::ObjectFile::createGOFFObjectFile(
411 MemoryBufferRef(Data
, "dummyGOFF"));
413 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
415 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
417 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
418 Expected
<StringRef
> SymbolNameOrErr
= GOFFObj
->getSymbolName(Symbol
);
419 ASSERT_THAT_EXPECTED(SymbolNameOrErr
, Succeeded());
420 StringRef SymbolName
= SymbolNameOrErr
.get();
421 EXPECT_EQ(SymbolName
, "Hello");
425 TEST(GOFFObjectFileTest
, InvalidSymbolType
) {
426 char GOFFData
[GOFF::RecordLength
* 3] = {0x00};
429 GOFFData
[0] = (char)0x03;
430 GOFFData
[1] = (char)0xF0;
433 GOFFData
[GOFF::RecordLength
] = (char)0x03;
434 GOFFData
[GOFF::RecordLength
+ 3] = (char)0x05;
435 GOFFData
[GOFF::RecordLength
+ 7] = (char)0x01;
436 GOFFData
[GOFF::RecordLength
+ 11] = (char)0x01;
437 GOFFData
[GOFF::RecordLength
+ 71] = (char)0x01; // Size of symbol name.
438 GOFFData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name.
441 GOFFData
[GOFF::RecordLength
* 2] = (char)0x03;
442 GOFFData
[GOFF::RecordLength
* 2 + 1] = (char)0x40;
444 StringRef
Data(GOFFData
, GOFF::RecordLength
* 3);
446 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
447 object::ObjectFile::createGOFFObjectFile(
448 MemoryBufferRef(Data
, "dummyGOFF"));
450 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
452 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
454 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
455 Expected
<SymbolRef::Type
> SymbolType
= Symbol
.getType();
456 EXPECT_THAT_EXPECTED(
458 FailedWithMessage("ESD record 1 has invalid symbol type 0x05"));
460 Expected
<section_iterator
> SymSI
= Symbol
.getSection();
461 ASSERT_THAT_EXPECTED(
464 "symbol with ESD id 1 refers to invalid section with ESD id 1"));
468 TEST(GOFFObjectFileTest
, InvalidERSymbolType
) {
469 char GOFFData
[GOFF::RecordLength
* 3] = {0x00};
472 GOFFData
[0] = (char)0x03;
473 GOFFData
[1] = (char)0xF0;
476 GOFFData
[GOFF::RecordLength
] = (char)0x03;
477 GOFFData
[GOFF::RecordLength
+ 3] = (char)0x04;
478 GOFFData
[GOFF::RecordLength
+ 7] = (char)0x01;
479 GOFFData
[GOFF::RecordLength
+ 11] = (char)0x01;
480 GOFFData
[GOFF::RecordLength
+ 63] = (char)0x03; // Unknown executable type.
481 GOFFData
[GOFF::RecordLength
+ 71] = (char)0x01; // Size of symbol name.
482 GOFFData
[GOFF::RecordLength
+ 72] = (char)0xC8; // Symbol name.
485 GOFFData
[GOFF::RecordLength
* 2] = (char)0x03;
486 GOFFData
[GOFF::RecordLength
* 2 + 1] = (char)0x40;
488 StringRef
Data(GOFFData
, GOFF::RecordLength
* 3);
490 Expected
<std::unique_ptr
<ObjectFile
>> GOFFObjOrErr
=
491 object::ObjectFile::createGOFFObjectFile(
492 MemoryBufferRef(Data
, "dummyGOFF"));
494 ASSERT_THAT_EXPECTED(GOFFObjOrErr
, Succeeded());
496 GOFFObjectFile
*GOFFObj
= dyn_cast
<GOFFObjectFile
>((*GOFFObjOrErr
).get());
498 for (SymbolRef Symbol
: GOFFObj
->symbols()) {
499 Expected
<SymbolRef::Type
> SymbolType
= Symbol
.getType();
500 EXPECT_THAT_EXPECTED(
502 FailedWithMessage("ESD record 1 has unknown Executable type 0x03"));