Static Value-Flow Analysis
Loading...
Searching...
No Matches
SVFIR.cpp
Go to the documentation of this file.
1//===- SVFIR.cpp -- IR of SVF ---------------------------------------------//
2//
3// SVF: Static Value-Flow Analysis
4//
5// Copyright (C) <2013-> <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: 31, 12, 2021
27 * Author: Yulei Sui
28 */
29
30#include "Util/Options.h"
31#include "SVFIR/SVFIR.h"
32#include "Graphs/CallGraph.h"
33
34using namespace SVF;
35using namespace SVFUtil;
36
37
38std::unique_ptr<SVFIR> SVFIR::pag;
39
40std::string SVFIR::pagReadFromTxt = "";
41
45
46
47const FunObjVar *SVFIR::getFunObjVar(const std::string &name)
48{
49 for (const auto &item: *callGraph)
50 {
51 if (item.second->getName() == name)
52 {
53 return item.second->getFunction();
54 }
55 }
56 return nullptr;
57}
58
59
64{
65 SVFVar* srcNode = getGNode(src);
66 SVFVar* dstNode = getGNode(dst);
68 return nullptr;
69 else
70 {
74 return addrPE;
75 }
76}
77
95
100{
102 SVFVar* resNode = getGNode(res);
103 PHINodeMap::iterator it = phiNodeMap.find(resNode);
104 if(it == phiNodeMap.end())
105 {
106 PhiStmt* phi = new PhiStmt(resNode, {opNode}, {pred});
110 return phi;
111 }
112 else
113 {
114 it->second->addOpVar(opNode,pred);
116 return nullptr;
117 }
118}
119
124{
127 SVFVar* dstNode = getGNode(res);
128 SVFVar* condNode = getGNode(cond);
130 return nullptr;
131 else
132 {
133 std::vector<SVFVar*> opnds = {op1Node, op2Node};
137 return select;
138 }
139}
140
145{
148 SVFVar* dstNode = getGNode(dst);
150 return nullptr;
151 else
152 {
153 std::vector<SVFVar*> opnds = {op1Node, op2Node};
154 CmpStmt* cmp = new CmpStmt(dstNode, opnds, predicate);
157 return cmp;
158 }
159}
160
161
166{
169 SVFVar* dstNode = getGNode(dst);
171 return nullptr;
172 else
173 {
174 std::vector<SVFVar*> opnds = {op1Node, op2Node};
178 return binaryOP;
179 }
180}
181
186{
187 SVFVar* srcNode = getGNode(src);
188 SVFVar* dstNode = getGNode(dst);
190 return nullptr;
191 else
192 {
196 return unaryOP;
197 }
198}
199
200/*
201* Add BranchStmt
202*/
217
222{
223 SVFVar* srcNode = getGNode(src);
224 SVFVar* dstNode = getGNode(dst);
226 return nullptr;
227 else
228 {
232 return loadPE;
233 }
234}
235
241{
242 SVFVar* srcNode = getGNode(src);
243 SVFVar* dstNode = getGNode(dst);
245 return nullptr;
246 else
247 {
248 StoreStmt* storePE = new StoreStmt(srcNode, dstNode, curVal);
251 return storePE;
252 }
253}
254
259{
260 SVFVar* srcNode = getGNode(src);
261 SVFVar* dstNode = getGNode(dst);
263 return nullptr;
264 else
265 {
266 CallPE* callPE = new CallPE(srcNode, dstNode, cs,entry);
269 return callPE;
270 }
271}
272
277{
278 SVFVar* srcNode = getGNode(src);
279 SVFVar* dstNode = getGNode(dst);
281 return nullptr;
282 else
283 {
284 RetPE* retPE = new RetPE(srcNode, dstNode, cs, exit);
285 addToStmt2TypeMap(retPE);
286 addEdge(srcNode,dstNode, retPE);
287 return retPE;
288 }
289}
290
295{
297 return pag->addAddrStmt(pag->getBlackHoleNode(), node);
298 else
299 return pag->addCopyStmt(pag->getNullPtr(), node, CopyStmt::COPYVAL);
300}
301
306{
307 SVFVar* srcNode = getGNode(src);
308 SVFVar* dstNode = getGNode(dst);
310 return nullptr;
311 else
312 {
313 TDForkPE* forkPE = new TDForkPE(srcNode, dstNode, cs, entry);
316 return forkPE;
317 }
318}
319
324{
325 SVFVar* srcNode = getGNode(src);
326 SVFVar* dstNode = getGNode(dst);
328 return nullptr;
329 else
330 {
331 TDJoinPE* joinPE = new TDJoinPE(srcNode, dstNode, cs, exit);
334 return joinPE;
335 }
336}
337
338
345{
346
347 SVFVar* node = getGNode(src);
348 if (!constGep || node->hasIncomingVariantGepEdge())
349 {
352 return addVariantGepStmt(src, dst, ap);
353 }
354 else
355 {
356 return addNormalGepStmt(src, dst, ap);
357 }
358}
359
364{
365 SVFVar* baseNode = getGNode(src);
366 SVFVar* dstNode = getGNode(dst);
368 return nullptr;
369 else
370 {
371 GepStmt* gepPE = new GepStmt(baseNode, dstNode, ap);
374 return gepPE;
375 }
376}
377
383{
384 SVFVar* baseNode = getGNode(src);
385 SVFVar* dstNode = getGNode(dst);
387 return nullptr;
388 else
389 {
390 GepStmt* gepPE = new GepStmt(baseNode, dstNode, ap, true);
393 return gepPE;
394 }
395}
396
397
398
404{
405 NodeID base = baseVar->getId();
406 //assert(findPAGNode(i) == false && "this node should not be created before");
407 assert(0==GepValObjMap[curInst].count(std::make_pair(base, ap))
408 && "this node should not be created before");
409 GepValObjMap[curInst][std::make_pair(base, ap)] = i;
410 GepValVar *node = new GepValVar(baseVar, i, ap, type, icn);
411 return addValNode(node);
412}
413
418{
419 SVFVar* node = pag->getGNode(id);
420 if (GepObjVar* gepNode = SVFUtil::dyn_cast<GepObjVar>(node))
421 return getGepObjVar(gepNode->getBaseObj(), gepNode->getConstantFieldIdx() + apOffset);
422 else if (BaseObjVar* baseNode = SVFUtil::dyn_cast<BaseObjVar>(node))
423 return getGepObjVar(baseNode, apOffset);
424 else if (DummyObjVar* baseNode = SVFUtil::dyn_cast<DummyObjVar>(node))
425 return getGepObjVar(baseNode, apOffset);
426 else
427 {
428 assert(false && "new gep obj node kind?");
429 return id;
430 }
431}
432
440{
441 NodeID base = baseObj->getId();
442
444 if (baseObj->isFieldInsensitive())
445 return getFIObjVar(baseObj);
446
447 APOffset newLS = pag->getModulusOffset(baseObj, apOffset);
448
449 // Base and first field are the same memory location.
450 if (Options::FirstFieldEqBase() && newLS == 0) return base;
451
452 NodeOffsetMap::iterator iter = GepObjVarMap.find(std::make_pair(base, newLS));
453 if (iter == GepObjVarMap.end())
454 {
457 }
458 else
459 return iter->second;
460
461}
462
467{
468 //assert(findPAGNode(i) == false && "this node should not be created before");
469 NodeID base = baseObj->getId();
470 assert(0==GepObjVarMap.count(std::make_pair(base, apOffset))
471 && "this node should not be created before");
472
473 GepObjVarMap[std::make_pair(base, apOffset)] = gepId;
474 //ABTest
475 GepObjVar *node = new GepObjVar(baseObj, gepId, apOffset);
476 memToFieldsMap[base].set(gepId);
477 return addObjNode(node);
478}
479
484{
485 NodeID base = obj->getId();
486 return memToFieldsMap[base];
487}
488
493{
494 const SVFVar* node = pag->getGNode(id);
495 assert(SVFUtil::isa<ObjVar>(node) && "need an object node");
497}
498
505{
506 const SVFVar* node = pag->getGNode(id);
507 assert(SVFUtil::isa<ObjVar>(node) && "need an object node");
508 const BaseObjVar* obj = getBaseObject(id);
509 if(obj->isFieldInsensitive())
510 {
511 NodeBS bs;
513 return bs;
514 }
515 else
516 return getAllFieldsObjVars(obj);
517}
518
523{
524 GepValueVarMap::const_iterator iter = GepValObjMap.find(curInst);
525 if(iter==GepValObjMap.end())
526 {
527 return UINT_MAX;
528 }
529 else
530 {
531 NodeAccessPathMap::const_iterator lit =
532 iter->second.find(std::make_pair(base, ap));
533 if (lit == iter->second.end())
534 return UINT_MAX;
535 else
536 return lit->second;
537 }
538}
539
540
545{
546 delete icfg;
547 icfg = nullptr;
548 delete chgraph;
549 chgraph = nullptr;
550 delete callGraph;
551 callGraph = nullptr;
552}
553
558{
559
560 outs() << "-------------------SVFIR------------------------------------\n";
561 SVFStmt::SVFStmtSetTy& addrs = pag->getSVFStmtSet(SVFStmt::Addr);
562 for (SVFStmt::SVFStmtSetTy::iterator iter = addrs.begin(), eiter =
563 addrs.end(); iter != eiter; ++iter)
564 {
565 outs() << (*iter)->getSrcID() << " -- Addr --> " << (*iter)->getDstID()
566 << "\n";
567 }
568
569 SVFStmt::SVFStmtSetTy& copys = pag->getSVFStmtSet(SVFStmt::Copy);
570 for (SVFStmt::SVFStmtSetTy::iterator iter = copys.begin(), eiter =
571 copys.end(); iter != eiter; ++iter)
572 {
573 outs() << (*iter)->getSrcID() << " -- Copy --> " << (*iter)->getDstID()
574 << "\n";
575 }
576
577 SVFStmt::SVFStmtSetTy& calls = pag->getSVFStmtSet(SVFStmt::Call);
578 for (SVFStmt::SVFStmtSetTy::iterator iter = calls.begin(), eiter =
579 calls.end(); iter != eiter; ++iter)
580 {
581 outs() << (*iter)->getSrcID() << " -- Call --> " << (*iter)->getDstID()
582 << "\n";
583 }
584
585 SVFStmt::SVFStmtSetTy& rets = pag->getSVFStmtSet(SVFStmt::Ret);
586 for (SVFStmt::SVFStmtSetTy::iterator iter = rets.begin(), eiter =
587 rets.end(); iter != eiter; ++iter)
588 {
589 outs() << (*iter)->getSrcID() << " -- Ret --> " << (*iter)->getDstID()
590 << "\n";
591 }
592
594 for (SVFStmt::SVFStmtSetTy::iterator iter = tdfks.begin(), eiter =
595 tdfks.end(); iter != eiter; ++iter)
596 {
597 outs() << (*iter)->getSrcID() << " -- ThreadFork --> "
598 << (*iter)->getDstID() << "\n";
599 }
600
602 for (SVFStmt::SVFStmtSetTy::iterator iter = tdjns.begin(), eiter =
603 tdjns.end(); iter != eiter; ++iter)
604 {
605 outs() << (*iter)->getSrcID() << " -- ThreadJoin --> "
606 << (*iter)->getDstID() << "\n";
607 }
608
609 SVFStmt::SVFStmtSetTy& ngeps = pag->getSVFStmtSet(SVFStmt::Gep);
610 for (SVFStmt::SVFStmtSetTy::iterator iter = ngeps.begin(), eiter =
611 ngeps.end(); iter != eiter; ++iter)
612 {
613 GepStmt* gep = SVFUtil::cast<GepStmt>(*iter);
614 if(gep->isVariantFieldGep())
615 outs() << (*iter)->getSrcID() << " -- VariantGep --> "
616 << (*iter)->getDstID() << "\n";
617 else
618 outs() << gep->getRHSVarID() << " -- Gep (" << gep->getConstantStructFldIdx()
619 << ") --> " << gep->getLHSVarID() << "\n";
620 }
621
622 SVFStmt::SVFStmtSetTy& loads = pag->getSVFStmtSet(SVFStmt::Load);
623 for (SVFStmt::SVFStmtSetTy::iterator iter = loads.begin(), eiter =
624 loads.end(); iter != eiter; ++iter)
625 {
626 outs() << (*iter)->getSrcID() << " -- Load --> " << (*iter)->getDstID()
627 << "\n";
628 }
629
631 for (SVFStmt::SVFStmtSetTy::iterator iter = stores.begin(), eiter =
632 stores.end(); iter != eiter; ++iter)
633 {
634 outs() << (*iter)->getSrcID() << " -- Store --> " << (*iter)->getDstID()
635 << "\n";
636 }
637 outs() << "----------------------------------------------------------\n";
638
639}
640
643{
644 // collect candidate pointers for demand-driven analysis
645 for (iterator nIter = begin(); nIter != end(); ++nIter)
646 {
647 NodeID nodeId = nIter->first;
648 // do not compute points-to for isolated node
649 if (isValidPointer(nodeId) == false)
650 continue;
652 }
653}
654/*
655 * If this is a dummy node or node does not have incoming edges and outgoing edges we assume it is not a pointer here.
656 * However, if it is a pointer and it is an argument of a function definition, we assume it is a pointer here.
657 */
659{
660 SVFVar* node = pag->getGNode(nodeId);
661
662 if(node->isPointer())
663 if (const ValVar* pVar = pag->getBaseValVar(nodeId))
664 if (const ArgValVar* arg = SVFUtil::dyn_cast<ArgValVar>(pVar))
665 if (!(arg->getParent()->isDeclaration()))
666 return true;
667
668 if ((node->getInEdges().empty() && node->getOutEdges().empty()))
669 return false;
670 return node->isPointer();
671}
672
674{
675 if (SVFUtil::isa<ValVar>(node))
676 {
677 if (isValidPointer(node->getId()))
678 {
679 const ValVar* baseVar = pag->getBaseValVar(node->getId());
680 if(!SVFUtil::isa<DummyValVar, BlackHoleValVar>(baseVar))
682 }
683 }
684 return false;
685}
686
694
695
696
newitem type
Definition cJSON.cpp:2739
const cJSON *const b
Definition cJSON.h:255
const char *const name
Definition cJSON.h:264
cJSON * item
Definition cJSON.h:222
int count
Definition cJSON.h:216
void setValue(T v)
Class representing a function argument variable in the SVFIR.
std::vector< std::pair< const ICFGNode *, s32_t > > SuccAndCondPairVec
IDToNodeMapTy::iterator iterator
Node Iterators.
NodeType * getGNode(NodeID id) const
Get a node.
const GEdgeSetTy & getOutEdges() const
const GEdgeSetTy & getInEdges() const
SVFStmt * hasLabeledEdge(SVFVar *src, SVFVar *dst, SVFStmt::PEDGEK kind, const ICFGNode *cs)
Definition IRGraph.cpp:296
bool addEdge(SVFVar *src, SVFVar *dst, SVFStmt *edge)
Add an edge into the graph.
Definition IRGraph.cpp:253
SVFStmt * hasNonlabeledEdge(SVFVar *src, SVFVar *dst, SVFStmt::PEDGEK kind)
Definition IRGraph.cpp:268
static NodeIDAllocator * get(void)
Return (singleton) allocator.
NodeID allocateGepObjectId(NodeID base, u32_t offset, u32_t maxFieldLimit)
static const Option< bool > FirstFieldEqBase
Definition Options.h:103
static Option< bool > HandBlackHole
Definition Options.h:102
static const Option< u32_t > MaxFieldLimit
Maximum number of field derivations for an object.
Definition Options.h:38
NodeID addGepObjNode(const BaseObjVar *baseObj, const APOffset &apOffset, const NodeID gepId)
Add a field obj node, this method can only invoked by getGepObjVar.
Definition SVFIR.cpp:466
NodeID addObjNode(NodeID i, ObjTypeInfo *ti, const SVFType *type, const ICFGNode *node)
Add a memory obj node.
Definition SVFIR.h:623
GepStmt * addVariantGepStmt(NodeID src, NodeID dst, const AccessPath &ap)
Add Variant(Gep) edge.
Definition SVFIR.cpp:382
NodeOffsetMap GepObjVarMap
Map a pair<base,off> to a gep obj node id.
Definition SVFIR.h:83
CopyStmt * addCopyStmt(NodeID src, NodeID dst, CopyStmt::CopyKind type)
Add Copy edge.
Definition SVFIR.cpp:81
NodeID getGepValVar(NodeID curInst, NodeID base, const AccessPath &ap) const
Due to constraint expression, curInst is used to distinguish different instructions (e....
Definition SVFIR.cpp:522
RetPE * addRetPE(NodeID src, NodeID dst, const CallICFGNode *cs, const FunExitICFGNode *exit)
Add Return edge.
Definition SVFIR.cpp:276
GepStmt * addGepStmt(NodeID src, NodeID dst, const AccessPath &ap, bool constGep)
Add Gep edge.
Definition SVFIR.cpp:344
void print()
Print SVFIR.
Definition SVFIR.cpp:557
static void handleBlackHole(bool b)
SVFIR build configurations.
Definition SVFIR.cpp:690
void addToStmt2TypeMap(SVFStmt *edge)
Map a SVFStatement type to a set of corresponding SVF statements.
Definition SVFIR.h:512
CommonCHGraph * chgraph
Definition SVFIR.h:97
SVFStmt * addBlackHoleAddrStmt(NodeID node)
Set a pointer points-to black hole (e.g. int2ptr)
Definition SVFIR.cpp:294
LoadStmt * addLoadStmt(NodeID src, NodeID dst)
Add Load edge.
Definition SVFIR.cpp:221
NodeID addGepValNode(NodeID curInst, const ValVar *base, const AccessPath &ap, NodeID i, const SVFType *type, const ICFGNode *node)
Add a temp field value node, this method can only invoked by getGepValVar.
Definition SVFIR.cpp:403
GepStmt * addNormalGepStmt(NodeID src, NodeID dst, const AccessPath &ap)
Add Offset(Gep) edge.
Definition SVFIR.cpp:363
MemObjToFieldsMap memToFieldsMap
Map a mem object id to all its fields.
Definition SVFIR.h:84
PHINodeMap phiNodeMap
A set of phi copy edges.
Definition SVFIR.h:86
NodeBS getFieldsAfterCollapse(NodeID id)
Definition SVFIR.cpp:504
NodeID getGepObjVar(const BaseObjVar *baseObj, const APOffset &ap)
Get a field SVFIR Object node according to base mem obj and offset.
Definition SVFIR.cpp:439
CallPE * addCallPE(NodeID src, NodeID dst, const CallICFGNode *cs, const FunEntryICFGNode *entry)
Add Call edge.
Definition SVFIR.cpp:258
bool isValidTopLevelPtr(const SVFVar *node)
Definition SVFIR.cpp:673
TDJoinPE * addThreadJoinPE(NodeID src, NodeID dst, const CallICFGNode *cs, const FunExitICFGNode *exit)
Add Thread join edge for parameter passing.
Definition SVFIR.cpp:323
CallGraph * callGraph
all the callsites of a program
Definition SVFIR.h:99
static std::string pagReadFromTxt
Definition SVFIR.h:102
void destroy()
Clean up memory.
Definition SVFIR.cpp:544
const BaseObjVar * getBaseObject(NodeID id) const
Definition SVFIR.h:423
const FunObjVar * getFunObjVar(const std::string &name)
Definition SVFIR.cpp:47
CmpStmt * addCmpStmt(NodeID op1, NodeID op2, NodeID dst, u32_t predict)
Add Copy edge.
Definition SVFIR.cpp:144
AddrStmt * addAddrStmt(NodeID src, NodeID dst)
Add an edge into SVFIR.
Definition SVFIR.cpp:63
UnaryOPStmt * addUnaryOPStmt(NodeID src, NodeID dst, u32_t opcode)
Add Unary edge.
Definition SVFIR.cpp:185
TDForkPE * addThreadForkPE(NodeID src, NodeID dst, const CallICFGNode *cs, const FunEntryICFGNode *entry)
Add Thread fork edge for parameter passing.
Definition SVFIR.cpp:305
BinaryOPStmt * addBinaryOPStmt(NodeID op1, NodeID op2, NodeID dst, u32_t opcode)
Add Copy edge.
Definition SVFIR.cpp:165
NodeBS & getAllFieldsObjVars(const BaseObjVar *obj)
Get all fields of an object.
Definition SVFIR.cpp:483
static std::unique_ptr< SVFIR > pag
call graph
Definition SVFIR.h:101
NodeID addValNode(NodeID i, const SVFType *type, const ICFGNode *icfgNode)
add node into SVFIR
Definition SVFIR.h:564
StoreStmt * addStoreStmt(NodeID src, NodeID dst, const ICFGNode *val)
Add Store edge.
Definition SVFIR.cpp:240
PhiStmt * addPhiStmt(NodeID res, NodeID opnd, const ICFGNode *pred)
Add phi node information.
Definition SVFIR.cpp:99
OrderedNodeSet candidatePointers
Definition SVFIR.h:95
ICFG * icfg
Definition SVFIR.h:96
SVFIR(bool buildFromFile)
Constructor.
Definition SVFIR.cpp:42
SelectStmt * addSelectStmt(NodeID res, NodeID op1, NodeID op2, NodeID cond)
Add SelectStmt.
Definition SVFIR.cpp:123
GepValueVarMap GepValObjMap
Map a pair<base,off> to a gep value node id.
Definition SVFIR.h:81
bool isValidPointer(NodeID nodeId) const
Whether a node is a valid pointer.
Definition SVFIR.cpp:658
BranchStmt * addBranchStmt(NodeID br, NodeID cond, const BranchStmt::SuccAndCondPairVec &succs)
Add BranchStmt.
Definition SVFIR.cpp:203
void initialiseCandidatePointers()
Initialize candidate pointers.
Definition SVFIR.cpp:642
NodeID getFIObjVar(const BaseObjVar *obj) const
Get a field-insensitive obj SVFIR node according to a mem obj.
Definition SVFIR.h:449
GenericNode< SVFVar, SVFStmt >::GEdgeSetTy SVFStmtSetTy
NodeID getId() const
Get ID.
Definition SVFValue.h:158
virtual bool isPointer() const
Check if this variable represents a pointer.
bool hasIncomingVariantGepEdge() const
Check for incoming variable field GEP edges.
void set(unsigned Idx)
bool isArgOfUncalledFunction(const SVFVar *svfvar)
Definition SVFUtil.cpp:422
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
u32_t NodeID
Definition GeneralType.h:56
s64_t APOffset
Definition GeneralType.h:60
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
unsigned u32_t
Definition GeneralType.h:47