blob: 2d915d4459824227c36ba5c44f488f895629f496 [file] [log] [blame]
Eugene Zelenko4e036ff2015-11-04 22:32:32 +00001//===- unittest/ProfileData/InstrProfTest.cpp -------------------*- C++ -*-===//
Justin Bogner09e5af72015-02-18 01:58:17 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Xinliang David Li6d3068a2016-01-20 01:26:34 +000010#include "llvm/IR/Function.h"
Xinliang David Lif45c13d2016-02-04 19:11:43 +000011#include "llvm/IR/IRBuilder.h"
Xinliang David Li6d3068a2016-01-20 01:26:34 +000012#include "llvm/IR/LLVMContext.h"
13#include "llvm/IR/Module.h"
Justin Bogner09e5af72015-02-18 01:58:17 +000014#include "llvm/ProfileData/InstrProfReader.h"
15#include "llvm/ProfileData/InstrProfWriter.h"
Xinliang David Lif2f39d62015-12-31 07:57:16 +000016#include "llvm/Support/Compression.h"
David Blaikie8c2dc922017-07-07 21:02:59 +000017#include "llvm/Testing/Support/Error.h"
18#include "llvm/Testing/Support/SupportHelpers.h"
Justin Bogner09e5af72015-02-18 01:58:17 +000019#include "gtest/gtest.h"
Justin Bogner09e5af72015-02-18 01:58:17 +000020#include <cstdarg>
21
22using namespace llvm;
23
David Blaikie8c2dc922017-07-07 21:02:59 +000024LLVM_NODISCARD static ::testing::AssertionResult
25ErrorEquals(instrprof_error Expected, Error E) {
Vedant Kumarc77570e2016-05-19 03:54:45 +000026 instrprof_error Found;
27 std::string FoundMsg;
28 handleAllErrors(std::move(E), [&](const InstrProfError &IPE) {
29 Found = IPE.get();
30 FoundMsg = IPE.message();
31 });
Justin Bogner09e5af72015-02-18 01:58:17 +000032 if (Expected == Found)
33 return ::testing::AssertionSuccess();
Vedant Kumarc77570e2016-05-19 03:54:45 +000034 return ::testing::AssertionFailure() << "error: " << FoundMsg << "\n";
Justin Bogner09e5af72015-02-18 01:58:17 +000035}
36
37namespace {
38
39struct InstrProfTest : ::testing::Test {
40 InstrProfWriter Writer;
41 std::unique_ptr<IndexedInstrProfReader> Reader;
42
Vedant Kumar0c94d7d2016-01-29 22:54:45 +000043 void SetUp() { Writer.setOutputSparse(false); }
44
Richard Smith3b1e4302018-10-10 21:09:37 +000045 void readProfile(std::unique_ptr<MemoryBuffer> Profile,
46 std::unique_ptr<MemoryBuffer> Remapping = nullptr) {
47 auto ReaderOrErr = IndexedInstrProfReader::create(std::move(Profile),
48 std::move(Remapping));
David Blaikie8c2dc922017-07-07 21:02:59 +000049 EXPECT_THAT_ERROR(ReaderOrErr.takeError(), Succeeded());
Justin Bogner09e5af72015-02-18 01:58:17 +000050 Reader = std::move(ReaderOrErr.get());
51 }
52};
53
Vedant Kumar0c94d7d2016-01-29 22:54:45 +000054struct SparseInstrProfTest : public InstrProfTest {
55 void SetUp() { Writer.setOutputSparse(true); }
56};
57
58struct MaybeSparseInstrProfTest : public InstrProfTest,
59 public ::testing::WithParamInterface<bool> {
Vedant Kumara9e82b42016-03-22 15:14:18 +000060 void SetUp() { Writer.setOutputSparse(GetParam()); }
Vedant Kumar0c94d7d2016-01-29 22:54:45 +000061};
62
63TEST_P(MaybeSparseInstrProfTest, write_and_read_empty_profile) {
Justin Bogner09e5af72015-02-18 01:58:17 +000064 auto Profile = Writer.writeBuffer();
65 readProfile(std::move(Profile));
66 ASSERT_TRUE(Reader->begin() == Reader->end());
67}
68
David Blaikiecb160612017-07-10 03:04:59 +000069static const auto Err = [](Error E) {
70 consumeError(std::move(E));
71 FAIL();
72};
73
Vedant Kumar0c94d7d2016-01-29 22:54:45 +000074TEST_P(MaybeSparseInstrProfTest, write_and_read_one_function) {
David Blaikiecb160612017-07-10 03:04:59 +000075 Writer.addRecord({"foo", 0x1234, {1, 2, 3, 4}}, Err);
Justin Bogner09e5af72015-02-18 01:58:17 +000076 auto Profile = Writer.writeBuffer();
77 readProfile(std::move(Profile));
78
79 auto I = Reader->begin(), E = Reader->end();
80 ASSERT_TRUE(I != E);
81 ASSERT_EQ(StringRef("foo"), I->Name);
82 ASSERT_EQ(0x1234U, I->Hash);
83 ASSERT_EQ(4U, I->Counts.size());
84 ASSERT_EQ(1U, I->Counts[0]);
85 ASSERT_EQ(2U, I->Counts[1]);
86 ASSERT_EQ(3U, I->Counts[2]);
87 ASSERT_EQ(4U, I->Counts[3]);
88 ASSERT_TRUE(++I == E);
89}
90
Vedant Kumar0c94d7d2016-01-29 22:54:45 +000091TEST_P(MaybeSparseInstrProfTest, get_instr_prof_record) {
David Blaikiecb160612017-07-10 03:04:59 +000092 Writer.addRecord({"foo", 0x1234, {1, 2}}, Err);
93 Writer.addRecord({"foo", 0x1235, {3, 4}}, Err);
Xinliang David Li61c7e682015-11-02 05:08:23 +000094 auto Profile = Writer.writeBuffer();
95 readProfile(std::move(Profile));
96
Vedant Kumarc77570e2016-05-19 03:54:45 +000097 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("foo", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +000098 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +000099 ASSERT_EQ(2U, R->Counts.size());
100 ASSERT_EQ(1U, R->Counts[0]);
101 ASSERT_EQ(2U, R->Counts[1]);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000102
103 R = Reader->getInstrProfRecord("foo", 0x1235);
David Blaikie8c2dc922017-07-07 21:02:59 +0000104 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000105 ASSERT_EQ(2U, R->Counts.size());
106 ASSERT_EQ(3U, R->Counts[0]);
107 ASSERT_EQ(4U, R->Counts[1]);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000108
109 R = Reader->getInstrProfRecord("foo", 0x5678);
Vedant Kumarc77570e2016-05-19 03:54:45 +0000110 ASSERT_TRUE(ErrorEquals(instrprof_error::hash_mismatch, R.takeError()));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000111
112 R = Reader->getInstrProfRecord("bar", 0x1234);
Vedant Kumarc77570e2016-05-19 03:54:45 +0000113 ASSERT_TRUE(ErrorEquals(instrprof_error::unknown_function, R.takeError()));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000114}
115
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000116TEST_P(MaybeSparseInstrProfTest, get_function_counts) {
David Blaikiecb160612017-07-10 03:04:59 +0000117 Writer.addRecord({"foo", 0x1234, {1, 2}}, Err);
118 Writer.addRecord({"foo", 0x1235, {3, 4}}, Err);
Justin Bogner09e5af72015-02-18 01:58:17 +0000119 auto Profile = Writer.writeBuffer();
120 readProfile(std::move(Profile));
121
122 std::vector<uint64_t> Counts;
David Blaikie8c2dc922017-07-07 21:02:59 +0000123 EXPECT_THAT_ERROR(Reader->getFunctionCounts("foo", 0x1234, Counts),
124 Succeeded());
Justin Bogner09e5af72015-02-18 01:58:17 +0000125 ASSERT_EQ(2U, Counts.size());
126 ASSERT_EQ(1U, Counts[0]);
127 ASSERT_EQ(2U, Counts[1]);
128
David Blaikie8c2dc922017-07-07 21:02:59 +0000129 EXPECT_THAT_ERROR(Reader->getFunctionCounts("foo", 0x1235, Counts),
130 Succeeded());
Justin Bognerfe3176f2015-06-22 23:56:53 +0000131 ASSERT_EQ(2U, Counts.size());
132 ASSERT_EQ(3U, Counts[0]);
133 ASSERT_EQ(4U, Counts[1]);
134
Vedant Kumarc77570e2016-05-19 03:54:45 +0000135 Error E1 = Reader->getFunctionCounts("foo", 0x5678, Counts);
136 ASSERT_TRUE(ErrorEquals(instrprof_error::hash_mismatch, std::move(E1)));
Justin Bogner09e5af72015-02-18 01:58:17 +0000137
Vedant Kumarc77570e2016-05-19 03:54:45 +0000138 Error E2 = Reader->getFunctionCounts("bar", 0x1234, Counts);
139 ASSERT_TRUE(ErrorEquals(instrprof_error::unknown_function, std::move(E2)));
Justin Bogner09e5af72015-02-18 01:58:17 +0000140}
141
Xinliang David Li8fd9cfb2016-02-03 04:08:18 +0000142// Profile data is copied from general.proftext
143TEST_F(InstrProfTest, get_profile_summary) {
David Blaikiecb160612017-07-10 03:04:59 +0000144 Writer.addRecord({"func1", 0x1234, {97531}}, Err);
145 Writer.addRecord({"func2", 0x1234, {0, 0}}, Err);
146 Writer.addRecord(
147 {"func3",
148 0x1234,
149 {2305843009213693952, 1152921504606846976, 576460752303423488,
150 288230376151711744, 144115188075855872, 72057594037927936}},
151 Err);
152 Writer.addRecord({"func4", 0x1234, {0}}, Err);
Xinliang David Li8fd9cfb2016-02-03 04:08:18 +0000153 auto Profile = Writer.writeBuffer();
154 readProfile(std::move(Profile));
155
Easwaran Raman30c760d2016-05-19 21:53:28 +0000156 auto VerifySummary = [](ProfileSummary &IPS) mutable {
157 ASSERT_EQ(ProfileSummary::PSK_Instr, IPS.getKind());
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000158 ASSERT_EQ(2305843009213693952U, IPS.getMaxFunctionCount());
Easwaran Raman30c760d2016-05-19 21:53:28 +0000159 ASSERT_EQ(2305843009213693952U, IPS.getMaxCount());
160 ASSERT_EQ(10U, IPS.getNumCounts());
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000161 ASSERT_EQ(4539628424389557499U, IPS.getTotalCount());
162 std::vector<ProfileSummaryEntry> &Details = IPS.getDetailedSummary();
163 uint32_t Cutoff = 800000;
164 auto Predicate = [&Cutoff](const ProfileSummaryEntry &PE) {
165 return PE.Cutoff == Cutoff;
166 };
David Majnemerb0353c62016-08-12 00:18:03 +0000167 auto EightyPerc = find_if(Details, Predicate);
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000168 Cutoff = 900000;
David Majnemerb0353c62016-08-12 00:18:03 +0000169 auto NinetyPerc = find_if(Details, Predicate);
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000170 Cutoff = 950000;
David Majnemerb0353c62016-08-12 00:18:03 +0000171 auto NinetyFivePerc = find_if(Details, Predicate);
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000172 Cutoff = 990000;
David Majnemerb0353c62016-08-12 00:18:03 +0000173 auto NinetyNinePerc = find_if(Details, Predicate);
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000174 ASSERT_EQ(576460752303423488U, EightyPerc->MinCount);
175 ASSERT_EQ(288230376151711744U, NinetyPerc->MinCount);
176 ASSERT_EQ(288230376151711744U, NinetyFivePerc->MinCount);
177 ASSERT_EQ(72057594037927936U, NinetyNinePerc->MinCount);
Xinliang David Li8fd9cfb2016-02-03 04:08:18 +0000178 };
Easwaran Raman30c760d2016-05-19 21:53:28 +0000179 ProfileSummary &PS = Reader->getSummary();
Easwaran Ramanab6fe182016-03-14 22:23:28 +0000180 VerifySummary(PS);
Easwaran Ramanaf640bf2016-03-18 21:29:30 +0000181
182 // Test that conversion of summary to and from Metadata works.
Mehdi Amini8be77072016-04-14 21:59:01 +0000183 LLVMContext Context;
184 Metadata *MD = PS.getMD(Context);
Easwaran Raman1283c1d2016-03-01 18:30:58 +0000185 ASSERT_TRUE(MD);
186 ProfileSummary *PSFromMD = ProfileSummary::getFromMD(MD);
187 ASSERT_TRUE(PSFromMD);
Easwaran Raman30c760d2016-05-19 21:53:28 +0000188 VerifySummary(*PSFromMD);
189 delete PSFromMD;
Easwaran Ramanaf640bf2016-03-18 21:29:30 +0000190
191 // Test that summary can be attached to and read back from module.
Mehdi Amini8be77072016-04-14 21:59:01 +0000192 Module M("my_module", Context);
Easwaran Ramanaf640bf2016-03-18 21:29:30 +0000193 M.setProfileSummary(MD);
194 MD = M.getProfileSummary();
195 ASSERT_TRUE(MD);
196 PSFromMD = ProfileSummary::getFromMD(MD);
197 ASSERT_TRUE(PSFromMD);
Easwaran Raman30c760d2016-05-19 21:53:28 +0000198 VerifySummary(*PSFromMD);
199 delete PSFromMD;
Xinliang David Li8fd9cfb2016-02-03 04:08:18 +0000200}
201
Vedant Kumard91d3782016-07-19 01:17:20 +0000202TEST_F(InstrProfTest, test_writer_merge) {
David Blaikiecb160612017-07-10 03:04:59 +0000203 Writer.addRecord({"func1", 0x1234, {42}}, Err);
Vedant Kumard91d3782016-07-19 01:17:20 +0000204
205 InstrProfWriter Writer2;
David Blaikiecb160612017-07-10 03:04:59 +0000206 Writer2.addRecord({"func2", 0x1234, {0, 0}}, Err);
Vedant Kumard91d3782016-07-19 01:17:20 +0000207
David Blaikiecb160612017-07-10 03:04:59 +0000208 Writer.mergeRecordsFromWriter(std::move(Writer2), Err);
Vedant Kumard91d3782016-07-19 01:17:20 +0000209
210 auto Profile = Writer.writeBuffer();
211 readProfile(std::move(Profile));
212
213 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("func1", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000214 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
Vedant Kumard91d3782016-07-19 01:17:20 +0000215 ASSERT_EQ(1U, R->Counts.size());
216 ASSERT_EQ(42U, R->Counts[0]);
217
218 R = Reader->getInstrProfRecord("func2", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000219 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
Vedant Kumard91d3782016-07-19 01:17:20 +0000220 ASSERT_EQ(2U, R->Counts.size());
221 ASSERT_EQ(0U, R->Counts[0]);
222 ASSERT_EQ(0U, R->Counts[1]);
223}
224
Xinliang David Li72c40752016-04-10 02:35:53 +0000225static const char callee1[] = "callee1";
226static const char callee2[] = "callee2";
227static const char callee3[] = "callee3";
228static const char callee4[] = "callee4";
229static const char callee5[] = "callee5";
230static const char callee6[] = "callee6";
231
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000232TEST_P(MaybeSparseInstrProfTest, get_icall_data_read_write) {
David Blaikie450ef2a2017-07-06 19:00:12 +0000233 NamedInstrProfRecord Record1("caller", 0x1234, {1, 2});
Xinliang David Li61c7e682015-11-02 05:08:23 +0000234
235 // 4 value sites.
236 Record1.reserveSites(IPVK_IndirectCallTarget, 4);
Xinliang David Li72c40752016-04-10 02:35:53 +0000237 InstrProfValueData VD0[] = {
238 {(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}, {(uint64_t)callee3, 3}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000239 Record1.addValueData(IPVK_IndirectCallTarget, 0, VD0, 3, nullptr);
Xinliang David Libcd8e0a2015-11-10 00:24:45 +0000240 // No value profile data at the second site.
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000241 Record1.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000242 InstrProfValueData VD2[] = {{(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000243 Record1.addValueData(IPVK_IndirectCallTarget, 2, VD2, 2, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000244 InstrProfValueData VD3[] = {{(uint64_t)callee1, 1}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000245 Record1.addValueData(IPVK_IndirectCallTarget, 3, VD3, 1, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000246
David Blaikiecb160612017-07-10 03:04:59 +0000247 Writer.addRecord(std::move(Record1), Err);
248 Writer.addRecord({"callee1", 0x1235, {3, 4}}, Err);
249 Writer.addRecord({"callee2", 0x1235, {3, 4}}, Err);
250 Writer.addRecord({"callee3", 0x1235, {3, 4}}, Err);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000251 auto Profile = Writer.writeBuffer();
252 readProfile(std::move(Profile));
253
Vedant Kumarc77570e2016-05-19 03:54:45 +0000254 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000255 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000256 ASSERT_EQ(4U, R->getNumValueSites(IPVK_IndirectCallTarget));
257 ASSERT_EQ(3U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
258 ASSERT_EQ(0U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 1));
259 ASSERT_EQ(2U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 2));
260 ASSERT_EQ(1U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 3));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000261
Xinliang David Li730d9342016-02-04 05:29:51 +0000262 uint64_t TotalC;
Xinliang David Li61c7e682015-11-02 05:08:23 +0000263 std::unique_ptr<InstrProfValueData[]> VD =
David Blaikie9f15b402016-02-09 01:02:24 +0000264 R->getValueForSite(IPVK_IndirectCallTarget, 0, &TotalC);
Xinliang David Li4f87d122016-01-08 03:49:59 +0000265
266 ASSERT_EQ(3U, VD[0].Count);
267 ASSERT_EQ(2U, VD[1].Count);
268 ASSERT_EQ(1U, VD[2].Count);
Xinliang David Li730d9342016-02-04 05:29:51 +0000269 ASSERT_EQ(6U, TotalC);
Xinliang David Li4f87d122016-01-08 03:49:59 +0000270
271 ASSERT_EQ(StringRef((const char *)VD[0].Value, 7), StringRef("callee3"));
272 ASSERT_EQ(StringRef((const char *)VD[1].Value, 7), StringRef("callee2"));
273 ASSERT_EQ(StringRef((const char *)VD[2].Value, 7), StringRef("callee1"));
274}
275
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000276TEST_P(MaybeSparseInstrProfTest, annotate_vp_data) {
David Blaikie450ef2a2017-07-06 19:00:12 +0000277 NamedInstrProfRecord Record("caller", 0x1234, {1, 2});
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000278 Record.reserveSites(IPVK_IndirectCallTarget, 1);
Rong Xu24657002016-02-10 22:19:43 +0000279 InstrProfValueData VD0[] = {{1000, 1}, {2000, 2}, {3000, 3}, {5000, 5},
280 {4000, 4}, {6000, 6}};
281 Record.addValueData(IPVK_IndirectCallTarget, 0, VD0, 6, nullptr);
David Blaikiecb160612017-07-10 03:04:59 +0000282 Writer.addRecord(std::move(Record), Err);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000283 auto Profile = Writer.writeBuffer();
284 readProfile(std::move(Profile));
Vedant Kumarc77570e2016-05-19 03:54:45 +0000285 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000286 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000287
288 LLVMContext Ctx;
289 std::unique_ptr<Module> M(new Module("MyModule", Ctx));
290 FunctionType *FTy = FunctionType::get(Type::getVoidTy(Ctx),
291 /*isVarArg=*/false);
292 Function *F =
293 Function::Create(FTy, Function::ExternalLinkage, "caller", M.get());
294 BasicBlock *BB = BasicBlock::Create(Ctx, "", F);
295
296 IRBuilder<> Builder(BB);
297 BasicBlock *TBB = BasicBlock::Create(Ctx, "", F);
298 BasicBlock *FBB = BasicBlock::Create(Ctx, "", F);
299
300 // Use branch instruction to annotate with value profile data for simplicity
301 Instruction *Inst = Builder.CreateCondBr(Builder.getTrue(), TBB, FBB);
302 Instruction *Inst2 = Builder.CreateCondBr(Builder.getTrue(), TBB, FBB);
David Blaikie9f15b402016-02-09 01:02:24 +0000303 annotateValueSite(*M, *Inst, R.get(), IPVK_IndirectCallTarget, 0);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000304
305 InstrProfValueData ValueData[5];
306 uint32_t N;
307 uint64_t T;
308 bool Res = getValueProfDataFromInst(*Inst, IPVK_IndirectCallTarget, 5,
309 ValueData, N, T);
310 ASSERT_TRUE(Res);
311 ASSERT_EQ(3U, N);
Rong Xu24657002016-02-10 22:19:43 +0000312 ASSERT_EQ(21U, T);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000313 // The result should be sorted already:
Rong Xu24657002016-02-10 22:19:43 +0000314 ASSERT_EQ(6000U, ValueData[0].Value);
315 ASSERT_EQ(6U, ValueData[0].Count);
316 ASSERT_EQ(5000U, ValueData[1].Value);
317 ASSERT_EQ(5U, ValueData[1].Count);
318 ASSERT_EQ(4000U, ValueData[2].Value);
319 ASSERT_EQ(4U, ValueData[2].Count);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000320 Res = getValueProfDataFromInst(*Inst, IPVK_IndirectCallTarget, 1, ValueData,
321 N, T);
322 ASSERT_TRUE(Res);
323 ASSERT_EQ(1U, N);
Rong Xu24657002016-02-10 22:19:43 +0000324 ASSERT_EQ(21U, T);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000325
326 Res = getValueProfDataFromInst(*Inst2, IPVK_IndirectCallTarget, 5, ValueData,
327 N, T);
328 ASSERT_FALSE(Res);
Rong Xu24657002016-02-10 22:19:43 +0000329
330 // Remove the MD_prof metadata
331 Inst->setMetadata(LLVMContext::MD_prof, 0);
332 // Annotate 5 records this time.
333 annotateValueSite(*M, *Inst, R.get(), IPVK_IndirectCallTarget, 0, 5);
334 Res = getValueProfDataFromInst(*Inst, IPVK_IndirectCallTarget, 5,
335 ValueData, N, T);
336 ASSERT_TRUE(Res);
337 ASSERT_EQ(5U, N);
338 ASSERT_EQ(21U, T);
339 ASSERT_EQ(6000U, ValueData[0].Value);
340 ASSERT_EQ(6U, ValueData[0].Count);
341 ASSERT_EQ(5000U, ValueData[1].Value);
342 ASSERT_EQ(5U, ValueData[1].Count);
343 ASSERT_EQ(4000U, ValueData[2].Value);
344 ASSERT_EQ(4U, ValueData[2].Count);
345 ASSERT_EQ(3000U, ValueData[3].Value);
346 ASSERT_EQ(3U, ValueData[3].Count);
347 ASSERT_EQ(2000U, ValueData[4].Value);
348 ASSERT_EQ(2U, ValueData[4].Count);
Rong Xu2ee5bb82016-02-12 21:36:17 +0000349
350 // Remove the MD_prof metadata
351 Inst->setMetadata(LLVMContext::MD_prof, 0);
352 // Annotate with 4 records.
353 InstrProfValueData VD0Sorted[] = {{1000, 6}, {2000, 5}, {3000, 4}, {4000, 3},
354 {5000, 2}, {6000, 1}};
Rong Xu34be7e62016-03-30 16:56:31 +0000355 annotateValueSite(*M, *Inst, makeArrayRef(VD0Sorted).slice(2), 10,
356 IPVK_IndirectCallTarget, 5);
Rong Xu2ee5bb82016-02-12 21:36:17 +0000357 Res = getValueProfDataFromInst(*Inst, IPVK_IndirectCallTarget, 5,
358 ValueData, N, T);
359 ASSERT_TRUE(Res);
360 ASSERT_EQ(4U, N);
361 ASSERT_EQ(10U, T);
362 ASSERT_EQ(3000U, ValueData[0].Value);
363 ASSERT_EQ(4U, ValueData[0].Count);
364 ASSERT_EQ(4000U, ValueData[1].Value);
365 ASSERT_EQ(3U, ValueData[1].Count);
366 ASSERT_EQ(5000U, ValueData[2].Value);
367 ASSERT_EQ(2U, ValueData[2].Count);
368 ASSERT_EQ(6000U, ValueData[3].Value);
369 ASSERT_EQ(1U, ValueData[3].Count);
Xinliang David Lif45c13d2016-02-04 19:11:43 +0000370}
371
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000372TEST_P(MaybeSparseInstrProfTest, get_icall_data_read_write_with_weight) {
David Blaikie450ef2a2017-07-06 19:00:12 +0000373 NamedInstrProfRecord Record1("caller", 0x1234, {1, 2});
Xinliang David Li4f87d122016-01-08 03:49:59 +0000374
375 // 4 value sites.
376 Record1.reserveSites(IPVK_IndirectCallTarget, 4);
Xinliang David Li72c40752016-04-10 02:35:53 +0000377 InstrProfValueData VD0[] = {
378 {(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}, {(uint64_t)callee3, 3}};
Xinliang David Li4f87d122016-01-08 03:49:59 +0000379 Record1.addValueData(IPVK_IndirectCallTarget, 0, VD0, 3, nullptr);
380 // No value profile data at the second site.
381 Record1.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000382 InstrProfValueData VD2[] = {{(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}};
Xinliang David Li4f87d122016-01-08 03:49:59 +0000383 Record1.addValueData(IPVK_IndirectCallTarget, 2, VD2, 2, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000384 InstrProfValueData VD3[] = {{(uint64_t)callee1, 1}};
Xinliang David Li4f87d122016-01-08 03:49:59 +0000385 Record1.addValueData(IPVK_IndirectCallTarget, 3, VD3, 1, nullptr);
386
David Blaikiecb160612017-07-10 03:04:59 +0000387 Writer.addRecord(std::move(Record1), 10, Err);
388 Writer.addRecord({"callee1", 0x1235, {3, 4}}, Err);
389 Writer.addRecord({"callee2", 0x1235, {3, 4}}, Err);
390 Writer.addRecord({"callee3", 0x1235, {3, 4}}, Err);
Xinliang David Li4f87d122016-01-08 03:49:59 +0000391 auto Profile = Writer.writeBuffer();
392 readProfile(std::move(Profile));
393
Vedant Kumarc77570e2016-05-19 03:54:45 +0000394 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000395 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000396 ASSERT_EQ(4U, R->getNumValueSites(IPVK_IndirectCallTarget));
397 ASSERT_EQ(3U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
398 ASSERT_EQ(0U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 1));
399 ASSERT_EQ(2U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 2));
400 ASSERT_EQ(1U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 3));
Xinliang David Li4f87d122016-01-08 03:49:59 +0000401
Xinliang David Li730d9342016-02-04 05:29:51 +0000402 uint64_t TotalC;
Xinliang David Li4f87d122016-01-08 03:49:59 +0000403 std::unique_ptr<InstrProfValueData[]> VD =
David Blaikie9f15b402016-02-09 01:02:24 +0000404 R->getValueForSite(IPVK_IndirectCallTarget, 0, &TotalC);
Xinliang David Li4f87d122016-01-08 03:49:59 +0000405 ASSERT_EQ(30U, VD[0].Count);
406 ASSERT_EQ(20U, VD[1].Count);
407 ASSERT_EQ(10U, VD[2].Count);
Xinliang David Li730d9342016-02-04 05:29:51 +0000408 ASSERT_EQ(60U, TotalC);
Xinliang David Li4f87d122016-01-08 03:49:59 +0000409
Xinliang David Li61c7e682015-11-02 05:08:23 +0000410 ASSERT_EQ(StringRef((const char *)VD[0].Value, 7), StringRef("callee3"));
411 ASSERT_EQ(StringRef((const char *)VD[1].Value, 7), StringRef("callee2"));
412 ASSERT_EQ(StringRef((const char *)VD[2].Value, 7), StringRef("callee1"));
413}
414
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000415TEST_P(MaybeSparseInstrProfTest, get_icall_data_read_write_big_endian) {
David Blaikie450ef2a2017-07-06 19:00:12 +0000416 NamedInstrProfRecord Record1("caller", 0x1234, {1, 2});
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000417
418 // 4 value sites.
419 Record1.reserveSites(IPVK_IndirectCallTarget, 4);
Xinliang David Li72c40752016-04-10 02:35:53 +0000420 InstrProfValueData VD0[] = {
421 {(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}, {(uint64_t)callee3, 3}};
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000422 Record1.addValueData(IPVK_IndirectCallTarget, 0, VD0, 3, nullptr);
423 // No value profile data at the second site.
424 Record1.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000425 InstrProfValueData VD2[] = {{(uint64_t)callee1, 1}, {(uint64_t)callee2, 2}};
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000426 Record1.addValueData(IPVK_IndirectCallTarget, 2, VD2, 2, nullptr);
Xinliang David Li72c40752016-04-10 02:35:53 +0000427 InstrProfValueData VD3[] = {{(uint64_t)callee1, 1}};
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000428 Record1.addValueData(IPVK_IndirectCallTarget, 3, VD3, 1, nullptr);
429
David Blaikiecb160612017-07-10 03:04:59 +0000430 Writer.addRecord(std::move(Record1), Err);
431 Writer.addRecord({"callee1", 0x1235, {3, 4}}, Err);
432 Writer.addRecord({"callee2", 0x1235, {3, 4}}, Err);
433 Writer.addRecord({"callee3", 0x1235, {3, 4}}, Err);
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000434
435 // Set big endian output.
436 Writer.setValueProfDataEndianness(support::big);
437
438 auto Profile = Writer.writeBuffer();
439 readProfile(std::move(Profile));
440
441 // Set big endian input.
442 Reader->setValueProfDataEndianness(support::big);
443
Vedant Kumarc77570e2016-05-19 03:54:45 +0000444 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000445 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000446 ASSERT_EQ(4U, R->getNumValueSites(IPVK_IndirectCallTarget));
447 ASSERT_EQ(3U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
448 ASSERT_EQ(0U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 1));
449 ASSERT_EQ(2U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 2));
450 ASSERT_EQ(1U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 3));
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000451
452 std::unique_ptr<InstrProfValueData[]> VD =
David Blaikie9f15b402016-02-09 01:02:24 +0000453 R->getValueForSite(IPVK_IndirectCallTarget, 0);
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000454 ASSERT_EQ(StringRef((const char *)VD[0].Value, 7), StringRef("callee3"));
455 ASSERT_EQ(StringRef((const char *)VD[1].Value, 7), StringRef("callee2"));
456 ASSERT_EQ(StringRef((const char *)VD[2].Value, 7), StringRef("callee1"));
457
458 // Restore little endian default:
459 Writer.setValueProfDataEndianness(support::little);
460}
461
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000462TEST_P(MaybeSparseInstrProfTest, get_icall_data_merge1) {
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000463 static const char caller[] = "caller";
David Blaikie450ef2a2017-07-06 19:00:12 +0000464 NamedInstrProfRecord Record11(caller, 0x1234, {1, 2});
465 NamedInstrProfRecord Record12(caller, 0x1234, {1, 2});
Xinliang David Li61c7e682015-11-02 05:08:23 +0000466
467 // 5 value sites.
468 Record11.reserveSites(IPVK_IndirectCallTarget, 5);
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000469 InstrProfValueData VD0[] = {{uint64_t(callee1), 1},
470 {uint64_t(callee2), 2},
471 {uint64_t(callee3), 3},
472 {uint64_t(callee4), 4}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000473 Record11.addValueData(IPVK_IndirectCallTarget, 0, VD0, 4, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000474
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000475 // No value profile data at the second site.
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000476 Record11.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000477
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000478 InstrProfValueData VD2[] = {
479 {uint64_t(callee1), 1}, {uint64_t(callee2), 2}, {uint64_t(callee3), 3}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000480 Record11.addValueData(IPVK_IndirectCallTarget, 2, VD2, 3, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000481
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000482 InstrProfValueData VD3[] = {{uint64_t(callee1), 1}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000483 Record11.addValueData(IPVK_IndirectCallTarget, 3, VD3, 1, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000484
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000485 InstrProfValueData VD4[] = {{uint64_t(callee1), 1},
486 {uint64_t(callee2), 2},
487 {uint64_t(callee3), 3}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000488 Record11.addValueData(IPVK_IndirectCallTarget, 4, VD4, 3, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000489
David Majnemerfd026c32016-04-22 06:37:48 +0000490 // A different record for the same caller.
Xinliang David Li61c7e682015-11-02 05:08:23 +0000491 Record12.reserveSites(IPVK_IndirectCallTarget, 5);
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000492 InstrProfValueData VD02[] = {{uint64_t(callee2), 5}, {uint64_t(callee3), 3}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000493 Record12.addValueData(IPVK_IndirectCallTarget, 0, VD02, 2, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000494
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000495 // No value profile data at the second site.
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000496 Record12.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000497
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000498 InstrProfValueData VD22[] = {
499 {uint64_t(callee2), 1}, {uint64_t(callee3), 3}, {uint64_t(callee4), 4}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000500 Record12.addValueData(IPVK_IndirectCallTarget, 2, VD22, 3, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000501
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000502 Record12.addValueData(IPVK_IndirectCallTarget, 3, nullptr, 0, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000503
Xinliang David Li8c37fa82016-01-22 19:53:31 +0000504 InstrProfValueData VD42[] = {{uint64_t(callee1), 1},
505 {uint64_t(callee2), 2},
506 {uint64_t(callee3), 3}};
Eugene Zelenko4e036ff2015-11-04 22:32:32 +0000507 Record12.addValueData(IPVK_IndirectCallTarget, 4, VD42, 3, nullptr);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000508
David Blaikiecb160612017-07-10 03:04:59 +0000509 Writer.addRecord(std::move(Record11), Err);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000510 // Merge profile data.
David Blaikiecb160612017-07-10 03:04:59 +0000511 Writer.addRecord(std::move(Record12), Err);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000512
David Blaikiecb160612017-07-10 03:04:59 +0000513 Writer.addRecord({callee1, 0x1235, {3, 4}}, Err);
514 Writer.addRecord({callee2, 0x1235, {3, 4}}, Err);
515 Writer.addRecord({callee3, 0x1235, {3, 4}}, Err);
516 Writer.addRecord({callee3, 0x1235, {3, 4}}, Err);
517 Writer.addRecord({callee4, 0x1235, {3, 5}}, Err);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000518 auto Profile = Writer.writeBuffer();
519 readProfile(std::move(Profile));
520
Vedant Kumarc77570e2016-05-19 03:54:45 +0000521 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000522 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000523 ASSERT_EQ(5U, R->getNumValueSites(IPVK_IndirectCallTarget));
524 ASSERT_EQ(4U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
525 ASSERT_EQ(0U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 1));
526 ASSERT_EQ(4U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 2));
527 ASSERT_EQ(1U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 3));
528 ASSERT_EQ(3U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 4));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000529
530 std::unique_ptr<InstrProfValueData[]> VD =
David Blaikie9f15b402016-02-09 01:02:24 +0000531 R->getValueForSite(IPVK_IndirectCallTarget, 0);
Xinliang David Li61c7e682015-11-02 05:08:23 +0000532 ASSERT_EQ(StringRef((const char *)VD[0].Value, 7), StringRef("callee2"));
533 ASSERT_EQ(7U, VD[0].Count);
534 ASSERT_EQ(StringRef((const char *)VD[1].Value, 7), StringRef("callee3"));
535 ASSERT_EQ(6U, VD[1].Count);
536 ASSERT_EQ(StringRef((const char *)VD[2].Value, 7), StringRef("callee4"));
537 ASSERT_EQ(4U, VD[2].Count);
538 ASSERT_EQ(StringRef((const char *)VD[3].Value, 7), StringRef("callee1"));
539 ASSERT_EQ(1U, VD[3].Count);
540
541 std::unique_ptr<InstrProfValueData[]> VD_2(
David Blaikie9f15b402016-02-09 01:02:24 +0000542 R->getValueForSite(IPVK_IndirectCallTarget, 2));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000543 ASSERT_EQ(StringRef((const char *)VD_2[0].Value, 7), StringRef("callee3"));
544 ASSERT_EQ(6U, VD_2[0].Count);
545 ASSERT_EQ(StringRef((const char *)VD_2[1].Value, 7), StringRef("callee4"));
546 ASSERT_EQ(4U, VD_2[1].Count);
547 ASSERT_EQ(StringRef((const char *)VD_2[2].Value, 7), StringRef("callee2"));
548 ASSERT_EQ(3U, VD_2[2].Count);
549 ASSERT_EQ(StringRef((const char *)VD_2[3].Value, 7), StringRef("callee1"));
550 ASSERT_EQ(1U, VD_2[3].Count);
551
552 std::unique_ptr<InstrProfValueData[]> VD_3(
David Blaikie9f15b402016-02-09 01:02:24 +0000553 R->getValueForSite(IPVK_IndirectCallTarget, 3));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000554 ASSERT_EQ(StringRef((const char *)VD_3[0].Value, 7), StringRef("callee1"));
555 ASSERT_EQ(1U, VD_3[0].Count);
556
557 std::unique_ptr<InstrProfValueData[]> VD_4(
David Blaikie9f15b402016-02-09 01:02:24 +0000558 R->getValueForSite(IPVK_IndirectCallTarget, 4));
Xinliang David Li61c7e682015-11-02 05:08:23 +0000559 ASSERT_EQ(StringRef((const char *)VD_4[0].Value, 7), StringRef("callee3"));
560 ASSERT_EQ(6U, VD_4[0].Count);
561 ASSERT_EQ(StringRef((const char *)VD_4[1].Value, 7), StringRef("callee2"));
562 ASSERT_EQ(4U, VD_4[1].Count);
563 ASSERT_EQ(StringRef((const char *)VD_4[2].Value, 7), StringRef("callee1"));
564 ASSERT_EQ(2U, VD_4[2].Count);
565}
566
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000567TEST_P(MaybeSparseInstrProfTest, get_icall_data_merge1_saturation) {
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000568 static const char bar[] = "bar";
569
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000570 const uint64_t Max = std::numeric_limits<uint64_t>::max();
571
David Blaikiecb160612017-07-10 03:04:59 +0000572 instrprof_error Result;
573 auto Err = [&](Error E) { Result = InstrProfError::take(std::move(E)); };
574 Result = instrprof_error::success;
575 Writer.addRecord({"foo", 0x1234, {1}}, Err);
576 ASSERT_EQ(Result, instrprof_error::success);
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000577
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000578 // Verify counter overflow.
David Blaikiecb160612017-07-10 03:04:59 +0000579 Result = instrprof_error::success;
580 Writer.addRecord({"foo", 0x1234, {Max}}, Err);
581 ASSERT_EQ(Result, instrprof_error::counter_overflow);
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000582
David Blaikiecb160612017-07-10 03:04:59 +0000583 Result = instrprof_error::success;
584 Writer.addRecord({bar, 0x9012, {8}}, Err);
585 ASSERT_EQ(Result, instrprof_error::success);
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000586
David Blaikie450ef2a2017-07-06 19:00:12 +0000587 NamedInstrProfRecord Record4("baz", 0x5678, {3, 4});
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000588 Record4.reserveSites(IPVK_IndirectCallTarget, 1);
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000589 InstrProfValueData VD4[] = {{uint64_t(bar), 1}};
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000590 Record4.addValueData(IPVK_IndirectCallTarget, 0, VD4, 1, nullptr);
David Blaikiecb160612017-07-10 03:04:59 +0000591 Result = instrprof_error::success;
592 Writer.addRecord(std::move(Record4), Err);
593 ASSERT_EQ(Result, instrprof_error::success);
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000594
595 // Verify value data counter overflow.
David Blaikie450ef2a2017-07-06 19:00:12 +0000596 NamedInstrProfRecord Record5("baz", 0x5678, {5, 6});
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000597 Record5.reserveSites(IPVK_IndirectCallTarget, 1);
NAKAMURA Takumiaee63ad2015-12-27 06:18:57 +0000598 InstrProfValueData VD5[] = {{uint64_t(bar), Max}};
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000599 Record5.addValueData(IPVK_IndirectCallTarget, 0, VD5, 1, nullptr);
David Blaikiecb160612017-07-10 03:04:59 +0000600 Result = instrprof_error::success;
601 Writer.addRecord(std::move(Record5), Err);
602 ASSERT_EQ(Result, instrprof_error::counter_overflow);
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000603
604 auto Profile = Writer.writeBuffer();
605 readProfile(std::move(Profile));
606
607 // Verify saturation of counts.
Vedant Kumarc77570e2016-05-19 03:54:45 +0000608 Expected<InstrProfRecord> ReadRecord1 =
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000609 Reader->getInstrProfRecord("foo", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000610 EXPECT_THAT_ERROR(ReadRecord1.takeError(), Succeeded());
David Blaikie9f15b402016-02-09 01:02:24 +0000611 ASSERT_EQ(Max, ReadRecord1->Counts[0]);
Nathan Slingerland1c2b9982015-12-02 18:19:24 +0000612
Vedant Kumarc77570e2016-05-19 03:54:45 +0000613 Expected<InstrProfRecord> ReadRecord2 =
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000614 Reader->getInstrProfRecord("baz", 0x5678);
Vedant Kumarc77570e2016-05-19 03:54:45 +0000615 ASSERT_TRUE(bool(ReadRecord2));
David Blaikie9f15b402016-02-09 01:02:24 +0000616 ASSERT_EQ(1U, ReadRecord2->getNumValueSites(IPVK_IndirectCallTarget));
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000617 std::unique_ptr<InstrProfValueData[]> VD =
David Blaikie9f15b402016-02-09 01:02:24 +0000618 ReadRecord2->getValueForSite(IPVK_IndirectCallTarget, 0);
Nathan Slingerlandfd568242015-12-16 21:45:43 +0000619 ASSERT_EQ(StringRef("bar"), StringRef((const char *)VD[0].Value, 3));
620 ASSERT_EQ(Max, VD[0].Count);
Daniel Sanders0bee4ed2015-11-20 13:13:53 +0000621}
622
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000623// This test tests that when there are too many values
624// for a given site, the merged results are properly
625// truncated.
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000626TEST_P(MaybeSparseInstrProfTest, get_icall_data_merge_site_trunc) {
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000627 static const char caller[] = "caller";
628
David Blaikie450ef2a2017-07-06 19:00:12 +0000629 NamedInstrProfRecord Record11(caller, 0x1234, {1, 2});
630 NamedInstrProfRecord Record12(caller, 0x1234, {1, 2});
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000631
632 // 2 value sites.
633 Record11.reserveSites(IPVK_IndirectCallTarget, 2);
634 InstrProfValueData VD0[255];
635 for (int I = 0; I < 255; I++) {
636 VD0[I].Value = 2 * I;
637 VD0[I].Count = 2 * I + 1000;
638 }
639
640 Record11.addValueData(IPVK_IndirectCallTarget, 0, VD0, 255, nullptr);
641 Record11.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
642
643 Record12.reserveSites(IPVK_IndirectCallTarget, 2);
644 InstrProfValueData VD1[255];
645 for (int I = 0; I < 255; I++) {
646 VD1[I].Value = 2 * I + 1;
647 VD1[I].Count = 2 * I + 1001;
648 }
649
650 Record12.addValueData(IPVK_IndirectCallTarget, 0, VD1, 255, nullptr);
651 Record12.addValueData(IPVK_IndirectCallTarget, 1, nullptr, 0, nullptr);
652
David Blaikiecb160612017-07-10 03:04:59 +0000653 Writer.addRecord(std::move(Record11), Err);
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000654 // Merge profile data.
David Blaikiecb160612017-07-10 03:04:59 +0000655 Writer.addRecord(std::move(Record12), Err);
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000656
657 auto Profile = Writer.writeBuffer();
658 readProfile(std::move(Profile));
659
Vedant Kumarc77570e2016-05-19 03:54:45 +0000660 Expected<InstrProfRecord> R = Reader->getInstrProfRecord("caller", 0x1234);
David Blaikie8c2dc922017-07-07 21:02:59 +0000661 EXPECT_THAT_ERROR(R.takeError(), Succeeded());
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000662 std::unique_ptr<InstrProfValueData[]> VD(
David Blaikie9f15b402016-02-09 01:02:24 +0000663 R->getValueForSite(IPVK_IndirectCallTarget, 0));
664 ASSERT_EQ(2U, R->getNumValueSites(IPVK_IndirectCallTarget));
665 ASSERT_EQ(255U, R->getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
NAKAMURA Takumi7203e502016-01-08 07:58:20 +0000666 for (unsigned I = 0; I < 255; I++) {
Xinliang David Li5dcdd412016-01-08 06:54:27 +0000667 ASSERT_EQ(VD[I].Value, 509 - I);
668 ASSERT_EQ(VD[I].Count, 1509 - I);
669 }
670}
671
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000672static void addValueProfData(InstrProfRecord &Record) {
673 Record.reserveSites(IPVK_IndirectCallTarget, 5);
674 InstrProfValueData VD0[] = {{uint64_t(callee1), 400},
675 {uint64_t(callee2), 1000},
676 {uint64_t(callee3), 500},
677 {uint64_t(callee4), 300},
678 {uint64_t(callee5), 100}};
679 Record.addValueData(IPVK_IndirectCallTarget, 0, VD0, 5, nullptr);
680 InstrProfValueData VD1[] = {{uint64_t(callee5), 800},
681 {uint64_t(callee3), 1000},
682 {uint64_t(callee2), 2500},
683 {uint64_t(callee1), 1300}};
684 Record.addValueData(IPVK_IndirectCallTarget, 1, VD1, 4, nullptr);
685 InstrProfValueData VD2[] = {{uint64_t(callee6), 800},
686 {uint64_t(callee3), 1000},
687 {uint64_t(callee4), 5500}};
688 Record.addValueData(IPVK_IndirectCallTarget, 2, VD2, 3, nullptr);
689 InstrProfValueData VD3[] = {{uint64_t(callee2), 1800},
690 {uint64_t(callee3), 2000}};
691 Record.addValueData(IPVK_IndirectCallTarget, 3, VD3, 2, nullptr);
692 Record.addValueData(IPVK_IndirectCallTarget, 4, nullptr, 0, nullptr);
693}
Xinliang David Li93121792015-11-25 23:31:18 +0000694
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000695TEST_P(MaybeSparseInstrProfTest, value_prof_data_read_write) {
David Blaikie450ef2a2017-07-06 19:00:12 +0000696 InstrProfRecord SrcRecord({1ULL << 31, 2});
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000697 addValueProfData(SrcRecord);
698 std::unique_ptr<ValueProfData> VPData =
699 ValueProfData::serializeFrom(SrcRecord);
Xinliang David Li93121792015-11-25 23:31:18 +0000700
David Blaikie450ef2a2017-07-06 19:00:12 +0000701 InstrProfRecord Record({1ULL << 31, 2});
Eugene Zelenko51ecde12016-01-26 18:48:36 +0000702 VPData->deserializeTo(Record, nullptr);
Xinliang David Li93121792015-11-25 23:31:18 +0000703
704 // Now read data from Record and sanity check the data
705 ASSERT_EQ(5U, Record.getNumValueSites(IPVK_IndirectCallTarget));
706 ASSERT_EQ(5U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
707 ASSERT_EQ(4U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 1));
708 ASSERT_EQ(3U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 2));
709 ASSERT_EQ(2U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 3));
710 ASSERT_EQ(0U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 4));
711
712 auto Cmp = [](const InstrProfValueData &VD1, const InstrProfValueData &VD2) {
713 return VD1.Count > VD2.Count;
714 };
715 std::unique_ptr<InstrProfValueData[]> VD_0(
716 Record.getValueForSite(IPVK_IndirectCallTarget, 0));
Mandeep Singh Grang77a1dcd2018-04-07 01:29:45 +0000717 llvm::sort(&VD_0[0], &VD_0[5], Cmp);
Xinliang David Li93121792015-11-25 23:31:18 +0000718 ASSERT_EQ(StringRef((const char *)VD_0[0].Value, 7), StringRef("callee2"));
719 ASSERT_EQ(1000U, VD_0[0].Count);
720 ASSERT_EQ(StringRef((const char *)VD_0[1].Value, 7), StringRef("callee3"));
721 ASSERT_EQ(500U, VD_0[1].Count);
722 ASSERT_EQ(StringRef((const char *)VD_0[2].Value, 7), StringRef("callee1"));
723 ASSERT_EQ(400U, VD_0[2].Count);
724 ASSERT_EQ(StringRef((const char *)VD_0[3].Value, 7), StringRef("callee4"));
725 ASSERT_EQ(300U, VD_0[3].Count);
726 ASSERT_EQ(StringRef((const char *)VD_0[4].Value, 7), StringRef("callee5"));
727 ASSERT_EQ(100U, VD_0[4].Count);
728
729 std::unique_ptr<InstrProfValueData[]> VD_1(
730 Record.getValueForSite(IPVK_IndirectCallTarget, 1));
Mandeep Singh Grang77a1dcd2018-04-07 01:29:45 +0000731 llvm::sort(&VD_1[0], &VD_1[4], Cmp);
Xinliang David Li93121792015-11-25 23:31:18 +0000732 ASSERT_EQ(StringRef((const char *)VD_1[0].Value, 7), StringRef("callee2"));
733 ASSERT_EQ(2500U, VD_1[0].Count);
734 ASSERT_EQ(StringRef((const char *)VD_1[1].Value, 7), StringRef("callee1"));
735 ASSERT_EQ(1300U, VD_1[1].Count);
736 ASSERT_EQ(StringRef((const char *)VD_1[2].Value, 7), StringRef("callee3"));
737 ASSERT_EQ(1000U, VD_1[2].Count);
738 ASSERT_EQ(StringRef((const char *)VD_1[3].Value, 7), StringRef("callee5"));
739 ASSERT_EQ(800U, VD_1[3].Count);
740
741 std::unique_ptr<InstrProfValueData[]> VD_2(
742 Record.getValueForSite(IPVK_IndirectCallTarget, 2));
Mandeep Singh Grang77a1dcd2018-04-07 01:29:45 +0000743 llvm::sort(&VD_2[0], &VD_2[3], Cmp);
Xinliang David Li93121792015-11-25 23:31:18 +0000744 ASSERT_EQ(StringRef((const char *)VD_2[0].Value, 7), StringRef("callee4"));
745 ASSERT_EQ(5500U, VD_2[0].Count);
746 ASSERT_EQ(StringRef((const char *)VD_2[1].Value, 7), StringRef("callee3"));
747 ASSERT_EQ(1000U, VD_2[1].Count);
748 ASSERT_EQ(StringRef((const char *)VD_2[2].Value, 7), StringRef("callee6"));
749 ASSERT_EQ(800U, VD_2[2].Count);
750
751 std::unique_ptr<InstrProfValueData[]> VD_3(
752 Record.getValueForSite(IPVK_IndirectCallTarget, 3));
Mandeep Singh Grang77a1dcd2018-04-07 01:29:45 +0000753 llvm::sort(&VD_3[0], &VD_3[2], Cmp);
Xinliang David Li93121792015-11-25 23:31:18 +0000754 ASSERT_EQ(StringRef((const char *)VD_3[0].Value, 7), StringRef("callee3"));
755 ASSERT_EQ(2000U, VD_3[0].Count);
756 ASSERT_EQ(StringRef((const char *)VD_3[1].Value, 7), StringRef("callee2"));
757 ASSERT_EQ(1800U, VD_3[1].Count);
Xinliang David Li93121792015-11-25 23:31:18 +0000758}
759
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000760TEST_P(MaybeSparseInstrProfTest, value_prof_data_read_write_mapping) {
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000761
David Blaikie450ef2a2017-07-06 19:00:12 +0000762 NamedInstrProfRecord SrcRecord("caller", 0x1234, {1ULL << 31, 2});
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000763 addValueProfData(SrcRecord);
764 std::unique_ptr<ValueProfData> VPData =
765 ValueProfData::serializeFrom(SrcRecord);
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000766
David Blaikie450ef2a2017-07-06 19:00:12 +0000767 NamedInstrProfRecord Record("caller", 0x1234, {1ULL << 31, 2});
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000768 InstrProfSymtab Symtab;
769 Symtab.mapAddress(uint64_t(callee1), 0x1000ULL);
770 Symtab.mapAddress(uint64_t(callee2), 0x2000ULL);
771 Symtab.mapAddress(uint64_t(callee3), 0x3000ULL);
772 Symtab.mapAddress(uint64_t(callee4), 0x4000ULL);
773 // Missing mapping for callee5
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000774
Mircea Trofin06b07832018-03-21 22:27:31 +0000775 VPData->deserializeTo(Record, &Symtab);
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000776
777 // Now read data from Record and sanity check the data
Xinliang David Lia4cc7ae2016-05-13 00:23:49 +0000778 ASSERT_EQ(5U, Record.getNumValueSites(IPVK_IndirectCallTarget));
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000779 ASSERT_EQ(5U, Record.getNumValueDataForSite(IPVK_IndirectCallTarget, 0));
780
781 auto Cmp = [](const InstrProfValueData &VD1, const InstrProfValueData &VD2) {
782 return VD1.Count > VD2.Count;
783 };
784 std::unique_ptr<InstrProfValueData[]> VD_0(
785 Record.getValueForSite(IPVK_IndirectCallTarget, 0));
Mandeep Singh Grang77a1dcd2018-04-07 01:29:45 +0000786 llvm::sort(&VD_0[0], &VD_0[5], Cmp);
Xinliang David Li6aa584f2016-04-10 03:32:02 +0000787 ASSERT_EQ(VD_0[0].Value, 0x2000ULL);
788 ASSERT_EQ(1000U, VD_0[0].Count);
789 ASSERT_EQ(VD_0[1].Value, 0x3000ULL);
790 ASSERT_EQ(500U, VD_0[1].Count);
791 ASSERT_EQ(VD_0[2].Value, 0x1000ULL);
792 ASSERT_EQ(400U, VD_0[2].Count);
793
794 // callee5 does not have a mapped value -- default to 0.
795 ASSERT_EQ(VD_0[4].Value, 0ULL);
796}
797
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000798TEST_P(MaybeSparseInstrProfTest, get_max_function_count) {
David Blaikiecb160612017-07-10 03:04:59 +0000799 Writer.addRecord({"foo", 0x1234, {1ULL << 31, 2}}, Err);
800 Writer.addRecord({"bar", 0, {1ULL << 63}}, Err);
801 Writer.addRecord({"baz", 0x5678, {0, 0, 0, 0}}, Err);
Justin Bogner09e5af72015-02-18 01:58:17 +0000802 auto Profile = Writer.writeBuffer();
803 readProfile(std::move(Profile));
804
805 ASSERT_EQ(1ULL << 63, Reader->getMaximumFunctionCount());
806}
807
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000808TEST_P(MaybeSparseInstrProfTest, get_weighted_function_counts) {
David Blaikiecb160612017-07-10 03:04:59 +0000809 Writer.addRecord({"foo", 0x1234, {1, 2}}, 3, Err);
810 Writer.addRecord({"foo", 0x1235, {3, 4}}, 5, Err);
Nathan Slingerland824c3ec2015-12-15 17:37:09 +0000811 auto Profile = Writer.writeBuffer();
812 readProfile(std::move(Profile));
813
814 std::vector<uint64_t> Counts;
David Blaikie8c2dc922017-07-07 21:02:59 +0000815 EXPECT_THAT_ERROR(Reader->getFunctionCounts("foo", 0x1234, Counts),
816 Succeeded());
Nathan Slingerland824c3ec2015-12-15 17:37:09 +0000817 ASSERT_EQ(2U, Counts.size());
818 ASSERT_EQ(3U, Counts[0]);
819 ASSERT_EQ(6U, Counts[1]);
820
David Blaikie8c2dc922017-07-07 21:02:59 +0000821 EXPECT_THAT_ERROR(Reader->getFunctionCounts("foo", 0x1235, Counts),
822 Succeeded());
Nathan Slingerland824c3ec2015-12-15 17:37:09 +0000823 ASSERT_EQ(2U, Counts.size());
824 ASSERT_EQ(15U, Counts[0]);
825 ASSERT_EQ(20U, Counts[1]);
826}
827
Xinliang David Lia55dfdc2016-02-09 05:47:08 +0000828// Testing symtab creator interface used by indexed profile reader.
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000829TEST_P(MaybeSparseInstrProfTest, instr_prof_symtab_test) {
Xinliang David Liab49f7c2015-12-19 07:44:57 +0000830 std::vector<StringRef> FuncNames;
831 FuncNames.push_back("func1");
832 FuncNames.push_back("func2");
833 FuncNames.push_back("func3");
834 FuncNames.push_back("bar1");
835 FuncNames.push_back("bar2");
836 FuncNames.push_back("bar3");
837 InstrProfSymtab Symtab;
David Blaikie8c2dc922017-07-07 21:02:59 +0000838 EXPECT_THAT_ERROR(Symtab.create(FuncNames), Succeeded());
Xinliang David Liab49f7c2015-12-19 07:44:57 +0000839 StringRef R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func1"));
840 ASSERT_EQ(StringRef("func1"), R);
841 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func2"));
842 ASSERT_EQ(StringRef("func2"), R);
843 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func3"));
844 ASSERT_EQ(StringRef("func3"), R);
845 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar1"));
846 ASSERT_EQ(StringRef("bar1"), R);
847 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar2"));
848 ASSERT_EQ(StringRef("bar2"), R);
849 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar3"));
850 ASSERT_EQ(StringRef("bar3"), R);
Xinliang David Lia6d12c92015-12-19 18:20:09 +0000851
Xinliang David Liebbb19e2016-02-10 06:36:55 +0000852 // negative tests
853 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar4"));
854 ASSERT_EQ(StringRef(), R);
855 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("foo4"));
856 ASSERT_EQ(StringRef(), R);
857
Xinliang David Lia6d12c92015-12-19 18:20:09 +0000858 // Now incrementally update the symtab
David Blaikie8c2dc922017-07-07 21:02:59 +0000859 EXPECT_THAT_ERROR(Symtab.addFuncName("blah_1"), Succeeded());
860 EXPECT_THAT_ERROR(Symtab.addFuncName("blah_2"), Succeeded());
861 EXPECT_THAT_ERROR(Symtab.addFuncName("blah_3"), Succeeded());
Xinliang David Lia6d12c92015-12-19 18:20:09 +0000862
863 // Check again
864 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("blah_1"));
865 ASSERT_EQ(StringRef("blah_1"), R);
866 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("blah_2"));
867 ASSERT_EQ(StringRef("blah_2"), R);
868 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("blah_3"));
869 ASSERT_EQ(StringRef("blah_3"), R);
870 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func1"));
871 ASSERT_EQ(StringRef("func1"), R);
872 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func2"));
873 ASSERT_EQ(StringRef("func2"), R);
874 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("func3"));
875 ASSERT_EQ(StringRef("func3"), R);
876 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar1"));
877 ASSERT_EQ(StringRef("bar1"), R);
878 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar2"));
879 ASSERT_EQ(StringRef("bar2"), R);
880 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash("bar3"));
881 ASSERT_EQ(StringRef("bar3"), R);
Xinliang David Liab49f7c2015-12-19 07:44:57 +0000882}
883
Vedant Kumarb9454632017-06-20 01:38:56 +0000884// Test that we get an error when creating a bogus symtab.
885TEST_P(MaybeSparseInstrProfTest, instr_prof_bogus_symtab_empty_func_name) {
886 InstrProfSymtab Symtab;
David Blaikie8c2dc922017-07-07 21:02:59 +0000887 EXPECT_TRUE(ErrorEquals(instrprof_error::malformed, Symtab.addFuncName("")));
Vedant Kumarb9454632017-06-20 01:38:56 +0000888}
889
Xinliang David Lia55dfdc2016-02-09 05:47:08 +0000890// Testing symtab creator interface used by value profile transformer.
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000891TEST_P(MaybeSparseInstrProfTest, instr_prof_symtab_module_test) {
Xinliang David Li6d3068a2016-01-20 01:26:34 +0000892 LLVMContext Ctx;
893 std::unique_ptr<Module> M = llvm::make_unique<Module>("MyModule.cpp", Ctx);
894 FunctionType *FTy = FunctionType::get(Type::getVoidTy(Ctx),
895 /*isVarArg=*/false);
896 Function::Create(FTy, Function::ExternalLinkage, "Gfoo", M.get());
897 Function::Create(FTy, Function::ExternalLinkage, "Gblah", M.get());
898 Function::Create(FTy, Function::ExternalLinkage, "Gbar", M.get());
899 Function::Create(FTy, Function::InternalLinkage, "Ifoo", M.get());
900 Function::Create(FTy, Function::InternalLinkage, "Iblah", M.get());
901 Function::Create(FTy, Function::InternalLinkage, "Ibar", M.get());
902 Function::Create(FTy, Function::PrivateLinkage, "Pfoo", M.get());
903 Function::Create(FTy, Function::PrivateLinkage, "Pblah", M.get());
904 Function::Create(FTy, Function::PrivateLinkage, "Pbar", M.get());
905 Function::Create(FTy, Function::WeakODRLinkage, "Wfoo", M.get());
906 Function::Create(FTy, Function::WeakODRLinkage, "Wblah", M.get());
907 Function::Create(FTy, Function::WeakODRLinkage, "Wbar", M.get());
908
909 InstrProfSymtab ProfSymtab;
David Blaikie8c2dc922017-07-07 21:02:59 +0000910 EXPECT_THAT_ERROR(ProfSymtab.create(*M), Succeeded());
Xinliang David Li6d3068a2016-01-20 01:26:34 +0000911
912 StringRef Funcs[] = {"Gfoo", "Gblah", "Gbar", "Ifoo", "Iblah", "Ibar",
913 "Pfoo", "Pblah", "Pbar", "Wfoo", "Wblah", "Wbar"};
914
915 for (unsigned I = 0; I < sizeof(Funcs) / sizeof(*Funcs); I++) {
916 Function *F = M->getFunction(Funcs[I]);
Eugene Zelenko51ecde12016-01-26 18:48:36 +0000917 ASSERT_TRUE(F != nullptr);
Xinliang David Li728b07f2016-01-20 02:49:53 +0000918 std::string PGOName = getPGOFuncName(*F);
Xinliang David Li0f05a462016-01-22 18:13:34 +0000919 uint64_t Key = IndexedInstrProf::ComputeHash(PGOName);
Xinliang David Li728b07f2016-01-20 02:49:53 +0000920 ASSERT_EQ(StringRef(PGOName),
Xinliang David Li0f05a462016-01-22 18:13:34 +0000921 ProfSymtab.getFuncName(Key));
922 ASSERT_EQ(StringRef(Funcs[I]), ProfSymtab.getOrigFuncName(Key));
Xinliang David Li6d3068a2016-01-20 01:26:34 +0000923 }
924}
925
Xinliang David Lia55dfdc2016-02-09 05:47:08 +0000926// Testing symtab serialization and creator/deserialization interface
927// used by coverage map reader, and raw profile reader.
Vedant Kumar0c94d7d2016-01-29 22:54:45 +0000928TEST_P(MaybeSparseInstrProfTest, instr_prof_symtab_compression_test) {
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000929 std::vector<std::string> FuncNames1;
930 std::vector<std::string> FuncNames2;
Xinliang David Lie8571872016-02-09 05:36:57 +0000931 for (int I = 0; I < 3; I++) {
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000932 std::string str;
933 raw_string_ostream OS(str);
934 OS << "func_" << I;
935 FuncNames1.push_back(OS.str());
936 str.clear();
Vedant Kumar76bf9912016-03-28 21:06:42 +0000937 OS << "f oooooooooooooo_" << I;
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000938 FuncNames1.push_back(OS.str());
939 str.clear();
940 OS << "BAR_" << I;
941 FuncNames2.push_back(OS.str());
942 str.clear();
943 OS << "BlahblahBlahblahBar_" << I;
944 FuncNames2.push_back(OS.str());
945 }
946
Xinliang David Lid1273882016-01-29 22:29:15 +0000947 for (bool DoCompression : {false, true}) {
948 // Compressing:
949 std::string FuncNameStrings1;
David Blaikie8c2dc922017-07-07 21:02:59 +0000950 EXPECT_THAT_ERROR(collectPGOFuncNameStrings(
951 FuncNames1, (DoCompression && zlib::isAvailable()),
952 FuncNameStrings1),
953 Succeeded());
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000954
Xinliang David Lid1273882016-01-29 22:29:15 +0000955 // Compressing:
956 std::string FuncNameStrings2;
David Blaikie8c2dc922017-07-07 21:02:59 +0000957 EXPECT_THAT_ERROR(collectPGOFuncNameStrings(
958 FuncNames2, (DoCompression && zlib::isAvailable()),
959 FuncNameStrings2),
960 Succeeded());
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000961
Xinliang David Lie8571872016-02-09 05:36:57 +0000962 for (int Padding = 0; Padding < 2; Padding++) {
963 // Join with paddings :
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000964 std::string FuncNameStrings = FuncNameStrings1;
965 for (int P = 0; P < Padding; P++) {
966 FuncNameStrings.push_back('\0');
967 }
968 FuncNameStrings += FuncNameStrings2;
969
Xinliang David Li2401ff62016-01-04 23:59:14 +0000970 // Now decompress:
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000971 InstrProfSymtab Symtab;
David Blaikie8c2dc922017-07-07 21:02:59 +0000972 EXPECT_THAT_ERROR(Symtab.create(StringRef(FuncNameStrings)), Succeeded());
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000973
Xinliang David Li2401ff62016-01-04 23:59:14 +0000974 // Now do the checks:
975 // First sampling some data points:
Xinliang David Li1626c622016-01-29 21:26:31 +0000976 StringRef R = Symtab.getFuncName(IndexedInstrProf::ComputeHash(FuncNames1[0]));
Xinliang David Li2401ff62016-01-04 23:59:14 +0000977 ASSERT_EQ(StringRef("func_0"), R);
978 R = Symtab.getFuncName(IndexedInstrProf::ComputeHash(FuncNames1[1]));
Vedant Kumar76bf9912016-03-28 21:06:42 +0000979 ASSERT_EQ(StringRef("f oooooooooooooo_0"), R);
Xinliang David Lie8571872016-02-09 05:36:57 +0000980 for (int I = 0; I < 3; I++) {
Xinliang David Lif2f39d62015-12-31 07:57:16 +0000981 std::string N[4];
982 N[0] = FuncNames1[2 * I];
983 N[1] = FuncNames1[2 * I + 1];
984 N[2] = FuncNames2[2 * I];
985 N[3] = FuncNames2[2 * I + 1];
986 for (int J = 0; J < 4; J++) {
987 StringRef R = Symtab.getFuncName(IndexedInstrProf::ComputeHash(N[J]));
988 ASSERT_EQ(StringRef(N[J]), R);
989 }
990 }
991 }
992 }
993}
994
Richard Smith3b1e4302018-10-10 21:09:37 +0000995TEST_P(MaybeSparseInstrProfTest, remapping_test) {
996 Writer.addRecord({"_Z3fooi", 0x1234, {1, 2, 3, 4}}, Err);
997 Writer.addRecord({"file:_Z3barf", 0x567, {5, 6, 7}}, Err);
998 auto Profile = Writer.writeBuffer();
999 readProfile(std::move(Profile), llvm::MemoryBuffer::getMemBuffer(R"(
1000 type i l
1001 name 3bar 4quux
1002 )"));
1003
1004 std::vector<uint64_t> Counts;
1005 for (StringRef FooName : {"_Z3fooi", "_Z3fool"}) {
1006 EXPECT_THAT_ERROR(Reader->getFunctionCounts(FooName, 0x1234, Counts),
1007 Succeeded());
1008 ASSERT_EQ(4u, Counts.size());
1009 EXPECT_EQ(1u, Counts[0]);
1010 EXPECT_EQ(2u, Counts[1]);
1011 EXPECT_EQ(3u, Counts[2]);
1012 EXPECT_EQ(4u, Counts[3]);
1013 }
1014
1015 for (StringRef BarName : {"file:_Z3barf", "file:_Z4quuxf"}) {
1016 EXPECT_THAT_ERROR(Reader->getFunctionCounts(BarName, 0x567, Counts),
1017 Succeeded());
1018 ASSERT_EQ(3u, Counts.size());
1019 EXPECT_EQ(5u, Counts[0]);
1020 EXPECT_EQ(6u, Counts[1]);
1021 EXPECT_EQ(7u, Counts[2]);
1022 }
1023
1024 for (StringRef BadName : {"_Z3foof", "_Z4quuxi", "_Z3barl", "", "_ZZZ",
1025 "_Z3barf", "otherfile:_Z4quuxf"}) {
1026 EXPECT_THAT_ERROR(Reader->getFunctionCounts(BadName, 0x1234, Counts),
1027 Failed());
1028 EXPECT_THAT_ERROR(Reader->getFunctionCounts(BadName, 0x567, Counts),
1029 Failed());
1030 }
1031}
1032
Vedant Kumar0c94d7d2016-01-29 22:54:45 +00001033TEST_F(SparseInstrProfTest, preserve_no_records) {
David Blaikiecb160612017-07-10 03:04:59 +00001034 Writer.addRecord({"foo", 0x1234, {0}}, Err);
1035 Writer.addRecord({"bar", 0x4321, {0, 0}}, Err);
Vedant Kumarcdcc59f2017-07-10 21:44:43 +00001036 Writer.addRecord({"baz", 0x4321, {0, 0, 0}}, Err);
Vedant Kumar0c94d7d2016-01-29 22:54:45 +00001037
Vedant Kumar0c94d7d2016-01-29 22:54:45 +00001038 auto Profile = Writer.writeBuffer();
1039 readProfile(std::move(Profile));
1040
1041 auto I = Reader->begin(), E = Reader->end();
1042 ASSERT_TRUE(I == E);
1043}
1044
1045INSTANTIATE_TEST_CASE_P(MaybeSparse, MaybeSparseInstrProfTest,
Galina Kistanova34d05a52017-06-04 05:30:26 +00001046 ::testing::Bool(),);
Vedant Kumar0c94d7d2016-01-29 22:54:45 +00001047
Justin Bogner09e5af72015-02-18 01:58:17 +00001048} // end anonymous namespace