blob: ed44efbf80b9690026b6c29f5c7f58a494c07c46 [file] [log] [blame]
Andrew Scull5e1ddfa2018-08-14 10:06:54 +01001//===- StackMapParser.h - StackMap Parsing Support --------------*- C++ -*-===//
2//
Andrew Walbran16937d02019-10-22 13:54:20 +01003// 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
Andrew Scull5e1ddfa2018-08-14 10:06:54 +01006//
7//===----------------------------------------------------------------------===//
8
9#ifndef LLVM_CODEGEN_STACKMAPPARSER_H
10#define LLVM_CODEGEN_STACKMAPPARSER_H
11
12#include "llvm/ADT/ArrayRef.h"
13#include "llvm/ADT/iterator_range.h"
14#include "llvm/Support/Endian.h"
15#include <cassert>
16#include <cstddef>
17#include <cstdint>
18#include <vector>
19
20namespace llvm {
21
Andrew Walbran3d2c1972020-04-07 12:24:26 +010022/// A parser for the latest stackmap format. At the moment, latest=V2.
Andrew Scull5e1ddfa2018-08-14 10:06:54 +010023template <support::endianness Endianness>
Andrew Walbran3d2c1972020-04-07 12:24:26 +010024class StackMapParser {
Andrew Scull5e1ddfa2018-08-14 10:06:54 +010025public:
26 template <typename AccessorT>
27 class AccessorIterator {
28 public:
29 AccessorIterator(AccessorT A) : A(A) {}
30
31 AccessorIterator& operator++() { A = A.next(); return *this; }
32 AccessorIterator operator++(int) {
33 auto tmp = *this;
34 ++*this;
35 return tmp;
36 }
37
38 bool operator==(const AccessorIterator &Other) {
39 return A.P == Other.A.P;
40 }
41
42 bool operator!=(const AccessorIterator &Other) { return !(*this == Other); }
43
44 AccessorT& operator*() { return A; }
45 AccessorT* operator->() { return &A; }
46
47 private:
48 AccessorT A;
49 };
50
51 /// Accessor for function records.
52 class FunctionAccessor {
Andrew Walbran3d2c1972020-04-07 12:24:26 +010053 friend class StackMapParser;
Andrew Scull5e1ddfa2018-08-14 10:06:54 +010054
55 public:
56 /// Get the function address.
57 uint64_t getFunctionAddress() const {
58 return read<uint64_t>(P);
59 }
60
61 /// Get the function's stack size.
62 uint64_t getStackSize() const {
63 return read<uint64_t>(P + sizeof(uint64_t));
64 }
65
66 /// Get the number of callsite records.
67 uint64_t getRecordCount() const {
68 return read<uint64_t>(P + (2 * sizeof(uint64_t)));
69 }
70
71 private:
72 FunctionAccessor(const uint8_t *P) : P(P) {}
73
74 const static int FunctionAccessorSize = 3 * sizeof(uint64_t);
75
76 FunctionAccessor next() const {
77 return FunctionAccessor(P + FunctionAccessorSize);
78 }
79
80 const uint8_t *P;
81 };
82
83 /// Accessor for constants.
84 class ConstantAccessor {
Andrew Walbran3d2c1972020-04-07 12:24:26 +010085 friend class StackMapParser;
Andrew Scull5e1ddfa2018-08-14 10:06:54 +010086
87 public:
88 /// Return the value of this constant.
89 uint64_t getValue() const { return read<uint64_t>(P); }
90
91 private:
92 ConstantAccessor(const uint8_t *P) : P(P) {}
93
94 const static int ConstantAccessorSize = sizeof(uint64_t);
95
96 ConstantAccessor next() const {
97 return ConstantAccessor(P + ConstantAccessorSize);
98 }
99
100 const uint8_t *P;
101 };
102
103 enum class LocationKind : uint8_t {
104 Register = 1, Direct = 2, Indirect = 3, Constant = 4, ConstantIndex = 5
105 };
106
107 /// Accessor for location records.
108 class LocationAccessor {
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100109 friend class StackMapParser;
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100110 friend class RecordAccessor;
111
112 public:
113 /// Get the Kind for this location.
114 LocationKind getKind() const {
115 return LocationKind(P[KindOffset]);
116 }
117
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100118 /// Get the Size for this location.
119 unsigned getSizeInBytes() const {
120 return read<uint16_t>(P + SizeOffset);
121
122 }
123
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100124 /// Get the Dwarf register number for this location.
125 uint16_t getDwarfRegNum() const {
126 return read<uint16_t>(P + DwarfRegNumOffset);
127 }
128
129 /// Get the small-constant for this location. (Kind must be Constant).
130 uint32_t getSmallConstant() const {
131 assert(getKind() == LocationKind::Constant && "Not a small constant.");
132 return read<uint32_t>(P + SmallConstantOffset);
133 }
134
135 /// Get the constant-index for this location. (Kind must be ConstantIndex).
136 uint32_t getConstantIndex() const {
137 assert(getKind() == LocationKind::ConstantIndex &&
138 "Not a constant-index.");
139 return read<uint32_t>(P + SmallConstantOffset);
140 }
141
142 /// Get the offset for this location. (Kind must be Direct or Indirect).
143 int32_t getOffset() const {
144 assert((getKind() == LocationKind::Direct ||
145 getKind() == LocationKind::Indirect) &&
146 "Not direct or indirect.");
147 return read<int32_t>(P + SmallConstantOffset);
148 }
149
150 private:
151 LocationAccessor(const uint8_t *P) : P(P) {}
152
153 LocationAccessor next() const {
154 return LocationAccessor(P + LocationAccessorSize);
155 }
156
157 static const int KindOffset = 0;
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100158 static const int SizeOffset = KindOffset + sizeof(uint16_t);
159 static const int DwarfRegNumOffset = SizeOffset + sizeof(uint16_t);
160 static const int SmallConstantOffset = DwarfRegNumOffset + sizeof(uint32_t);
161 static const int LocationAccessorSize = sizeof(uint64_t) + sizeof(uint32_t);
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100162
163 const uint8_t *P;
164 };
165
166 /// Accessor for stackmap live-out fields.
167 class LiveOutAccessor {
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100168 friend class StackMapParser;
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100169 friend class RecordAccessor;
170
171 public:
172 /// Get the Dwarf register number for this live-out.
173 uint16_t getDwarfRegNum() const {
174 return read<uint16_t>(P + DwarfRegNumOffset);
175 }
176
177 /// Get the size in bytes of live [sub]register.
178 unsigned getSizeInBytes() const {
179 return read<uint8_t>(P + SizeOffset);
180 }
181
182 private:
183 LiveOutAccessor(const uint8_t *P) : P(P) {}
184
185 LiveOutAccessor next() const {
186 return LiveOutAccessor(P + LiveOutAccessorSize);
187 }
188
189 static const int DwarfRegNumOffset = 0;
190 static const int SizeOffset =
191 DwarfRegNumOffset + sizeof(uint16_t) + sizeof(uint8_t);
192 static const int LiveOutAccessorSize = sizeof(uint32_t);
193
194 const uint8_t *P;
195 };
196
197 /// Accessor for stackmap records.
198 class RecordAccessor {
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100199 friend class StackMapParser;
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100200
201 public:
202 using location_iterator = AccessorIterator<LocationAccessor>;
203 using liveout_iterator = AccessorIterator<LiveOutAccessor>;
204
205 /// Get the patchpoint/stackmap ID for this record.
206 uint64_t getID() const {
207 return read<uint64_t>(P + PatchpointIDOffset);
208 }
209
210 /// Get the instruction offset (from the start of the containing function)
211 /// for this record.
212 uint32_t getInstructionOffset() const {
213 return read<uint32_t>(P + InstructionOffsetOffset);
214 }
215
216 /// Get the number of locations contained in this record.
217 uint16_t getNumLocations() const {
218 return read<uint16_t>(P + NumLocationsOffset);
219 }
220
221 /// Get the location with the given index.
222 LocationAccessor getLocation(unsigned LocationIndex) const {
223 unsigned LocationOffset =
224 LocationListOffset + LocationIndex * LocationSize;
225 return LocationAccessor(P + LocationOffset);
226 }
227
228 /// Begin iterator for locations.
229 location_iterator location_begin() const {
230 return location_iterator(getLocation(0));
231 }
232
233 /// End iterator for locations.
234 location_iterator location_end() const {
235 return location_iterator(getLocation(getNumLocations()));
236 }
237
238 /// Iterator range for locations.
239 iterator_range<location_iterator> locations() const {
240 return make_range(location_begin(), location_end());
241 }
242
243 /// Get the number of liveouts contained in this record.
244 uint16_t getNumLiveOuts() const {
245 return read<uint16_t>(P + getNumLiveOutsOffset());
246 }
247
248 /// Get the live-out with the given index.
249 LiveOutAccessor getLiveOut(unsigned LiveOutIndex) const {
250 unsigned LiveOutOffset =
251 getNumLiveOutsOffset() + sizeof(uint16_t) + LiveOutIndex * LiveOutSize;
252 return LiveOutAccessor(P + LiveOutOffset);
253 }
254
255 /// Begin iterator for live-outs.
256 liveout_iterator liveouts_begin() const {
257 return liveout_iterator(getLiveOut(0));
258 }
259
260 /// End iterator for live-outs.
261 liveout_iterator liveouts_end() const {
262 return liveout_iterator(getLiveOut(getNumLiveOuts()));
263 }
264
265 /// Iterator range for live-outs.
266 iterator_range<liveout_iterator> liveouts() const {
267 return make_range(liveouts_begin(), liveouts_end());
268 }
269
270 private:
271 RecordAccessor(const uint8_t *P) : P(P) {}
272
273 unsigned getNumLiveOutsOffset() const {
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100274 unsigned LocOffset =
275 ((LocationListOffset + LocationSize * getNumLocations()) + 7) & ~0x7;
276 return LocOffset + sizeof(uint16_t);
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100277 }
278
279 unsigned getSizeInBytes() const {
280 unsigned RecordSize =
281 getNumLiveOutsOffset() + sizeof(uint16_t) + getNumLiveOuts() * LiveOutSize;
282 return (RecordSize + 7) & ~0x7;
283 }
284
285 RecordAccessor next() const {
286 return RecordAccessor(P + getSizeInBytes());
287 }
288
289 static const unsigned PatchpointIDOffset = 0;
290 static const unsigned InstructionOffsetOffset =
291 PatchpointIDOffset + sizeof(uint64_t);
292 static const unsigned NumLocationsOffset =
293 InstructionOffsetOffset + sizeof(uint32_t) + sizeof(uint16_t);
294 static const unsigned LocationListOffset =
295 NumLocationsOffset + sizeof(uint16_t);
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100296 static const unsigned LocationSize = sizeof(uint64_t) + sizeof(uint32_t);
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100297 static const unsigned LiveOutSize = sizeof(uint32_t);
298
299 const uint8_t *P;
300 };
301
302 /// Construct a parser for a version-2 stackmap. StackMap data will be read
303 /// from the given array.
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100304 StackMapParser(ArrayRef<uint8_t> StackMapSection)
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100305 : StackMapSection(StackMapSection) {
306 ConstantsListOffset = FunctionListOffset + getNumFunctions() * FunctionSize;
307
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100308 assert(StackMapSection[0] == 3 &&
309 "StackMapParser can only parse version 3 stackmaps");
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100310
311 unsigned CurrentRecordOffset =
312 ConstantsListOffset + getNumConstants() * ConstantSize;
313
314 for (unsigned I = 0, E = getNumRecords(); I != E; ++I) {
315 StackMapRecordOffsets.push_back(CurrentRecordOffset);
316 CurrentRecordOffset +=
317 RecordAccessor(&StackMapSection[CurrentRecordOffset]).getSizeInBytes();
318 }
319 }
320
321 using function_iterator = AccessorIterator<FunctionAccessor>;
322 using constant_iterator = AccessorIterator<ConstantAccessor>;
323 using record_iterator = AccessorIterator<RecordAccessor>;
324
Andrew Walbran3d2c1972020-04-07 12:24:26 +0100325 /// Get the version number of this stackmap. (Always returns 3).
326 unsigned getVersion() const { return 3; }
Andrew Scull5e1ddfa2018-08-14 10:06:54 +0100327
328 /// Get the number of functions in the stack map.
329 uint32_t getNumFunctions() const {
330 return read<uint32_t>(&StackMapSection[NumFunctionsOffset]);
331 }
332
333 /// Get the number of large constants in the stack map.
334 uint32_t getNumConstants() const {
335 return read<uint32_t>(&StackMapSection[NumConstantsOffset]);
336 }
337
338 /// Get the number of stackmap records in the stackmap.
339 uint32_t getNumRecords() const {
340 return read<uint32_t>(&StackMapSection[NumRecordsOffset]);
341 }
342
343 /// Return an FunctionAccessor for the given function index.
344 FunctionAccessor getFunction(unsigned FunctionIndex) const {
345 return FunctionAccessor(StackMapSection.data() +
346 getFunctionOffset(FunctionIndex));
347 }
348
349 /// Begin iterator for functions.
350 function_iterator functions_begin() const {
351 return function_iterator(getFunction(0));
352 }
353
354 /// End iterator for functions.
355 function_iterator functions_end() const {
356 return function_iterator(
357 FunctionAccessor(StackMapSection.data() +
358 getFunctionOffset(getNumFunctions())));
359 }
360
361 /// Iterator range for functions.
362 iterator_range<function_iterator> functions() const {
363 return make_range(functions_begin(), functions_end());
364 }
365
366 /// Return the large constant at the given index.
367 ConstantAccessor getConstant(unsigned ConstantIndex) const {
368 return ConstantAccessor(StackMapSection.data() +
369 getConstantOffset(ConstantIndex));
370 }
371
372 /// Begin iterator for constants.
373 constant_iterator constants_begin() const {
374 return constant_iterator(getConstant(0));
375 }
376
377 /// End iterator for constants.
378 constant_iterator constants_end() const {
379 return constant_iterator(
380 ConstantAccessor(StackMapSection.data() +
381 getConstantOffset(getNumConstants())));
382 }
383
384 /// Iterator range for constants.
385 iterator_range<constant_iterator> constants() const {
386 return make_range(constants_begin(), constants_end());
387 }
388
389 /// Return a RecordAccessor for the given record index.
390 RecordAccessor getRecord(unsigned RecordIndex) const {
391 std::size_t RecordOffset = StackMapRecordOffsets[RecordIndex];
392 return RecordAccessor(StackMapSection.data() + RecordOffset);
393 }
394
395 /// Begin iterator for records.
396 record_iterator records_begin() const {
397 if (getNumRecords() == 0)
398 return record_iterator(RecordAccessor(nullptr));
399 return record_iterator(getRecord(0));
400 }
401
402 /// End iterator for records.
403 record_iterator records_end() const {
404 // Records need to be handled specially, since we cache the start addresses
405 // for them: We can't just compute the 1-past-the-end address, we have to
406 // look at the last record and use the 'next' method.
407 if (getNumRecords() == 0)
408 return record_iterator(RecordAccessor(nullptr));
409 return record_iterator(getRecord(getNumRecords() - 1).next());
410 }
411
412 /// Iterator range for records.
413 iterator_range<record_iterator> records() const {
414 return make_range(records_begin(), records_end());
415 }
416
417private:
418 template <typename T>
419 static T read(const uint8_t *P) {
420 return support::endian::read<T, Endianness, 1>(P);
421 }
422
423 static const unsigned HeaderOffset = 0;
424 static const unsigned NumFunctionsOffset = HeaderOffset + sizeof(uint32_t);
425 static const unsigned NumConstantsOffset = NumFunctionsOffset + sizeof(uint32_t);
426 static const unsigned NumRecordsOffset = NumConstantsOffset + sizeof(uint32_t);
427 static const unsigned FunctionListOffset = NumRecordsOffset + sizeof(uint32_t);
428
429 static const unsigned FunctionSize = 3 * sizeof(uint64_t);
430 static const unsigned ConstantSize = sizeof(uint64_t);
431
432 std::size_t getFunctionOffset(unsigned FunctionIndex) const {
433 return FunctionListOffset + FunctionIndex * FunctionSize;
434 }
435
436 std::size_t getConstantOffset(unsigned ConstantIndex) const {
437 return ConstantsListOffset + ConstantIndex * ConstantSize;
438 }
439
440 ArrayRef<uint8_t> StackMapSection;
441 unsigned ConstantsListOffset;
442 std::vector<unsigned> StackMapRecordOffsets;
443};
444
445} // end namespace llvm
446
447#endif // LLVM_CODEGEN_STACKMAPPARSER_H