1 //===- unittest/ProfileData/InstrProfTest.cpp -------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "llvm/IR/Function.h"
11 #include "llvm/IR/IRBuilder.h"
12 #include "llvm/IR/LLVMContext.h"
13 #include "llvm/IR/Module.h"
14 #include "llvm/ProfileData/InstrProfReader.h"
15 #include "llvm/ProfileData/InstrProfWriter.h"
16 #include "llvm/Support/Compression.h"
17 #include "llvm/Testing/Support/Error.h"
18 #include "llvm/Testing/Support/SupportHelpers.h"
19 #include "gtest/gtest.h"
24 LLVM_NODISCARD
static ::testing::AssertionResult
25 ErrorEquals(instrprof_error Expected
, Error E
) {
26 instrprof_error Found
;
28 handleAllErrors(std::move(E
), [&](const InstrProfError
&IPE
) {
30 FoundMsg
= IPE
.message();
32 if (Expected
== Found
)
33 return ::testing::AssertionSuccess();
34 return ::testing::AssertionFailure() << "error: " << FoundMsg
<< "\n";
39 struct InstrProfTest
: ::testing::Test
{
40 InstrProfWriter Writer
;
41 std::unique_ptr
<IndexedInstrProfReader
> Reader
;
43 void SetUp() { Writer
.setOutputSparse(false); }
45 void readProfile(std::unique_ptr
<MemoryBuffer
> Profile
) {
46 auto ReaderOrErr
= IndexedInstrProfReader::create(std::move(Profile
));
47 EXPECT_THAT_ERROR(ReaderOrErr
.takeError(), Succeeded());
48 Reader
= std::move(ReaderOrErr
.get());
52 struct SparseInstrProfTest
: public InstrProfTest
{
53 void SetUp() { Writer
.setOutputSparse(true); }
56 struct MaybeSparseInstrProfTest
: public InstrProfTest
,
57 public ::testing::WithParamInterface
<bool> {
58 void SetUp() { Writer
.setOutputSparse(GetParam()); }
61 TEST_P(MaybeSparseInstrProfTest
, write_and_read_empty_profile
) {
62 auto Profile
= Writer
.writeBuffer();
63 readProfile(std::move(Profile
));
64 ASSERT_TRUE(Reader
->begin() == Reader
->end());
67 static const auto Err
= [](Error E
) {
68 consumeError(std::move(E
));
72 TEST_P(MaybeSparseInstrProfTest
, write_and_read_one_function
) {
73 Writer
.addRecord({"foo", 0x1234, {1, 2, 3, 4}}, Err
);
74 auto Profile
= Writer
.writeBuffer();
75 readProfile(std::move(Profile
));
77 auto I
= Reader
->begin(), E
= Reader
->end();
79 ASSERT_EQ(StringRef("foo"), I
->Name
);
80 ASSERT_EQ(0x1234U
, I
->Hash
);
81 ASSERT_EQ(4U, I
->Counts
.size());
82 ASSERT_EQ(1U, I
->Counts
[0]);
83 ASSERT_EQ(2U, I
->Counts
[1]);
84 ASSERT_EQ(3U, I
->Counts
[2]);
85 ASSERT_EQ(4U, I
->Counts
[3]);
86 ASSERT_TRUE(++I
== E
);
89 TEST_P(MaybeSparseInstrProfTest
, get_instr_prof_record
) {
90 Writer
.addRecord({"foo", 0x1234, {1, 2}}, Err
);
91 Writer
.addRecord({"foo", 0x1235, {3, 4}}, Err
);
92 auto Profile
= Writer
.writeBuffer();
93 readProfile(std::move(Profile
));
95 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("foo", 0x1234);
96 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
97 ASSERT_EQ(2U, R
->Counts
.size());
98 ASSERT_EQ(1U, R
->Counts
[0]);
99 ASSERT_EQ(2U, R
->Counts
[1]);
101 R
= Reader
->getInstrProfRecord("foo", 0x1235);
102 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
103 ASSERT_EQ(2U, R
->Counts
.size());
104 ASSERT_EQ(3U, R
->Counts
[0]);
105 ASSERT_EQ(4U, R
->Counts
[1]);
107 R
= Reader
->getInstrProfRecord("foo", 0x5678);
108 ASSERT_TRUE(ErrorEquals(instrprof_error::hash_mismatch
, R
.takeError()));
110 R
= Reader
->getInstrProfRecord("bar", 0x1234);
111 ASSERT_TRUE(ErrorEquals(instrprof_error::unknown_function
, R
.takeError()));
114 TEST_P(MaybeSparseInstrProfTest
, get_function_counts
) {
115 Writer
.addRecord({"foo", 0x1234, {1, 2}}, Err
);
116 Writer
.addRecord({"foo", 0x1235, {3, 4}}, Err
);
117 auto Profile
= Writer
.writeBuffer();
118 readProfile(std::move(Profile
));
120 std::vector
<uint64_t> Counts
;
121 EXPECT_THAT_ERROR(Reader
->getFunctionCounts("foo", 0x1234, Counts
),
123 ASSERT_EQ(2U, Counts
.size());
124 ASSERT_EQ(1U, Counts
[0]);
125 ASSERT_EQ(2U, Counts
[1]);
127 EXPECT_THAT_ERROR(Reader
->getFunctionCounts("foo", 0x1235, Counts
),
129 ASSERT_EQ(2U, Counts
.size());
130 ASSERT_EQ(3U, Counts
[0]);
131 ASSERT_EQ(4U, Counts
[1]);
133 Error E1
= Reader
->getFunctionCounts("foo", 0x5678, Counts
);
134 ASSERT_TRUE(ErrorEquals(instrprof_error::hash_mismatch
, std::move(E1
)));
136 Error E2
= Reader
->getFunctionCounts("bar", 0x1234, Counts
);
137 ASSERT_TRUE(ErrorEquals(instrprof_error::unknown_function
, std::move(E2
)));
140 // Profile data is copied from general.proftext
141 TEST_F(InstrProfTest
, get_profile_summary
) {
142 Writer
.addRecord({"func1", 0x1234, {97531}}, Err
);
143 Writer
.addRecord({"func2", 0x1234, {0, 0}}, Err
);
147 {2305843009213693952, 1152921504606846976, 576460752303423488,
148 288230376151711744, 144115188075855872, 72057594037927936}},
150 Writer
.addRecord({"func4", 0x1234, {0}}, Err
);
151 auto Profile
= Writer
.writeBuffer();
152 readProfile(std::move(Profile
));
154 auto VerifySummary
= [](ProfileSummary
&IPS
) mutable {
155 ASSERT_EQ(ProfileSummary::PSK_Instr
, IPS
.getKind());
156 ASSERT_EQ(2305843009213693952U, IPS
.getMaxFunctionCount());
157 ASSERT_EQ(2305843009213693952U, IPS
.getMaxCount());
158 ASSERT_EQ(10U, IPS
.getNumCounts());
159 ASSERT_EQ(4539628424389557499U, IPS
.getTotalCount());
160 std::vector
<ProfileSummaryEntry
> &Details
= IPS
.getDetailedSummary();
161 uint32_t Cutoff
= 800000;
162 auto Predicate
= [&Cutoff
](const ProfileSummaryEntry
&PE
) {
163 return PE
.Cutoff
== Cutoff
;
165 auto EightyPerc
= find_if(Details
, Predicate
);
167 auto NinetyPerc
= find_if(Details
, Predicate
);
169 auto NinetyFivePerc
= find_if(Details
, Predicate
);
171 auto NinetyNinePerc
= find_if(Details
, Predicate
);
172 ASSERT_EQ(576460752303423488U, EightyPerc
->MinCount
);
173 ASSERT_EQ(288230376151711744U, NinetyPerc
->MinCount
);
174 ASSERT_EQ(288230376151711744U, NinetyFivePerc
->MinCount
);
175 ASSERT_EQ(72057594037927936U, NinetyNinePerc
->MinCount
);
177 ProfileSummary
&PS
= Reader
->getSummary();
180 // Test that conversion of summary to and from Metadata works.
182 Metadata
*MD
= PS
.getMD(Context
);
184 ProfileSummary
*PSFromMD
= ProfileSummary::getFromMD(MD
);
185 ASSERT_TRUE(PSFromMD
);
186 VerifySummary(*PSFromMD
);
189 // Test that summary can be attached to and read back from module.
190 Module
M("my_module", Context
);
191 M
.setProfileSummary(MD
);
192 MD
= M
.getProfileSummary();
194 PSFromMD
= ProfileSummary::getFromMD(MD
);
195 ASSERT_TRUE(PSFromMD
);
196 VerifySummary(*PSFromMD
);
200 TEST_F(InstrProfTest
, test_writer_merge
) {
201 Writer
.addRecord({"func1", 0x1234, {42}}, Err
);
203 InstrProfWriter Writer2
;
204 Writer2
.addRecord({"func2", 0x1234, {0, 0}}, Err
);
206 Writer
.mergeRecordsFromWriter(std::move(Writer2
), Err
);
208 auto Profile
= Writer
.writeBuffer();
209 readProfile(std::move(Profile
));
211 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("func1", 0x1234);
212 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
213 ASSERT_EQ(1U, R
->Counts
.size());
214 ASSERT_EQ(42U, R
->Counts
[0]);
216 R
= Reader
->getInstrProfRecord("func2", 0x1234);
217 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
218 ASSERT_EQ(2U, R
->Counts
.size());
219 ASSERT_EQ(0U, R
->Counts
[0]);
220 ASSERT_EQ(0U, R
->Counts
[1]);
223 static const char callee1
[] = "callee1";
224 static const char callee2
[] = "callee2";
225 static const char callee3
[] = "callee3";
226 static const char callee4
[] = "callee4";
227 static const char callee5
[] = "callee5";
228 static const char callee6
[] = "callee6";
230 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_read_write
) {
231 NamedInstrProfRecord
Record1("caller", 0x1234, {1, 2});
234 Record1
.reserveSites(IPVK_IndirectCallTarget
, 4);
235 InstrProfValueData VD0
[] = {
236 {(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}, {(uint64_t)callee3
, 3}};
237 Record1
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 3, nullptr);
238 // No value profile data at the second site.
239 Record1
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
240 InstrProfValueData VD2
[] = {{(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}};
241 Record1
.addValueData(IPVK_IndirectCallTarget
, 2, VD2
, 2, nullptr);
242 InstrProfValueData VD3
[] = {{(uint64_t)callee1
, 1}};
243 Record1
.addValueData(IPVK_IndirectCallTarget
, 3, VD3
, 1, nullptr);
245 Writer
.addRecord(std::move(Record1
), Err
);
246 Writer
.addRecord({"callee1", 0x1235, {3, 4}}, Err
);
247 Writer
.addRecord({"callee2", 0x1235, {3, 4}}, Err
);
248 Writer
.addRecord({"callee3", 0x1235, {3, 4}}, Err
);
249 auto Profile
= Writer
.writeBuffer();
250 readProfile(std::move(Profile
));
252 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
253 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
254 ASSERT_EQ(4U, R
->getNumValueSites(IPVK_IndirectCallTarget
));
255 ASSERT_EQ(3U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
256 ASSERT_EQ(0U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 1));
257 ASSERT_EQ(2U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 2));
258 ASSERT_EQ(1U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 3));
261 std::unique_ptr
<InstrProfValueData
[]> VD
=
262 R
->getValueForSite(IPVK_IndirectCallTarget
, 0, &TotalC
);
264 ASSERT_EQ(3U, VD
[0].Count
);
265 ASSERT_EQ(2U, VD
[1].Count
);
266 ASSERT_EQ(1U, VD
[2].Count
);
267 ASSERT_EQ(6U, TotalC
);
269 ASSERT_EQ(StringRef((const char *)VD
[0].Value
, 7), StringRef("callee3"));
270 ASSERT_EQ(StringRef((const char *)VD
[1].Value
, 7), StringRef("callee2"));
271 ASSERT_EQ(StringRef((const char *)VD
[2].Value
, 7), StringRef("callee1"));
274 TEST_P(MaybeSparseInstrProfTest
, annotate_vp_data
) {
275 NamedInstrProfRecord
Record("caller", 0x1234, {1, 2});
276 Record
.reserveSites(IPVK_IndirectCallTarget
, 1);
277 InstrProfValueData VD0
[] = {{1000, 1}, {2000, 2}, {3000, 3}, {5000, 5},
278 {4000, 4}, {6000, 6}};
279 Record
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 6, nullptr);
280 Writer
.addRecord(std::move(Record
), Err
);
281 auto Profile
= Writer
.writeBuffer();
282 readProfile(std::move(Profile
));
283 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
284 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
287 std::unique_ptr
<Module
> M(new Module("MyModule", Ctx
));
288 FunctionType
*FTy
= FunctionType::get(Type::getVoidTy(Ctx
),
291 Function::Create(FTy
, Function::ExternalLinkage
, "caller", M
.get());
292 BasicBlock
*BB
= BasicBlock::Create(Ctx
, "", F
);
294 IRBuilder
<> Builder(BB
);
295 BasicBlock
*TBB
= BasicBlock::Create(Ctx
, "", F
);
296 BasicBlock
*FBB
= BasicBlock::Create(Ctx
, "", F
);
298 // Use branch instruction to annotate with value profile data for simplicity
299 Instruction
*Inst
= Builder
.CreateCondBr(Builder
.getTrue(), TBB
, FBB
);
300 Instruction
*Inst2
= Builder
.CreateCondBr(Builder
.getTrue(), TBB
, FBB
);
301 annotateValueSite(*M
, *Inst
, R
.get(), IPVK_IndirectCallTarget
, 0);
303 InstrProfValueData ValueData
[5];
306 bool Res
= getValueProfDataFromInst(*Inst
, IPVK_IndirectCallTarget
, 5,
311 // The result should be sorted already:
312 ASSERT_EQ(6000U, ValueData
[0].Value
);
313 ASSERT_EQ(6U, ValueData
[0].Count
);
314 ASSERT_EQ(5000U, ValueData
[1].Value
);
315 ASSERT_EQ(5U, ValueData
[1].Count
);
316 ASSERT_EQ(4000U, ValueData
[2].Value
);
317 ASSERT_EQ(4U, ValueData
[2].Count
);
318 Res
= getValueProfDataFromInst(*Inst
, IPVK_IndirectCallTarget
, 1, ValueData
,
324 Res
= getValueProfDataFromInst(*Inst2
, IPVK_IndirectCallTarget
, 5, ValueData
,
328 // Remove the MD_prof metadata
329 Inst
->setMetadata(LLVMContext::MD_prof
, 0);
330 // Annotate 5 records this time.
331 annotateValueSite(*M
, *Inst
, R
.get(), IPVK_IndirectCallTarget
, 0, 5);
332 Res
= getValueProfDataFromInst(*Inst
, IPVK_IndirectCallTarget
, 5,
337 ASSERT_EQ(6000U, ValueData
[0].Value
);
338 ASSERT_EQ(6U, ValueData
[0].Count
);
339 ASSERT_EQ(5000U, ValueData
[1].Value
);
340 ASSERT_EQ(5U, ValueData
[1].Count
);
341 ASSERT_EQ(4000U, ValueData
[2].Value
);
342 ASSERT_EQ(4U, ValueData
[2].Count
);
343 ASSERT_EQ(3000U, ValueData
[3].Value
);
344 ASSERT_EQ(3U, ValueData
[3].Count
);
345 ASSERT_EQ(2000U, ValueData
[4].Value
);
346 ASSERT_EQ(2U, ValueData
[4].Count
);
348 // Remove the MD_prof metadata
349 Inst
->setMetadata(LLVMContext::MD_prof
, 0);
350 // Annotate with 4 records.
351 InstrProfValueData VD0Sorted
[] = {{1000, 6}, {2000, 5}, {3000, 4}, {4000, 3},
352 {5000, 2}, {6000, 1}};
353 annotateValueSite(*M
, *Inst
, makeArrayRef(VD0Sorted
).slice(2), 10,
354 IPVK_IndirectCallTarget
, 5);
355 Res
= getValueProfDataFromInst(*Inst
, IPVK_IndirectCallTarget
, 5,
360 ASSERT_EQ(3000U, ValueData
[0].Value
);
361 ASSERT_EQ(4U, ValueData
[0].Count
);
362 ASSERT_EQ(4000U, ValueData
[1].Value
);
363 ASSERT_EQ(3U, ValueData
[1].Count
);
364 ASSERT_EQ(5000U, ValueData
[2].Value
);
365 ASSERT_EQ(2U, ValueData
[2].Count
);
366 ASSERT_EQ(6000U, ValueData
[3].Value
);
367 ASSERT_EQ(1U, ValueData
[3].Count
);
370 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_read_write_with_weight
) {
371 NamedInstrProfRecord
Record1("caller", 0x1234, {1, 2});
374 Record1
.reserveSites(IPVK_IndirectCallTarget
, 4);
375 InstrProfValueData VD0
[] = {
376 {(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}, {(uint64_t)callee3
, 3}};
377 Record1
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 3, nullptr);
378 // No value profile data at the second site.
379 Record1
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
380 InstrProfValueData VD2
[] = {{(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}};
381 Record1
.addValueData(IPVK_IndirectCallTarget
, 2, VD2
, 2, nullptr);
382 InstrProfValueData VD3
[] = {{(uint64_t)callee1
, 1}};
383 Record1
.addValueData(IPVK_IndirectCallTarget
, 3, VD3
, 1, nullptr);
385 Writer
.addRecord(std::move(Record1
), 10, Err
);
386 Writer
.addRecord({"callee1", 0x1235, {3, 4}}, Err
);
387 Writer
.addRecord({"callee2", 0x1235, {3, 4}}, Err
);
388 Writer
.addRecord({"callee3", 0x1235, {3, 4}}, Err
);
389 auto Profile
= Writer
.writeBuffer();
390 readProfile(std::move(Profile
));
392 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
393 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
394 ASSERT_EQ(4U, R
->getNumValueSites(IPVK_IndirectCallTarget
));
395 ASSERT_EQ(3U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
396 ASSERT_EQ(0U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 1));
397 ASSERT_EQ(2U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 2));
398 ASSERT_EQ(1U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 3));
401 std::unique_ptr
<InstrProfValueData
[]> VD
=
402 R
->getValueForSite(IPVK_IndirectCallTarget
, 0, &TotalC
);
403 ASSERT_EQ(30U, VD
[0].Count
);
404 ASSERT_EQ(20U, VD
[1].Count
);
405 ASSERT_EQ(10U, VD
[2].Count
);
406 ASSERT_EQ(60U, TotalC
);
408 ASSERT_EQ(StringRef((const char *)VD
[0].Value
, 7), StringRef("callee3"));
409 ASSERT_EQ(StringRef((const char *)VD
[1].Value
, 7), StringRef("callee2"));
410 ASSERT_EQ(StringRef((const char *)VD
[2].Value
, 7), StringRef("callee1"));
413 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_read_write_big_endian
) {
414 NamedInstrProfRecord
Record1("caller", 0x1234, {1, 2});
417 Record1
.reserveSites(IPVK_IndirectCallTarget
, 4);
418 InstrProfValueData VD0
[] = {
419 {(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}, {(uint64_t)callee3
, 3}};
420 Record1
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 3, nullptr);
421 // No value profile data at the second site.
422 Record1
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
423 InstrProfValueData VD2
[] = {{(uint64_t)callee1
, 1}, {(uint64_t)callee2
, 2}};
424 Record1
.addValueData(IPVK_IndirectCallTarget
, 2, VD2
, 2, nullptr);
425 InstrProfValueData VD3
[] = {{(uint64_t)callee1
, 1}};
426 Record1
.addValueData(IPVK_IndirectCallTarget
, 3, VD3
, 1, nullptr);
428 Writer
.addRecord(std::move(Record1
), Err
);
429 Writer
.addRecord({"callee1", 0x1235, {3, 4}}, Err
);
430 Writer
.addRecord({"callee2", 0x1235, {3, 4}}, Err
);
431 Writer
.addRecord({"callee3", 0x1235, {3, 4}}, Err
);
433 // Set big endian output.
434 Writer
.setValueProfDataEndianness(support::big
);
436 auto Profile
= Writer
.writeBuffer();
437 readProfile(std::move(Profile
));
439 // Set big endian input.
440 Reader
->setValueProfDataEndianness(support::big
);
442 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
443 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
444 ASSERT_EQ(4U, R
->getNumValueSites(IPVK_IndirectCallTarget
));
445 ASSERT_EQ(3U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
446 ASSERT_EQ(0U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 1));
447 ASSERT_EQ(2U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 2));
448 ASSERT_EQ(1U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 3));
450 std::unique_ptr
<InstrProfValueData
[]> VD
=
451 R
->getValueForSite(IPVK_IndirectCallTarget
, 0);
452 ASSERT_EQ(StringRef((const char *)VD
[0].Value
, 7), StringRef("callee3"));
453 ASSERT_EQ(StringRef((const char *)VD
[1].Value
, 7), StringRef("callee2"));
454 ASSERT_EQ(StringRef((const char *)VD
[2].Value
, 7), StringRef("callee1"));
456 // Restore little endian default:
457 Writer
.setValueProfDataEndianness(support::little
);
460 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_merge1
) {
461 static const char caller
[] = "caller";
462 NamedInstrProfRecord
Record11(caller
, 0x1234, {1, 2});
463 NamedInstrProfRecord
Record12(caller
, 0x1234, {1, 2});
466 Record11
.reserveSites(IPVK_IndirectCallTarget
, 5);
467 InstrProfValueData VD0
[] = {{uint64_t(callee1
), 1},
468 {uint64_t(callee2
), 2},
469 {uint64_t(callee3
), 3},
470 {uint64_t(callee4
), 4}};
471 Record11
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 4, nullptr);
473 // No value profile data at the second site.
474 Record11
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
476 InstrProfValueData VD2
[] = {
477 {uint64_t(callee1
), 1}, {uint64_t(callee2
), 2}, {uint64_t(callee3
), 3}};
478 Record11
.addValueData(IPVK_IndirectCallTarget
, 2, VD2
, 3, nullptr);
480 InstrProfValueData VD3
[] = {{uint64_t(callee1
), 1}};
481 Record11
.addValueData(IPVK_IndirectCallTarget
, 3, VD3
, 1, nullptr);
483 InstrProfValueData VD4
[] = {{uint64_t(callee1
), 1},
484 {uint64_t(callee2
), 2},
485 {uint64_t(callee3
), 3}};
486 Record11
.addValueData(IPVK_IndirectCallTarget
, 4, VD4
, 3, nullptr);
488 // A different record for the same caller.
489 Record12
.reserveSites(IPVK_IndirectCallTarget
, 5);
490 InstrProfValueData VD02
[] = {{uint64_t(callee2
), 5}, {uint64_t(callee3
), 3}};
491 Record12
.addValueData(IPVK_IndirectCallTarget
, 0, VD02
, 2, nullptr);
493 // No value profile data at the second site.
494 Record12
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
496 InstrProfValueData VD22
[] = {
497 {uint64_t(callee2
), 1}, {uint64_t(callee3
), 3}, {uint64_t(callee4
), 4}};
498 Record12
.addValueData(IPVK_IndirectCallTarget
, 2, VD22
, 3, nullptr);
500 Record12
.addValueData(IPVK_IndirectCallTarget
, 3, nullptr, 0, nullptr);
502 InstrProfValueData VD42
[] = {{uint64_t(callee1
), 1},
503 {uint64_t(callee2
), 2},
504 {uint64_t(callee3
), 3}};
505 Record12
.addValueData(IPVK_IndirectCallTarget
, 4, VD42
, 3, nullptr);
507 Writer
.addRecord(std::move(Record11
), Err
);
508 // Merge profile data.
509 Writer
.addRecord(std::move(Record12
), Err
);
511 Writer
.addRecord({callee1
, 0x1235, {3, 4}}, Err
);
512 Writer
.addRecord({callee2
, 0x1235, {3, 4}}, Err
);
513 Writer
.addRecord({callee3
, 0x1235, {3, 4}}, Err
);
514 Writer
.addRecord({callee3
, 0x1235, {3, 4}}, Err
);
515 Writer
.addRecord({callee4
, 0x1235, {3, 5}}, Err
);
516 auto Profile
= Writer
.writeBuffer();
517 readProfile(std::move(Profile
));
519 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
520 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
521 ASSERT_EQ(5U, R
->getNumValueSites(IPVK_IndirectCallTarget
));
522 ASSERT_EQ(4U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
523 ASSERT_EQ(0U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 1));
524 ASSERT_EQ(4U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 2));
525 ASSERT_EQ(1U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 3));
526 ASSERT_EQ(3U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 4));
528 std::unique_ptr
<InstrProfValueData
[]> VD
=
529 R
->getValueForSite(IPVK_IndirectCallTarget
, 0);
530 ASSERT_EQ(StringRef((const char *)VD
[0].Value
, 7), StringRef("callee2"));
531 ASSERT_EQ(7U, VD
[0].Count
);
532 ASSERT_EQ(StringRef((const char *)VD
[1].Value
, 7), StringRef("callee3"));
533 ASSERT_EQ(6U, VD
[1].Count
);
534 ASSERT_EQ(StringRef((const char *)VD
[2].Value
, 7), StringRef("callee4"));
535 ASSERT_EQ(4U, VD
[2].Count
);
536 ASSERT_EQ(StringRef((const char *)VD
[3].Value
, 7), StringRef("callee1"));
537 ASSERT_EQ(1U, VD
[3].Count
);
539 std::unique_ptr
<InstrProfValueData
[]> VD_2(
540 R
->getValueForSite(IPVK_IndirectCallTarget
, 2));
541 ASSERT_EQ(StringRef((const char *)VD_2
[0].Value
, 7), StringRef("callee3"));
542 ASSERT_EQ(6U, VD_2
[0].Count
);
543 ASSERT_EQ(StringRef((const char *)VD_2
[1].Value
, 7), StringRef("callee4"));
544 ASSERT_EQ(4U, VD_2
[1].Count
);
545 ASSERT_EQ(StringRef((const char *)VD_2
[2].Value
, 7), StringRef("callee2"));
546 ASSERT_EQ(3U, VD_2
[2].Count
);
547 ASSERT_EQ(StringRef((const char *)VD_2
[3].Value
, 7), StringRef("callee1"));
548 ASSERT_EQ(1U, VD_2
[3].Count
);
550 std::unique_ptr
<InstrProfValueData
[]> VD_3(
551 R
->getValueForSite(IPVK_IndirectCallTarget
, 3));
552 ASSERT_EQ(StringRef((const char *)VD_3
[0].Value
, 7), StringRef("callee1"));
553 ASSERT_EQ(1U, VD_3
[0].Count
);
555 std::unique_ptr
<InstrProfValueData
[]> VD_4(
556 R
->getValueForSite(IPVK_IndirectCallTarget
, 4));
557 ASSERT_EQ(StringRef((const char *)VD_4
[0].Value
, 7), StringRef("callee3"));
558 ASSERT_EQ(6U, VD_4
[0].Count
);
559 ASSERT_EQ(StringRef((const char *)VD_4
[1].Value
, 7), StringRef("callee2"));
560 ASSERT_EQ(4U, VD_4
[1].Count
);
561 ASSERT_EQ(StringRef((const char *)VD_4
[2].Value
, 7), StringRef("callee1"));
562 ASSERT_EQ(2U, VD_4
[2].Count
);
565 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_merge1_saturation
) {
566 static const char bar
[] = "bar";
568 const uint64_t Max
= std::numeric_limits
<uint64_t>::max();
570 instrprof_error Result
;
571 auto Err
= [&](Error E
) { Result
= InstrProfError::take(std::move(E
)); };
572 Result
= instrprof_error::success
;
573 Writer
.addRecord({"foo", 0x1234, {1}}, Err
);
574 ASSERT_EQ(Result
, instrprof_error::success
);
576 // Verify counter overflow.
577 Result
= instrprof_error::success
;
578 Writer
.addRecord({"foo", 0x1234, {Max
}}, Err
);
579 ASSERT_EQ(Result
, instrprof_error::counter_overflow
);
581 Result
= instrprof_error::success
;
582 Writer
.addRecord({bar
, 0x9012, {8}}, Err
);
583 ASSERT_EQ(Result
, instrprof_error::success
);
585 NamedInstrProfRecord
Record4("baz", 0x5678, {3, 4});
586 Record4
.reserveSites(IPVK_IndirectCallTarget
, 1);
587 InstrProfValueData VD4
[] = {{uint64_t(bar
), 1}};
588 Record4
.addValueData(IPVK_IndirectCallTarget
, 0, VD4
, 1, nullptr);
589 Result
= instrprof_error::success
;
590 Writer
.addRecord(std::move(Record4
), Err
);
591 ASSERT_EQ(Result
, instrprof_error::success
);
593 // Verify value data counter overflow.
594 NamedInstrProfRecord
Record5("baz", 0x5678, {5, 6});
595 Record5
.reserveSites(IPVK_IndirectCallTarget
, 1);
596 InstrProfValueData VD5
[] = {{uint64_t(bar
), Max
}};
597 Record5
.addValueData(IPVK_IndirectCallTarget
, 0, VD5
, 1, nullptr);
598 Result
= instrprof_error::success
;
599 Writer
.addRecord(std::move(Record5
), Err
);
600 ASSERT_EQ(Result
, instrprof_error::counter_overflow
);
602 auto Profile
= Writer
.writeBuffer();
603 readProfile(std::move(Profile
));
605 // Verify saturation of counts.
606 Expected
<InstrProfRecord
> ReadRecord1
=
607 Reader
->getInstrProfRecord("foo", 0x1234);
608 EXPECT_THAT_ERROR(ReadRecord1
.takeError(), Succeeded());
609 ASSERT_EQ(Max
, ReadRecord1
->Counts
[0]);
611 Expected
<InstrProfRecord
> ReadRecord2
=
612 Reader
->getInstrProfRecord("baz", 0x5678);
613 ASSERT_TRUE(bool(ReadRecord2
));
614 ASSERT_EQ(1U, ReadRecord2
->getNumValueSites(IPVK_IndirectCallTarget
));
615 std::unique_ptr
<InstrProfValueData
[]> VD
=
616 ReadRecord2
->getValueForSite(IPVK_IndirectCallTarget
, 0);
617 ASSERT_EQ(StringRef("bar"), StringRef((const char *)VD
[0].Value
, 3));
618 ASSERT_EQ(Max
, VD
[0].Count
);
621 // This test tests that when there are too many values
622 // for a given site, the merged results are properly
624 TEST_P(MaybeSparseInstrProfTest
, get_icall_data_merge_site_trunc
) {
625 static const char caller
[] = "caller";
627 NamedInstrProfRecord
Record11(caller
, 0x1234, {1, 2});
628 NamedInstrProfRecord
Record12(caller
, 0x1234, {1, 2});
631 Record11
.reserveSites(IPVK_IndirectCallTarget
, 2);
632 InstrProfValueData VD0
[255];
633 for (int I
= 0; I
< 255; I
++) {
634 VD0
[I
].Value
= 2 * I
;
635 VD0
[I
].Count
= 2 * I
+ 1000;
638 Record11
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 255, nullptr);
639 Record11
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
641 Record12
.reserveSites(IPVK_IndirectCallTarget
, 2);
642 InstrProfValueData VD1
[255];
643 for (int I
= 0; I
< 255; I
++) {
644 VD1
[I
].Value
= 2 * I
+ 1;
645 VD1
[I
].Count
= 2 * I
+ 1001;
648 Record12
.addValueData(IPVK_IndirectCallTarget
, 0, VD1
, 255, nullptr);
649 Record12
.addValueData(IPVK_IndirectCallTarget
, 1, nullptr, 0, nullptr);
651 Writer
.addRecord(std::move(Record11
), Err
);
652 // Merge profile data.
653 Writer
.addRecord(std::move(Record12
), Err
);
655 auto Profile
= Writer
.writeBuffer();
656 readProfile(std::move(Profile
));
658 Expected
<InstrProfRecord
> R
= Reader
->getInstrProfRecord("caller", 0x1234);
659 EXPECT_THAT_ERROR(R
.takeError(), Succeeded());
660 std::unique_ptr
<InstrProfValueData
[]> VD(
661 R
->getValueForSite(IPVK_IndirectCallTarget
, 0));
662 ASSERT_EQ(2U, R
->getNumValueSites(IPVK_IndirectCallTarget
));
663 ASSERT_EQ(255U, R
->getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
664 for (unsigned I
= 0; I
< 255; I
++) {
665 ASSERT_EQ(VD
[I
].Value
, 509 - I
);
666 ASSERT_EQ(VD
[I
].Count
, 1509 - I
);
670 static void addValueProfData(InstrProfRecord
&Record
) {
671 Record
.reserveSites(IPVK_IndirectCallTarget
, 5);
672 InstrProfValueData VD0
[] = {{uint64_t(callee1
), 400},
673 {uint64_t(callee2
), 1000},
674 {uint64_t(callee3
), 500},
675 {uint64_t(callee4
), 300},
676 {uint64_t(callee5
), 100}};
677 Record
.addValueData(IPVK_IndirectCallTarget
, 0, VD0
, 5, nullptr);
678 InstrProfValueData VD1
[] = {{uint64_t(callee5
), 800},
679 {uint64_t(callee3
), 1000},
680 {uint64_t(callee2
), 2500},
681 {uint64_t(callee1
), 1300}};
682 Record
.addValueData(IPVK_IndirectCallTarget
, 1, VD1
, 4, nullptr);
683 InstrProfValueData VD2
[] = {{uint64_t(callee6
), 800},
684 {uint64_t(callee3
), 1000},
685 {uint64_t(callee4
), 5500}};
686 Record
.addValueData(IPVK_IndirectCallTarget
, 2, VD2
, 3, nullptr);
687 InstrProfValueData VD3
[] = {{uint64_t(callee2
), 1800},
688 {uint64_t(callee3
), 2000}};
689 Record
.addValueData(IPVK_IndirectCallTarget
, 3, VD3
, 2, nullptr);
690 Record
.addValueData(IPVK_IndirectCallTarget
, 4, nullptr, 0, nullptr);
693 TEST_P(MaybeSparseInstrProfTest
, value_prof_data_read_write
) {
694 InstrProfRecord
SrcRecord({1ULL << 31, 2});
695 addValueProfData(SrcRecord
);
696 std::unique_ptr
<ValueProfData
> VPData
=
697 ValueProfData::serializeFrom(SrcRecord
);
699 InstrProfRecord
Record({1ULL << 31, 2});
700 VPData
->deserializeTo(Record
, nullptr);
702 // Now read data from Record and sanity check the data
703 ASSERT_EQ(5U, Record
.getNumValueSites(IPVK_IndirectCallTarget
));
704 ASSERT_EQ(5U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
705 ASSERT_EQ(4U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 1));
706 ASSERT_EQ(3U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 2));
707 ASSERT_EQ(2U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 3));
708 ASSERT_EQ(0U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 4));
710 auto Cmp
= [](const InstrProfValueData
&VD1
, const InstrProfValueData
&VD2
) {
711 return VD1
.Count
> VD2
.Count
;
713 std::unique_ptr
<InstrProfValueData
[]> VD_0(
714 Record
.getValueForSite(IPVK_IndirectCallTarget
, 0));
715 llvm::sort(&VD_0
[0], &VD_0
[5], Cmp
);
716 ASSERT_EQ(StringRef((const char *)VD_0
[0].Value
, 7), StringRef("callee2"));
717 ASSERT_EQ(1000U, VD_0
[0].Count
);
718 ASSERT_EQ(StringRef((const char *)VD_0
[1].Value
, 7), StringRef("callee3"));
719 ASSERT_EQ(500U, VD_0
[1].Count
);
720 ASSERT_EQ(StringRef((const char *)VD_0
[2].Value
, 7), StringRef("callee1"));
721 ASSERT_EQ(400U, VD_0
[2].Count
);
722 ASSERT_EQ(StringRef((const char *)VD_0
[3].Value
, 7), StringRef("callee4"));
723 ASSERT_EQ(300U, VD_0
[3].Count
);
724 ASSERT_EQ(StringRef((const char *)VD_0
[4].Value
, 7), StringRef("callee5"));
725 ASSERT_EQ(100U, VD_0
[4].Count
);
727 std::unique_ptr
<InstrProfValueData
[]> VD_1(
728 Record
.getValueForSite(IPVK_IndirectCallTarget
, 1));
729 llvm::sort(&VD_1
[0], &VD_1
[4], Cmp
);
730 ASSERT_EQ(StringRef((const char *)VD_1
[0].Value
, 7), StringRef("callee2"));
731 ASSERT_EQ(2500U, VD_1
[0].Count
);
732 ASSERT_EQ(StringRef((const char *)VD_1
[1].Value
, 7), StringRef("callee1"));
733 ASSERT_EQ(1300U, VD_1
[1].Count
);
734 ASSERT_EQ(StringRef((const char *)VD_1
[2].Value
, 7), StringRef("callee3"));
735 ASSERT_EQ(1000U, VD_1
[2].Count
);
736 ASSERT_EQ(StringRef((const char *)VD_1
[3].Value
, 7), StringRef("callee5"));
737 ASSERT_EQ(800U, VD_1
[3].Count
);
739 std::unique_ptr
<InstrProfValueData
[]> VD_2(
740 Record
.getValueForSite(IPVK_IndirectCallTarget
, 2));
741 llvm::sort(&VD_2
[0], &VD_2
[3], Cmp
);
742 ASSERT_EQ(StringRef((const char *)VD_2
[0].Value
, 7), StringRef("callee4"));
743 ASSERT_EQ(5500U, VD_2
[0].Count
);
744 ASSERT_EQ(StringRef((const char *)VD_2
[1].Value
, 7), StringRef("callee3"));
745 ASSERT_EQ(1000U, VD_2
[1].Count
);
746 ASSERT_EQ(StringRef((const char *)VD_2
[2].Value
, 7), StringRef("callee6"));
747 ASSERT_EQ(800U, VD_2
[2].Count
);
749 std::unique_ptr
<InstrProfValueData
[]> VD_3(
750 Record
.getValueForSite(IPVK_IndirectCallTarget
, 3));
751 llvm::sort(&VD_3
[0], &VD_3
[2], Cmp
);
752 ASSERT_EQ(StringRef((const char *)VD_3
[0].Value
, 7), StringRef("callee3"));
753 ASSERT_EQ(2000U, VD_3
[0].Count
);
754 ASSERT_EQ(StringRef((const char *)VD_3
[1].Value
, 7), StringRef("callee2"));
755 ASSERT_EQ(1800U, VD_3
[1].Count
);
758 TEST_P(MaybeSparseInstrProfTest
, value_prof_data_read_write_mapping
) {
760 NamedInstrProfRecord
SrcRecord("caller", 0x1234, {1ULL << 31, 2});
761 addValueProfData(SrcRecord
);
762 std::unique_ptr
<ValueProfData
> VPData
=
763 ValueProfData::serializeFrom(SrcRecord
);
765 NamedInstrProfRecord
Record("caller", 0x1234, {1ULL << 31, 2});
766 InstrProfSymtab Symtab
;
767 Symtab
.mapAddress(uint64_t(callee1
), 0x1000ULL
);
768 Symtab
.mapAddress(uint64_t(callee2
), 0x2000ULL
);
769 Symtab
.mapAddress(uint64_t(callee3
), 0x3000ULL
);
770 Symtab
.mapAddress(uint64_t(callee4
), 0x4000ULL
);
771 // Missing mapping for callee5
773 VPData
->deserializeTo(Record
, &Symtab
);
775 // Now read data from Record and sanity check the data
776 ASSERT_EQ(5U, Record
.getNumValueSites(IPVK_IndirectCallTarget
));
777 ASSERT_EQ(5U, Record
.getNumValueDataForSite(IPVK_IndirectCallTarget
, 0));
779 auto Cmp
= [](const InstrProfValueData
&VD1
, const InstrProfValueData
&VD2
) {
780 return VD1
.Count
> VD2
.Count
;
782 std::unique_ptr
<InstrProfValueData
[]> VD_0(
783 Record
.getValueForSite(IPVK_IndirectCallTarget
, 0));
784 llvm::sort(&VD_0
[0], &VD_0
[5], Cmp
);
785 ASSERT_EQ(VD_0
[0].Value
, 0x2000ULL
);
786 ASSERT_EQ(1000U, VD_0
[0].Count
);
787 ASSERT_EQ(VD_0
[1].Value
, 0x3000ULL
);
788 ASSERT_EQ(500U, VD_0
[1].Count
);
789 ASSERT_EQ(VD_0
[2].Value
, 0x1000ULL
);
790 ASSERT_EQ(400U, VD_0
[2].Count
);
792 // callee5 does not have a mapped value -- default to 0.
793 ASSERT_EQ(VD_0
[4].Value
, 0ULL);
796 TEST_P(MaybeSparseInstrProfTest
, get_max_function_count
) {
797 Writer
.addRecord({"foo", 0x1234, {1ULL << 31, 2}}, Err
);
798 Writer
.addRecord({"bar", 0, {1ULL << 63}}, Err
);
799 Writer
.addRecord({"baz", 0x5678, {0, 0, 0, 0}}, Err
);
800 auto Profile
= Writer
.writeBuffer();
801 readProfile(std::move(Profile
));
803 ASSERT_EQ(1ULL << 63, Reader
->getMaximumFunctionCount());
806 TEST_P(MaybeSparseInstrProfTest
, get_weighted_function_counts
) {
807 Writer
.addRecord({"foo", 0x1234, {1, 2}}, 3, Err
);
808 Writer
.addRecord({"foo", 0x1235, {3, 4}}, 5, Err
);
809 auto Profile
= Writer
.writeBuffer();
810 readProfile(std::move(Profile
));
812 std::vector
<uint64_t> Counts
;
813 EXPECT_THAT_ERROR(Reader
->getFunctionCounts("foo", 0x1234, Counts
),
815 ASSERT_EQ(2U, Counts
.size());
816 ASSERT_EQ(3U, Counts
[0]);
817 ASSERT_EQ(6U, Counts
[1]);
819 EXPECT_THAT_ERROR(Reader
->getFunctionCounts("foo", 0x1235, Counts
),
821 ASSERT_EQ(2U, Counts
.size());
822 ASSERT_EQ(15U, Counts
[0]);
823 ASSERT_EQ(20U, Counts
[1]);
826 // Testing symtab creator interface used by indexed profile reader.
827 TEST_P(MaybeSparseInstrProfTest
, instr_prof_symtab_test
) {
828 std::vector
<StringRef
> FuncNames
;
829 FuncNames
.push_back("func1");
830 FuncNames
.push_back("func2");
831 FuncNames
.push_back("func3");
832 FuncNames
.push_back("bar1");
833 FuncNames
.push_back("bar2");
834 FuncNames
.push_back("bar3");
835 InstrProfSymtab Symtab
;
836 EXPECT_THAT_ERROR(Symtab
.create(FuncNames
), Succeeded());
837 StringRef R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func1"));
838 ASSERT_EQ(StringRef("func1"), R
);
839 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func2"));
840 ASSERT_EQ(StringRef("func2"), R
);
841 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func3"));
842 ASSERT_EQ(StringRef("func3"), R
);
843 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar1"));
844 ASSERT_EQ(StringRef("bar1"), R
);
845 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar2"));
846 ASSERT_EQ(StringRef("bar2"), R
);
847 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar3"));
848 ASSERT_EQ(StringRef("bar3"), R
);
851 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar4"));
852 ASSERT_EQ(StringRef(), R
);
853 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("foo4"));
854 ASSERT_EQ(StringRef(), R
);
856 // Now incrementally update the symtab
857 EXPECT_THAT_ERROR(Symtab
.addFuncName("blah_1"), Succeeded());
858 EXPECT_THAT_ERROR(Symtab
.addFuncName("blah_2"), Succeeded());
859 EXPECT_THAT_ERROR(Symtab
.addFuncName("blah_3"), Succeeded());
862 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("blah_1"));
863 ASSERT_EQ(StringRef("blah_1"), R
);
864 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("blah_2"));
865 ASSERT_EQ(StringRef("blah_2"), R
);
866 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("blah_3"));
867 ASSERT_EQ(StringRef("blah_3"), R
);
868 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func1"));
869 ASSERT_EQ(StringRef("func1"), R
);
870 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func2"));
871 ASSERT_EQ(StringRef("func2"), R
);
872 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("func3"));
873 ASSERT_EQ(StringRef("func3"), R
);
874 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar1"));
875 ASSERT_EQ(StringRef("bar1"), R
);
876 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar2"));
877 ASSERT_EQ(StringRef("bar2"), R
);
878 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash("bar3"));
879 ASSERT_EQ(StringRef("bar3"), R
);
882 // Test that we get an error when creating a bogus symtab.
883 TEST_P(MaybeSparseInstrProfTest
, instr_prof_bogus_symtab_empty_func_name
) {
884 InstrProfSymtab Symtab
;
885 EXPECT_TRUE(ErrorEquals(instrprof_error::malformed
, Symtab
.addFuncName("")));
888 // Testing symtab creator interface used by value profile transformer.
889 TEST_P(MaybeSparseInstrProfTest
, instr_prof_symtab_module_test
) {
891 std::unique_ptr
<Module
> M
= llvm::make_unique
<Module
>("MyModule.cpp", Ctx
);
892 FunctionType
*FTy
= FunctionType::get(Type::getVoidTy(Ctx
),
894 Function::Create(FTy
, Function::ExternalLinkage
, "Gfoo", M
.get());
895 Function::Create(FTy
, Function::ExternalLinkage
, "Gblah", M
.get());
896 Function::Create(FTy
, Function::ExternalLinkage
, "Gbar", M
.get());
897 Function::Create(FTy
, Function::InternalLinkage
, "Ifoo", M
.get());
898 Function::Create(FTy
, Function::InternalLinkage
, "Iblah", M
.get());
899 Function::Create(FTy
, Function::InternalLinkage
, "Ibar", M
.get());
900 Function::Create(FTy
, Function::PrivateLinkage
, "Pfoo", M
.get());
901 Function::Create(FTy
, Function::PrivateLinkage
, "Pblah", M
.get());
902 Function::Create(FTy
, Function::PrivateLinkage
, "Pbar", M
.get());
903 Function::Create(FTy
, Function::WeakODRLinkage
, "Wfoo", M
.get());
904 Function::Create(FTy
, Function::WeakODRLinkage
, "Wblah", M
.get());
905 Function::Create(FTy
, Function::WeakODRLinkage
, "Wbar", M
.get());
907 InstrProfSymtab ProfSymtab
;
908 EXPECT_THAT_ERROR(ProfSymtab
.create(*M
), Succeeded());
910 StringRef Funcs
[] = {"Gfoo", "Gblah", "Gbar", "Ifoo", "Iblah", "Ibar",
911 "Pfoo", "Pblah", "Pbar", "Wfoo", "Wblah", "Wbar"};
913 for (unsigned I
= 0; I
< sizeof(Funcs
) / sizeof(*Funcs
); I
++) {
914 Function
*F
= M
->getFunction(Funcs
[I
]);
915 ASSERT_TRUE(F
!= nullptr);
916 std::string PGOName
= getPGOFuncName(*F
);
917 uint64_t Key
= IndexedInstrProf::ComputeHash(PGOName
);
918 ASSERT_EQ(StringRef(PGOName
),
919 ProfSymtab
.getFuncName(Key
));
920 ASSERT_EQ(StringRef(Funcs
[I
]), ProfSymtab
.getOrigFuncName(Key
));
924 // Testing symtab serialization and creator/deserialization interface
925 // used by coverage map reader, and raw profile reader.
926 TEST_P(MaybeSparseInstrProfTest
, instr_prof_symtab_compression_test
) {
927 std::vector
<std::string
> FuncNames1
;
928 std::vector
<std::string
> FuncNames2
;
929 for (int I
= 0; I
< 3; I
++) {
931 raw_string_ostream
OS(str
);
933 FuncNames1
.push_back(OS
.str());
935 OS
<< "f oooooooooooooo_" << I
;
936 FuncNames1
.push_back(OS
.str());
939 FuncNames2
.push_back(OS
.str());
941 OS
<< "BlahblahBlahblahBar_" << I
;
942 FuncNames2
.push_back(OS
.str());
945 for (bool DoCompression
: {false, true}) {
947 std::string FuncNameStrings1
;
948 EXPECT_THAT_ERROR(collectPGOFuncNameStrings(
949 FuncNames1
, (DoCompression
&& zlib::isAvailable()),
954 std::string FuncNameStrings2
;
955 EXPECT_THAT_ERROR(collectPGOFuncNameStrings(
956 FuncNames2
, (DoCompression
&& zlib::isAvailable()),
960 for (int Padding
= 0; Padding
< 2; Padding
++) {
961 // Join with paddings :
962 std::string FuncNameStrings
= FuncNameStrings1
;
963 for (int P
= 0; P
< Padding
; P
++) {
964 FuncNameStrings
.push_back('\0');
966 FuncNameStrings
+= FuncNameStrings2
;
969 InstrProfSymtab Symtab
;
970 EXPECT_THAT_ERROR(Symtab
.create(StringRef(FuncNameStrings
)), Succeeded());
972 // Now do the checks:
973 // First sampling some data points:
974 StringRef R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash(FuncNames1
[0]));
975 ASSERT_EQ(StringRef("func_0"), R
);
976 R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash(FuncNames1
[1]));
977 ASSERT_EQ(StringRef("f oooooooooooooo_0"), R
);
978 for (int I
= 0; I
< 3; I
++) {
980 N
[0] = FuncNames1
[2 * I
];
981 N
[1] = FuncNames1
[2 * I
+ 1];
982 N
[2] = FuncNames2
[2 * I
];
983 N
[3] = FuncNames2
[2 * I
+ 1];
984 for (int J
= 0; J
< 4; J
++) {
985 StringRef R
= Symtab
.getFuncName(IndexedInstrProf::ComputeHash(N
[J
]));
986 ASSERT_EQ(StringRef(N
[J
]), R
);
993 TEST_F(SparseInstrProfTest
, preserve_no_records
) {
994 Writer
.addRecord({"foo", 0x1234, {0}}, Err
);
995 Writer
.addRecord({"bar", 0x4321, {0, 0}}, Err
);
996 Writer
.addRecord({"baz", 0x4321, {0, 0, 0}}, Err
);
998 auto Profile
= Writer
.writeBuffer();
999 readProfile(std::move(Profile
));
1001 auto I
= Reader
->begin(), E
= Reader
->end();
1002 ASSERT_TRUE(I
== E
);
1005 INSTANTIATE_TEST_CASE_P(MaybeSparse
, MaybeSparseInstrProfTest
,
1006 ::testing::Bool(),);
1008 } // end anonymous namespace