Static Value-Flow Analysis
Loading...
Searching...
No Matches
DCHG.h
Go to the documentation of this file.
1//===----- DCHG.h -- CHG using DWARF debug info ---------------------------//
2// This is based upon the original CHA.h (now CHG.h).
3
4/*
5 * DCHG.h
6 *
7 * Created on: Aug 23, 2019
8 * Author: Mohamad Barbar
9 */
10
11// TODO: add a flag such that getCanonicalType returns its arg so
12// that the impl. does not "node collapsing" based on teq.
13
14#ifndef DCHG_H_
15#define DCHG_H_
16
17#include "Graphs/GenericGraph.h"
18#include "Graphs/CHG.h"
19#include "SVF-LLVM/BasicTypes.h"
20#include "Util/SVFUtil.h"
21#include "Util/WorkList.h"
22
23namespace SVF
24{
25
26class DCHNode;
27
28class DCHEdge : public GenericEdge<DCHNode>
29{
30public:
31 enum
32 {
33 INHERITANCE, // inheritance relation
34 INSTANCE, // template-instance relation
35 FIRST_FIELD, // src -ff-> dst => dst is first field of src
36 STD_DEF // Edges defined by the standard like (int -std-> char)
37 // We also make the char --> void edge a STD_DEF edge.
38 };
39
41
46
48 {
49 return offset;
50 }
51
53 {
54 this->offset = offset;
55 }
56
57private:
59};
60
61class DCHNode : public GenericNode<DCHNode, DCHEdge>
62{
63public:
64 typedef enum
65 {
66 PURE_ABSTRACT = 0x1, // pure virtual abstract class
67 MULTI_INHERITANCE = 0x2, // multi inheritance class
68 TEMPLATE = 0x04, // template class
69 SCALAR = 0x08 // non-class scalar type
71
72 typedef std::vector<const Function*> FuncVector;
73
76 {
77 this->diType = diType;
78 if (diType == nullptr)
79 {
80 typeName = "null-void";
81 }
82 else if (diType->getRawName() != nullptr)
83 {
84 typeName = diType->getName().str();
85 }
86 else
87 {
88 typeName = "unnamed!";
89 }
90 }
91
93
94 const DIType * getDIType(void) const
95 {
96 return diType;
97 }
98
99 virtual const std::string& getName() const
100 {
101 return typeName;
102 }
103
105
106 inline void setFlag(CLASSATTR mask)
107 {
108 flags |= mask;
109 }
110 inline bool hasFlag(CLASSATTR mask) const
111 {
112 return (flags & mask) == mask;
113 }
115
117
118 inline void setPureAbstract()
119 {
121 }
123 {
125 }
126 inline void setTemplate()
127 {
129 }
130 inline void setScalar()
131 {
133 }
134 inline bool isPureAbstract() const
135 {
136 return hasFlag(PURE_ABSTRACT);
137 }
138 inline bool isMultiInheritance() const
139 {
141 }
142 inline bool isTemplate() const
143 {
144 return hasFlag(TEMPLATE);
145 }
146 inline bool isScalar() const
147 {
148 return hasFlag(SCALAR);
149 }
151
153 {
154 typedefs.insert(diTypedef);
155 }
156
158 {
159 return typedefs;
160 }
161
163 {
164 vtable = vtbl;
165 }
166
167 const GlobalObjVar *getVTable() const
168 {
169 return vtable;
170 }
171
173 const std::vector<std::vector<const Function* >> &getVfnVectors(void) const
174 {
175 return vfnVectors;
176 }
177
179 std::vector<const Function* > &getVfnVector(unsigned n)
180 {
181 if (vfnVectors.size() < n + 1)
182 {
183 vfnVectors.resize(n + 1);
184 }
185
186 return vfnVectors[n];
187 }
188
189private:
195 std::string typeName;
196 size_t flags;
198 std::vector<const Function* > primaryVTable;
202 std::vector<std::vector<const Function*>> vfnVectors;
203};
204
206class DCHGraph : public CommonCHGraph, public GenericGraph<DCHNode, DCHEdge>
207{
208public:
210 static const DIType* stripQualifiers(const DIType*);
211
213 static const DIType* stripArray(const DIType*);
214
222 static bool teq(const DIType* t1, const DIType* t2);
223
225 static std::string diTypeToStr(const DIType *);
226
227 // Returns whether t is an array, a struct, a class, a union, or neither.
228 static bool isAgg(const DIType* t);
229
230public:
232 :numTypes(0) // vfID(0), buildingCHGTime(0) {
233 {
234 this->kind = DI;
235 }
236
237 virtual ~DCHGraph() { };
238
241 virtual void buildCHG(bool extend);
242
243 void dump(const std::string& filename)
244 {
246 }
247
248 void print(void);
249
250 virtual bool csHasVFnsBasedonCHA(const CallICFGNode* cs) override
251 {
252 return csHasVtblsBasedonCHA(cs);
253 }
254
255 virtual const VFunSet &getCSVFsBasedonCHA(const CallICFGNode* cs) override;
256
258 {
259 assert(false && "not supported");
261 if (!hasNode(type))
262 {
263 return false;
264 }
265
266 return getNode(type)->getVTable() != nullptr;
267 }
268
269 virtual bool csHasVtblsBasedonCHA(const CallICFGNode* cs) override
270 {
271 assert(false && "not supported!");
272 abort();
273 }
274
275 virtual const VTableSet &getCSVtblsBasedonCHA(const CallICFGNode* cs) override;
276 virtual void getVFnsFromVtbls(const CallICFGNode* cs, const VTableSet &vtbls, VFunSet &virtualFunctions) override;
277
280 virtual bool isBase(const DIType *a, const DIType *b, bool firstField);
281
283 virtual bool isFieldOf(const DIType *f, const DIType *b);
284
285 static inline bool classof(const CommonCHGraph *chg)
286 {
287 return chg->getKind() == DI;
288 }
289
293 const DIType* getCanonicalType(const DIType* t);
294
296 const DIType *getFieldType(const DIType *base, unsigned idx)
297 {
298 base = getCanonicalType(base);
299 if (base == nullptr)
300 {
301 // Conservative; the base object is untyped, sadly.
302 return nullptr;
303 }
304
305 // For TBHC this is conservative because the union type is lower in the DCHG
306 // than its fields. TODO: make more precise.
307 if (base->getTag() == dwarf::DW_TAG_union_type)
308 {
309 return base;
310 }
311
312 if (base->getTag() == dwarf::DW_TAG_array_type)
313 {
314 const DICompositeType* cbase =
315 SVFUtil::dyn_cast<DICompositeType>(base);
316 assert(cbase && "DCHG: bad DIComposite case");
317 return cbase->getBaseType();
318 }
319
320 if (!(base->getTag() == dwarf::DW_TAG_class_type ||
321 base->getTag() == dwarf::DW_TAG_structure_type))
322 {
323 return nullptr;
324 }
325
326 assert(fieldTypes.find(base) != fieldTypes.end() && "DCHG: base not flattened!");
327 std::vector<const DIType *> &fields = fieldTypes[base];
328 assert(fields.size() > idx && "DCHG: idx into struct larger than # fields!");
329 return getCanonicalType(fields[idx]);
330 }
331
333 const std::vector<const DIType *> &getFieldTypes(const DIType *base)
334 {
335 base = getCanonicalType(base);
336 assert(fieldTypes.find(base) != fieldTypes.end() && "DCHG: base not flattened!");
337 return fieldTypes[base];
338 }
339
340 // Returns the number of fields in base (length of getFieldTypes).
341 unsigned getNumFields(const DIType *base)
342 {
343 base = getCanonicalType(base);
344 assert(fieldTypes.find(base) != fieldTypes.end() && "DCHG: base not flattened!");
345 return fieldTypes[base].size();
346 }
347
350 {
351 base = getCanonicalType(base);
352 assert(containingAggs.find(base) != containingAggs.end() && "DCHG: aggregates not gathered for base!");
353 return containingAggs[base];
354 }
355
356 bool isFirstField(const DIType* f, const DIType* b);
357
358protected:
360 bool extended = false;
381
382private:
391
393 void buildVTables();
394
396 const NodeBS& cha(const DIType* type, bool firstField);
397
399 void handleTypedef(const DIType* typedefType);
400
402 void flatten(const DICompositeType* type);
403
405 void gatherAggs(const DICompositeType* type);
406
409
411 const DIType* getCSStaticType(CallBase* cs) const;
412
413 const DIType *getCSStaticType(const CallICFGNode* cs) const
414 {
415 assert(false && "not supported!");
416 abort();
417 }
418
420 bool hasNode(const DIType* type)
421 {
423 return diTypeToNodeMap.find(type) != diTypeToNodeMap.end();
424 }
425
428 {
430 if (hasNode(type))
431 {
432 return diTypeToNodeMap.at(type);
433 }
434
435 return nullptr;
436 }
437
438
443
446};
447
448} // End namespace SVF
449
450namespace SVF
451{
452/* !
453 * GenericGraphTraits specializations for generic graph algorithms.
454 * Provide graph traits for traversing from a constraint node using standard graph traversals.
455 */
456template<> struct GenericGraphTraits<SVF::DCHNode*> : public GenericGraphTraits<SVF::GenericNode<SVF::DCHNode,SVF::DCHEdge>* >
457{
458};
459
461template<>
462struct GenericGraphTraits<Inverse<SVF::DCHNode*> > : public GenericGraphTraits<Inverse<SVF::GenericNode<SVF::DCHNode,SVF::DCHEdge>* > >
463{
464};
465
466template<> struct GenericGraphTraits<SVF::DCHGraph*> : public GenericGraphTraits<SVF::GenericGraph<SVF::DCHNode,SVF::DCHEdge>* >
467{
469};
470
471} // End namespace llvm
472
473#endif /* DCHG_H_ */
newitem type
Definition cJSON.cpp:2739
cJSON * a
Definition cJSON.cpp:2560
cJSON * n
Definition cJSON.cpp:2558
const cJSON *const b
Definition cJSON.h:255
Common base for class hierarchy graph. Only implements what PointerAnalysis needs.
Definition CHG.h:51
CHGKind kind
Definition CHG.h:73
CHGKind getKind(void) const
Definition CHG.h:67
@ FIRST_FIELD
Definition DCHG.h:35
@ INSTANCE
Definition DCHG.h:34
@ STD_DEF
Definition DCHG.h:36
@ INHERITANCE
Definition DCHG.h:33
DCHEdge(DCHNode *src, DCHNode *dst, GEdgeFlag k=0)
Definition DCHG.h:42
u32_t offset
Definition DCHG.h:58
u32_t getConstantFieldIdx(void) const
Definition DCHG.h:47
void setOffset(u32_t offset)
Definition DCHG.h:52
GenericNode< DCHNode, DCHEdge >::GEdgeSetTy DCHEdgeSetTy
Definition DCHG.h:40
Dwarf based CHG.
Definition DCHG.h:207
const DIType * getCSStaticType(CallBase *cs) const
Retrieves the metadata associated with a virtual callsite.
Definition DCHG.cpp:426
unsigned getNumFields(const DIType *base)
Definition DCHG.h:341
const DIType * getCanonicalType(const DIType *t)
Definition DCHG.cpp:724
const NodeBS & cha(const DIType *type, bool firstField)
Returns a set of all children of type (CHA). Also gradually builds chaMap.
Definition DCHG.cpp:225
const Set< const DIType * > & getAggs(const DIType *base)
Returns all the aggregates contained (transitively) in base.
Definition DCHG.h:349
const DIType * getFieldType(const DIType *base, unsigned idx)
Returns the type of field number idx (flattened) in base.
Definition DCHG.h:296
static const DIType * stripQualifiers(const DIType *)
Returns the DIType beneath the qualifiers. Does not strip away "DW_TAG_members".
Definition DCHG.cpp:765
void handleDIDerivedType(const DIDerivedType *derivedType)
Construction helper to process DIDerivedTypes.
Definition DCHG.cpp:97
DCHEdge * addEdge(const DIType *t1, const DIType *t2, DCHEdge::GEdgeKind et)
Creates an edge between from t1 to t2.
Definition DCHG.cpp:435
bool isFirstField(const DIType *f, const DIType *b)
Definition DCHG.cpp:942
static std::string diTypeToStr(const DIType *)
Returns a human-readable version of the DIType.
Definition DCHG.cpp:969
void buildVTables()
Finds all defined virtual functions and attaches them to nodes.
Definition DCHG.cpp:166
void handleDICompositeType(const DICompositeType *compositeType)
Construction helper to process DICompositeTypes.
Definition DCHG.cpp:27
Map< const CallICFGNode *, VFunSet > csCHAMap
Maps callsites to a set of potential virtual functions based on CHA.
Definition DCHG.h:372
bool hasNode(const DIType *type)
Checks if a node exists for type.
Definition DCHG.h:420
void dump(const std::string &filename)
Definition DCHG.h:243
void print(void)
Definition DCHG.cpp:1142
Map< const DIType *, NodeBS > chaFFMap
Maps types to all children but also considering first field.
Definition DCHG.h:368
DCHNode * getNode(const DIType *type)
Returns the node for type (nullptr if it doesn't exist).
Definition DCHG.h:427
Map< const DIType *, const DIType * > canonicalTypeMap
Maps types to their canonical type (many-to-one).
Definition DCHG.h:374
void handleDISubroutineType(const DISubroutineType *subroutineType)
Construction helper to process DISubroutineTypes.
Definition DCHG.cpp:133
virtual void getVFnsFromVtbls(const CallICFGNode *cs, const VTableSet &vtbls, VFunSet &virtualFunctions) override
Definition DCHG.cpp:594
static const DIType * stripArray(const DIType *)
Returns the DIType beneath all qualifiers and arrays.
Definition DCHG.cpp:818
void handleTypedef(const DIType *typedefType)
Attaches the typedef(s) to the base node.
Definition DCHG.cpp:138
virtual void buildCHG(bool extend)
Definition DCHG.cpp:471
virtual bool csHasVFnsBasedonCHA(const CallICFGNode *cs) override
Definition DCHG.h:250
DCHEdge * hasEdge(const DIType *t1, const DIType *t2, DCHEdge::GEdgeKind et)
Returns the edge between t1 and t2 if it exists, returns nullptr otherwise.
Definition DCHG.cpp:452
NodeID numTypes
Number of types (nodes) in the graph.
Definition DCHG.h:445
void handleDIBasicType(const DIBasicType *basicType)
Construction helper to process DIBasicTypes.
Definition DCHG.cpp:22
virtual const VTableSet & getCSVtblsBasedonCHA(const CallICFGNode *cs) override
Definition DCHG.cpp:566
void gatherAggs(const DICompositeType *type)
Populates containingAggs for type and all its elements.
Definition DCHG.cpp:355
static bool isAgg(const DIType *t)
Definition DCHG.cpp:347
DCHNode * getOrCreateNode(const DIType *type)
Creates a node from type, or returns it if it exists.
Definition DCHG.cpp:408
Map< const DIType *, VTableSet > vtblCHAMap
Maps types to a set with their vtable and all their children's.
Definition DCHG.h:370
bool extended
Whether this CHG is an extended CHG (first-field). Set by buildCHG.
Definition DCHG.h:360
const std::vector< const DIType * > & getFieldTypes(const DIType *base)
Returns a vector of the types of all fields in base.
Definition DCHG.h:333
Map< const DIType *, Set< const DIType * > > containingAggs
Maps aggregate types to all the aggregate types it transitively contains.
Definition DCHG.h:380
virtual bool isBase(const DIType *a, const DIType *b, bool firstField)
Definition DCHG.cpp:677
void flatten(const DICompositeType *type)
Populates fieldTypes for type and all its elements.
Definition DCHG.cpp:270
Map< const GlobalObjVar *, const DIType * > vtblToTypeMap
Maps VTables to the DIType associated with them.
Definition DCHG.h:364
const DIType * getCSStaticType(const CallICFGNode *cs) const
Definition DCHG.h:413
virtual const VFunSet & getCSVFsBasedonCHA(const CallICFGNode *cs) override
Definition DCHG.cpp:549
static bool teq(const DIType *t1, const DIType *t2)
Definition DCHG.cpp:830
Map< const DIType *, std::vector< const DIType * > > fieldTypes
Maps types to their flattened fields' types.
Definition DCHG.h:378
Map< const DIType *, NodeBS > chaMap
Maps types to all children (i.e. CHA).
Definition DCHG.h:366
Map< const DIType *, DCHNode * > diTypeToNodeMap
Maps DITypes to their nodes.
Definition DCHG.h:362
virtual ~DCHGraph()
Definition DCHG.h:237
virtual bool csHasVtblsBasedonCHA(const CallICFGNode *cs) override
Definition DCHG.h:269
virtual bool isFieldOf(const DIType *f, const DIType *b)
Returns true if f is a field of b (fields from getFieldTypes).
Definition DCHG.cpp:688
static bool classof(const CommonCHGraph *chg)
Definition DCHG.h:285
Set< const DIType * > canonicalTypes
Set of all possible canonical types (i.e. values of canonicalTypeMap).
Definition DCHG.h:376
virtual bool csHasVtblsBasedonCHA(CallBase *cs)
Definition DCHG.h:257
@ PURE_ABSTRACT
Definition DCHG.h:66
@ MULTI_INHERITANCE
Definition DCHG.h:67
@ TEMPLATE
Definition DCHG.h:68
const DIType * diType
Type of this node.
Definition DCHG.h:191
size_t flags
Definition DCHG.h:196
void setTemplate()
Definition DCHG.h:126
bool isMultiInheritance() const
Definition DCHG.h:138
void setFlag(CLASSATTR mask)
Flags.
Definition DCHG.h:106
bool isScalar() const
Definition DCHG.h:146
std::vector< const Function * > primaryVTable
The virtual functions which this class actually defines/overrides.
Definition DCHG.h:198
bool hasFlag(CLASSATTR mask) const
Definition DCHG.h:110
void setMultiInheritance()
Definition DCHG.h:122
const std::vector< std::vector< const Function * > > & getVfnVectors(void) const
Returns the vector of virtual function vectors.
Definition DCHG.h:173
DCHNode(const DIType *diType, NodeID i=0, GNodeK k=GNodeK::DCHNodeKd)
Definition DCHG.h:74
std::string typeName
Definition DCHG.h:195
void setScalar()
Definition DCHG.h:130
void addTypedef(const DIDerivedType *diTypedef)
Definition DCHG.h:152
const GlobalObjVar * vtable
Definition DCHG.h:194
const GlobalObjVar * getVTable() const
Definition DCHG.h:167
std::vector< const Function * > & getVfnVector(unsigned n)
Return the nth virtual function vector in the vtable.
Definition DCHG.h:179
~DCHNode()
Definition DCHG.h:92
Set< const DIDerivedType * > typedefs
Typedefs which map to this type.
Definition DCHG.h:193
const DIType * getDIType(void) const
Definition DCHG.h:94
bool isTemplate() const
Definition DCHG.h:142
const Set< const DIDerivedType * > & getTypedefs(void) const
Definition DCHG.h:157
std::vector< const Function * > FuncVector
Definition DCHG.h:72
bool isPureAbstract() const
Definition DCHG.h:134
std::vector< std::vector< const Function * > > vfnVectors
Definition DCHG.h:202
virtual const std::string & getName() const
Definition DCHG.h:99
void setPureAbstract()
Attribute.
Definition DCHG.h:118
void setVTable(const GlobalObjVar *vtbl)
Definition DCHG.h:162
DCHNode * src
source node
DCHNode * dst
destination node
OrderedSet< EdgeType *, typename EdgeType::equalGEdge > GEdgeSetTy
Edge kind.
static void WriteGraphToFile(SVF::OutStream &O, const std::string &GraphName, const GraphType &GT, bool simple=false)
virtual const std::string & getName() const
Definition SVFValue.h:184
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
llvm::DISubroutineType DISubroutineType
Definition BasicTypes.h:240
llvm::CallBase CallBase
Definition BasicTypes.h:146
u32_t NodeID
Definition GeneralType.h:56
Set< const GlobalObjVar * > VTableSet
Definition CHG.h:46
llvm::DIDerivedType DIDerivedType
Definition BasicTypes.h:238
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
llvm::DIType DIType
Definition BasicTypes.h:236
unsigned u32_t
Definition GeneralType.h:47
llvm::DIBasicType DIBasicType
Definition BasicTypes.h:241
llvm::DICompositeType DICompositeType
Definition BasicTypes.h:237
Set< const FunObjVar * > VFunSet
Definition CHG.h:47