1 //===- StackMapParser.h - StackMap Parsing Support --------------*- C++ -*-===//
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 #ifndef LLVM_CODEGEN_STACKMAPPARSER_H
10 #define LLVM_CODEGEN_STACKMAPPARSER_H
12 #include "llvm/ADT/ArrayRef.h"
13 #include "llvm/ADT/iterator_range.h"
14 #include "llvm/Support/Endian.h"
22 /// A parser for the latest stackmap format. At the moment, latest=V3.
23 template <support::endianness Endianness
>
24 class StackMapParser
{
26 template <typename AccessorT
>
27 class AccessorIterator
{
29 AccessorIterator(AccessorT A
) : A(A
) {}
31 AccessorIterator
& operator++() { A
= A
.next(); return *this; }
32 AccessorIterator
operator++(int) {
38 bool operator==(const AccessorIterator
&Other
) {
39 return A
.P
== Other
.A
.P
;
42 bool operator!=(const AccessorIterator
&Other
) { return !(*this == Other
); }
44 AccessorT
& operator*() { return A
; }
45 AccessorT
* operator->() { return &A
; }
51 /// Accessor for function records.
52 class FunctionAccessor
{
53 friend class StackMapParser
;
56 /// Get the function address.
57 uint64_t getFunctionAddress() const {
58 return read
<uint64_t>(P
);
61 /// Get the function's stack size.
62 uint64_t getStackSize() const {
63 return read
<uint64_t>(P
+ sizeof(uint64_t));
66 /// Get the number of callsite records.
67 uint64_t getRecordCount() const {
68 return read
<uint64_t>(P
+ (2 * sizeof(uint64_t)));
72 FunctionAccessor(const uint8_t *P
) : P(P
) {}
74 const static int FunctionAccessorSize
= 3 * sizeof(uint64_t);
76 FunctionAccessor
next() const {
77 return FunctionAccessor(P
+ FunctionAccessorSize
);
83 /// Accessor for constants.
84 class ConstantAccessor
{
85 friend class StackMapParser
;
88 /// Return the value of this constant.
89 uint64_t getValue() const { return read
<uint64_t>(P
); }
92 ConstantAccessor(const uint8_t *P
) : P(P
) {}
94 const static int ConstantAccessorSize
= sizeof(uint64_t);
96 ConstantAccessor
next() const {
97 return ConstantAccessor(P
+ ConstantAccessorSize
);
103 enum class LocationKind
: uint8_t {
104 Register
= 1, Direct
= 2, Indirect
= 3, Constant
= 4, ConstantIndex
= 5
107 /// Accessor for location records.
108 class LocationAccessor
{
109 friend class StackMapParser
;
110 friend class RecordAccessor
;
113 /// Get the Kind for this location.
114 LocationKind
getKind() const {
115 return LocationKind(P
[KindOffset
]);
118 /// Get the Size for this location.
119 unsigned getSizeInBytes() const {
120 return read
<uint16_t>(P
+ SizeOffset
);
124 /// Get the Dwarf register number for this location.
125 uint16_t getDwarfRegNum() const {
126 return read
<uint16_t>(P
+ DwarfRegNumOffset
);
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
);
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
);
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
);
151 LocationAccessor(const uint8_t *P
) : P(P
) {}
153 LocationAccessor
next() const {
154 return LocationAccessor(P
+ LocationAccessorSize
);
157 static const int KindOffset
= 0;
158 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);
166 /// Accessor for stackmap live-out fields.
167 class LiveOutAccessor
{
168 friend class StackMapParser
;
169 friend class RecordAccessor
;
172 /// Get the Dwarf register number for this live-out.
173 uint16_t getDwarfRegNum() const {
174 return read
<uint16_t>(P
+ DwarfRegNumOffset
);
177 /// Get the size in bytes of live [sub]register.
178 unsigned getSizeInBytes() const {
179 return read
<uint8_t>(P
+ SizeOffset
);
183 LiveOutAccessor(const uint8_t *P
) : P(P
) {}
185 LiveOutAccessor
next() const {
186 return LiveOutAccessor(P
+ LiveOutAccessorSize
);
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);
197 /// Accessor for stackmap records.
198 class RecordAccessor
{
199 friend class StackMapParser
;
202 using location_iterator
= AccessorIterator
<LocationAccessor
>;
203 using liveout_iterator
= AccessorIterator
<LiveOutAccessor
>;
205 /// Get the patchpoint/stackmap ID for this record.
206 uint64_t getID() const {
207 return read
<uint64_t>(P
+ PatchpointIDOffset
);
210 /// Get the instruction offset (from the start of the containing function)
212 uint32_t getInstructionOffset() const {
213 return read
<uint32_t>(P
+ InstructionOffsetOffset
);
216 /// Get the number of locations contained in this record.
217 uint16_t getNumLocations() const {
218 return read
<uint16_t>(P
+ NumLocationsOffset
);
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
);
228 /// Begin iterator for locations.
229 location_iterator
location_begin() const {
230 return location_iterator(getLocation(0));
233 /// End iterator for locations.
234 location_iterator
location_end() const {
235 return location_iterator(getLocation(getNumLocations()));
238 /// Iterator range for locations.
239 iterator_range
<location_iterator
> locations() const {
240 return make_range(location_begin(), location_end());
243 /// Get the number of liveouts contained in this record.
244 uint16_t getNumLiveOuts() const {
245 return read
<uint16_t>(P
+ getNumLiveOutsOffset());
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
);
255 /// Begin iterator for live-outs.
256 liveout_iterator
liveouts_begin() const {
257 return liveout_iterator(getLiveOut(0));
260 /// End iterator for live-outs.
261 liveout_iterator
liveouts_end() const {
262 return liveout_iterator(getLiveOut(getNumLiveOuts()));
265 /// Iterator range for live-outs.
266 iterator_range
<liveout_iterator
> liveouts() const {
267 return make_range(liveouts_begin(), liveouts_end());
271 RecordAccessor(const uint8_t *P
) : P(P
) {}
273 unsigned getNumLiveOutsOffset() const {
275 ((LocationListOffset
+ LocationSize
* getNumLocations()) + 7) & ~0x7;
276 return LocOffset
+ sizeof(uint16_t);
279 unsigned getSizeInBytes() const {
280 unsigned RecordSize
=
281 getNumLiveOutsOffset() + sizeof(uint16_t) + getNumLiveOuts() * LiveOutSize
;
282 return (RecordSize
+ 7) & ~0x7;
285 RecordAccessor
next() const {
286 return RecordAccessor(P
+ getSizeInBytes());
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);
296 static const unsigned LocationSize
= sizeof(uint64_t) + sizeof(uint32_t);
297 static const unsigned LiveOutSize
= sizeof(uint32_t);
302 /// Construct a parser for a version-3 stackmap. StackMap data will be read
303 /// from the given array.
304 StackMapParser(ArrayRef
<uint8_t> StackMapSection
)
305 : StackMapSection(StackMapSection
) {
306 ConstantsListOffset
= FunctionListOffset
+ getNumFunctions() * FunctionSize
;
308 assert(StackMapSection
[0] == 3 &&
309 "StackMapParser can only parse version 3 stackmaps");
311 unsigned CurrentRecordOffset
=
312 ConstantsListOffset
+ getNumConstants() * ConstantSize
;
314 for (unsigned I
= 0, E
= getNumRecords(); I
!= E
; ++I
) {
315 StackMapRecordOffsets
.push_back(CurrentRecordOffset
);
316 CurrentRecordOffset
+=
317 RecordAccessor(&StackMapSection
[CurrentRecordOffset
]).getSizeInBytes();
321 using function_iterator
= AccessorIterator
<FunctionAccessor
>;
322 using constant_iterator
= AccessorIterator
<ConstantAccessor
>;
323 using record_iterator
= AccessorIterator
<RecordAccessor
>;
325 /// Get the version number of this stackmap. (Always returns 3).
326 unsigned getVersion() const { return 3; }
328 /// Get the number of functions in the stack map.
329 uint32_t getNumFunctions() const {
330 return read
<uint32_t>(&StackMapSection
[NumFunctionsOffset
]);
333 /// Get the number of large constants in the stack map.
334 uint32_t getNumConstants() const {
335 return read
<uint32_t>(&StackMapSection
[NumConstantsOffset
]);
338 /// Get the number of stackmap records in the stackmap.
339 uint32_t getNumRecords() const {
340 return read
<uint32_t>(&StackMapSection
[NumRecordsOffset
]);
343 /// Return an FunctionAccessor for the given function index.
344 FunctionAccessor
getFunction(unsigned FunctionIndex
) const {
345 return FunctionAccessor(StackMapSection
.data() +
346 getFunctionOffset(FunctionIndex
));
349 /// Begin iterator for functions.
350 function_iterator
functions_begin() const {
351 return function_iterator(getFunction(0));
354 /// End iterator for functions.
355 function_iterator
functions_end() const {
356 return function_iterator(
357 FunctionAccessor(StackMapSection
.data() +
358 getFunctionOffset(getNumFunctions())));
361 /// Iterator range for functions.
362 iterator_range
<function_iterator
> functions() const {
363 return make_range(functions_begin(), functions_end());
366 /// Return the large constant at the given index.
367 ConstantAccessor
getConstant(unsigned ConstantIndex
) const {
368 return ConstantAccessor(StackMapSection
.data() +
369 getConstantOffset(ConstantIndex
));
372 /// Begin iterator for constants.
373 constant_iterator
constants_begin() const {
374 return constant_iterator(getConstant(0));
377 /// End iterator for constants.
378 constant_iterator
constants_end() const {
379 return constant_iterator(
380 ConstantAccessor(StackMapSection
.data() +
381 getConstantOffset(getNumConstants())));
384 /// Iterator range for constants.
385 iterator_range
<constant_iterator
> constants() const {
386 return make_range(constants_begin(), constants_end());
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
);
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));
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());
412 /// Iterator range for records.
413 iterator_range
<record_iterator
> records() const {
414 return make_range(records_begin(), records_end());
418 template <typename T
>
419 static T
read(const uint8_t *P
) {
420 return support::endian::read
<T
, Endianness
, 1>(P
);
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);
429 static const unsigned FunctionSize
= 3 * sizeof(uint64_t);
430 static const unsigned ConstantSize
= sizeof(uint64_t);
432 std::size_t getFunctionOffset(unsigned FunctionIndex
) const {
433 return FunctionListOffset
+ FunctionIndex
* FunctionSize
;
436 std::size_t getConstantOffset(unsigned ConstantIndex
) const {
437 return ConstantsListOffset
+ ConstantIndex
* ConstantSize
;
440 ArrayRef
<uint8_t> StackMapSection
;
441 unsigned ConstantsListOffset
;
442 std::vector
<unsigned> StackMapRecordOffsets
;
445 } // end namespace llvm
447 #endif // LLVM_CODEGEN_STACKMAPPARSER_H