1 //===- bolt/Profile/Heatmap.cpp -------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "bolt/Profile/Heatmap.h"
10 #include "bolt/Utils/CommandLineOpts.h"
11 #include "llvm/ADT/StringMap.h"
12 #include "llvm/ADT/Twine.h"
13 #include "llvm/Support/Debug.h"
14 #include "llvm/Support/FileSystem.h"
15 #include "llvm/Support/Format.h"
16 #include "llvm/Support/FormatVariadic.h"
17 #include "llvm/Support/MathExtras.h"
18 #include "llvm/Support/raw_ostream.h"
24 #define DEBUG_TYPE "bolt-heatmap"
31 void Heatmap::registerAddressRange(uint64_t StartAddress
, uint64_t EndAddress
,
33 if (ignoreAddress(StartAddress
)) {
38 if (StartAddress
> EndAddress
|| EndAddress
- StartAddress
> 64 * 1024) {
39 LLVM_DEBUG(dbgs() << "invalid range : 0x" << Twine::utohexstr(StartAddress
)
40 << " -> 0x" << Twine::utohexstr(EndAddress
) << '\n');
45 for (uint64_t Bucket
= StartAddress
/ BucketSize
;
46 Bucket
<= EndAddress
/ BucketSize
; ++Bucket
)
50 void Heatmap::print(StringRef FileName
) const {
52 raw_fd_ostream
OS(FileName
, EC
, sys::fs::OpenFlags::OF_None
);
54 errs() << "error opening output file: " << EC
.message() << '\n';
60 void Heatmap::print(raw_ostream
&OS
) const {
61 const char FillChar
= '.';
63 const auto DefaultColor
= raw_ostream::WHITE
;
64 auto changeColor
= [&](raw_ostream::Colors Color
) -> void {
65 static auto CurrentColor
= raw_ostream::BLACK
;
66 if (CurrentColor
== Color
)
68 OS
.changeColor(Color
);
72 const uint64_t BytesPerLine
= opts::BucketsPerLine
* BucketSize
;
74 // Calculate the max value for scaling.
75 uint64_t MaxValue
= 0;
76 for (const std::pair
<const uint64_t, uint64_t> &Entry
: Map
)
77 MaxValue
= std::max
<uint64_t>(MaxValue
, Entry
.second
);
79 // Print start of the line and fill it with an empty space right before
81 auto startLine
= [&](uint64_t Address
, bool Empty
= false) {
82 changeColor(DefaultColor
);
83 const uint64_t LineAddress
= Address
/ BytesPerLine
* BytesPerLine
;
85 if (MaxAddress
> 0xffffffff)
86 OS
<< format("0x%016" PRIx64
": ", LineAddress
);
88 OS
<< format("0x%08" PRIx64
": ", LineAddress
);
91 Address
= LineAddress
+ BytesPerLine
;
92 for (uint64_t Fill
= LineAddress
; Fill
< Address
; Fill
+= BucketSize
)
96 // Finish line after \p Address was printed.
97 auto finishLine
= [&](uint64_t Address
) {
98 const uint64_t End
= alignTo(Address
+ 1, BytesPerLine
);
99 for (uint64_t Fill
= Address
+ BucketSize
; Fill
< End
; Fill
+= BucketSize
)
104 // Fill empty space in (Start, End) range.
105 auto fillRange
= [&](uint64_t Start
, uint64_t End
) {
106 if ((Start
/ BytesPerLine
) == (End
/ BytesPerLine
)) {
107 for (uint64_t Fill
= Start
+ BucketSize
; Fill
< End
; Fill
+= BucketSize
) {
108 changeColor(DefaultColor
);
114 changeColor(DefaultColor
);
116 Start
= alignTo(Start
, BytesPerLine
);
118 uint64_t NumEmptyLines
= (End
- Start
) / BytesPerLine
;
120 if (NumEmptyLines
> 32) {
123 while (NumEmptyLines
--) {
124 startLine(Start
, /*Empty=*/true);
126 Start
+= BytesPerLine
;
133 static raw_ostream::Colors Colors
[] = {
134 raw_ostream::WHITE
, raw_ostream::WHITE
, raw_ostream::CYAN
,
135 raw_ostream::GREEN
, raw_ostream::YELLOW
, raw_ostream::RED
};
136 constexpr size_t NumRanges
= sizeof(Colors
) / sizeof(Colors
[0]);
138 uint64_t Range
[NumRanges
];
139 for (uint64_t I
= 0; I
< NumRanges
; ++I
)
140 Range
[I
] = std::max(I
+ 1, (uint64_t)std::pow((double)MaxValue
,
141 (double)(I
+ 1) / NumRanges
));
142 Range
[NumRanges
- 1] = std::max((uint64_t)NumRanges
, MaxValue
);
144 // Print scaled value
145 auto printValue
= [&](uint64_t Value
, char Character
, bool ResetColor
) {
146 assert(Value
&& "should only print positive values");
147 for (unsigned I
= 0; I
< sizeof(Range
) / sizeof(Range
[0]); ++I
) {
148 if (Value
<= Range
[I
]) {
149 changeColor(Colors
[I
]);
153 if (Value
<= Range
[0])
154 OS
<< static_cast<char>(std::tolower(Character
));
156 OS
<< static_cast<char>(std::toupper(Character
));
159 changeColor(DefaultColor
);
162 // Print against black background
163 OS
.changeColor(raw_ostream::BLACK
, /*Bold=*/false, /*Background=*/true);
164 changeColor(DefaultColor
);
169 uint64_t PrevValue
= 0;
170 for (unsigned I
= 0; I
< sizeof(Range
) / sizeof(Range
[0]); ++I
) {
171 const uint64_t Value
= Range
[I
];
173 printValue(Value
, 'o', /*ResetColor=*/true);
174 OS
<< " : (" << PrevValue
<< ", " << Value
<< "]\n";
177 if (opts::HeatmapPrintMappings
) {
178 OS
<< "\nSections:\n";
179 unsigned SectionIdx
= 0;
180 for (auto TxtSeg
: TextSections
) {
181 const char Upper
= static_cast<char>('A' + ((SectionIdx
++) % 26));
182 const char Lower
= static_cast<char>(std::tolower(Upper
));
183 OS
<< formatv(" {0}/{1} : {2,-10} ", Lower
, Upper
, TxtSeg
.Name
);
184 if (MaxAddress
> 0xffffffff)
185 OS
<< format("0x%016" PRIx64
, TxtSeg
.BeginAddress
) << "-"
186 << format("0x%016" PRIx64
, TxtSeg
.EndAddress
) << "\n";
188 OS
<< format("0x%08" PRIx64
, TxtSeg
.BeginAddress
) << "-"
189 << format("0x%08" PRIx64
, TxtSeg
.EndAddress
) << "\n";
194 // Pos - character position from right in hex form.
195 auto printHeader
= [&](unsigned Pos
) {
197 if (MaxAddress
> 0xffffffff)
199 unsigned PrevValue
= unsigned(-1);
200 for (unsigned I
= 0; I
< BytesPerLine
; I
+= BucketSize
) {
201 const unsigned Value
= (I
& ((1 << Pos
* 4) - 1)) >> (Pos
- 1) * 4;
202 if (Value
!= PrevValue
) {
203 OS
<< Twine::utohexstr(Value
);
211 for (unsigned I
= 5; I
> 0; --I
)
214 auto SectionStart
= TextSections
.begin();
215 uint64_t PrevAddress
= 0;
216 for (auto MI
= Map
.begin(), ME
= Map
.end(); MI
!= ME
; ++MI
) {
217 const std::pair
<const uint64_t, uint64_t> &Entry
= *MI
;
218 uint64_t Address
= Entry
.first
* BucketSize
;
219 char Character
= 'o';
221 // Check if address is in the current or any later section.
222 auto Section
= std::find_if(
223 SectionStart
, TextSections
.end(), [&](const SectionNameAndRange
&S
) {
224 return Address
>= S
.BeginAddress
&& Address
< S
.EndAddress
;
226 if (Section
!= TextSections
.end()) {
227 // Shift the section forward (if SectionStart is different from Section).
228 // This works, because TextSections is sorted by start address.
229 SectionStart
= Section
;
230 Character
= 'a' + ((Section
- TextSections
.begin()) % 26);
234 fillRange(PrevAddress
, Address
);
238 printValue(Entry
.second
, Character
, /*ResetColor=*/false);
240 PrevAddress
= Address
;
244 changeColor(DefaultColor
);
245 finishLine(PrevAddress
);
249 void Heatmap::printCDF(StringRef FileName
) const {
251 raw_fd_ostream
OS(FileName
, EC
, sys::fs::OpenFlags::OF_None
);
253 errs() << "error opening output file: " << EC
.message() << '\n';
259 void Heatmap::printCDF(raw_ostream
&OS
) const {
260 uint64_t NumTotalCounts
= 0;
261 std::vector
<uint64_t> Counts
;
263 for (const std::pair
<const uint64_t, uint64_t> &KV
: Map
) {
264 Counts
.push_back(KV
.second
);
265 NumTotalCounts
+= KV
.second
;
268 llvm::sort(Counts
, std::greater
<uint64_t>());
270 double RatioLeftInKB
= (1.0 * BucketSize
) / 1024;
271 assert(NumTotalCounts
> 0 &&
272 "total number of heatmap buckets should be greater than 0");
273 double RatioRightInPercent
= 100.0 / NumTotalCounts
;
274 uint64_t RunningCount
= 0;
276 OS
<< "Bucket counts, Size (KB), CDF (%)\n";
277 for (uint64_t I
= 0; I
< Counts
.size(); I
++) {
278 RunningCount
+= Counts
[I
];
279 OS
<< format("%llu", (I
+ 1)) << ", "
280 << format("%.4f", RatioLeftInKB
* (I
+ 1)) << ", "
281 << format("%.4f", RatioRightInPercent
* (RunningCount
)) << "\n";
287 void Heatmap::printSectionHotness(StringRef FileName
) const {
289 raw_fd_ostream
OS(FileName
, EC
, sys::fs::OpenFlags::OF_None
);
291 errs() << "error opening output file: " << EC
.message() << '\n';
294 printSectionHotness(OS
);
297 void Heatmap::printSectionHotness(raw_ostream
&OS
) const {
298 uint64_t NumTotalCounts
= 0;
299 StringMap
<uint64_t> SectionHotness
;
300 unsigned TextSectionIndex
= 0;
302 if (TextSections
.empty())
305 uint64_t UnmappedHotness
= 0;
306 auto RecordUnmappedBucket
= [&](uint64_t Address
, uint64_t Frequency
) {
307 errs() << "Couldn't map the address bucket [0x" << Twine::utohexstr(Address
)
308 << ", 0x" << Twine::utohexstr(Address
+ BucketSize
)
309 << "] containing " << Frequency
310 << " samples to a text section in the binary.";
311 UnmappedHotness
+= Frequency
;
314 for (const std::pair
<const uint64_t, uint64_t> &KV
: Map
) {
315 NumTotalCounts
+= KV
.second
;
316 // We map an address bucket to the first section (lowest address)
317 // overlapping with that bucket.
318 auto Address
= KV
.first
* BucketSize
;
319 while (TextSectionIndex
< TextSections
.size() &&
320 Address
>= TextSections
[TextSectionIndex
].EndAddress
)
322 if (TextSectionIndex
>= TextSections
.size() ||
323 Address
+ BucketSize
< TextSections
[TextSectionIndex
].BeginAddress
) {
324 RecordUnmappedBucket(Address
, KV
.second
);
327 SectionHotness
[TextSections
[TextSectionIndex
].Name
] += KV
.second
;
330 assert(NumTotalCounts
> 0 &&
331 "total number of heatmap buckets should be greater than 0");
333 OS
<< "Section Name, Begin Address, End Address, Percentage Hotness\n";
334 for (auto &TextSection
: TextSections
) {
335 OS
<< TextSection
.Name
<< ", 0x"
336 << Twine::utohexstr(TextSection
.BeginAddress
) << ", 0x"
337 << Twine::utohexstr(TextSection
.EndAddress
) << ", "
339 100.0 * SectionHotness
[TextSection
.Name
] / NumTotalCounts
)
342 if (UnmappedHotness
> 0)
343 OS
<< "[unmapped], 0x0, 0x0, "
344 << format("%.4f", 100.0 * UnmappedHotness
/ NumTotalCounts
) << "\n";