diff options
Diffstat (limited to 'include/llvm/Object/StackMapParser.h')
| -rw-r--r-- | include/llvm/Object/StackMapParser.h | 442 | 
1 files changed, 442 insertions, 0 deletions
diff --git a/include/llvm/Object/StackMapParser.h b/include/llvm/Object/StackMapParser.h new file mode 100644 index 0000000000000..276eab6c294e5 --- /dev/null +++ b/include/llvm/Object/StackMapParser.h @@ -0,0 +1,442 @@ +//===-------- StackMapParser.h - StackMap Parsing Support -------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CODEGEN_STACKMAPPARSER_H +#define LLVM_CODEGEN_STACKMAPPARSER_H + +#include "llvm/Support/Debug.h" +#include "llvm/Support/Endian.h" +#include <map> +#include <vector> + +namespace llvm { + +template <support::endianness Endianness> +class StackMapV1Parser { +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 StackMapV1Parser; +  public: + +    /// Get the function address. +    uint64_t getFunctionAddress() const { +      return read<uint64_t>(P); +    } + +    /// Get the function's stack size. +    uint32_t getStackSize() const { +      return read<uint64_t>(P + sizeof(uint64_t)); +    } + +  private: +    FunctionAccessor(const uint8_t *P) : P(P) {} + +    const static int FunctionAccessorSize = 2 * sizeof(uint64_t); + +    FunctionAccessor next() const { +      return FunctionAccessor(P + FunctionAccessorSize); +    } + +    const uint8_t *P; +  }; + +  /// Accessor for constants. +  class ConstantAccessor { +    friend class StackMapV1Parser; +  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; +  }; + +  // Forward-declare RecordAccessor so we can friend it below. +  class RecordAccessor; + +  enum class LocationKind : uint8_t { +    Register = 1, Direct = 2, Indirect = 3, Constant = 4, ConstantIndex = 5 +  }; + + +  /// Accessor for location records. +  class LocationAccessor { +    friend class StackMapV1Parser; +    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 StackMapV1Parser; +    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 StackMapV1Parser; +  public: + +    typedef AccessorIterator<LocationAccessor> location_iterator; +    typedef AccessorIterator<LiveOutAccessor> liveout_iterator; + +    /// 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-1 stackmap. StackMap data will be read +  /// from the given array. +  StackMapV1Parser(ArrayRef<uint8_t> StackMapSection) +      : StackMapSection(StackMapSection) { +    ConstantsListOffset = FunctionListOffset + getNumFunctions() * FunctionSize; + +    assert(StackMapSection[0] == 1 && +           "StackMapV1Parser can only parse version 1 stackmaps"); + +    unsigned CurrentRecordOffset = +      ConstantsListOffset + getNumConstants() * ConstantSize; + +    for (unsigned I = 0, E = getNumRecords(); I != E; ++I) { +      StackMapRecordOffsets.push_back(CurrentRecordOffset); +      CurrentRecordOffset += +        RecordAccessor(&StackMapSection[CurrentRecordOffset]).getSizeInBytes(); +    } +  } + +  typedef AccessorIterator<FunctionAccessor> function_iterator; +  typedef AccessorIterator<ConstantAccessor> constant_iterator; +  typedef AccessorIterator<RecordAccessor> record_iterator; + +  /// Get the version number of this stackmap. (Always returns 1). +  unsigned getVersion() const { return 1; } + +  /// 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 = 2 * 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; +}; + +} + +#endif  | 
