Static Value-Flow Analysis
Loading...
Searching...
No Matches
FlowSensitive.cpp
Go to the documentation of this file.
1//===- FlowSensitive.cpp -- Sparse flow-sensitive pointer analysis------------//
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 * FlowSensitive.cpp
25 *
26 * Created on: Oct 28, 2013
27 * Author: Yulei Sui
28 */
29
30#include "Util/Options.h"
31#include "WPA/WPAStat.h"
32#include "WPA/FlowSensitive.h"
33#include "WPA/Andersen.h"
35
36using namespace SVF;
37using namespace SVFUtil;
38
39std::unique_ptr<FlowSensitive> FlowSensitive::fspta;
40
45{
47
48 stat = new FlowSensitiveStat(this);
49
50 // TODO: support clustered aux. Andersen's.
51 assert(!Options::ClusterAnder() && "FlowSensitive::initialize: clustering auxiliary Andersen's unsupported.");
53
54 // If cluster option is not set, it will give us a no-mapping points-to set.
56 && "FS::init: plain-mapping and cluster-fs are mutually exclusive.");
58 {
59 cluster();
60 // Reset the points-to cache although empty so the new mapping could
61 // be applied to the inserted empty set.
62 getPtCache().reset();
63 }
64 else if (Options::PlainMappingFs())
65 {
66 plainMap();
67 // As above.
68 getPtCache().reset();
69 }
70
72
74 //AndersenWaveDiff::releaseAndersenWaveDiff();
75}
77{
79
80 double start = stat->getClk(true);
82 DBOUT(DGENERAL, outs() << SVFUtil::pasMsg("Start Solving Constraints\n"));
83
84 do
85 {
87
89 dumpStat();
90
91 callGraphSCC->find();
92
95 }
97
98 DBOUT(DGENERAL, outs() << SVFUtil::pasMsg("Finish Solving Constraints\n"));
99
100 // Reset the time-up alarm; analysis is done.
102
103 double end = stat->getClk(true);
104 solveTime += (end - start) / TIMEINTERVAL;
105
106}
107
112{
114 initialize();
115 if(!filename.empty())
118 if(!filename.empty())
121 finalize();
122}
123
128{
129 if(!Options::ReadAnder().empty())
130 {
132 }
133 else
134 {
135 if(Options::WriteAnder().empty())
136 {
137 initialize();
139 finalize();
140 }
141 else
142 {
144 }
145 }
146}
147
149{
151 initialize();
153 if(!filename.empty())
154 this->readFromFile(filename);
156 finalize();
157}
158
163{
164 if(Options::DumpVFG())
165 svfg->dump("fs_solved", true);
166
168 while (nodeStack.empty() == false)
169 {
170 NodeID rep = nodeStack.top();
171 nodeStack.pop();
172 const NodeBS& subNodes = getSCCDetector()->subNodes(rep);
173 if (subNodes.count() > maxSCCSize)
174 maxSCCSize = subNodes.count();
175 if (subNodes.count() > 1)
176 {
177 numOfNodesInSCC += subNodes.count();
178 numOfSCC++;
179 }
180 }
181
182 // TODO: check -stat too.
183 if (Options::ClusterFs())
184 {
186 const PTDataTy *ptd = getPTDataTy();
187 // TODO: should we use liveOnly?
188 Map<PointsTo, unsigned> allPts = ptd->getAllPts(true);
189 // TODO: parameterise final arg.
192
193 // Do the same for the candidates. TODO: probably temporary for eval. purposes.
194 for (std::pair<hclust_fast_methods, std::vector<NodeID>> &candidate : candidateMappings)
195 {
196 // Can reuse stats, since we're always filling it with `evaluate`, it will always be overwritten.
199 }
200 }
201
203}
204
209{
210 double start = stat->getClk();
212 double end = stat->getClk();
213 sccTime += (end - start) / TIMEINTERVAL;
214 return nodeStack;
215}
216
221{
223 if (processSVFGNode(node))
224 propagate(&node);
225
227}
228
233{
234 double start = stat->getClk();
235 bool changed = false;
236 if (AddrSVFGNode* addr = SVFUtil::dyn_cast<AddrSVFGNode>(node))
237 {
239 if (processAddr(addr))
240 changed = true;
241 }
242 else if (CopySVFGNode* copy = SVFUtil::dyn_cast<CopySVFGNode>(node))
243 {
245 if (processCopy(copy))
246 changed = true;
247 }
248 else if (GepSVFGNode* gep = SVFUtil::dyn_cast<GepSVFGNode>(node))
249 {
251 if(processGep(gep))
252 changed = true;
253 }
254 else if (LoadSVFGNode* load = SVFUtil::dyn_cast<LoadSVFGNode>(node))
255 {
257 if(processLoad(load))
258 changed = true;
259 }
260 else if (StoreSVFGNode* store = SVFUtil::dyn_cast<StoreSVFGNode>(node))
261 {
263 if (processStore(store))
264 changed = true;
265 }
266 else if (PHISVFGNode* phi = SVFUtil::dyn_cast<PHISVFGNode>(node))
267 {
269 if (processPhi(phi))
270 changed = true;
271 }
274 ActualOUTSVFGNode>(node))
275 {
277 changed = true;
278 }
281 NullPtrSVFGNode>(node))
282 {
283 changed = true;
284 }
285 else if (SVFUtil::isa<CmpVFGNode, BinaryOPVFGNode>(node) ||
286 SVFUtil::dyn_cast<UnaryOPVFGNode>(node))
287 {
288 }
289 else
290 {
291 assert(false && "unexpected kind of SVFG nodes");
292 }
293
294 double end = stat->getClk();
295 processTime += (end - start) / TIMEINTERVAL;
296
297 return changed;
298}
299
309{
310 double start = stat->getClk();
311 bool changed = false;
312
313 if (DirectSVFGEdge* dirEdge = SVFUtil::dyn_cast<DirectSVFGEdge>(edge))
315 else if (IndirectSVFGEdge* indEdge = SVFUtil::dyn_cast<IndirectSVFGEdge>(edge))
317 else
318 assert(false && "new kind of svfg edge?");
319
320 double end = stat->getClk();
322 return changed;
323}
324
329{
330 double start = stat->getClk();
331 bool changed = false;
332
333 SVFGNode* src = edge->getSrcNode();
334 SVFGNode* dst = edge->getDstNode();
335 // If this is an actual-param or formal-ret, top-level pointer's pts must be
336 // propagated from src to dst.
337 if (ActualParmSVFGNode* ap = SVFUtil::dyn_cast<ActualParmSVFGNode>(src))
338 changed = propagateFromAPToFP(ap, dst);
339 else if (FormalRetSVFGNode* fp = SVFUtil::dyn_cast<FormalRetSVFGNode>(src))
341 else
342 {
343 // Direct SVFG edge links between def and use of a top-level pointer.
344 // There's no points-to information propagated along direct edge.
345 // Since the top-level pointer's value has been changed at src node,
346 // return TRUE to put dst node into the work list.
347 changed = true;
348 }
349
350 double end = stat->getClk();
352 return changed;
353}
354
360{
361 const FormalParmSVFGNode* fp = SVFUtil::dyn_cast<FormalParmSVFGNode>(dst);
362 assert(fp && "expecting a formal param node");
363
364 NodeID pagDst = fp->getParam()->getId();
365 const PointsTo &srcCPts = getPts(ap->getParam()->getId());
367
368 return changed;
369}
370
376{
377 const ActualRetSVFGNode* ar = SVFUtil::dyn_cast<ActualRetSVFGNode>(dst);
378 assert(ar && "expecting an actual return node");
379
380 NodeID pagDst = ar->getRev()->getId();
381 const PointsTo & srcCPts = getPts(fr->getRet()->getId());
383
384 return changed;
385}
386
391{
392 double start = stat->getClk();
393
394 SVFGNode* src = edge->getSrcNode();
395 SVFGNode* dst = edge->getDstNode();
396
397 bool changed = false;
398
399 // Get points-to targets may be used by next SVFG node.
400 // Propagate points-to set for node used in dst.
401 const NodeBS& pts = edge->getPointsTo();
402 for (NodeBS::iterator ptdIt = pts.begin(), ptdEit = pts.end(); ptdIt != ptdEit; ++ptdIt)
403 {
404 NodeID ptd = *ptdIt;
405
406 if (propVarPtsFromSrcToDst(ptd, src, dst))
407 changed = true;
408
410 {
413 for (NodeBS::iterator fieldIt = allFields.begin(), fieldEit = allFields.end();
415 {
416 if (propVarPtsFromSrcToDst(*fieldIt, src, dst))
417 changed = true;
418 }
419 }
420 }
421
422 double end = stat->getClk();
424 return changed;
425}
426
431{
432 bool changed = false;
433 if (SVFUtil::isa<StoreSVFGNode>(src))
434 {
435 if (updateInFromOut(src, var, dst, var))
436 changed = true;
437 }
438 else
439 {
440 if (updateInFromIn(src, var, dst, var))
441 changed = true;
442 }
443 return changed;
444}
445
450{
451 double start = stat->getClk();
452 NodeID srcID = addr->getPAGSrcNodeID();
457 bool changed = addPts(addr->getPAGDstNodeID(), srcID);
458 double end = stat->getClk();
459 addrTime += (end - start) / TIMEINTERVAL;
460 return changed;
461}
462
467{
468 double start = stat->getClk();
469 bool changed = unionPts(copy->getPAGDstNodeID(), copy->getPAGSrcNodeID());
470 double end = stat->getClk();
471 copyTime += (end - start) / TIMEINTERVAL;
472 return changed;
473}
474
479{
480 double start = stat->getClk();
481 bool changed = false;
482 NodeID pagDst = phi->getRes()->getId();
483 for (PHISVFGNode::OPVers::const_iterator it = phi->opVerBegin(), eit = phi->opVerEnd(); it != eit; ++it)
484 {
485 NodeID src = it->second->getId();
486 const PointsTo& srcPts = getPts(src);
487 if (unionPts(pagDst, srcPts))
488 changed = true;
489 }
490
491 double end = stat->getClk();
492 phiTime += (end - start) / TIMEINTERVAL;
493 return changed;
494}
495
500{
501 double start = stat->getClk();
502 bool changed = false;
503 const PointsTo& srcPts = getPts(edge->getPAGSrcNodeID());
504
506 const GepStmt* gepStmt = SVFUtil::cast<GepStmt>(edge->getPAGEdge());
507 if (gepStmt->isVariantFieldGep())
508 {
509 for (NodeID o : srcPts)
510 {
512 {
513 tmpDstPts.set(o);
514 continue;
515 }
516
518 tmpDstPts.set(getFIObjVar(o));
519 }
520 }
521 else
522 {
523 for (NodeID o : srcPts)
524 {
526 {
527 tmpDstPts.set(o);
528 continue;
529 }
530
531 NodeID fieldSrcPtdNode = getGepObjVar(o, gepStmt->getAccessPath().getConstantStructFldIdx());
533 }
534 }
535
536 if (unionPts(edge->getPAGDstNodeID(), tmpDstPts))
537 changed = true;
538
539 double end = stat->getClk();
540 gepTime += (end - start) / TIMEINTERVAL;
541 return changed;
542}
543
544
552{
553 double start = stat->getClk();
554 bool changed = false;
555
556 NodeID dstVar = load->getPAGDstNodeID();
557
558 const PointsTo& srcPts = getPts(load->getPAGSrcNodeID());
559
560 // p = *q, the type of p must be a pointer
561 if(load->getPAGDstNode()->isPointer())
562 {
563 for (PointsTo::iterator ptdIt = srcPts.begin(); ptdIt != srcPts.end(); ++ptdIt)
564 {
565 NodeID ptd = *ptdIt;
566
567 if (pag->isConstantObj(ptd))
568 continue;
569
570 if (unionPtsFromIn(load, ptd, dstVar))
571 changed = true;
572
574 {
578 for (NodeBS::iterator fieldIt = allFields.begin(), fieldEit = allFields.end();
580 {
581 if (unionPtsFromIn(load, *fieldIt, dstVar))
582 changed = true;
583 }
584 }
585 }
586 }
587 double end = stat->getClk();
588 loadTime += (end - start) / TIMEINTERVAL;
589 return changed;
590}
591
599{
600
601 const PointsTo & dstPts = getPts(store->getPAGDstNodeID());
602
609 if (dstPts.empty())
610 return false;
611
612 double start = stat->getClk();
613 bool changed = false;
614
615 // *p = q, the type of q must be a pointer
616 if(getPts(store->getPAGSrcNodeID()).empty() == false && store->getPAGSrcNode()->isPointer())
617 {
618 for (PointsTo::iterator it = dstPts.begin(), eit = dstPts.end(); it != eit; ++it)
619 {
620 NodeID ptd = *it;
621
622 if (pag->isConstantObj(ptd))
623 continue;
624
625 if (unionPtsFromTop(store, store->getPAGSrcNodeID(), ptd))
626 changed = true;
627 }
628 }
629
630 double end = stat->getClk();
631 storeTime += (end - start) / TIMEINTERVAL;
632
633 double updateStart = stat->getClk();
634 // also merge the DFInSet to DFOutSet.
637 bool isSU = isStrongUpdate(store, singleton);
638 if (isSU)
639 {
640 svfgHasSU.set(store->getId());
642 changed = true;
643 }
644 else
645 {
646 svfgHasSU.reset(store->getId());
647 if (weakUpdateOutFromIn(store))
648 changed = true;
649 }
650 double updateEnd = stat->getClk();
652
653 return changed;
654}
655
660{
661 bool isSU = false;
662 if (const StoreSVFGNode* store = SVFUtil::dyn_cast<StoreSVFGNode>(node))
663 {
664 const PointsTo& dstCPSet = getPts(store->getPAGDstNodeID());
665 if (dstCPSet.count() == 1)
666 {
669 singleton = *it;
670
671 // Strong update can be made if this points-to target is not heap, array or field-insensitive.
673 {
677 {
678 isSU = true;
679 }
680 }
681 }
682 }
683 return isSU;
684}
685
690{
691 double start = stat->getClk();
694
695 // Bound the new edges by the Andersen's call graph.
696 // TODO: we want this to be an assertion eventually.
698 for (typename CallEdgeMap::value_type &csfs : newEdges)
699 {
700 const CallICFGNode *potentialCallSite = csfs.first;
702
703 // Check this callsite even calls anything per Andersen's.
704 typename CallEdgeMap::const_iterator andersFunctionSetIt
707 {
708 potentialFunctionSet.clear();
709 }
710
712 for (FunctionSet::iterator potentialFunctionIt = potentialFunctionSet.begin();
714 {
717 {
718 // potentialFunction is not in the Andersen's call graph -- remove it.
720 }
721 else
722 {
723 // potentialFunction is in the Andersen's call graph -- keep it..
725 }
726 }
727 }
728
731
733
734 double end = stat->getClk();
736 return (!newEdges.empty());
737}
738
743{
744 CallEdgeMap::const_iterator iter = newEdges.begin();
745 CallEdgeMap::const_iterator eiter = newEdges.end();
746 for (; iter != eiter; iter++)
747 {
748 const CallICFGNode* cs = iter->first;
749 const FunctionSet & functions = iter->second;
750 for (FunctionSet::const_iterator func_iter = functions.begin(); func_iter != functions.end(); func_iter++)
751 {
752 const FunObjVar* func = *func_iter;
754 }
755 }
756}
757
763{
764 for (const SVFGEdge* edge : edges)
765 {
766 SVFGNode* dstNode = edge->getDstNode();
767 if (SVFUtil::isa<PHISVFGNode>(dstNode))
768 {
771 pushIntoWorklist(dstNode->getId());
772 }
773 else if (SVFUtil::isa<FormalINSVFGNode, ActualOUTSVFGNode>(dstNode))
774 {
777 bool changed = false;
778
779 SVFGNode* srcNode = edge->getSrcNode();
780
781 const NodeBS& pts = SVFUtil::cast<IndirectSVFGEdge>(edge)->getPointsTo();
782 for (NodeBS::iterator ptdIt = pts.begin(), ptdEit = pts.end(); ptdIt != ptdEit; ++ptdIt)
783 {
784 NodeID ptd = *ptdIt;
785
787 changed = true;
788
790 {
793 for (NodeBS::iterator fieldIt = allFields.begin(), fieldEit = allFields.end();
795 {
797 changed = true;
798 }
799 }
800 }
801
802 if (changed)
803 pushIntoWorklist(dstNode->getId());
804 }
805 }
806}
807
808
813{
814 if (SVFUtil::isa<StoreSVFGNode>(src))
815 {
816 if (propDFOutToIn(src, var, dst, var))
817 return true;
818 }
819 else
820 {
821 if (propDFInToIn(src, var, dst, var))
822 return true;
823 }
824 return false;
825}
826
828{
829 std::vector<std::pair<unsigned, unsigned>> keys;
830 for (const auto& pair : *pag)
831 keys.emplace_back(pair.first, 1);
832
833 PointsTo::MappingPtr nodeMapping =
834 std::make_shared<std::vector<NodeID>>(NodeIDAllocator::Clusterer::cluster(ander, keys, candidateMappings, "aux-ander"));
835 PointsTo::MappingPtr reverseNodeMapping =
836 std::make_shared<std::vector<NodeID>>(NodeIDAllocator::Clusterer::getReverseNodeMapping(*nodeMapping));
837
838 PointsTo::setCurrentBestNodeMapping(nodeMapping, reverseNodeMapping);
839}
840
842{
844 && "FS::cluster: plain mapping requires dense allocation strategy.");
845
846 const size_t numObjects = NodeIDAllocator::get()->getNumObjects();
847 PointsTo::MappingPtr plainMapping = std::make_shared<std::vector<NodeID>>(numObjects);
848 PointsTo::MappingPtr reversePlainMapping = std::make_shared<std::vector<NodeID>>(numObjects);
849 for (NodeID i = 0; i < plainMapping->size(); ++i)
850 {
851 plainMapping->at(i) = i;
852 reversePlainMapping->at(i) = i;
853 }
854
856}
857
858void FlowSensitive::countAliases(Set<std::pair<NodeID, NodeID>> cmp, unsigned *mayAliases, unsigned *noAliases)
859{
860 for (std::pair<NodeID, NodeID> locPA : cmp)
861 {
862 // loc doesn't make a difference for FSPTA.
863 NodeID p = locPA.second;
864 for (std::pair<NodeID, NodeID> locPB : cmp)
865 {
866 if (locPB == locPA) continue;
867
868 NodeID q = locPB.second;
869
870 switch (alias(p, q))
871 {
873 ++(*noAliases);
874 break;
876 ++(*mayAliases);
877 break;
878 default:
879 assert("Not May/NoAlias?");
880 }
881 }
882 }
883
884}
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
Definition SVFType.h:498
#define TIMEINTERVAL
Definition SVFType.h:526
#define DGENERAL
Definition SVFType.h:504
cJSON * p
Definition cJSON.cpp:2559
copy
Definition cJSON.cpp:414
const PAGNode * getParam() const
Return parameter.
Definition VFGNode.h:909
static AndersenWaveDiff * createAndersenWaveDiff(SVFIR *_pag)
Create an singleton instance directly instead of invoking llvm pass manager.
Definition Andersen.h:407
virtual void writeToFile(const std::string &filename)
Interface for analysis result storage on filesystem.
virtual bool readFromFile(const std::string &filename)
virtual void writeObjVarToFile(const std::string &filename)
AliasResult alias(const SVFVar *V1, const SVFVar *V2) override
Interface expose to users of our pointer analysis, given Value infos.
void finalize() override
Finalization of pointer analysis, and normalize points-to information to Bit Vector representation.
virtual void onTheFlyCallGraphSolve(const CallSiteToFunPtrMap &callsites, CallEdgeMap &newEdges)
On the fly call graph construction.
PersistentPointsToCache< PointsTo > & getPtCache()
PTData< NodeID, NodeSet, NodeID, PointsTo > PTDataTy
const PointsTo & getPts(NodeID id) override
virtual bool unionPts(NodeID id, const PointsTo &target)
PTDataTy * getPTDataTy() const
Get points-to data structure.
virtual bool addPts(NodeID id, NodeID ptd)
bool isFieldInsensitive() const
Return true if its field limit is 0.
void processNode(NodeID nodeId) override
Handle various constraints.
u32_t numOfProcessedLoad
Number of processed Phi node.
virtual void solveConstraints()
virtual bool unionPtsFromIn(const SVFGNode *stmt, NodeID srcVar, NodeID dstVar)
virtual void updateConnectedNodes(const SVFGEdgeSetTy &edges)
Update nodes connected during updating call graph.
u32_t numOfProcessedCopy
Number of processed Addr node.
virtual void countAliases(Set< std::pair< NodeID, NodeID > > cmp, unsigned *mayAliases, unsigned *noAliases)
Fills may/noAliases for the location/pointer pairs in cmp.
double gepTime
time of handling gep edges
static std::unique_ptr< FlowSensitive > fspta
double indirectPropaTime
time of points-to propagation of top-level pointers
virtual bool propagateFromAPToFP(const ActualParmSVFGNode *ap, const SVFGNode *dst)
double addrTime
time of handling address edges
virtual bool propagateFromFRToAR(const FormalRetSVFGNode *fr, const SVFGNode *dst)
virtual bool propDFInToIn(const SVFGNode *srcStmt, NodeID srcVar, const SVFGNode *dstStmt, NodeID dstVar)
NodeStack & SCCDetect() override
SCC detection.
double solveTime
time of solve.
bool isStrongUpdate(const SVFGNode *node, NodeID &singleton)
Return TRUE if this is a strong update STORE statement.
virtual void readPtsFromFile(const std::string &filename)
virtual bool unionPtsFromTop(const SVFGNode *stmt, NodeID srcVar, NodeID dstVar)
virtual void plainMap(void) const
Sets the global best mapping as a plain mapping, i.e. n -> n.
AndersenWaveDiff * ander
virtual bool propDFOutToIn(const SVFGNode *srcStmt, NodeID srcVar, const SVFGNode *dstStmt, NodeID dstVar)
u32_t numOfProcessedStore
Number of processed Load node.
SVFG::SVFGEdgeSetTy SVFGEdgeSetTy
void analyze() override
Flow sensitive analysis.
double storeTime
time of store edges
virtual bool updateInFromIn(const SVFGNode *srcStmt, NodeID srcVar, const SVFGNode *dstStmt, NodeID dstVar)
double copyTime
time of handling copy edges
bool propFromSrcToDst(SVFGEdge *edge) override
Propagation.
u32_t numOfProcessedGep
Number of processed Copy node.
friend class FlowSensitiveStat
virtual void cluster(void)
virtual bool strongUpdateOutFromIn(const SVFGNode *node, NodeID singleton)
Handle strong updates.
void finalize() override
Finalize analysis.
void clearAllDFOutVarFlag(const SVFGNode *stmt)
bool processSVFGNode(SVFGNode *node)
virtual bool processLoad(const LoadSVFGNode *load)
bool propVarPtsAfterCGUpdated(NodeID var, const SVFGNode *src, const SVFGNode *dst)
virtual bool processPhi(const PHISVFGNode *phi)
virtual bool processStore(const StoreSVFGNode *store)
u32_t maxSCCSize
Number of processed mssa node.
virtual bool processCopy(const CopySVFGNode *copy)
double loadTime
time of load edges
bool updateCallGraph(const CallSiteToFunPtrMap &callsites) override
Update call graph.
virtual bool weakUpdateOutFromIn(const SVFGNode *node)
Handle weak updates.
virtual bool processAddr(const AddrSVFGNode *addr)
u32_t numOfProcessedPhi
Number of processed Gep node.
double propagationTime
time of points-to propagation.
virtual bool propAlongDirectEdge(const DirectSVFGEdge *edge)
Propagate points-to information along a DIRECT SVFG edge.
void initialize() override
Initialize analysis.
void connectCallerAndCallee(const CallEdgeMap &newEdges, SVFGEdgeSetTy &edges)
Connect nodes in SVFG.
std::vector< std::pair< hclust_fast_methods, std::vector< NodeID > > > candidateMappings
Save candidate mappings for evaluation's sake.
virtual bool processGep(const GepSVFGNode *edge)
double directPropaTime
time of points-to propagation of address-taken objects
double processTime
time of processNode.
u32_t numOfProcessedAddr
Statistics.
virtual bool propVarPtsFromSrcToDst(NodeID var, const SVFGNode *src, const SVFGNode *dst)
Propagate points-to information of a certain variable from src to dst.
virtual bool propAlongIndirectEdge(const IndirectSVFGEdge *edge)
Propagate points-to information along an INDIRECT SVFG edge.
double phiTime
time of phi nodes.
double sccTime
time of SCC detection.
double updateTime
time of strong/weak updates.
virtual bool updateInFromOut(const SVFGNode *srcStmt, NodeID srcVar, const SVFGNode *dstStmt, NodeID dstVar)
u32_t numOfProcessedMSSANode
Number of processed formal ret node.
virtual void solveAndwritePtsToFile(const std::string &filename)
double updateCallGraphTime
time of updating call graph
GEdgeSetTy::const_iterator const_iterator
static std::vector< NodeID > cluster(BVDataPTAImpl *pta, const std::vector< std::pair< NodeID, unsigned > > keys, std::vector< std::pair< hclust_fast_methods, std::vector< NodeID > > > &candidates, std::string evalSubtitle="")
static std::vector< NodeID > getReverseNodeMapping(const std::vector< NodeID > &nodeMapping)
static void printStats(std::string title, Map< std::string, std::string > &stats)
static void evaluate(const std::vector< NodeID > &nodeMap, const Map< PointsTo, unsigned > pointsToSets, Map< std::string, std::string > &stats, bool accountForOcc)
Fills in *NumWords statistics in stats..
static NodeIDAllocator * get(void)
Return (singleton) allocator.
NodeID getNumObjects(void) const
Returns the total number of memory objects.
static const Option< bool > PlainMappingFs
Use an explicitly plain mapping with flow-sensitive (not null).
Definition Options.h:47
static const OptionMap< SVF::NodeIDAllocator::Strategy > NodeAllocStrat
Definition Options.h:35
static const Option< std::string > ReadAnder
Definition Options.h:214
static const Option< bool > ClusterAnder
Whether to stage Andersen's with Steensgaard and cluster based on that data.
Definition Options.h:41
static const Option< u32_t > FsTimeLimit
Time limit for the main phase (i.e., the actual solving) of FS analyses.
Definition Options.h:72
static const Option< bool > ClusterFs
Whether to cluster FS or VFS with the auxiliary Andersen's.
Definition Options.h:44
static const Option< std::string > WriteAnder
Definition Options.h:212
static const Option< bool > DumpVFG
Definition Options.h:111
bool isFieldInsensitive(NodeID id) const
bool isLocalVarInRecursiveFun(NodeID id) const
Whether a local variable is in function recursions.
OrderedMap< const CallICFGNode *, FunctionSet > CallEdgeMap
virtual void initialize()
Initialization of a pointer analysis, including building symbol table and SVFIR etc.
virtual bool isBlkObjOrConstantObj(NodeID ptd) const
PTAStat * stat
Statistics.
NodeID getFIObjVar(NodeID id)
SVFIR * getPAG() const
Set< const FunObjVar * > FunctionSet
const CallSiteToFunPtrMap & getIndirectCallsites() const
Return all indirect callsites.
bool isArrayMemObj(NodeID id) const
NodeID getGepObjVar(NodeID id, const APOffset &ap)
void dumpStat()
Dump the statistics.
CallEdgeMap & getIndCallMap()
Get callees from an indirect callsite.
void setObjFieldInsensitive(NodeID id)
SVFIR::CallSiteToFunPtrMap CallSiteToFunPtrMap
static SVFIR * pag
SVFIR.
CallGraphSCC * callGraphSCC
SCC for PTACallGraph.
bool isHeapMemObj(NodeID id) const
Whether this object is heap or array.
virtual const NodeBS & getAllFieldsObjVars(NodeID id)
u32_t OnTheFlyIterBudgetForStat
Flag for iteration budget for on-the-fly statistics.
bool empty() const
Returns true if set is empty.
Definition PointsTo.cpp:98
std::shared_ptr< std::vector< NodeID > > MappingPtr
Definition PointsTo.h:42
static void setCurrentBestNodeMapping(MappingPtr newCurrentBestNodeMapping, MappingPtr newCurrentBestReverseNodeMapping)
Definition PointsTo.cpp:371
static MappingPtr getCurrentBestNodeMapping()
Definition PointsTo.cpp:361
SVFG * buildPTROnlySVFG(BVDataPTAImpl *pta)
SVFGNode * getSVFGNode(NodeID id) const
Get a SVFG node.
Definition SVFG.h:150
void dump(const std::string &file, bool simple=false)
Dump graph into dot file.
Definition SVFG.cpp:576
virtual void connectCallerAndCallee(const CallICFGNode *cs, const FunObjVar *callee, SVFGEdgeSetTy &edges)
Connect SVFG nodes between caller and callee for indirect call site.
Definition SVFG.cpp:658
bool isConstantObj(NodeID id) const
Definition SVFIR.h:467
const BaseObjVar * getBaseObject(NodeID id) const
Definition SVFIR.h:423
static double getClk(bool mark=false)
Definition SVFStat.cpp:48
NodeID getId() const
Get ID.
Definition SVFValue.h:158
virtual bool isPointer() const
Check if this variable represents a pointer.
void set(unsigned Idx)
unsigned count() const
void reset(unsigned Idx)
NodeID getPAGDstNodeID() const
Definition VFGNode.h:157
PAGNode * getPAGSrcNode() const
Definition VFGNode.h:162
PAGNode * getPAGDstNode() const
Definition VFGNode.h:167
NodeID getPAGSrcNodeID() const
Definition VFGNode.h:152
NodeStack nodeStack
stack used for processing nodes.
Definition WPAFSSolver.h:64
virtual NodeStack & SCCDetect()
SCC detection.
Definition WPAFSSolver.h:67
SCC * getSCCDetector() const
Get SCC detector.
Definition WPASolver.h:67
virtual void pushIntoWorklist(NodeID id)
Definition WPASolver.h:156
virtual void propagate(GNODE *v)
Definition WPASolver.h:127
virtual void initWorklist()
Definition WPASolver.h:97
void setGraph(GraphType g)
Definition WPASolver.h:78
virtual NodeStack & SCCDetect()
SCC detection.
Definition WPASolver.h:86
u32_t numOfIteration
num of iterations during constraint solving
Definition WPASolver.h:200
virtual void solveWorklist()
Definition WPASolver.h:108
hclust_fast_methods
Definition fastcluster.h:66
std::string hclustMethodToString(hclust_fast_methods method)
Returns a string representation of a hclust method.
Definition SVFUtil.cpp:248
void stopAnalysisLimitTimer(bool limitTimerSet)
Definition SVFUtil.cpp:298
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
Definition SVFUtil.cpp:101
LLVM_NODISCARD bool isa(const Y &Val)
Definition Casting.h:241
bool startAnalysisLimitTimer(unsigned timeLimit)
Definition SVFUtil.cpp:277
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
std::stack< NodeID > NodeStack
u32_t NodeID
Definition GeneralType.h:56
@ MayAlias
Definition SVFType.h:543
@ NoAlias
Definition SVFType.h:542
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
std::unordered_set< Key, Hash, KeyEqual, Allocator > Set
Definition GeneralType.h:96