aboutsummaryrefslogtreecommitdiff
path: root/llvm/unittests/Object/BuildIDTest.cpp
blob: 04ca6364fbc4dc000c95e372e329652130c9f5a6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
//===- BuildIDTest.cpp - Tests for getBuildID ----------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

#include "llvm/Object/BuildID.h"
#include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/Object/ELFObjectFile.h"
#include "llvm/ObjectYAML/yaml2obj.h"
#include "llvm/Support/YAMLTraits.h"
#include "llvm/Testing/Support/Error.h"

#include "gtest/gtest.h"

using namespace llvm;
using namespace llvm::object;

template <class ELFT>
static Expected<ELFObjectFile<ELFT>> toBinary(SmallVectorImpl<char> &Storage,
                                              StringRef Yaml) {
  raw_svector_ostream OS(Storage);
  yaml::Input YIn(Yaml);
  if (!yaml::convertYAML(YIn, OS, [](const Twine &Msg) {}))
    return createStringError(std::errc::invalid_argument,
                             "unable to convert YAML");
  return ELFObjectFile<ELFT>::create(MemoryBufferRef(OS.str(), "dummyELF"));
}

static StringRef getInvalidNoteELF(bool WithShdr) {
  static std::string WithSection(R"(
--- !ELF
FileHeader:
  Class:          ELFCLASS64
  Data:           ELFDATA2LSB
  Type:           ET_EXEC
  Machine:        EM_X86_64
ProgramHeaders:
  - Type:         PT_NOTE
    FileSize:     0x1a
    FirstSec:     .note.gnu.build-id
    LastSec:      .note.gnu.build-id
Sections:
  - Name:         .note.gnu.build-id
    Type:         SHT_NOTE
    AddressAlign: 0x04
    Notes:
      - Name:     "GNU"
        Desc:     "abb50d82b6bdc861"
        Type:     3
)");
  static std::string WithoutSection(WithSection + R"(
  - Type:         SectionHeaderTable
    NoHeaders:    true
)");
  if (WithShdr)
    return WithSection;
  return WithoutSection;
}

// The BuildID can be looked up from a section header, if there is no program
// header.
TEST(BuildIDTest, InvalidPhdrFileSizeWithShdrs) {
  SmallString<0> Storage;
  Expected<ELFObjectFile<ELF64LE>> ElfOrErr =
      toBinary<ELF64LE>(Storage, getInvalidNoteELF(true));
  ASSERT_THAT_EXPECTED(ElfOrErr, Succeeded());
  BuildIDRef BuildID = getBuildID(&ElfOrErr.get());
  EXPECT_EQ(
      StringRef(reinterpret_cast<const char *>(BuildID.data()), BuildID.size()),
      "\xAB\xB5\x0D\x82\xB6\xBD\xC8\x61");
}

// The code handles a malformed program header that points at data outside the
// file.
TEST(BuildIDTest, InvalidPhdrFileSizeNoShdrs) {
  SmallString<0> Storage;
  Expected<ELFObjectFile<ELF64LE>> ElfOrErr =
      toBinary<ELF64LE>(Storage, getInvalidNoteELF(false));
  ASSERT_THAT_EXPECTED(ElfOrErr, Succeeded());
  BuildIDRef BuildID = getBuildID(&ElfOrErr.get());
  EXPECT_EQ(
      StringRef(reinterpret_cast<const char *>(BuildID.data()), BuildID.size()),
      "");
}

// The code handles a malformed section header that points at data outside the
// file.
TEST(BuildIDTest, InvalidSectionHeader) {
  SmallString<0> Storage;
  Expected<ELFObjectFile<ELF64LE>> ElfOrErr = toBinary<ELF64LE>(Storage, R"(
--- !ELF
FileHeader:
  Class:          ELFCLASS64
  Data:           ELFDATA2LSB
  Type:           ET_EXEC
  Machine:        EM_X86_64
ProgramHeaders:
  - Type:         PT_NOTE
    FirstSec:     .note.gnu.build-id
    LastSec:      .note.gnu.build-id
Sections:
  - Name:         .note.gnu.build-id
    Type:         SHT_NOTE
    AddressAlign: 0x04
    ShOffset:     0x1a1
    Notes:
      - Name:     "GNU"
        Desc:     "abb50d82b6bdc861"
        Type:     3
)");
  ASSERT_THAT_EXPECTED(ElfOrErr, Succeeded());
  BuildIDRef BuildID = getBuildID(&ElfOrErr.get());
  EXPECT_EQ(
      StringRef(reinterpret_cast<const char *>(BuildID.data()), BuildID.size()),
      "\xAB\xB5\x0D\x82\xB6\xBD\xC8\x61");
}