[sanitizer] Improve FreeBSD ASLR detection
[llvm-project.git] / llvm / unittests / InterfaceStub / ELFYAMLTest.cpp
blobf6418f8f2a25191536b83260a320f835038dc1c5
1 //===- llvm/unittests/TextAPI/YAMLTest.cpp --------------------------------===//
2 //
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
6 //
7 //===-----------------------------------------------------------------------===/
9 #include "llvm/ADT/StringRef.h"
10 #include "llvm/BinaryFormat/ELF.h"
11 #include "llvm/InterfaceStub/IFSHandler.h"
12 #include "llvm/InterfaceStub/IFSStub.h"
13 #include "llvm/Support/Error.h"
14 #include "llvm/Testing/Support/Error.h"
15 #include "gtest/gtest.h"
16 #include <string>
18 using namespace llvm;
19 using namespace llvm::ELF;
20 using namespace llvm::ifs;
22 void compareByLine(StringRef LHS, StringRef RHS) {
23 StringRef Line1;
24 StringRef Line2;
25 while (LHS.size() > 0 && RHS.size() > 0) {
26 std::tie(Line1, LHS) = LHS.split('\n');
27 std::tie(Line2, RHS) = RHS.split('\n');
28 // Comparing StringRef objects works, but has messy output when not equal.
29 // Using STREQ on StringRef.data() doesn't work since these substrings are
30 // not null terminated.
31 // This is inefficient, but forces null terminated strings that can be
32 // cleanly compared.
33 EXPECT_STREQ(Line1.str().data(), Line2.str().data());
37 TEST(ElfYamlTextAPI, YAMLReadableTBE) {
38 const char Data[] = "--- !ifs-v1\n"
39 "IfsVersion: 1.0\n"
40 "Target: { ObjectFormat: ELF, Arch: x86_64, Endianness: "
41 "little, BitWidth: 64 }\n"
42 "NeededLibs: [libc.so, libfoo.so, libbar.so]\n"
43 "Symbols:\n"
44 " - { Name: foo, Type: Func, Undefined: true }\n"
45 "...\n";
46 Expected<std::unique_ptr<IFSStub>> StubOrErr = readIFSFromBuffer(Data);
47 ASSERT_THAT_ERROR(StubOrErr.takeError(), Succeeded());
48 std::unique_ptr<IFSStub> Stub = std::move(StubOrErr.get());
49 EXPECT_NE(Stub.get(), nullptr);
50 EXPECT_FALSE(Stub->SoName.hasValue());
51 EXPECT_TRUE(Stub->Target.Arch.hasValue());
52 EXPECT_EQ(Stub->Target.Arch.getValue(), (uint16_t)llvm::ELF::EM_X86_64);
53 EXPECT_EQ(Stub->NeededLibs.size(), 3u);
54 EXPECT_STREQ(Stub->NeededLibs[0].c_str(), "libc.so");
55 EXPECT_STREQ(Stub->NeededLibs[1].c_str(), "libfoo.so");
56 EXPECT_STREQ(Stub->NeededLibs[2].c_str(), "libbar.so");
59 TEST(ElfYamlTextAPI, YAMLReadsTBESymbols) {
60 const char Data[] =
61 "--- !ifs-v1\n"
62 "IfsVersion: 1.0\n"
63 "SoName: test.so\n"
64 "Target: { ObjectFormat: ELF, Arch: x86_64, Endianness: little, "
65 "BitWidth: 64 }\n"
66 "Symbols:\n"
67 " - { Name: bar, Type: Object, Size: 42 }\n"
68 " - { Name: baz, Type: TLS, Size: 3 }\n"
69 " - { Name: foo, Type: Func, Warning: \"Deprecated!\" }\n"
70 " - { Name: nor, Type: NoType, Undefined: true }\n"
71 " - { Name: not, Type: File, Undefined: true, Size: 111, "
72 "Weak: true, Warning: \'All fields populated!\' }\n"
73 "...\n";
74 Expected<std::unique_ptr<IFSStub>> StubOrErr = readIFSFromBuffer(Data);
75 ASSERT_THAT_ERROR(StubOrErr.takeError(), Succeeded());
76 std::unique_ptr<IFSStub> Stub = std::move(StubOrErr.get());
77 EXPECT_NE(Stub.get(), nullptr);
78 EXPECT_TRUE(Stub->SoName.hasValue());
79 EXPECT_STREQ(Stub->SoName->c_str(), "test.so");
80 EXPECT_EQ(Stub->Symbols.size(), 5u);
82 auto Iterator = Stub->Symbols.begin();
83 IFSSymbol const &SymBar = *Iterator++;
84 EXPECT_STREQ(SymBar.Name.c_str(), "bar");
85 EXPECT_EQ(SymBar.Size, 42u);
86 EXPECT_EQ(SymBar.Type, IFSSymbolType::Object);
87 EXPECT_FALSE(SymBar.Undefined);
88 EXPECT_FALSE(SymBar.Weak);
89 EXPECT_FALSE(SymBar.Warning.hasValue());
91 IFSSymbol const &SymBaz = *Iterator++;
92 EXPECT_STREQ(SymBaz.Name.c_str(), "baz");
93 EXPECT_EQ(SymBaz.Size, 3u);
94 EXPECT_EQ(SymBaz.Type, IFSSymbolType::TLS);
95 EXPECT_FALSE(SymBaz.Undefined);
96 EXPECT_FALSE(SymBaz.Weak);
97 EXPECT_FALSE(SymBaz.Warning.hasValue());
99 IFSSymbol const &SymFoo = *Iterator++;
100 EXPECT_STREQ(SymFoo.Name.c_str(), "foo");
101 EXPECT_EQ(SymFoo.Size, 0u);
102 EXPECT_EQ(SymFoo.Type, IFSSymbolType::Func);
103 EXPECT_FALSE(SymFoo.Undefined);
104 EXPECT_FALSE(SymFoo.Weak);
105 EXPECT_TRUE(SymFoo.Warning.hasValue());
106 EXPECT_STREQ(SymFoo.Warning->c_str(), "Deprecated!");
108 IFSSymbol const &SymNor = *Iterator++;
109 EXPECT_STREQ(SymNor.Name.c_str(), "nor");
110 EXPECT_EQ(SymNor.Size, 0u);
111 EXPECT_EQ(SymNor.Type, IFSSymbolType::NoType);
112 EXPECT_TRUE(SymNor.Undefined);
113 EXPECT_FALSE(SymNor.Weak);
114 EXPECT_FALSE(SymNor.Warning.hasValue());
116 IFSSymbol const &SymNot = *Iterator++;
117 EXPECT_STREQ(SymNot.Name.c_str(), "not");
118 EXPECT_EQ(SymNot.Size, 111u);
119 EXPECT_EQ(SymNot.Type, IFSSymbolType::Unknown);
120 EXPECT_TRUE(SymNot.Undefined);
121 EXPECT_TRUE(SymNot.Weak);
122 EXPECT_TRUE(SymNot.Warning.hasValue());
123 EXPECT_STREQ(SymNot.Warning->c_str(), "All fields populated!");
126 TEST(ElfYamlTextAPI, YAMLReadsNoTBESyms) {
127 const char Data[] = "--- !ifs-v1\n"
128 "IfsVersion: 1.0\n"
129 "SoName: test.so\n"
130 "Target: { ObjectFormat: ELF, Arch: x86_64, Endianness: "
131 "little, BitWidth: 64 }\n"
132 "Symbols: []\n"
133 "...\n";
134 Expected<std::unique_ptr<IFSStub>> StubOrErr = readIFSFromBuffer(Data);
135 ASSERT_THAT_ERROR(StubOrErr.takeError(), Succeeded());
136 std::unique_ptr<IFSStub> Stub = std::move(StubOrErr.get());
137 EXPECT_NE(Stub.get(), nullptr);
138 EXPECT_EQ(0u, Stub->Symbols.size());
141 TEST(ElfYamlTextAPI, YAMLUnreadableTBE) {
142 // Can't read: wrong format/version.
143 const char Data[] = "--- !tapi-tbz\n"
144 "IfsVersion: z.3\n"
145 "SoName: test.so\n"
146 "Target: { ObjectFormat: ELF, Arch: x86_64, Endianness: "
147 "little, BitWidth: 64 }\n"
148 "Symbols:\n"
149 " foo: { Type: Func, Undefined: true }\n";
150 Expected<std::unique_ptr<IFSStub>> StubOrErr = readIFSFromBuffer(Data);
151 ASSERT_THAT_ERROR(StubOrErr.takeError(), Failed());
154 TEST(ElfYamlTextAPI, YAMLUnsupportedVersion) {
155 const char Data[] = "--- !ifs-v1\n"
156 "IfsVersion: 9.9.9\n"
157 "SoName: test.so\n"
158 "Target: { ObjectFormat: ELF, Arch: x86_64, Endianness: "
159 "little, BitWidth: 64 }\n"
160 "Symbols: []\n"
161 "...\n";
162 Expected<std::unique_ptr<IFSStub>> StubOrErr = readIFSFromBuffer(Data);
163 std::string ErrorMessage = toString(StubOrErr.takeError());
164 EXPECT_EQ("IFS version 9.9.9 is unsupported.", ErrorMessage);
167 TEST(ElfYamlTextAPI, YAMLWritesTBESymbols) {
168 const char Expected[] =
169 "--- !ifs-v1\n"
170 "IfsVersion: 1.0\n"
171 "Target: { ObjectFormat: ELF, Arch: AArch64, Endianness: "
172 "little, BitWidth: 64 }\n"
173 "Symbols:\n"
174 " - { Name: bar, Type: Func, Weak: true }\n"
175 " - { Name: foo, Type: NoType, Size: 99, Warning: Does nothing }\n"
176 " - { Name: nor, Type: Func, Undefined: true }\n"
177 " - { Name: not, Type: Unknown, Size: 12345678901234 }\n"
178 "...\n";
179 IFSStub Stub;
180 Stub.IfsVersion = VersionTuple(1, 0);
181 Stub.Target.Arch = ELF::EM_AARCH64;
182 Stub.Target.BitWidth = IFSBitWidthType::IFS64;
183 Stub.Target.Endianness = IFSEndiannessType::Little;
184 Stub.Target.ObjectFormat = "ELF";
186 IFSSymbol SymBar("bar");
187 SymBar.Size = 128u;
188 SymBar.Type = IFSSymbolType::Func;
189 SymBar.Undefined = false;
190 SymBar.Weak = true;
192 IFSSymbol SymFoo("foo");
193 SymFoo.Size = 99u;
194 SymFoo.Type = IFSSymbolType::NoType;
195 SymFoo.Undefined = false;
196 SymFoo.Weak = false;
197 SymFoo.Warning = "Does nothing";
199 IFSSymbol SymNor("nor");
200 SymNor.Size = 1234u;
201 SymNor.Type = IFSSymbolType::Func;
202 SymNor.Undefined = true;
203 SymNor.Weak = false;
205 IFSSymbol SymNot("not");
206 SymNot.Size = 12345678901234u;
207 SymNot.Type = IFSSymbolType::Unknown;
208 SymNot.Undefined = false;
209 SymNot.Weak = false;
211 // Symbol order is preserved instead of being sorted.
212 Stub.Symbols.push_back(SymBar);
213 Stub.Symbols.push_back(SymFoo);
214 Stub.Symbols.push_back(SymNor);
215 Stub.Symbols.push_back(SymNot);
217 // Ensure move constructor works as expected.
218 IFSStub Moved = std::move(Stub);
220 std::string Result;
221 raw_string_ostream OS(Result);
222 ASSERT_THAT_ERROR(writeIFSToOutputStream(OS, Moved), Succeeded());
223 Result = OS.str();
224 compareByLine(Result.c_str(), Expected);
227 TEST(ElfYamlTextAPI, YAMLWritesNoTBESyms) {
228 const char Expected[] = "--- !ifs-v1\n"
229 "IfsVersion: 1.0\n"
230 "SoName: nosyms.so\n"
231 "Target: { ObjectFormat: ELF, Arch: x86_64, "
232 "Endianness: little, BitWidth: 64 }\n"
233 "NeededLibs:\n"
234 " - libc.so\n"
235 " - libfoo.so\n"
236 " - libbar.so\n"
237 "Symbols: []\n"
238 "...\n";
239 IFSStub Stub;
240 Stub.IfsVersion = VersionTuple(1, 0);
241 Stub.SoName = "nosyms.so";
242 Stub.Target.Arch = ELF::EM_X86_64;
243 Stub.Target.BitWidth = IFSBitWidthType::IFS64;
244 Stub.Target.Endianness = IFSEndiannessType::Little;
245 Stub.Target.ObjectFormat = "ELF";
246 Stub.NeededLibs.push_back("libc.so");
247 Stub.NeededLibs.push_back("libfoo.so");
248 Stub.NeededLibs.push_back("libbar.so");
250 std::string Result;
251 raw_string_ostream OS(Result);
252 ASSERT_THAT_ERROR(writeIFSToOutputStream(OS, Stub), Succeeded());
253 Result = OS.str();
254 compareByLine(Result.c_str(), Expected);