//===- StackMapParser.h - StackMap Parsing Support --------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLVM_CODEGEN_STACKMAPPARSER_H #define LLVM_CODEGEN_STACKMAPPARSER_H #include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/iterator_range.h" #include "llvm/Support/Endian.h" #include <cassert> #include <cstddef> #include <cstdint> #include <vector> namespace llvm { template <support::endianness Endianness> class StackMapV2Parser { public: template <typename AccessorT> class AccessorIterator { public: AccessorIterator(AccessorT A) : A(A) {} AccessorIterator& operator++() { A = A.next(); return *this; } AccessorIterator operator++(int) { auto tmp = *this; ++*this; return tmp; } bool operator==(const AccessorIterator &Other) { return A.P == Other.A.P; } bool operator!=(const AccessorIterator &Other) { return !(*this == Other); } AccessorT& operator*() { return A; } AccessorT* operator->() { return &A; } private: AccessorT A; }; /// Accessor for function records. class FunctionAccessor { friend class StackMapV2Parser; public: /// Get the function address. uint64_t getFunctionAddress() const { return read<uint64_t>(P); } /// Get the function's stack size. uint64_t getStackSize() const { return read<uint64_t>(P + sizeof(uint64_t)); } /// Get the number of callsite records. uint64_t getRecordCount() const { return read<uint64_t>(P + (2 * sizeof(uint64_t))); } private: FunctionAccessor(const uint8_t *P) : P(P) {} const static int FunctionAccessorSize = 3 * sizeof(uint64_t); FunctionAccessor next() const { return FunctionAccessor(P + FunctionAccessorSize); } const uint8_t *P; }; /// Accessor for constants. class ConstantAccessor { friend class StackMapV2Parser; public: /// Return the value of this constant. uint64_t getValue() const { return read<uint64_t>(P); } private: ConstantAccessor(const uint8_t *P) : P(P) {} const static int ConstantAccessorSize = sizeof(uint64_t); ConstantAccessor next() const { return ConstantAccessor(P + ConstantAccessorSize); } const uint8_t *P; }; enum class LocationKind : uint8_t { Register = 1, Direct = 2, Indirect = 3, Constant = 4, ConstantIndex = 5 }; /// Accessor for location records. class LocationAccessor { friend class StackMapV2Parser; friend class RecordAccessor; public: /// Get the Kind for this location. LocationKind getKind() const { return LocationKind(P[KindOffset]); } /// Get the Dwarf register number for this location. uint16_t getDwarfRegNum() const { return read<uint16_t>(P + DwarfRegNumOffset); } /// Get the small-constant for this location. (Kind must be Constant). uint32_t getSmallConstant() const { assert(getKind() == LocationKind::Constant && "Not a small constant."); return read<uint32_t>(P + SmallConstantOffset); } /// Get the constant-index for this location. (Kind must be ConstantIndex). uint32_t getConstantIndex() const { assert(getKind() == LocationKind::ConstantIndex && "Not a constant-index."); return read<uint32_t>(P + SmallConstantOffset); } /// Get the offset for this location. (Kind must be Direct or Indirect). int32_t getOffset() const { assert((getKind() == LocationKind::Direct || getKind() == LocationKind::Indirect) && "Not direct or indirect."); return read<int32_t>(P + SmallConstantOffset); } private: LocationAccessor(const uint8_t *P) : P(P) {} LocationAccessor next() const { return LocationAccessor(P + LocationAccessorSize); } static const int KindOffset = 0; static const int DwarfRegNumOffset = KindOffset + sizeof(uint16_t); static const int SmallConstantOffset = DwarfRegNumOffset + sizeof(uint16_t); static const int LocationAccessorSize = sizeof(uint64_t); const uint8_t *P; }; /// Accessor for stackmap live-out fields. class LiveOutAccessor { friend class StackMapV2Parser; friend class RecordAccessor; public: /// Get the Dwarf register number for this live-out. uint16_t getDwarfRegNum() const { return read<uint16_t>(P + DwarfRegNumOffset); } /// Get the size in bytes of live [sub]register. unsigned getSizeInBytes() const { return read<uint8_t>(P + SizeOffset); } private: LiveOutAccessor(const uint8_t *P) : P(P) {} LiveOutAccessor next() const { return LiveOutAccessor(P + LiveOutAccessorSize); } static const int DwarfRegNumOffset = 0; static const int SizeOffset = DwarfRegNumOffset + sizeof(uint16_t) + sizeof(uint8_t); static const int LiveOutAccessorSize = sizeof(uint32_t); const uint8_t *P; }; /// Accessor for stackmap records. class RecordAccessor { friend class StackMapV2Parser; public: using location_iterator = AccessorIterator<LocationAccessor>; using liveout_iterator = AccessorIterator<LiveOutAccessor>; /// Get the patchpoint/stackmap ID for this record. uint64_t getID() const { return read<uint64_t>(P + PatchpointIDOffset); } /// Get the instruction offset (from the start of the containing function) /// for this record. uint32_t getInstructionOffset() const { return read<uint32_t>(P + InstructionOffsetOffset); } /// Get the number of locations contained in this record. uint16_t getNumLocations() const { return read<uint16_t>(P + NumLocationsOffset); } /// Get the location with the given index. LocationAccessor getLocation(unsigned LocationIndex) const { unsigned LocationOffset = LocationListOffset + LocationIndex * LocationSize; return LocationAccessor(P + LocationOffset); } /// Begin iterator for locations. location_iterator location_begin() const { return location_iterator(getLocation(0)); } /// End iterator for locations. location_iterator location_end() const { return location_iterator(getLocation(getNumLocations())); } /// Iterator range for locations. iterator_range<location_iterator> locations() const { return make_range(location_begin(), location_end()); } /// Get the number of liveouts contained in this record. uint16_t getNumLiveOuts() const { return read<uint16_t>(P + getNumLiveOutsOffset()); } /// Get the live-out with the given index. LiveOutAccessor getLiveOut(unsigned LiveOutIndex) const { unsigned LiveOutOffset = getNumLiveOutsOffset() + sizeof(uint16_t) + LiveOutIndex * LiveOutSize; return LiveOutAccessor(P + LiveOutOffset); } /// Begin iterator for live-outs. liveout_iterator liveouts_begin() const { return liveout_iterator(getLiveOut(0)); } /// End iterator for live-outs. liveout_iterator liveouts_end() const { return liveout_iterator(getLiveOut(getNumLiveOuts())); } /// Iterator range for live-outs. iterator_range<liveout_iterator> liveouts() const { return make_range(liveouts_begin(), liveouts_end()); } private: RecordAccessor(const uint8_t *P) : P(P) {} unsigned getNumLiveOutsOffset() const { return LocationListOffset + LocationSize * getNumLocations() + sizeof(uint16_t); } unsigned getSizeInBytes() const { unsigned RecordSize = getNumLiveOutsOffset() + sizeof(uint16_t) + getNumLiveOuts() * LiveOutSize; return (RecordSize + 7) & ~0x7; } RecordAccessor next() const { return RecordAccessor(P + getSizeInBytes()); } static const unsigned PatchpointIDOffset = 0; static const unsigned InstructionOffsetOffset = PatchpointIDOffset + sizeof(uint64_t); static const unsigned NumLocationsOffset = InstructionOffsetOffset + sizeof(uint32_t) + sizeof(uint16_t); static const unsigned LocationListOffset = NumLocationsOffset + sizeof(uint16_t); static const unsigned LocationSize = sizeof(uint64_t); static const unsigned LiveOutSize = sizeof(uint32_t); const uint8_t *P; }; /// Construct a parser for a version-2 stackmap. StackMap data will be read /// from the given array. StackMapV2Parser(ArrayRef<uint8_t> StackMapSection) : StackMapSection(StackMapSection) { ConstantsListOffset = FunctionListOffset + getNumFunctions() * FunctionSize; assert(StackMapSection[0] == 2 && "StackMapV2Parser can only parse version 2 stackmaps"); unsigned CurrentRecordOffset = ConstantsListOffset + getNumConstants() * ConstantSize; for (unsigned I = 0, E = getNumRecords(); I != E; ++I) { StackMapRecordOffsets.push_back(CurrentRecordOffset); CurrentRecordOffset += RecordAccessor(&StackMapSection[CurrentRecordOffset]).getSizeInBytes(); } } using function_iterator = AccessorIterator<FunctionAccessor>; using constant_iterator = AccessorIterator<ConstantAccessor>; using record_iterator = AccessorIterator<RecordAccessor>; /// Get the version number of this stackmap. (Always returns 2). unsigned getVersion() const { return 2; } /// Get the number of functions in the stack map. uint32_t getNumFunctions() const { return read<uint32_t>(&StackMapSection[NumFunctionsOffset]); } /// Get the number of large constants in the stack map. uint32_t getNumConstants() const { return read<uint32_t>(&StackMapSection[NumConstantsOffset]); } /// Get the number of stackmap records in the stackmap. uint32_t getNumRecords() const { return read<uint32_t>(&StackMapSection[NumRecordsOffset]); } /// Return an FunctionAccessor for the given function index. FunctionAccessor getFunction(unsigned FunctionIndex) const { return FunctionAccessor(StackMapSection.data() + getFunctionOffset(FunctionIndex)); } /// Begin iterator for functions. function_iterator functions_begin() const { return function_iterator(getFunction(0)); } /// End iterator for functions. function_iterator functions_end() const { return function_iterator( FunctionAccessor(StackMapSection.data() + getFunctionOffset(getNumFunctions()))); } /// Iterator range for functions. iterator_range<function_iterator> functions() const { return make_range(functions_begin(), functions_end()); } /// Return the large constant at the given index. ConstantAccessor getConstant(unsigned ConstantIndex) const { return ConstantAccessor(StackMapSection.data() + getConstantOffset(ConstantIndex)); } /// Begin iterator for constants. constant_iterator constants_begin() const { return constant_iterator(getConstant(0)); } /// End iterator for constants. constant_iterator constants_end() const { return constant_iterator( ConstantAccessor(StackMapSection.data() + getConstantOffset(getNumConstants()))); } /// Iterator range for constants. iterator_range<constant_iterator> constants() const { return make_range(constants_begin(), constants_end()); } /// Return a RecordAccessor for the given record index. RecordAccessor getRecord(unsigned RecordIndex) const { std::size_t RecordOffset = StackMapRecordOffsets[RecordIndex]; return RecordAccessor(StackMapSection.data() + RecordOffset); } /// Begin iterator for records. record_iterator records_begin() const { if (getNumRecords() == 0) return record_iterator(RecordAccessor(nullptr)); return record_iterator(getRecord(0)); } /// End iterator for records. record_iterator records_end() const { // Records need to be handled specially, since we cache the start addresses // for them: We can't just compute the 1-past-the-end address, we have to // look at the last record and use the 'next' method. if (getNumRecords() == 0) return record_iterator(RecordAccessor(nullptr)); return record_iterator(getRecord(getNumRecords() - 1).next()); } /// Iterator range for records. iterator_range<record_iterator> records() const { return make_range(records_begin(), records_end()); } private: template <typename T> static T read(const uint8_t *P) { return support::endian::read<T, Endianness, 1>(P); } static const unsigned HeaderOffset = 0; static const unsigned NumFunctionsOffset = HeaderOffset + sizeof(uint32_t); static const unsigned NumConstantsOffset = NumFunctionsOffset + sizeof(uint32_t); static const unsigned NumRecordsOffset = NumConstantsOffset + sizeof(uint32_t); static const unsigned FunctionListOffset = NumRecordsOffset + sizeof(uint32_t); static const unsigned FunctionSize = 3 * sizeof(uint64_t); static const unsigned ConstantSize = sizeof(uint64_t); std::size_t getFunctionOffset(unsigned FunctionIndex) const { return FunctionListOffset + FunctionIndex * FunctionSize; } std::size_t getConstantOffset(unsigned ConstantIndex) const { return ConstantsListOffset + ConstantIndex * ConstantSize; } ArrayRef<uint8_t> StackMapSection; unsigned ConstantsListOffset; std::vector<unsigned> StackMapRecordOffsets; }; } // end namespace llvm #endif // LLVM_CODEGEN_STACKMAPPARSER_H