Static Value-Flow Analysis
Loading...
Searching...
No Matches
ThreadCallGraph.h
Go to the documentation of this file.
1//===- ThreadCallGraph.h -- 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.h
25 *
26 * Created on: Jul 7, 2014
27 * Authors: Peng Di, Yulei Sui, Ding Ye
28 */
29
30#ifndef RCG_H_
31#define RCG_H_
32
33#include "Graphs/CallGraph.h"
34
35namespace SVF
36{
37
38class ThreadAPI;
39class PointerAnalysis;
44{
45
46public:
54 {
55 }
56
58
59 static inline bool classof(const ThreadForkEdge*)
60 {
61 return true;
62 }
63 static inline bool classof(const CallGraphEdge*edge)
64 {
65 return edge->getEdgeKind() == CallGraphEdge::TDForkEdge;
66 }
68
69 virtual const std::string toString() const;
70
72};
73
78{
79
80public:
88 {
89 }
90
91 static inline bool classof(const ThreadJoinEdge*)
92 {
93 return true;
94 }
95 static inline bool classof(const CallGraphEdge*edge)
96 {
97 return edge->getEdgeKind() == CallGraphEdge::TDJoinEdge;
98 }
99
100 virtual const std::string toString() const;
101
103};
104
109{
110
111public:
119 {
120 }
121
123
124 static inline bool classof(const HareParForEdge*)
125 {
126 return true;
127 }
128 static inline bool classof(const CallGraphEdge*edge)
129 {
130 return edge->getEdgeKind() == CallGraphEdge::HareParForEdge;
131 }
133
135};
136
137
142{
143
144public:
154
157
159
162 {
163 }
164
166
167 static inline bool classof(const ThreadCallGraph *)
168 {
169 return true;
170 }
171 static inline bool classof(const CallGraph*g)
172 {
173 return g->getKind() == CallGraph::ThdCallGraph;
174 }
176
181
182
184
185
186 inline bool hasThreadForkEdge(const CallICFGNode* cs) const
187 {
188 return callinstToThreadForkEdgesMap.find(cs) !=
190 }
191 inline ForkEdgeSet::const_iterator getForkEdgeBegin(const CallICFGNode* cs) const
192 {
193 CallInstToForkEdgesMap::const_iterator it = callinstToThreadForkEdgesMap.find(cs);
194 assert(it != callinstToThreadForkEdgesMap.end() && "call instruction not found");
195 return it->second.begin();
196 }
197 inline ForkEdgeSet::const_iterator getForkEdgeEnd(const CallICFGNode* cs) const
198 {
199 CallInstToForkEdgesMap::const_iterator it = callinstToThreadForkEdgesMap.find(cs);
200 assert(it != callinstToThreadForkEdgesMap.end() && "call instruction not found");
201 return it->second.end();
202 }
203
205
206
207 inline bool hasThreadJoinEdge(const CallICFGNode* cs) const
208 {
210 }
211 inline JoinEdgeSet::const_iterator getJoinEdgeBegin(const CallICFGNode* cs) const
212 {
213 CallInstToJoinEdgesMap::const_iterator it = callinstToThreadJoinEdgesMap.find(cs);
214 assert(it != callinstToThreadJoinEdgesMap.end() && "call instruction does not have a valid callee");
215 return it->second.begin();
216 }
217 inline JoinEdgeSet::const_iterator getJoinEdgeEnd(const CallICFGNode* cs) const
218 {
219 CallInstToJoinEdgesMap::const_iterator it = callinstToThreadJoinEdgesMap.find(cs);
220 assert(it != callinstToThreadJoinEdgesMap.end() && "call instruction does not have a valid callee");
221 return it->second.end();
222 }
224 {
225 for(CallInstToJoinEdgesMap::const_iterator it = callinstToThreadJoinEdgesMap.begin(), eit = callinstToThreadJoinEdgesMap.end(); it!=eit; ++it)
226 {
227 for(JoinEdgeSet::const_iterator jit = it->second.begin(), ejit = it->second.end(); jit!=ejit; ++jit)
228 {
229 if((*jit)->getDstNode() == routine)
230 {
231 csSet.insert(it->first);
232 }
233 }
234 }
235 }
237
240 inline bool isForksite(const CallICFGNode* csInst)
241 {
242 return forksites.find(csInst) != forksites.end();
243 }
244 inline bool isJoinsite(const CallICFGNode* csInst)
245 {
246 return joinsites.find(csInst) != joinsites.end();
247 }
248 inline bool isParForSite(const CallICFGNode* csInst)
249 {
250 return parForSites.find(csInst) != parForSites.end();
251 }
253
255
256 inline CallSiteSet::const_iterator forksitesBegin() const
257 {
258 return forksites.begin();
259 }
260 inline CallSiteSet::const_iterator forksitesEnd() const
261 {
262 return forksites.end();
263 }
265
267
268 inline CallSiteSet::const_iterator joinsitesBegin() const
269 {
270 return joinsites.begin();
271 }
272 inline CallSiteSet::const_iterator joinsitesEnd() const
273 {
274 return joinsites.end();
275 }
277
279
280 inline CallSiteSet::const_iterator parForSitesBegin() const
281 {
282 return parForSites.begin();
283 }
284 inline CallSiteSet::const_iterator parForSitesEnd() const
285 {
286 return parForSites.end();
287 }
289
291
292 inline u32_t getNumOfForksite() const
293 {
294 return forksites.size();
295 }
296 inline u32_t getNumOfJoinsite() const
297 {
298 return joinsites.size();
299 }
301 {
302 return parForSites.size();
303 }
305
307 inline ThreadAPI* getThreadAPI() const
308 {
309 return tdAPI;
310 }
311
313
314 inline bool addForksite(const CallICFGNode* cs)
315 {
317 return forksites.insert(cs).second;
318 }
319 inline bool addJoinsite(const CallICFGNode* cs)
320 {
322 return joinsites.insert(cs).second;
323 }
324 inline bool addParForSite(const CallICFGNode* cs)
325 {
327 return parForSites.insert(cs).second;
328 }
330
332
333 bool addDirectForkEdge(const CallICFGNode* cs);
334 bool addIndirectForkEdge(const CallICFGNode* cs, const FunObjVar* callee);
336
338
339 void addDirectJoinEdge(const CallICFGNode* cs,const CallSiteSet& forksite);
341
342
345 {
346 if(edge!=nullptr)
347 {
350 }
351 }
352
355 {
356 if(edge!=nullptr)
357 {
360 }
361 }
362
365 {
366 if(edge!=nullptr)
367 {
370 }
371 }
372
375 {
377 CallInstToJoinEdgesMap::const_iterator it = callinstToThreadJoinEdgesMap.find(call);
379 {
380 JoinEdgeSet::const_iterator jit = it->second.find(&joinEdge);
381 if(jit!=it->second.end())
382 return *jit;
383 }
384 return nullptr;
385 }
386
387private:
395};
396
397} // End namespace SVF
398
399#endif /* RCG_H_ */
CallSiteID csId
Definition CallGraph.h:65
CallInstToCallGraphEdgesMap callinstToCallGraphEdgesMap
Map a call instruction to its corresponding call edges.
Definition CallGraph.h:265
OrderedSet< EdgeType *, typename EdgeType::equalGEdge > GEdgeSetTy
Edge kind.
GenericNode< CallGraphNode, HareParForEdge >::GEdgeSetTy ParForEdgeSet
HareParForEdge(CallGraphNode *s, CallGraphNode *d, CallSiteID csId)
Constructor.
virtual ~HareParForEdge()
Destructor.
static bool classof(const HareParForEdge *)
ClassOf.
static bool classof(const CallGraphEdge *edge)
bool isForksite(const CallICFGNode *csInst)
CallSiteSet::const_iterator forksitesEnd() const
CallSiteSet::const_iterator parForSitesBegin() const
hare_parallel_for sites iterators
bool addDirectForkEdge(const CallICFGNode *cs)
Add direct/indirect thread fork edges.
Set< CallSiteSet * > CtxSet
CallSiteSet::const_iterator forksitesBegin() const
Fork sites iterators.
CallInstToParForEdgesMap callinstToHareParForEdgesMap
Map a call instruction to its corresponding hare_parallel_for edges.
void updateJoinEdge(PointerAnalysis *pta)
Update join edge using pointer analysis results.
static bool classof(const ThreadCallGraph *)
ClassOf.
CallSiteSet::const_iterator joinsitesEnd() const
CallSiteSet::const_iterator parForSitesEnd() const
bool addForksite(const CallICFGNode *cs)
Add fork sites which directly or indirectly create a thread.
Set< const CallICFGNode * > InstSet
bool hasThreadJoinEdge(const CallICFGNode *cs) const
Get call graph edge via call instruction.
bool addParForSite(const CallICFGNode *cs)
CallSiteSet forksites
all thread fork sites
CallSiteSet parForSites
all parallel for sites
void addThreadJoinEdgeSetMap(const CallICFGNode *cs, ThreadJoinEdge *edge)
map call instruction to its PTACallGraphEdge map
u32_t getNumOfJoinsite() const
bool addJoinsite(const CallICFGNode *cs)
virtual ~ThreadCallGraph()
Destructor.
CallInstToForkEdgesMap callinstToThreadForkEdgesMap
Map a call instruction to its corresponding fork edges.
JoinEdgeSet::const_iterator getJoinEdgeEnd(const CallICFGNode *cs) const
ForkEdgeSet::const_iterator getForkEdgeEnd(const CallICFGNode *cs) const
ForkEdgeSet::const_iterator getForkEdgeBegin(const CallICFGNode *cs) const
void addDirectJoinEdge(const CallICFGNode *cs, const CallSiteSet &forksite)
Add thread join edges.
bool addIndirectForkEdge(const CallICFGNode *cs, const FunObjVar *callee)
bool hasThreadForkEdge(const CallICFGNode *cs) const
Get call graph edge via call instruction.
CallSiteSet::const_iterator joinsitesBegin() const
Join sites iterators.
ThreadJoinEdge * hasThreadJoinEdge(const CallICFGNode *call, CallGraphNode *joinFunNode, CallGraphNode *threadRoutineFunNode, CallSiteID csId) const
has thread join edge
ThreadForkEdge::ForkEdgeSet ForkEdgeSet
u32_t getNumOfForksite() const
Num of fork/join sites.
ThreadAPI * getThreadAPI() const
Thread API.
ThreadJoinEdge::JoinEdgeSet JoinEdgeSet
CallSiteSet joinsites
all thread fork sites
void addHareParForEdgeSetMap(const CallICFGNode *cs, HareParForEdge *edge)
map call instruction to its PTACallGraphEdge map
void addThreadForkEdgeSetMap(const CallICFGNode *cs, ThreadForkEdge *edge)
map call instruction to its PTACallGraphEdge map
Map< const CallICFGNode *, JoinEdgeSet > CallInstToJoinEdgesMap
Map< const CallICFGNode *, ParForEdgeSet > CallInstToParForEdgesMap
HareParForEdge::ParForEdgeSet ParForEdgeSet
bool isJoinsite(const CallICFGNode *csInst)
static bool classof(const CallGraph *g)
ThreadCallGraph(ThreadCallGraph &cg)=delete
u32_t getNumOfParForSite() const
bool isParForSite(const CallICFGNode *csInst)
void updateCallGraph(PointerAnalysis *pta)
Update call graph using pointer results.
Map< const CallICFGNode *, ForkEdgeSet > CallInstToForkEdgesMap
ThreadAPI * tdAPI
Thread API.
CallInstToJoinEdgesMap callinstToThreadJoinEdgesMap
Map a call instruction to its corresponding join edges.
void getJoinSites(const CallGraphNode *routine, InstSet &csSet)
JoinEdgeSet::const_iterator getJoinEdgeBegin(const CallICFGNode *cs) const
static bool classof(const ThreadForkEdge *)
ClassOf.
virtual ~ThreadForkEdge()
Destructor.
ThreadForkEdge(CallGraphNode *s, CallGraphNode *d, CallSiteID csId)
Constructor.
static bool classof(const CallGraphEdge *edge)
GenericNode< CallGraphNode, ThreadForkEdge >::GEdgeSetTy ForkEdgeSet
virtual const std::string toString() const
virtual ~ThreadJoinEdge()
Destructor.
static bool classof(const ThreadJoinEdge *)
virtual const std::string toString() const
static bool classof(const CallGraphEdge *edge)
ThreadJoinEdge(CallGraphNode *s, CallGraphNode *d, CallSiteID csId)
Constructor.
GenericNode< CallGraphNode, ThreadJoinEdge >::GEdgeSetTy JoinEdgeSet
for isBitcode
Definition BasicTypes.h:68
unsigned CallSiteID
Definition GeneralType.h:58
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
unsigned u32_t
Definition GeneralType.h:47