Static Value-Flow Analysis
Loading...
Searching...
No Matches
PointerAnalysis.cpp
Go to the documentation of this file.
1//===- PointerAnalysis.cpp -- Base class of pointer analyses------------------//
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 * PointerAnalysis.cpp
25 *
26 * Created on: May 14, 2013
27 * Author: Yulei Sui
28 */
29
30#include "Util/Options.h"
31#include "Util/SVFUtil.h"
32
35#include "Util/PTAStat.h"
37#include "Graphs/ICFG.h"
38#include "Graphs/CallGraph.h"
40
41#include <iomanip>
42#include <iostream>
43#include <fstream>
44#include <sstream>
45
46using namespace SVF;
47using namespace SVFUtil;
48
49
51
52const std::string PointerAnalysis::aliasTestMayAlias = "MAYALIAS";
53const std::string PointerAnalysis::aliasTestMayAliasMangled = "_Z8MAYALIASPvS_";
54const std::string PointerAnalysis::aliasTestNoAlias = "NOALIAS";
55const std::string PointerAnalysis::aliasTestNoAliasMangled = "_Z7NOALIASPvS_";
56const std::string PointerAnalysis::aliasTestPartialAlias = "PARTIALALIAS";
57const std::string PointerAnalysis::aliasTestPartialAliasMangled = "_Z12PARTIALALIASPvS_";
58const std::string PointerAnalysis::aliasTestMustAlias = "MUSTALIAS";
59const std::string PointerAnalysis::aliasTestMustAliasMangled = "_Z9MUSTALIASPvS_";
60const std::string PointerAnalysis::aliasTestFailMayAlias = "EXPECTEDFAIL_MAYALIAS";
61const std::string PointerAnalysis::aliasTestFailMayAliasMangled = "_Z21EXPECTEDFAIL_MAYALIASPvS_";
62const std::string PointerAnalysis::aliasTestFailNoAlias = "EXPECTEDFAIL_NOALIAS";
63const std::string PointerAnalysis::aliasTestFailNoAliasMangled = "_Z20EXPECTEDFAIL_NOALIASPvS_";
64
78
83{
84 destroy();
85 // do not delete the SVFIR for now
86 //delete pag;
87}
88
89
91{
92 delete callgraph;
93 callgraph = nullptr;
94
95 delete callGraphSCC;
96 callGraphSCC = nullptr;
97
98 delete stat;
99 stat = nullptr;
100}
101
106{
107 assert(pag && "SVFIR has not been built!");
108
109 chgraph = pag->getCHG();
110
113 {
115 callgraph = bd.buildThreadCallGraph();
116 }
117 else
118 {
120 callgraph = bd.buildPTACallGraph();
121 }
123
124 // dump callgraph
126 getCallGraph()->dump("callgraph_initial");
127}
128
129
134{
136 assert(baseObjVar && "base object not found!!");
137 if(SVFUtil::isa<StackObjVar>(baseObjVar))
138 {
139 if(const FunObjVar* svffun = pag->getGNode(id)->getFunction())
140 {
141 return callGraphSCC->isInCycle(getCallGraph()->getCallGraphNode(svffun)->getId());
142 }
143 }
144 return false;
145}
146
151{
152 for (SVFIR::iterator nIter = pag->begin(); nIter != pag->end(); ++nIter)
153 {
154 if(ObjVar* node = SVFUtil::dyn_cast<ObjVar>(nIter->second))
155 const_cast<BaseObjVar*>(pag->getBaseObject(node->getId()))->setFieldSensitive();
156 }
157}
158
159/*
160 * Dump statistics
161 */
162
164{
165
166 if(print_stat && stat)
167 {
168 stat->performStat();
169 }
170}
171
177{
178
180 dumpStat();
181
183 if (Options::PTSPrint())
184 {
186 //dumpAllPts();
187 //dumpCPts();
188 }
189
190 if (Options::TypePrint())
191 dumpAllTypes();
192
194 dumpAllPts();
195
198
200
202 getCallGraph()->dump("callgraph_final");
203
206
209}
210
230
231
233{
234 for (OrderedNodeSet::iterator nIter = this->getAllValidPtrs().begin();
235 nIter != this->getAllValidPtrs().end(); ++nIter)
236 {
237 const PAGNode* node = getPAG()->getGNode(*nIter);
238 if (SVFUtil::isa<DummyObjVar, DummyValVar>(node))
239 continue;
240
241 outs() << "##<" << node->getName() << "> ";
242 outs() << "Source Loc: " << node->getSourceLoc();
243 outs() << "\nNodeID " << node->getId() << "\n";
244
245 const SVFType* type = node->getType();
247 }
248}
249
254{
255
256 const PAGNode* node = pag->getGNode(ptr);
258 if (SVFUtil::isa<DummyObjVar> (node))
259 {
260 outs() << "##<Dummy Obj > id:" << node->getId();
261 }
262 else if (!SVFUtil::isa<DummyValVar>(node) && !SVFIR::pagReadFromTXT())
263 {
264 outs() << "##<" << node->getName() << "> ";
265 outs() << "Source Loc: " << node->getSourceLoc();
266 }
267 outs() << "\nPtr " << node->getId() << " ";
268
269 if (pts.empty())
270 {
271 outs() << "\t\tPointsTo: {empty}\n\n";
272 }
273 else
274 {
275 outs() << "\t\tPointsTo: { ";
276 for (PointsTo::iterator it = pts.begin(), eit = pts.end(); it != eit;
277 ++it)
278 outs() << *it << " ";
279 outs() << "}\n\n";
280 }
281
282 outs() << "";
283
284 for (PointsTo::iterator it = pts.begin(), eit = pts.end(); it != eit; ++it)
285 {
286 const PAGNode* node = pag->getGNode(*it);
287 if(SVFUtil::isa<ObjVar>(node) == false)
288 continue;
289 NodeID ptd = node->getId();
290 outs() << "!!Target NodeID " << ptd << "\t [";
291 const PAGNode* pagNode = pag->getGNode(ptd);
292 if (SVFUtil::isa<DummyValVar>(node))
293 outs() << "DummyVal\n";
294 else if (SVFUtil::isa<DummyObjVar>(node))
295 outs() << "Dummy Obj id: " << node->getId() << "]\n";
296 else
297 {
299 {
300 outs() << "<" << pagNode->getName() << "> ";
301 outs() << "Source Loc: "
302 << pagNode->getSourceLoc() << "] \n";
303 }
304 }
305 }
306}
307
312{
313 outs() << "\nNodeID: " << getFunPtr(cs);
314 outs() << "\nCallSite: ";
315 outs() << cs->toString();
316 outs() << "\tLocation: " << cs->getSourceLoc();
317 outs() << "\t with Targets: ";
318
319 if (!targets.empty())
320 {
321 FunctionSet::const_iterator fit = targets.begin();
322 FunctionSet::const_iterator feit = targets.end();
323 for (; fit != feit; ++fit)
324 {
325 const FunObjVar* callee = *fit;
326 outs() << "\n\t" << callee->getName();
327 }
328 }
329 else
330 {
331 outs() << "\n\tNo Targets!";
332 }
333
334 outs() << "\n";
335}
336
341{
342 outs() << "==================Function Pointer Targets==================\n";
344 CallEdgeMap::const_iterator it = callEdges.begin();
345 CallEdgeMap::const_iterator eit = callEdges.end();
346 for (; it != eit; ++it)
347 {
348 const CallICFGNode* cs = it->first;
349 const FunctionSet& targets = it->second;
351 }
352
354 CallSiteToFunPtrMap::const_iterator csIt = indCS.begin();
355 CallSiteToFunPtrMap::const_iterator csEit = indCS.end();
356 for (; csIt != csEit; ++csIt)
357 {
358 const CallICFGNode* cs = csIt->first;
359 if (hasIndCSCallees(cs) == false)
360 {
361 outs() << "\nNodeID: " << csIt->second;
362 outs() << "\nCallSite: ";
363 outs() << cs->toString();
364 outs() << "\tLocation: " << cs->getSourceLoc();
365 outs() << "\n\t!!!has no targets!!!\n";
366 }
367 }
368}
369
370
371
376{
377
378 assert(pag->isIndirectCallSites(cs) && "not an indirect callsite?");
380 for (PointsTo::iterator ii = target.begin(), ie = target.end();
381 ii != ie; ii++)
382 {
383
385 {
386 wrnMsg("Resolved Indirect Call Edges are Out-Of-Budget, please increase the limit");
387 return;
388 }
389
390 if(ObjVar* objPN = SVFUtil::dyn_cast<ObjVar>(pag->getGNode(*ii)))
391 {
392 const BaseObjVar* obj = pag->getBaseObject(objPN->getId());
393
394 if(obj->isFunction())
395 {
396 const FunObjVar* calleefun = SVFUtil::cast<FunObjVar>(obj)->getFunction();
398
399 if(SVFUtil::matchArgs(cs, callee) == false)
400 continue;
401
402 if(0 == getIndCallMap()[cs].count(callee))
403 {
404 newEdges[cs].insert(callee);
405 getIndCallMap()[cs].insert(callee);
406
408 // FIXME: do we need to update llvm call graph here?
409 // The indirect call is maintained by ourself, We may update llvm's when we need to
410 //PTACallGraphNode* callgraphNode = callgraph->getOrInsertFunction(cs.getCaller());
411 //callgraphNode->addCalledFunction(cs,callgraph->getOrInsertFunction(callee));
412 }
413 }
414 }
415 }
416}
417
418/*
419 * Get virtual functions "vfns" based on CHA
420 */
426
427/*
428 * Get virtual functions "vfns" from PoninsTo set "target" for callsite "cs"
429 */
431{
432
434 {
437 for (PointsTo::iterator it = target.begin(), eit = target.end(); it != eit; ++it)
438 {
439 const PAGNode *ptdnode = pag->getGNode(*it);
440 const GlobalObjVar* pVar = nullptr;
442 {
444
445 }
446 else if (isa<ValVar>(ptdnode) &&
448 pag->getBaseValVar(ptdnode->getId())))
449 {
452 pag->getBaseValVar(ptdnode->getId()))));
453 }
454
455 if (pVar && chaVtbls.find(pVar) != chaVtbls.end())
456 vtbls.insert(pVar);
457 }
459 }
460}
461
462/*
463 * Connect callsite "cs" to virtual functions in "vfns"
464 */
466{
468 for (VFunSet::const_iterator fit = vfns.begin(),
469 feit = vfns.end(); fit != feit; ++fit)
470 {
471 const FunObjVar* callee = *fit;
473 if (getIndCallMap()[cs].count(callee) > 0)
474 continue;
475 if(cs->arg_size() == callee->arg_size() ||
476 (cs->isVarArg() && callee->isVarArg()))
477 {
478 newEdges[cs].insert(callee);
479 getIndCallMap()[cs].insert(callee);
480 const CallICFGNode* callBlockNode = cs;
481 callgraph->addIndirectCallGraphEdge(callBlockNode, cs->getCaller(),callee);
482 }
483 }
484}
485
488{
489 assert(cs->isVirtualCall() && "not cpp virtual call");
490
493 getVFnsFromCHA(cs, vfns);
494 else
497}
498
504{
505 // check for must alias cases, whether our alias analysis produce the correct results
506 if (const FunObjVar* checkFun = pag->getFunObjVar(fun))
507 {
508 if(!checkFun->isUncalledFunction())
509 outs() << "[" << this->PTAName() << "] Checking " << fun << "\n";
510
511 for(const CallICFGNode* callNode : pag->getCallSiteSet())
512 {
513 if (callNode->getCalledFunction() == checkFun)
514 {
515 assert(callNode->getNumArgOperands() == 2
516 && "arguments should be two pointers!!");
517 const SVFVar* V1 = callNode->getArgument(0);
518 const SVFVar* V2 = callNode->getArgument(1);
519 AliasResult aliasRes = alias(V1->getId(), V2->getId());
520
521 bool checkSuccessful = false;
522 if (fun == aliasTestMayAlias || fun == aliasTestMayAliasMangled)
523 {
525 checkSuccessful = true;
526 }
527 else if (fun == aliasTestNoAlias || fun == aliasTestNoAliasMangled)
528 {
530 checkSuccessful = true;
531 }
532 else if (fun == aliasTestMustAlias || fun == aliasTestMustAliasMangled)
533 {
534 // change to must alias when our analysis support it
536 checkSuccessful = true;
537 }
538 else if (fun == aliasTestPartialAlias || fun == aliasTestPartialAliasMangled)
539 {
540 // change to partial alias when our analysis support it
542 checkSuccessful = true;
543 }
544 else
545 assert(false && "not supported alias check!!");
546
547 NodeID id1 = V1->getId();
548 NodeID id2 = V2->getId();
549
550 if (checkSuccessful)
551 outs() << sucMsg("\t SUCCESS :") << fun << " check <id:" << id1 << ", id:" << id2 << "> at ("
552 << callNode->getSourceLoc() << ")\n";
553 else
554 {
555 SVFUtil::errs() << errMsg("\t FAILURE :") << fun
556 << " check <id:" << id1 << ", id:" << id2
557 << "> at (" << callNode->getSourceLoc() << ")\n";
558 assert(false && "test case failed!");
559 }
560 }
561 }
562 }
563}
564
569{
570
571 if (const FunObjVar* checkFun = pag->getFunObjVar(fun))
572 {
573 if(!checkFun->isUncalledFunction())
574 outs() << "[" << this->PTAName() << "] Checking " << fun << "\n";
575
576 for(const CallICFGNode* callNode : pag->getCallSiteSet())
577 {
578 if (callNode->getCalledFunction() == checkFun)
579 {
580 assert(callNode->arg_size() == 2
581 && "arguments should be two pointers!!");
582 const SVFVar* V1 = callNode->getArgument(0);
583 const SVFVar* V2 = callNode->getArgument(1);
584 AliasResult aliasRes = alias(V1->getId(), V2->getId());
585
586 bool expectedFailure = false;
588 {
589 // change to must alias when our analysis support it
591 expectedFailure = true;
592 }
593 else if (fun == aliasTestFailNoAlias || fun == aliasTestFailNoAliasMangled)
594 {
595 // change to partial alias when our analysis support it
597 expectedFailure = true;
598 }
599 else
600 assert(false && "not supported alias check!!");
601
602 NodeID id1 = V1->getId();
603 NodeID id2 = V2->getId();
604
605 if (expectedFailure)
606 outs() << sucMsg("\t EXPECTED-FAILURE :") << fun << " check <id:" << id1 << ", id:" << id2 << "> at ("
607 << callNode->getSourceLoc() << ")\n";
608 else
609 {
610 SVFUtil::errs() << errMsg("\t UNEXPECTED FAILURE :") << fun << " check <id:" << id1 << ", id:" << id2 << "> at ("
611 << callNode->getSourceLoc() << ")\n";
612 assert(false && "test case failed!");
613 }
614 }
615 }
616 }
617}
cJSON * p
Definition cJSON.cpp:2559
newitem type
Definition cJSON.cpp:2739
int count
Definition cJSON.h:216
void addIndirectCallGraphEdge(const CallICFGNode *cs, const FunObjVar *callerFun, const FunObjVar *calleeFun)
Add indirect call edges.
void dump(const std::string &filename)
Dump the graph.
void verifyCallGraph()
Issue a warning if the function which has indirect call sites can not be reached from program entry.
const std::string toString() const override
Definition ICFG.cpp:139
bool isVarArg() const
Definition ICFGNode.h:517
const std::string getSourceLoc() const override
Definition ICFGNode.h:582
bool isVirtualCall() const
Definition ICFGNode.h:521
const FunObjVar * getCaller() const
Return callsite.
Definition ICFGNode.h:464
u32_t arg_size() const
Definition ICFGNode.h:499
virtual const VFunSet & getCSVFsBasedonCHA(const CallICFGNode *cs)=0
virtual bool csHasVFnsBasedonCHA(const CallICFGNode *cs)=0
virtual const VTableSet & getCSVtblsBasedonCHA(const CallICFGNode *cs)=0
virtual void getVFnsFromVtbls(const CallICFGNode *cs, const VTableSet &vtbls, VFunSet &virtualFunctions)=0
virtual bool csHasVtblsBasedonCHA(const CallICFGNode *cs)=0
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
const FunObjVar * getDefFunForMultipleModule() const
iterator begin()
Iterators.
IDToNodeMapTy::iterator iterator
Node Iterators.
NodeType * getGNode(NodeID id) const
Get a node.
void printFlattenFields(const SVFType *type)
Debug method.
Definition IRGraph.cpp:73
bool isBuiltFromFile()
Whether this SVFIR built from a txt file.
Definition IRGraph.h:150
static const Option< bool > CallGraphDotGraph
Definition Options.h:126
static const Option< bool > EnableThreadCallGraph
Definition Options.h:132
static const Option< bool > PTSPrint
Definition Options.h:116
static const Option< bool > EnableAliasCheck
Definition Options.h:130
static const Option< bool > PTSAllPrint
Definition Options.h:117
static const Option< bool > TypePrint
Definition Options.h:114
static Option< bool > UsePreCompFieldSensitive
Definition Options.h:129
static const Option< u32_t > IndirectCallLimit
Definition Options.h:128
static const Option< bool > FuncPointerPrint
Definition Options.h:115
static const Option< bool > ConnectVCallOnCHA
Definition Options.h:133
static const Option< bool > PStat
Definition Options.h:119
static const Option< u32_t > StatBudget
Definition Options.h:120
void performStat() override
Definition PTAStat.cpp:52
void getVFnsFromPts(const CallICFGNode *cs, const PointsTo &target, VFunSet &vfns)
void destroy()
Release the memory.
virtual void validateTests()
Alias check functions to verify correctness of pointer analysis.
PTATY
Pointer analysis type list.
CommonCHGraph * chgraph
CHGraph.
static const std::string aliasTestNoAliasMangled
bool isLocalVarInRecursiveFun(NodeID id) const
Whether a local variable is in function recursions.
virtual void finalize()
Finalization of a pointer analysis, including checking alias correctness.
static const std::string aliasTestMayAliasMangled
PointerAnalysis(SVFIR *pag, PTATY ty=Default_PTA, bool alias_check=true)
Constructor.
static const std::string aliasTestFailNoAlias
virtual void dumpPts(NodeID ptr, const PointsTo &pts)
static const std::string aliasTestFailMayAlias
bool print_stat
User input flags.
OrderedMap< const CallICFGNode *, FunctionSet > CallEdgeMap
Set< const GlobalObjVar * > VTableSet
virtual void initialize()
Initialization of a pointer analysis, including building symbol table and SVFIR etc.
virtual ~PointerAnalysis()
Destructor.
PTAImplTy ptaImplTy
PTA implementation type.
PTAStat * stat
Statistics.
virtual void dumpTopLevelPtsTo()
static const std::string aliasTestFailMayAliasMangled
SVFIR * getPAG() const
void connectVCallToVFns(const CallICFGNode *cs, const VFunSet &vfns, CallEdgeMap &newEdges)
CallGraph * getCallGraph() const
Return call graph.
OrderedNodeSet & getAllValidPtrs()
Get all Valid Pointers for resolution.
void resetObjFieldSensitive()
Reset all object node as field-sensitive.
static const std::string aliasTestMustAlias
static const std::string aliasTestMayAlias
Set< const FunObjVar * > FunctionSet
virtual void validateSuccessTests(std::string fun)
const CallSiteToFunPtrMap & getIndirectCallsites() const
Return all indirect callsites.
static const std::string aliasTestPartialAlias
virtual void dumpAllPts()
virtual AliasResult alias(const SVFVar *V1, const SVFVar *V2)=0
Interface exposed to users of our pointer analysis, given Value infos.
virtual void resolveIndCalls(const CallICFGNode *cs, const PointsTo &target, CallEdgeMap &newEdges)
Resolve indirect call edges.
CallGraph * callgraph
Call graph used for pointer analysis.
bool alias_validation
Flag for validating points-to/alias results.
void callGraphSCCDetection()
PTACallGraph SCC related methods.
void dumpStat()
Dump the statistics.
virtual void validateExpectedFailureTests(std::string fun)
bool hasIndCSCallees(const CallICFGNode *cs) const
virtual void resolveCPPIndCalls(const CallICFGNode *cs, const PointsTo &target, CallEdgeMap &newEdges)
Resolve cpp indirect call edges.
@ BaseImpl
Represents PointerAnalaysis.
CallEdgeMap & getIndCallMap()
Get callees from an indirect callsite.
static const std::string aliasTestNoAlias
static const std::string aliasTestPartialAliasMangled
u32_t getNumOfResolvedIndCallEdge() const
Return number of resolved indirect call edges.
SVFIR::CallSiteToFunPtrMap CallSiteToFunPtrMap
NodeID getFunPtr(const CallICFGNode *cs) const
Return function pointer PAGNode at a callsite cs.
static SVFIR * pag
SVFIR.
void getVFnsFromCHA(const CallICFGNode *cs, VFunSet &vfns)
CallGraphSCC * callGraphSCC
SCC for PTACallGraph.
static const std::string aliasTestMustAliasMangled
virtual const std::string PTAName() const
Return PTA name.
static const std::string aliasTestFailNoAliasMangled
Set< const FunObjVar * > VFunSet
u32_t OnTheFlyIterBudgetForStat
Flag for iteration budget for on-the-fly statistics.
const CallSiteSet & getCallSiteSet() const
Get all callsites.
Definition SVFIR.h:282
bool isIndirectCallSites(const CallICFGNode *cs) const
Definition SVFIR.h:394
static bool pagReadFromTXT()
Definition SVFIR.h:211
const BaseObjVar * getBaseObject(NodeID id) const
Definition SVFIR.h:423
const FunObjVar * getFunObjVar(const std::string &name)
Definition SVFIR.cpp:47
CommonCHGraph * getCHG()
Definition SVFIR.h:173
const ValVar * getBaseValVar(NodeID id) const
Definition SVFIR.h:433
NodeID getId() const
Get ID.
Definition SVFValue.h:158
virtual const SVFType * getType() const
Definition SVFValue.h:169
virtual const std::string getSourceLoc() const
Definition SVFValue.h:194
virtual const std::string & getName() const
Definition SVFValue.h:184
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
std::string sucMsg(const std::string &msg)
Returns successful message by converting a string into green string output.
Definition SVFUtil.cpp:55
const ObjVar * getObjVarOfValVar(const ValVar *valVar)
Definition SVFUtil.cpp:431
std::string errMsg(const std::string &msg)
Print error message by converting a string into red string output.
Definition SVFUtil.cpp:78
std::ostream & errs()
Overwrite llvm::errs()
Definition SVFUtil.h:58
bool matchArgs(const CallICFGNode *cs, const FunObjVar *callee)
Definition SVFUtil.cpp:308
std::string wrnMsg(const std::string &msg)
Returns warning message by converting a string into yellow string output.
Definition SVFUtil.cpp:63
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
u32_t NodeID
Definition GeneralType.h:56
AliasResult
Definition SVFType.h:541
@ PartialAlias
Definition SVFType.h:545
@ MustAlias
Definition SVFType.h:544
@ MayAlias
Definition SVFType.h:543
@ NoAlias
Definition SVFType.h:542
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74