Static Value-Flow Analysis
Loading...
Searching...
No Matches
IRGraph.cpp
Go to the documentation of this file.
1//===- SVFIR.cpp -- Program assignment graph------------------------------------//
2//
3// SVF: Static Value-Flow Analysis
4//
5// Copyright (C) <2013-2017> <Yulei Sui>
6//
7
8// This program is free software: you can redistribute it and/or modify
9// it under the terms of the GNU Affero General Public License as published by
10// the Free Software Foundation, either version 3 of the License, or
11// (at your option) any later version.
12
13// This program is distributed in the hope that it will be useful,
14// but WITHOUT ANY WARRANTY; without even the implied warranty of
15// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16// GNU Affero General Public License for more details.
17
18// You should have received a copy of the GNU Affero General Public License
19// along with this program. If not, see <http://www.gnu.org/licenses/>.
20//
21//===----------------------------------------------------------------------===//
22
23/*
24 * SVFIR.cpp
25 *
26 * Created on: Nov 1, 2013
27 * Author: Yulei Sui
28 */
29
30#include "Graphs/IRGraph.h"
31#include "Util/Options.h"
32
33using namespace SVF;
34using namespace SVFUtil;
35
37{
38
39 for (auto &pair: objTypeInfoMap)
40 {
41 if (ObjTypeInfo* ti = pair.second)
42 delete ti;
43 }
44
45 for (const SVFType* type : svfTypes)
46 delete type;
47 svfTypes.clear();
48
49 for (const StInfo* st : stInfos)
50 delete st;
51 stInfos.clear();
52}
53
58
59
61{
62 FunObjVarToIDMapTy::const_iterator iter = returnFunObjSymMap.find(func);
63 assert(iter!=returnFunObjSymMap.end() && "ret sym not found");
64 return iter->second;
65}
66
68{
69 FunObjVarToIDMapTy::const_iterator iter = varargFunObjSymMap.find(func);
70 assert(iter!=varargFunObjSymMap.end() && "vararg sym not found");
71 return iter->second;
72}
74{
75 if (const SVFArrayType* at = SVFUtil::dyn_cast<SVFArrayType>(type))
76 {
77 outs() << " {Type: " << *at << "}\n"
78 << "\tarray type "
79 << "\t [element size = " << getNumOfFlattenElements(at) << "]\n"
80 << "\n";
81 }
82 else if (const SVFStructType *st = SVFUtil::dyn_cast<SVFStructType>(type))
83 {
84 outs() <<" {Type: " << *st << "}\n";
85 const std::vector<const SVFType*>& finfo = getTypeInfo(st)->getFlattenFieldTypes();
86 int field_idx = 0;
87 for(const SVFType* type : finfo)
88 {
89 outs() << " \tField_idx = " << ++field_idx
90 << ", field type: " << *type << "\n";
91 }
92 outs() << "\n";
93 }
94 else if (const SVFPointerType* pt= SVFUtil::dyn_cast<SVFPointerType>(type))
95 {
96 outs() << *pt << "\n";
97 }
98 else if (const SVFFunctionType* fu =
99 SVFUtil::dyn_cast<SVFFunctionType>(type))
100 {
101 outs() << " {Type: " << *fu << "}\n\n";
102 }
103 else if (const SVFOtherType* ot = SVFUtil::dyn_cast<SVFOtherType>(type))
104 {
105 outs() << " {Type: "<< *ot << "(SVFOtherType)}\n\n";
106 }
107 else
108 {
109 assert(type->isSingleValueType() && "not a single value type, then what else!!");
112 outs() << " {Type: " << *type << "}\n"
113 << "\t [object size = " << eSize << "]\n"
114 << "\n";
115 }
116}
117
118const std::vector<const SVFType *> &IRGraph::getFlattenFieldTypes(const SVFStructType *T)
119{
121}
122
124{
126 {
127 const std::vector<const SVFType*>& so = getTypeInfo(baseType)->getFlattenElementTypes();
128 assert (flatten_idx < so.size() && !so.empty() && "element index out of bounds or struct opaque type, can't get element type!");
129 return so[flatten_idx];
130 }
131 else
132 {
133 const std::vector<const SVFType*>& so = getTypeInfo(baseType)->getFlattenFieldTypes();
134 assert (flatten_idx < so.size() && !so.empty() && "element index out of bounds or struct opaque type, can't get element type!");
135 return so[flatten_idx];
136 }
137}
138
143
145{
147 {
148 const std::vector<u32_t>& so = getTypeInfo(T)->getFlattenedElemIdxVec();
149 assert ((unsigned)origId < so.size() && !so.empty() && "element index out of bounds, can't get flattened index!");
150 return so[origId];
151 }
152 else
153 {
154 if(SVFUtil::isa<SVFStructType>(T))
155 {
156 const std::vector<u32_t>& so = getTypeInfo(T)->getFlattenedFieldIdxVec();
157 assert ((unsigned)origId < so.size() && !so.empty() && "Struct index out of bounds, can't get flattened index!");
158 return so[origId];
159 }
160 else
161 {
163 assert(SVFUtil::isa<SVFArrayType>(T) && "Only accept struct or array type if Options::ModelArrays is disabled!");
164 return 0;
165 }
166 }
167}
168
176
187
189{
193
194 APOffset offset = apOffset;
195 if(offset < 0)
196 {
197 writeWrnMsg("try to create a gep node with negative offset.");
198 offset = std::abs(offset);
199 }
200 u32_t maxOffset = baseObj->getMaxFieldOffsetLimit();
201
207 if (maxOffset == 0)
208 offset = 0;
214 else if ((u32_t)offset > maxOffset - 1)
215 {
221 else
225 offset = maxOffset - 1;
226 }
227
228 return offset;
229}
230
232{
233
235 if(type && type->isPointerTy())
236 {
238 }
239 return typeInfo;
240}
241
243{
244 assert(T);
245 SVFTypeSet::const_iterator it = svfTypes.find(T);
246 assert(it != svfTypes.end() && "type info not found? collect them first during SVFIR Building");
247 return (*it)->getTypeInfo();
248}
249
254{
255
257 outs() << "add edge from " << src->getId() << " kind :"
258 << src->getNodeKind() << " to " << dst->getId()
259 << " kind :" << dst->getNodeKind() << "\n");
260 src->addOutEdge(edge);
261 dst->addInEdge(edge);
262 return true;
263}
264
269{
270 SVFStmt edge(src,dst,kind, false);
271 SVFStmt::SVFStmtSetTy::iterator it = KindToSVFStmtSetMap[kind].find(&edge);
272 if (it != KindToSVFStmtSetMap[kind].end())
273 {
274 return *it;
275 }
276 return nullptr;
277}
278
283{
285 SVFStmt::SVFStmtSetTy::iterator it = KindToSVFStmtSetMap[kind].find(&edge);
286 if (it != KindToSVFStmtSetMap[kind].end())
287 {
288 return *it;
289 }
290 return nullptr;
291}
292
297{
298 SVFStmt edge(src,dst,SVFStmt::makeEdgeFlagWithCallInst(kind,callInst), false);
299 SVFStmt::SVFStmtSetTy::iterator it = KindToSVFStmtSetMap[kind].find(&edge);
300 if (it != KindToSVFStmtSetMap[kind].end())
301 {
302 return *it;
303 }
304 return nullptr;
305}
306
310void IRGraph::dump(std::string name)
311{
313}
314
319{
320 SVF::ViewGraph(this, "ProgramAssignmentGraph");
321}
322
323
325{
326 if (valVarNum != 0) return valVarNum;
327 u32_t num = 0;
328 for (const auto& item: *this)
329 {
330 if (SVFUtil::isa<ValVar>(item.second))
331 num++;
332 }
333 return valVarNum = num;
334}
335
336
338{
339 if (objVarNum != 0) return objVarNum;
340 u32_t num = 0;
341 for (const auto& item: *this)
342 {
343 if (SVFUtil::isa<ObjVar>(item.second))
344 num++;
345 }
346 return objVarNum = num;
347}
348
349
350
351namespace SVF
352{
356template<>
358{
359
362 DOTGraphTraits(bool isSimple = false) :
363 DefaultDOTGraphTraits(isSimple)
364 {
365 }
366
368 static std::string getGraphName(IRGraph *graph)
369 {
370 return graph->getGraphName();
371 }
372
375 static bool isNodeHidden(SVFVar *node, IRGraph *)
376 {
377 if (Options::ShowHiddenNode()) return false;
378 else return node->isIsolatedNode();
379 }
380
383 static std::string getNodeLabel(SVFVar *node, IRGraph*)
384 {
385 std::string str;
386 std::stringstream rawstr(str);
387 // print function info
388 if (node->getFunction())
389 rawstr << "[" << node->getFunction()->getName() << "] ";
390
391 rawstr << node->toString();
392
393 return rawstr.str();
394
395 }
396
397 static std::string getNodeAttributes(SVFVar *node, IRGraph*)
398 {
399 if (SVFUtil::isa<ValVar>(node))
400 {
401 if(SVFUtil::isa<GepValVar>(node))
402 return "shape=hexagon";
403 else if (SVFUtil::isa<DummyValVar>(node))
404 return "shape=diamond";
405 else
406 return "shape=box";
407 }
408 else if (SVFUtil::isa<ObjVar>(node))
409 {
410 if(SVFUtil::isa<GepObjVar>(node))
411 return "shape=doubleoctagon";
412 else if(SVFUtil::isa<BaseObjVar>(node))
413 return "shape=box3d";
414 else if (SVFUtil::isa<DummyObjVar>(node))
415 return "shape=tab";
416 else
417 return "shape=component";
418 }
419 else if (SVFUtil::isa<RetValPN>(node))
420 {
421 return "shape=Mrecord";
422 }
423 else if (SVFUtil::isa<VarArgValPN>(node))
424 {
425 return "shape=octagon";
426 }
427 else
428 {
429 assert(0 && "no such kind!!");
430 }
431 return "";
432 }
433
434 template<class EdgeIter>
436 {
437 const SVFStmt* edge = *(EI.getCurrent());
438 assert(edge && "No edge found!!");
439 if (SVFUtil::isa<AddrStmt>(edge))
440 {
441 return "color=green";
442 }
443 else if (SVFUtil::isa<CopyStmt>(edge))
444 {
445 return "color=black";
446 }
447 else if (SVFUtil::isa<GepStmt>(edge))
448 {
449 return "color=purple";
450 }
451 else if (SVFUtil::isa<StoreStmt>(edge))
452 {
453 return "color=blue";
454 }
455 else if (SVFUtil::isa<LoadStmt>(edge))
456 {
457 return "color=red";
458 }
459 else if (SVFUtil::isa<PhiStmt>(edge))
460 {
461 return "color=grey";
462 }
463 else if (SVFUtil::isa<SelectStmt>(edge))
464 {
465 return "color=grey";
466 }
467 else if (SVFUtil::isa<CmpStmt>(edge))
468 {
469 return "color=grey";
470 }
471 else if (SVFUtil::isa<BinaryOPStmt>(edge))
472 {
473 return "color=grey";
474 }
475 else if (SVFUtil::isa<UnaryOPStmt>(edge))
476 {
477 return "color=grey";
478 }
479 else if (SVFUtil::isa<BranchStmt>(edge))
480 {
481 return "color=grey";
482 }
483 else if (SVFUtil::isa<TDForkPE>(edge))
484 {
485 return "color=Turquoise";
486 }
487 else if (SVFUtil::isa<TDJoinPE>(edge))
488 {
489 return "color=Turquoise";
490 }
491 else if (SVFUtil::isa<CallPE>(edge))
492 {
493 return "color=black,style=dashed";
494 }
495 else if (SVFUtil::isa<RetPE>(edge))
496 {
497 return "color=black,style=dotted";
498 }
499
500 assert(false && "No such kind edge!!");
501 exit(1);
502 }
503
504 template<class EdgeIter>
505 static std::string getEdgeSourceLabel(SVFVar*, EdgeIter EI)
506 {
507 const SVFStmt* edge = *(EI.getCurrent());
508 assert(edge && "No edge found!!");
509 if(const CallPE* calledge = SVFUtil::dyn_cast<CallPE>(edge))
510 {
511 return calledge->getCallSite()->getSourceLoc();
512 }
513 else if(const RetPE* retedge = SVFUtil::dyn_cast<RetPE>(edge))
514 {
515 return retedge->getCallSite()->getSourceLoc();
516 }
517 return "";
518 }
519};
520} // End namespace llvm
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
Definition SVFType.h:498
#define DPAGBuild
Definition SVFType.h:506
newitem type
Definition cJSON.cpp:2739
buffer offset
Definition cJSON.cpp:1113
const char *const name
Definition cJSON.h:264
cJSON * item
Definition cJSON.h:222
GEdgeSetTy::iterator iterator
static void WriteGraphToFile(SVF::OutStream &O, const std::string &GraphName, const GraphType &GT, bool simple=false)
u32_t getFlattenedElemIdx(const SVFType *T, u32_t origId)
Flattened element idx of an array or struct by considering stride.
Definition IRGraph.cpp:144
u32_t getNumOfFlattenElements(const SVFType *T)
Definition IRGraph.cpp:169
const std::vector< const SVFType * > & getFlattenFieldTypes(const SVFStructType *T)
Return the flattened field type for struct type only.
Definition IRGraph.cpp:118
void destorySymTable()
Definition IRGraph.cpp:36
SVFStmt * hasLabeledEdge(SVFVar *src, SVFVar *dst, SVFStmt::PEDGEK kind, const ICFGNode *cs)
Definition IRGraph.cpp:296
void printFlattenFields(const SVFType *type)
Debug method.
Definition IRGraph.cpp:73
void dump(std::string name)
Dump SVFIR.
Definition IRGraph.cpp:310
virtual ~IRGraph()
Definition IRGraph.cpp:54
Set< const StInfo * > stInfos
(owned) All StInfo
Definition IRGraph.h:98
u32_t getObjectNodeNum()
Definition IRGraph.cpp:337
void view()
View graph from the debugger.
Definition IRGraph.cpp:318
virtual APOffset getModulusOffset(const BaseObjVar *baseObj, const APOffset &apOffset)
Given an offset from a Gep Instruction, return it modulus offset by considering memory layout.
Definition IRGraph.cpp:188
SVFStmt::KindToSVFStmtMapTy KindToSVFStmtSetMap
SVFIR edge map containing all PAGEdges.
Definition IRGraph.h:109
const SVFType * getFlatternedElemType(const SVFType *baseType, u32_t flatten_idx)
Return the type of a flattened element given a flattened index.
Definition IRGraph.cpp:123
ObjTypeInfo * createObjTypeInfo(const SVFType *type)
Create an objectInfo based on LLVM type (value is null, and type could be null, representing a dummy ...
Definition IRGraph.cpp:231
std::string getGraphName() const
Return graph name.
Definition IRGraph.h:335
FunObjVarToIDMapTy varargFunObjSymMap
vararg map
Definition IRGraph.h:87
NodeID getReturnNode(const FunObjVar *func) const
GetReturnNode - Return the unique node representing the return value of a function.
Definition IRGraph.cpp:60
const SVFType * getOriginalElemType(const SVFType *baseType, u32_t origId) const
Definition IRGraph.cpp:139
u32_t valVarNum
Definition IRGraph.h:114
IDToTypeInfoMapTy objTypeInfoMap
map a memory sym id to its obj
Definition IRGraph.h:88
bool addEdge(SVFVar *src, SVFVar *dst, SVFStmt *edge)
Add an edge into the graph.
Definition IRGraph.cpp:253
u32_t objVarNum
Definition IRGraph.h:115
NodeID getVarargNode(const FunObjVar *func) const
getVarargNode - Return the unique node representing the variadic argument of a variadic function.
Definition IRGraph.cpp:67
SVFStmt * hasNonlabeledEdge(SVFVar *src, SVFVar *dst, SVFStmt::PEDGEK kind)
Definition IRGraph.cpp:268
u32_t getValueNodeNum()
Definition IRGraph.cpp:324
SVFTypeSet svfTypes
Definition IRGraph.h:95
const StInfo * getTypeInfo(const SVFType *T) const
Get struct info.
Definition IRGraph.cpp:242
FunObjVarToIDMapTy returnFunObjSymMap
return map
Definition IRGraph.h:86
const ObjTypeInfo * createDummyObjTypeInfo(NodeID symId, const SVFType *type)
Definition IRGraph.cpp:177
void setFlag(MEMTYPE mask)
Flag for this object type.
static Option< bool > ModelArrays
Definition Options.h:188
static const Option< bool > CyclicFldIdx
Definition Options.h:189
static const Option< u32_t > MaxFieldLimit
Maximum number of field derivations for an object.
Definition Options.h:38
static const Option< bool > ShowHiddenNode
Definition Options.h:228
static GEdgeFlag makeEdgeFlagWithAddionalOpnd(GEdgeKind k, const SVFVar *var)
static GEdgeFlag makeEdgeFlagWithCallInst(GEdgeKind k, const ICFGNode *cs)
NodeID getId() const
Get ID.
Definition SVFValue.h:158
GNodeK getNodeKind() const
Get node kind.
Definition SVFValue.h:164
virtual const std::string & getName() const
Definition SVFValue.h:184
void addOutEdge(SVFStmt *outEdge)
void addInEdge(SVFStmt *inEdge)
Edge management methods.
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
virtual bool isIsolatedNode() const
Check if this node is isolated (no edges) in the SVFIR graph.
virtual const std::string toString() const
Get string representation.
const SVFType * getOriginalElemType(u32_t fldIdx) const
Definition SVFValue.cpp:31
std::vector< const SVFType * > & getFlattenElementTypes()
Definition SVFType.h:102
std::vector< u32_t > & getFlattenedElemIdxVec()
Definition SVFType.h:98
u32_t getNumOfFlattenElements() const
Return number of elements after flattening (including array elements)
Definition SVFType.h:139
std::vector< const SVFType * > & getFlattenFieldTypes()
Definition SVFType.h:106
std::vector< u32_t > & getFlattenedFieldIdxVec()
Definition SVFType.h:94
u32_t getNumOfFlattenFields() const
Return the number of fields after flattening (ignoring array elements)
Definition SVFType.h:145
void writeWrnMsg(const std::string &msg)
Writes a message run through wrnMsg.
Definition SVFUtil.cpp:68
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
u32_t NodeID
Definition GeneralType.h:56
void ViewGraph(const GraphType &G, const std::string &name, bool ShortNames=false, GraphProgram::Name Program=GraphProgram::DOT)
s64_t APOffset
Definition GeneralType.h:60
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
unsigned u32_t
Definition GeneralType.h:47
static std::string getEdgeAttributes(SVFVar *, EdgeIter EI, IRGraph *)
Definition IRGraph.cpp:435
DOTGraphTraits(bool isSimple=false)
Definition IRGraph.cpp:362
static std::string getGraphName(IRGraph *graph)
Return name of the graph.
Definition IRGraph.cpp:368
static std::string getNodeLabel(SVFVar *node, IRGraph *)
Definition IRGraph.cpp:383
static std::string getNodeAttributes(SVFVar *node, IRGraph *)
Definition IRGraph.cpp:397
static bool isNodeHidden(SVFVar *node, IRGraph *)
Definition IRGraph.cpp:375
NodeType::iterator ChildIteratorType
Definition IRGraph.cpp:361
static std::string getEdgeSourceLabel(SVFVar *, EdgeIter EI)
Definition IRGraph.cpp:505