Static Value-Flow Analysis
Loading...
Searching...
No Matches
ThreadCallGraph.cpp
Go to the documentation of this file.
1//===- ThreadCallGraph.cpp -- Call graph considering thread fork/join---------//
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 * ThreadCallGraph.cpp
25 *
26 * Created on: Jul 12, 2014
27 * Author: Yulei Sui, Peng Di, Ding Ye
28 */
29
31#include "Util/ThreadAPI.h"
32#include "SVFIR/SVFIR.h"
34#include "Graphs/CallGraph.h"
35
36using namespace SVF;
37using namespace SVFUtil;
38
43 CallGraph(cg), tdAPI(ThreadAPI::getThreadAPI())
44{
46 DBOUT(DGENERAL, outs() << SVFUtil::pasMsg("Building ThreadCallGraph\n"));
47}
48
49const std::string ThreadForkEdge::toString() const
50{
51 std::string str;
52 std::stringstream rawstr(str);
53 rawstr << "ThreadForkEdge ";
54 rawstr << "CallSiteID: " << getCallSiteID();
55 rawstr << " srcNodeID " << getSrcID() << " (fun: " << getSrcNode()->getFunction()->getName() << ")";
56 rawstr << " dstNodeID " << getDstID() << " (fun: " << getDstNode()->getFunction()->getName() << ")";
57 return rawstr.str();
58}
59
60const std::string ThreadJoinEdge::toString() const
61{
62 std::string str;
63 std::stringstream rawstr(str);
64 rawstr << "ThreadJoinEdge ";
65 rawstr << "CallSiteID: " << getCallSiteID();
66 rawstr << " srcNodeID " << getSrcID() << " (fun: " << getSrcNode()->getFunction()->getName() << ")";
67 rawstr << " dstNodeID " << getDstID() << " (fun: " << getDstNode()->getFunction()->getName() << ")";
68 return rawstr.str();
69}
70
71/*
72 * Update call graph using pointer analysis results
73 * (1) resolve function pointers for non-fork calls
74 * (2) resolve function pointers for fork sites
75 * (3) resolve function pointers for parallel_for sites
76 */
78{
79
80 PointerAnalysis::CallEdgeMap::const_iterator iter = pta->getIndCallMap().begin();
81 PointerAnalysis::CallEdgeMap::const_iterator eiter = pta->getIndCallMap().end();
82 for (; iter != eiter; iter++)
83 {
84 const CallICFGNode* cs = iter->first;
85 const CallGraph::FunctionSet &functions = iter->second;
86 for (CallGraph::FunctionSet::const_iterator func_iter =
87 functions.begin(); func_iter != functions.end(); func_iter++)
88 {
89 const FunObjVar* callee = *func_iter;
91 }
92 }
93
94 // Fork sites
95 for (CallSiteSet::const_iterator it = forksitesBegin(), eit = forksitesEnd(); it != eit; ++it)
96 {
98 if(SVFUtil::dyn_cast<FunValVar>(forkedval)==nullptr)
99 {
100 SVFIR* pag = pta->getPAG();
101 const NodeBS targets = pta->getPts(forkedval->getId()).toNodeBS();
102 for (NodeBS::iterator ii = targets.begin(), ie = targets.end(); ii != ie; ii++)
103 {
104 if(ObjVar* objPN = SVFUtil::dyn_cast<ObjVar>(pag->getGNode(*ii)))
105 {
106 const BaseObjVar* obj = pag->getBaseObject(objPN->getId());
107 if(obj->isFunction())
108 {
109 const FunObjVar* svfCallee = SVFUtil::cast<FunObjVar>(obj)->getFunction();
111 }
112 }
113 }
114 }
115 }
116}
117
118
123{
124
125 for (CallSiteSet::const_iterator it = joinsitesBegin(), eit = joinsitesEnd(); it != eit; ++it)
126 {
128 // find its corresponding fork sites first
130 for (CallSiteSet::const_iterator it = forksitesBegin(), eit = forksitesEnd(); it != eit; ++it)
131 {
133 if (pta->alias(jointhread->getId(), forkthread->getId()))
134 {
135 forkset.insert(*it);
136 }
137 }
138 assert(!forkset.empty() && "Can't find a forksite for this join!!");
140 }
141}
142
147{
148
150 const FunValVar* funValvar = SVFUtil::dyn_cast<FunValVar>(tdAPI->getForkedFun(cs));
152 assert(forkee && "callee does not exist");
153 CallGraphNode* callee = getCallGraphNode(forkee->getDefFunForMultipleModule());
154 CallSiteID csId = addCallSite(cs, callee->getFunction());
155
157 {
158 assert(cs->getCaller() == caller->getFunction() && "callee instruction not inside caller??");
159
161 edge->addDirectCallSite(cs);
162
163 addEdge(edge);
165 return true;
166 }
167 else
168 return false;
169}
170
175{
178
179 CallSiteID csId = addCallSite(cs, callee->getFunction());
180
182 {
183 assert(cs->getCaller() == caller->getFunction() && "callee instruction not inside caller??");
184
186 edge->addInDirectCallSite(cs);
187
188 addEdge(edge);
190 return true;
191 }
192 else
193 return false;
194}
195
203{
204
206
207 for (CallSiteSet::const_iterator it = forkset.begin(), eit = forkset.end(); it != eit; ++it)
208 {
209
211 SVFUtil::dyn_cast<FunValVar>(tdAPI->getForkedFun(*it))->getFunction();
212 assert(threadRoutineFun && "thread routine function does not exist");
215
217 {
218 assert(cs->getCaller() == joinFunNode->getFunction() && "callee instruction not inside caller??");
220 edge->addDirectCallSite(cs);
221
223 }
224 }
225}
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
Definition SVFType.h:498
#define DGENERAL
Definition SVFType.h:504
CallSiteID getCallSiteID() const
Get direct and indirect calls.
Definition CallGraph.h:83
CallGraphEdge * hasGraphEdge(CallGraphNode *src, CallGraphNode *dst, CallGraphEdge::CEDGEK kind, CallSiteID csId) const
Whether we have already created this call graph edge.
void addIndirectCallGraphEdge(const CallICFGNode *cs, const FunObjVar *callerFun, const FunObjVar *calleeFun)
Add indirect call edges.
void addEdge(CallGraphEdge *edge)
Add call graph edge.
Definition CallGraph.h:292
CallSiteID addCallSite(const CallICFGNode *cs, const FunObjVar *callee)
Add CallSiteID.
Definition CallGraph.h:276
Set< const FunObjVar * > FunctionSet
Definition CallGraph.h:244
const CallGraphNode * getCallGraphNode(const std::string &name)
Get call graph node.
const FunObjVar * getCaller() const
Return callsite.
Definition ICFGNode.h:464
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
NodeType * getSrcNode() const
NodeType * getDstNode() const
NodeID getDstID() const
NodeID getSrcID() const
get methods of the components
NodeType * getGNode(NodeID id) const
Get a node.
virtual const PointsTo & getPts(NodeID ptr)=0
Get points-to targets of a pointer. It needs to be implemented in child class.
SVFIR * getPAG() const
virtual AliasResult alias(const SVFVar *V1, const SVFVar *V2)=0
Interface exposed to users of our pointer analysis, given Value infos.
CallEdgeMap & getIndCallMap()
Get callees from an indirect callsite.
NodeBS toNodeBS() const
Returns this points-to set as a NodeBS.
Definition PointsTo.cpp:313
const BaseObjVar * getBaseObject(NodeID id) const
Definition SVFIR.h:423
const ValVar * getForkedFun(const CallICFGNode *inst) const
const ValVar * getForkedThread(const CallICFGNode *inst) const
Return arguments/attributes of pthread_create / hare_parallel_for.
const SVFVar * getJoinedThread(const CallICFGNode *inst) const
Return arguments/attributes of pthread_join.
CallSiteSet::const_iterator forksitesEnd() const
bool addDirectForkEdge(const CallICFGNode *cs)
Add direct/indirect thread fork edges.
CallSiteSet::const_iterator forksitesBegin() const
Fork sites iterators.
void updateJoinEdge(PointerAnalysis *pta)
Update join edge using pointer analysis results.
CallSiteSet::const_iterator joinsitesEnd() const
bool hasThreadJoinEdge(const CallICFGNode *cs) const
Get call graph edge via call instruction.
void addThreadJoinEdgeSetMap(const CallICFGNode *cs, ThreadJoinEdge *edge)
map call instruction to its PTACallGraphEdge map
void addDirectJoinEdge(const CallICFGNode *cs, const CallSiteSet &forksite)
Add thread join edges.
bool addIndirectForkEdge(const CallICFGNode *cs, const FunObjVar *callee)
CallSiteSet::const_iterator joinsitesBegin() const
Join sites iterators.
void addThreadForkEdgeSetMap(const CallICFGNode *cs, ThreadForkEdge *edge)
map call instruction to its PTACallGraphEdge map
void updateCallGraph(PointerAnalysis *pta)
Update call graph using pointer results.
ThreadCallGraph(const CallGraph &cg)
Constructor.
ThreadAPI * tdAPI
Thread API.
virtual const std::string toString() const
virtual const std::string toString() const
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
Definition SVFUtil.cpp:101
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
unsigned CallSiteID
Definition GeneralType.h:58
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74