1 //===- CVTypeVisitor.cpp ----------------------------------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
11 
12 #include "llvm/DebugInfo/CodeView/CodeViewError.h"
13 #include "llvm/DebugInfo/CodeView/TypeCollection.h"
14 #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
15 #include "llvm/DebugInfo/CodeView/TypeRecordMapping.h"
16 #include "llvm/DebugInfo/CodeView/TypeVisitorCallbackPipeline.h"
17 #include "llvm/Support/BinaryByteStream.h"
18 #include "llvm/Support/BinaryStreamReader.h"
19 
20 using namespace llvm;
21 using namespace llvm::codeview;
22 
23 
24 template <typename T>
visitKnownRecord(CVType & Record,TypeVisitorCallbacks & Callbacks)25 static Error visitKnownRecord(CVType &Record, TypeVisitorCallbacks &Callbacks) {
26   TypeRecordKind RK = static_cast<TypeRecordKind>(Record.Type);
27   T KnownRecord(RK);
28   if (auto EC = Callbacks.visitKnownRecord(Record, KnownRecord))
29     return EC;
30   return Error::success();
31 }
32 
33 template <typename T>
visitKnownMember(CVMemberRecord & Record,TypeVisitorCallbacks & Callbacks)34 static Error visitKnownMember(CVMemberRecord &Record,
35                               TypeVisitorCallbacks &Callbacks) {
36   TypeRecordKind RK = static_cast<TypeRecordKind>(Record.Kind);
37   T KnownRecord(RK);
38   if (auto EC = Callbacks.visitKnownMember(Record, KnownRecord))
39     return EC;
40   return Error::success();
41 }
42 
visitMemberRecord(CVMemberRecord & Record,TypeVisitorCallbacks & Callbacks)43 static Error visitMemberRecord(CVMemberRecord &Record,
44                                TypeVisitorCallbacks &Callbacks) {
45   if (auto EC = Callbacks.visitMemberBegin(Record))
46     return EC;
47 
48   switch (Record.Kind) {
49   default:
50     if (auto EC = Callbacks.visitUnknownMember(Record))
51       return EC;
52     break;
53 #define MEMBER_RECORD(EnumName, EnumVal, Name)                                 \
54   case EnumName: {                                                             \
55     if (auto EC = visitKnownMember<Name##Record>(Record, Callbacks))           \
56       return EC;                                                               \
57     break;                                                                     \
58   }
59 #define MEMBER_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)                \
60   MEMBER_RECORD(EnumVal, EnumVal, AliasName)
61 #define TYPE_RECORD(EnumName, EnumVal, Name)
62 #define TYPE_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)
63 #include "llvm/DebugInfo/CodeView/CodeViewTypes.def"
64   }
65 
66   if (auto EC = Callbacks.visitMemberEnd(Record))
67     return EC;
68 
69   return Error::success();
70 }
71 
72 namespace {
73 
74 class CVTypeVisitor {
75 public:
76   explicit CVTypeVisitor(TypeVisitorCallbacks &Callbacks);
77 
78   Error visitTypeRecord(CVType &Record, TypeIndex Index);
79   Error visitTypeRecord(CVType &Record);
80 
81   /// Visits the type records in Data. Sets the error flag on parse failures.
82   Error visitTypeStream(const CVTypeArray &Types);
83   Error visitTypeStream(CVTypeRange Types);
84   Error visitTypeStream(TypeCollection &Types);
85 
86   Error visitMemberRecord(CVMemberRecord Record);
87   Error visitFieldListMemberStream(BinaryStreamReader &Stream);
88 
89 private:
90   Error finishVisitation(CVType &Record);
91 
92   /// The interface to the class that gets notified of each visitation.
93   TypeVisitorCallbacks &Callbacks;
94 };
95 
CVTypeVisitor(TypeVisitorCallbacks & Callbacks)96 CVTypeVisitor::CVTypeVisitor(TypeVisitorCallbacks &Callbacks)
97     : Callbacks(Callbacks) {}
98 
finishVisitation(CVType & Record)99 Error CVTypeVisitor::finishVisitation(CVType &Record) {
100   switch (Record.Type) {
101   default:
102     if (auto EC = Callbacks.visitUnknownType(Record))
103       return EC;
104     break;
105 #define TYPE_RECORD(EnumName, EnumVal, Name)                                   \
106   case EnumName: {                                                             \
107     if (auto EC = visitKnownRecord<Name##Record>(Record, Callbacks))           \
108       return EC;                                                               \
109     break;                                                                     \
110   }
111 #define TYPE_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)                  \
112   TYPE_RECORD(EnumVal, EnumVal, AliasName)
113 #define MEMBER_RECORD(EnumName, EnumVal, Name)
114 #define MEMBER_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)
115 #include "llvm/DebugInfo/CodeView/CodeViewTypes.def"
116   }
117 
118   if (auto EC = Callbacks.visitTypeEnd(Record))
119     return EC;
120 
121   return Error::success();
122 }
123 
visitTypeRecord(CVType & Record,TypeIndex Index)124 Error CVTypeVisitor::visitTypeRecord(CVType &Record, TypeIndex Index) {
125   if (auto EC = Callbacks.visitTypeBegin(Record, Index))
126     return EC;
127 
128   return finishVisitation(Record);
129 }
130 
visitTypeRecord(CVType & Record)131 Error CVTypeVisitor::visitTypeRecord(CVType &Record) {
132   if (auto EC = Callbacks.visitTypeBegin(Record))
133     return EC;
134 
135   return finishVisitation(Record);
136 }
137 
visitMemberRecord(CVMemberRecord Record)138 Error CVTypeVisitor::visitMemberRecord(CVMemberRecord Record) {
139   return ::visitMemberRecord(Record, Callbacks);
140 }
141 
142 /// Visits the type records in Data. Sets the error flag on parse failures.
visitTypeStream(const CVTypeArray & Types)143 Error CVTypeVisitor::visitTypeStream(const CVTypeArray &Types) {
144   for (auto I : Types) {
145     if (auto EC = visitTypeRecord(I))
146       return EC;
147   }
148   return Error::success();
149 }
150 
visitTypeStream(CVTypeRange Types)151 Error CVTypeVisitor::visitTypeStream(CVTypeRange Types) {
152   for (auto I : Types) {
153     if (auto EC = visitTypeRecord(I))
154       return EC;
155   }
156   return Error::success();
157 }
158 
visitTypeStream(TypeCollection & Types)159 Error CVTypeVisitor::visitTypeStream(TypeCollection &Types) {
160   Optional<TypeIndex> I = Types.getFirst();
161   while (I) {
162     CVType Type = Types.getType(*I);
163     if (auto EC = visitTypeRecord(Type, *I))
164       return EC;
165     I = Types.getNext(*I);
166   }
167   return Error::success();
168 }
169 
visitFieldListMemberStream(BinaryStreamReader & Reader)170 Error CVTypeVisitor::visitFieldListMemberStream(BinaryStreamReader &Reader) {
171   TypeLeafKind Leaf;
172   while (!Reader.empty()) {
173     if (auto EC = Reader.readEnum(Leaf))
174       return EC;
175 
176     CVMemberRecord Record;
177     Record.Kind = Leaf;
178     if (auto EC = ::visitMemberRecord(Record, Callbacks))
179       return EC;
180   }
181 
182   return Error::success();
183 }
184 
185 struct FieldListVisitHelper {
FieldListVisitHelper__anon83306b1c0111::FieldListVisitHelper186   FieldListVisitHelper(TypeVisitorCallbacks &Callbacks, ArrayRef<uint8_t> Data,
187                        VisitorDataSource Source)
188       : Stream(Data, llvm::support::little), Reader(Stream),
189         Deserializer(Reader),
190         Visitor((Source == VDS_BytesPresent) ? Pipeline : Callbacks) {
191     if (Source == VDS_BytesPresent) {
192       Pipeline.addCallbackToPipeline(Deserializer);
193       Pipeline.addCallbackToPipeline(Callbacks);
194     }
195   }
196 
197   BinaryByteStream Stream;
198   BinaryStreamReader Reader;
199   FieldListDeserializer Deserializer;
200   TypeVisitorCallbackPipeline Pipeline;
201   CVTypeVisitor Visitor;
202 };
203 
204 struct VisitHelper {
VisitHelper__anon83306b1c0111::VisitHelper205   VisitHelper(TypeVisitorCallbacks &Callbacks, VisitorDataSource Source)
206       : Visitor((Source == VDS_BytesPresent) ? Pipeline : Callbacks) {
207     if (Source == VDS_BytesPresent) {
208       Pipeline.addCallbackToPipeline(Deserializer);
209       Pipeline.addCallbackToPipeline(Callbacks);
210     }
211   }
212 
213   TypeDeserializer Deserializer;
214   TypeVisitorCallbackPipeline Pipeline;
215   CVTypeVisitor Visitor;
216 };
217 }
218 
visitTypeRecord(CVType & Record,TypeIndex Index,TypeVisitorCallbacks & Callbacks,VisitorDataSource Source)219 Error llvm::codeview::visitTypeRecord(CVType &Record, TypeIndex Index,
220                                       TypeVisitorCallbacks &Callbacks,
221                                       VisitorDataSource Source) {
222   VisitHelper V(Callbacks, Source);
223   return V.Visitor.visitTypeRecord(Record, Index);
224 }
225 
visitTypeRecord(CVType & Record,TypeVisitorCallbacks & Callbacks,VisitorDataSource Source)226 Error llvm::codeview::visitTypeRecord(CVType &Record,
227                                       TypeVisitorCallbacks &Callbacks,
228                                       VisitorDataSource Source) {
229   VisitHelper V(Callbacks, Source);
230   return V.Visitor.visitTypeRecord(Record);
231 }
232 
visitTypeStream(const CVTypeArray & Types,TypeVisitorCallbacks & Callbacks,VisitorDataSource Source)233 Error llvm::codeview::visitTypeStream(const CVTypeArray &Types,
234                                       TypeVisitorCallbacks &Callbacks,
235                                       VisitorDataSource Source) {
236   VisitHelper V(Callbacks, Source);
237   return V.Visitor.visitTypeStream(Types);
238 }
239 
visitTypeStream(CVTypeRange Types,TypeVisitorCallbacks & Callbacks)240 Error llvm::codeview::visitTypeStream(CVTypeRange Types,
241                                       TypeVisitorCallbacks &Callbacks) {
242   VisitHelper V(Callbacks, VDS_BytesPresent);
243   return V.Visitor.visitTypeStream(Types);
244 }
245 
visitTypeStream(TypeCollection & Types,TypeVisitorCallbacks & Callbacks)246 Error llvm::codeview::visitTypeStream(TypeCollection &Types,
247                                       TypeVisitorCallbacks &Callbacks) {
248   // When the internal visitor calls Types.getType(Index) the interface is
249   // required to return a CVType with the bytes filled out.  So we can assume
250   // that the bytes will be present when individual records are visited.
251   VisitHelper V(Callbacks, VDS_BytesPresent);
252   return V.Visitor.visitTypeStream(Types);
253 }
254 
visitMemberRecord(CVMemberRecord Record,TypeVisitorCallbacks & Callbacks,VisitorDataSource Source)255 Error llvm::codeview::visitMemberRecord(CVMemberRecord Record,
256                                         TypeVisitorCallbacks &Callbacks,
257                                         VisitorDataSource Source) {
258   FieldListVisitHelper V(Callbacks, Record.Data, Source);
259   return V.Visitor.visitMemberRecord(Record);
260 }
261 
visitMemberRecord(TypeLeafKind Kind,ArrayRef<uint8_t> Record,TypeVisitorCallbacks & Callbacks)262 Error llvm::codeview::visitMemberRecord(TypeLeafKind Kind,
263                                         ArrayRef<uint8_t> Record,
264                                         TypeVisitorCallbacks &Callbacks) {
265   CVMemberRecord R;
266   R.Data = Record;
267   R.Kind = Kind;
268   return visitMemberRecord(R, Callbacks, VDS_BytesPresent);
269 }
270 
visitMemberRecordStream(ArrayRef<uint8_t> FieldList,TypeVisitorCallbacks & Callbacks)271 Error llvm::codeview::visitMemberRecordStream(ArrayRef<uint8_t> FieldList,
272                                               TypeVisitorCallbacks &Callbacks) {
273   FieldListVisitHelper V(Callbacks, FieldList, VDS_BytesPresent);
274   return V.Visitor.visitFieldListMemberStream(V.Reader);
275 }
276