123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===- 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/Object/ELF.h"
- #include "llvm/Support/Endian.h"
- #include <cassert>
- #include <cstddef>
- #include <cstdint>
- #include <vector>
- namespace llvm {
- /// A parser for the latest stackmap format. At the moment, latest=V3.
- template <support::endianness Endianness>
- class StackMapParser {
- 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) const {
- return A.P == Other.A.P;
- }
- bool operator!=(const AccessorIterator &Other) const {
- return !(*this == Other);
- }
- AccessorT& operator*() { return A; }
- AccessorT* operator->() { return &A; }
- private:
- AccessorT A;
- };
- /// Accessor for function records.
- class FunctionAccessor {
- friend class StackMapParser;
- 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 StackMapParser;
- 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 StackMapParser;
- friend class RecordAccessor;
- public:
- /// Get the Kind for this location.
- LocationKind getKind() const {
- return LocationKind(P[KindOffset]);
- }
- /// Get the Size for this location.
- unsigned getSizeInBytes() const {
- return read<uint16_t>(P + SizeOffset);
- }
- /// 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 SizeOffset = KindOffset + sizeof(uint16_t);
- static const int DwarfRegNumOffset = SizeOffset + sizeof(uint16_t);
- static const int SmallConstantOffset = DwarfRegNumOffset + sizeof(uint32_t);
- static const int LocationAccessorSize = sizeof(uint64_t) + sizeof(uint32_t);
- const uint8_t *P;
- };
- /// Accessor for stackmap live-out fields.
- class LiveOutAccessor {
- friend class StackMapParser;
- 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 StackMapParser;
- 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 {
- unsigned LocOffset =
- ((LocationListOffset + LocationSize * getNumLocations()) + 7) & ~0x7;
- return LocOffset + 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) + sizeof(uint32_t);
- static const unsigned LiveOutSize = sizeof(uint32_t);
- const uint8_t *P;
- };
- /// Construct a parser for a version-3 stackmap. StackMap data will be read
- /// from the given array.
- StackMapParser(ArrayRef<uint8_t> StackMapSection)
- : StackMapSection(StackMapSection) {
- ConstantsListOffset = FunctionListOffset + getNumFunctions() * FunctionSize;
- assert(StackMapSection[0] == 3 &&
- "StackMapParser can only parse version 3 stackmaps");
- unsigned CurrentRecordOffset =
- ConstantsListOffset + getNumConstants() * ConstantSize;
- for (unsigned I = 0, E = getNumRecords(); I != E; ++I) {
- StackMapRecordOffsets.push_back(CurrentRecordOffset);
- CurrentRecordOffset +=
- RecordAccessor(&StackMapSection[CurrentRecordOffset]).getSizeInBytes();
- }
- }
- /// Validates the header of the specified stack map section.
- static Error validateHeader(ArrayRef<uint8_t> StackMapSection) {
- // See the comment for StackMaps::emitStackmapHeader().
- if (StackMapSection.size() < 16)
- return object::createError(
- "the stack map section size (" + Twine(StackMapSection.size()) +
- ") is less than the minimum possible size of its header (16)");
- unsigned Version = StackMapSection[0];
- if (Version != 3)
- return object::createError(
- "the version (" + Twine(Version) +
- ") of the stack map section is unsupported, the "
- "supported version is 3");
- return Error::success();
- }
- using function_iterator = AccessorIterator<FunctionAccessor>;
- using constant_iterator = AccessorIterator<ConstantAccessor>;
- using record_iterator = AccessorIterator<RecordAccessor>;
- /// Get the version number of this stackmap. (Always returns 3).
- unsigned getVersion() const { return 3; }
- /// 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
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|