xref: /freebsd/contrib/llvm-project/llvm/include/llvm/DebugInfo/CodeView/CVRecord.h (revision 349cc55c9796c4596a5b9904cd3281af295f878f)
1  //===- CVRecord.h -----------------------------------------------*- C++ -*-===//
2  //
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
6  //
7  //===----------------------------------------------------------------------===//
8  
9  #ifndef LLVM_DEBUGINFO_CODEVIEW_CVRECORD_H
10  #define LLVM_DEBUGINFO_CODEVIEW_CVRECORD_H
11  
12  #include "llvm/ADT/ArrayRef.h"
13  #include "llvm/DebugInfo/CodeView/CodeView.h"
14  #include "llvm/DebugInfo/CodeView/CodeViewError.h"
15  #include "llvm/DebugInfo/CodeView/RecordSerialization.h"
16  #include "llvm/Support/BinaryStreamReader.h"
17  #include "llvm/Support/BinaryStreamRef.h"
18  #include "llvm/Support/Endian.h"
19  #include "llvm/Support/Error.h"
20  #include <cstdint>
21  
22  namespace llvm {
23  
24  namespace codeview {
25  
26  /// CVRecord is a fat pointer (base + size pair) to a symbol or type record.
27  /// Carrying the size separately instead of trusting the size stored in the
28  /// record prefix provides some extra safety and flexibility.
29  template <typename Kind> class CVRecord {
30  public:
31    CVRecord() = default;
32  
CVRecord(ArrayRef<uint8_t> Data)33    CVRecord(ArrayRef<uint8_t> Data) : RecordData(Data) {}
34  
CVRecord(const RecordPrefix * P,size_t Size)35    CVRecord(const RecordPrefix *P, size_t Size)
36        : RecordData(reinterpret_cast<const uint8_t *>(P), Size) {}
37  
valid()38    bool valid() const { return kind() != Kind(0); }
39  
length()40    uint32_t length() const { return RecordData.size(); }
41  
kind()42    Kind kind() const {
43      if (RecordData.size() < sizeof(RecordPrefix))
44        return Kind(0);
45      return static_cast<Kind>(static_cast<uint16_t>(
46          reinterpret_cast<const RecordPrefix *>(RecordData.data())->RecordKind));
47    }
48  
data()49    ArrayRef<uint8_t> data() const { return RecordData; }
50  
str_data()51    StringRef str_data() const {
52      return StringRef(reinterpret_cast<const char *>(RecordData.data()),
53                       RecordData.size());
54    }
55  
content()56    ArrayRef<uint8_t> content() const {
57      return RecordData.drop_front(sizeof(RecordPrefix));
58    }
59  
60    ArrayRef<uint8_t> RecordData;
61  };
62  
63  // There are two kinds of codeview records: type and symbol records.
64  using CVType = CVRecord<TypeLeafKind>;
65  using CVSymbol = CVRecord<SymbolKind>;
66  
67  template <typename Record, typename Func>
forEachCodeViewRecord(ArrayRef<uint8_t> StreamBuffer,Func F)68  Error forEachCodeViewRecord(ArrayRef<uint8_t> StreamBuffer, Func F) {
69    while (!StreamBuffer.empty()) {
70      if (StreamBuffer.size() < sizeof(RecordPrefix))
71        return make_error<CodeViewError>(cv_error_code::corrupt_record);
72  
73      const RecordPrefix *Prefix =
74          reinterpret_cast<const RecordPrefix *>(StreamBuffer.data());
75  
76      size_t RealLen = Prefix->RecordLen + 2;
77      if (StreamBuffer.size() < RealLen)
78        return make_error<CodeViewError>(cv_error_code::corrupt_record);
79  
80      ArrayRef<uint8_t> Data = StreamBuffer.take_front(RealLen);
81      StreamBuffer = StreamBuffer.drop_front(RealLen);
82  
83      Record R(Data);
84      if (auto EC = F(R))
85        return EC;
86    }
87    return Error::success();
88  }
89  
90  /// Read a complete record from a stream at a random offset.
91  template <typename Kind>
readCVRecordFromStream(BinaryStreamRef Stream,uint32_t Offset)92  inline Expected<CVRecord<Kind>> readCVRecordFromStream(BinaryStreamRef Stream,
93                                                         uint32_t Offset) {
94    const RecordPrefix *Prefix = nullptr;
95    BinaryStreamReader Reader(Stream);
96    Reader.setOffset(Offset);
97  
98    if (auto EC = Reader.readObject(Prefix))
99      return std::move(EC);
100    if (Prefix->RecordLen < 2)
101      return make_error<CodeViewError>(cv_error_code::corrupt_record);
102  
103    Reader.setOffset(Offset);
104    ArrayRef<uint8_t> RawData;
105    if (auto EC = Reader.readBytes(RawData, Prefix->RecordLen + sizeof(uint16_t)))
106      return std::move(EC);
107    return codeview::CVRecord<Kind>(RawData);
108  }
109  
110  } // end namespace codeview
111  
112  template <typename Kind>
113  struct VarStreamArrayExtractor<codeview::CVRecord<Kind>> {
114    Error operator()(BinaryStreamRef Stream, uint32_t &Len,
115                     codeview::CVRecord<Kind> &Item) {
116      auto ExpectedRec = codeview::readCVRecordFromStream<Kind>(Stream, 0);
117      if (!ExpectedRec)
118        return ExpectedRec.takeError();
119      Item = *ExpectedRec;
120      Len = ExpectedRec->length();
121      return Error::success();
122    }
123  };
124  
125  namespace codeview {
126  using CVSymbolArray = VarStreamArray<CVSymbol>;
127  using CVTypeArray = VarStreamArray<CVType>;
128  using CVTypeRange = iterator_range<CVTypeArray::Iterator>;
129  } // namespace codeview
130  
131  } // end namespace llvm
132  
133  #endif // LLVM_DEBUGINFO_CODEVIEW_CVRECORD_H
134