Static Value-Flow Analysis
Loading...
Searching...
No Matches
CallGraph.cpp
Go to the documentation of this file.
1//===- PTACallGraph.cpp -- Call graph used internally in SVF------------------//
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/*
25 * PTACallGraph.cpp
26 *
27 * Created on: Nov 7, 2013
28 * Author: Yulei Sui
29 */
30
31#include "Graphs/CallGraph.h"
32#include "SVFIR/SVFIR.h"
33#include "Util/Options.h"
34#include "Util/SVFUtil.h"
35#include <sstream>
36
37using namespace SVF;
38using namespace SVFUtil;
39
43
44
45const std::string &CallGraphNode::getName() const
46{
47 return fun->getName();
48}
49
50
52
54{
55 assert(call->getCalledFunction() && "not a direct callsite??");
56 directCalls.insert(call);
57}
58
60{
61 assert((nullptr == call->getCalledFunction() || !SVFUtil::isa<FunValVar>(SVFUtil::getForkedFun(call))) &&
62 "not an indirect callsite??");
63 indirectCalls.insert(call);
64}
66
67const std::string CallGraphEdge::toString() const
68{
69 std::string str;
70 std::stringstream rawstr(str);
71 rawstr << "CallSite ID: " << getCallSiteID();
73 rawstr << "direct call";
74 else
75 rawstr << "indirect call";
76 rawstr << "[" << getDstID() << "<--" << getSrcID() << "]\t";
77 return rawstr.str();
78}
79
80const std::string CallGraphNode::toString() const
81{
82 std::string str;
83 std::stringstream rawstr(str);
84 rawstr << "PTACallGraphNode ID: " << getId() << " {fun: " << fun->getName() << "}";
85 return rawstr.str();
86}
87
89{
90 return getCallSite(id)->getCaller();
91}
92
94{
95 std::function<bool(const CallGraphNode*)> dfs =
97 {
98 NodeID id = v->getId();
99 if (!visitedNodes.test_and_set(id))
100 return reachableFromEntry[id];
101
102 if (SVFUtil::isProgEntryFunction(v->getFunction()))
103 return reachableFromEntry[id] = true;
104
105 bool result = false;
106 for (const_iterator it = v->InEdgeBegin(), eit = v->InEdgeEnd(); it != eit; ++it)
107 {
109 result |= dfs(edge->getSrcNode());
110 if (result)
111 break;
112 }
113 return reachableFromEntry[id] = result;
114 };
115
116 return dfs(this);
117}
118
119
126
129{
130 callGraphNodeNum = other.getTotalNodeNum();
133
135 for (const auto& item : other)
136 {
137 const CallGraphNode* cgn = item.second;
138 CallGraphNode* callGraphNode = new CallGraphNode(cgn->getId(), cgn->getFunction());
139 addGNode(cgn->getId(),callGraphNode);
140 funToCallGraphNodeMap[cgn->getFunction()] = callGraphNode;
141 }
142
144 for (const auto& item : other.callinstToCallGraphEdgesMap)
145 {
146 const CallICFGNode* cs = item.first;
147 for (const CallGraphEdge* edge : item.second)
148 {
149 CallGraphNode* src = getCallGraphNode(edge->getSrcID());
150 CallGraphNode* dst = getCallGraphNode(edge->getDstID());
151 CallSiteID csId = addCallSite(cs, dst->getFunction());
152
154 newEdge->addDirectCallSite(cs);
157 }
158 }
159
160}
161
166{
167}
168
173 CallGraphNode* dst,
174 CallGraphEdge::CEDGEK kind, CallSiteID csId) const
175{
176 CallGraphEdge edge(src,dst,kind,csId);
179 if (outEdge && inEdge)
180 {
181 assert(outEdge == inEdge && "edges not match");
182 return outEdge;
183 }
184 else
185 return nullptr;
186}
187
192 CallGraphNode* dst,
194{
195 for (CallGraphEdge::CallGraphEdgeSet::iterator iter = src->OutEdgeBegin();
196 iter != src->OutEdgeEnd(); ++iter)
197 {
198 CallGraphEdge* edge = (*iter);
199 if (edge->getEdgeKind() == kind && edge->getDstID() == dst->getId())
200 return edge;
201 }
202 return nullptr;
203}
204
205
227
232{
233 CallGraphNode* callGraphNode = getCallGraphNode(callee);
234 for(CallGraphNode::iterator it = callGraphNode->InEdgeBegin(), eit = callGraphNode->InEdgeEnd();
235 it!=eit; ++it)
236 {
237 for(CallGraphEdge::CallInstSet::const_iterator cit = (*it)->directCallsBegin(),
238 ecit = (*it)->directCallsEnd(); cit!=ecit; ++cit)
239 {
240 csSet.insert((*cit));
241 }
242 for(CallGraphEdge::CallInstSet::const_iterator cit = (*it)->indirectCallsBegin(),
243 ecit = (*it)->indirectCallsEnd(); cit!=ecit; ++cit)
244 {
245 csSet.insert((*cit));
246 }
247 }
248}
249
254{
255 CallGraphNode* callGraphNode = getCallGraphNode(callee);
256 for(CallGraphNode::iterator it = callGraphNode->InEdgeBegin(), eit = callGraphNode->InEdgeEnd();
257 it!=eit; ++it)
258 {
259 for(CallGraphEdge::CallInstSet::const_iterator cit = (*it)->directCallsBegin(),
260 ecit = (*it)->directCallsEnd(); cit!=ecit; ++cit)
261 {
262 csSet.insert((*cit));
263 }
264 }
265}
266
271{
272 CallGraphNode* callGraphNode = getCallGraphNode(callee);
273 for(CallGraphNode::iterator it = callGraphNode->InEdgeBegin(), eit = callGraphNode->InEdgeEnd();
274 it!=eit; ++it)
275 {
276 for(CallGraphEdge::CallInstSet::const_iterator cit = (*it)->indirectCallsBegin(),
277 ecit = (*it)->indirectCallsEnd(); cit!=ecit; ++cit)
278 {
279 csSet.insert((*cit));
280 }
281 }
282}
283
288{
290 return;
291
294 CallEdgeMap::const_iterator it = indirectCallMap.begin();
295 CallEdgeMap::const_iterator eit = indirectCallMap.end();
296 for (; it != eit; ++it)
297 {
298 const FunctionSet& targets = it->second;
299 if (targets.empty() == false)
300 {
301 const CallICFGNode* cs = it->first;
302 const FunObjVar* func = cs->getCaller();
304 isReachableFromProgEntry(reachableFromEntry, visitedNodes) == false)
305 writeWrnMsg(func->getName() + " has indirect call site but not reachable from main");
306 }
307 }
308}
309
314{
316
317 std::stack<const CallGraphNode*> nodeStack;
319 nodeStack.push(dstNode);
320 visitedNodes.set(dstNode->getId());
321
322 while (nodeStack.empty() == false)
323 {
324 CallGraphNode* node = const_cast<CallGraphNode*>(nodeStack.top());
325 nodeStack.pop();
326
327 if (node->getFunction() == srcFn)
328 return true;
329
330 for (CallGraphEdgeConstIter it = node->InEdgeBegin(), eit = node->InEdgeEnd(); it != eit; ++it)
331 {
333 if (visitedNodes.test_and_set(edge->getSrcID()))
334 nodeStack.push(edge->getSrcNode());
335 }
336 }
337
338 return false;
339}
340
344void CallGraph::dump(const std::string& filename)
345{
347}
348
350{
351 SVF::ViewGraph(this, "Call Graph");
352}
353
354
359{
361 CallGraphNode*callGraphNode = new CallGraphNode(id, fun);
362 addGNode(id, callGraphNode);
363 funToCallGraphNodeMap[callGraphNode->getFunction()] = callGraphNode;
365}
366
368{
369 for (const auto& item : *this)
370 {
371 if (item.second->getName() == name)
372 return item.second;
373 }
374 return nullptr;
375}
376
395
396namespace SVF
397{
398
402template<>
404{
405
408 DOTGraphTraits(bool isSimple = false) :
409 DefaultDOTGraphTraits(isSimple)
410 {
411 }
412
414 static std::string getGraphName(CallGraph*)
415 {
416 return "Call Graph";
417 }
419 static std::string getNodeLabel(CallGraphNode*node, CallGraph*)
420 {
421 return node->toString();
422 }
423
424 static std::string getNodeAttributes(CallGraphNode*node, CallGraph*)
425 {
426 const FunObjVar* fun = node->getFunction();
427 if (!SVFUtil::isExtCall(fun))
428 {
429 return "shape=box";
430 }
431 else
432 return "shape=Mrecord";
433 }
434
435 template<class EdgeIter>
437 CallGraph*)
438 {
439
440 //TODO: mark indirect call of Fork with different color
441 CallGraphEdge* edge = *(EI.getCurrent());
442 assert(edge && "No edge found!!");
443
444 std::string color;
445
446 if (edge->getEdgeKind() == CallGraphEdge::TDJoinEdge)
447 {
448 color = "color=green";
449 }
450 else if (edge->getEdgeKind() == CallGraphEdge::TDForkEdge)
451 {
452 color = "color=blue";
453 }
454 else
455 {
456 color = "color=black";
457 }
458 if (0 != edge->getIndirectCalls().size())
459 {
460 color = "color=red";
461 }
462 return color;
463 }
464
465 template<class EdgeIter>
467 {
468 CallGraphEdge* edge = *(EI.getCurrent());
469 assert(edge && "No edge found!!");
470
471 std::string str;
472 std::stringstream rawstr(str);
473 rawstr << edge->getCallSiteID();
474
475 return rawstr.str();
476 }
477};
478} // End namespace llvm
const char *const name
Definition cJSON.h:264
cJSON * item
Definition cJSON.h:222
NodeID getId() const
Get the memory object id.
void addInDirectCallSite(const CallICFGNode *call)
Definition CallGraph.cpp:59
void addDirectCallSite(const CallICFGNode *call)
Add direct and indirect callsite.
Definition CallGraph.cpp:53
CallInstSet indirectCalls
Definition CallGraph.h:64
virtual const std::string toString() const
Definition CallGraph.cpp:67
bool isDirectCallEdge() const
Definition CallGraph.h:87
Set< const CallICFGNode * > CallInstSet
Definition CallGraph.h:55
CallInstSet directCalls
Definition CallGraph.h:63
CallSiteID getCallSiteID() const
Get direct and indirect calls.
Definition CallGraph.h:83
const FunObjVar * getFunction() const
Get function of this call node.
Definition CallGraph.h:191
const std::string & getName() const
Definition CallGraph.cpp:45
virtual const std::string toString() const
Definition CallGraph.cpp:80
bool isReachableFromProgEntry(Map< NodeID, bool > &reachableFromEntry, NodeBS &visitedNodes) const
Return TRUE if this function can be reached from main.
Definition CallGraph.cpp:93
const FunObjVar * fun
Definition CallGraph.h:179
CallInstToCallGraphEdgesMap callinstToCallGraphEdgesMap
Map a call instruction to its corresponding call edges.
Definition CallGraph.h:265
void getAllCallSitesInvokingCallee(const FunObjVar *callee, CallGraphEdge::CallInstSet &csSet)
Get callsites invoking the callee.
void addCallGraphNode(const FunObjVar *fun)
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.
CallEdgeMap indirectCallMap
Indirect call map.
Definition CallGraph.h:256
void addEdge(CallGraphEdge *edge)
Add call graph edge.
Definition CallGraph.h:292
const FunObjVar * getCallerOfCallSite(CallSiteID id) const
Definition CallGraph.cpp:88
void getIndCallSitesInvokingCallee(const FunObjVar *callee, CallGraphEdge::CallInstSet &csSet)
static IdToCallSiteMap idToCSMap
Map a callsite ID to a pair of call instruction and callee.
Definition CallGraph.h:260
static CallSiteID totalCallSiteNum
CallSiteIDs, start from 1;.
Definition CallGraph.h:261
Map< CallSiteID, CallSitePair > IdToCallSiteMap
Definition CallGraph.h:243
NodeID callGraphNodeNum
Definition CallGraph.h:267
void addDirectCallGraphEdge(const CallICFGNode *call, const FunObjVar *callerFun, const FunObjVar *calleeFun)
Add direct call edges.
void destroy()
Clean up memory.
CallSiteID addCallSite(const CallICFGNode *cs, const FunObjVar *callee)
Add CallSiteID.
Definition CallGraph.h:276
FunToCallGraphNodeMap funToCallGraphNodeMap
Call Graph node map.
Definition CallGraph.h:264
CallGraphEdgeSet::const_iterator CallGraphEdgeConstIter
Definition CallGraph.h:247
CallGraph(CGEK k=NormCallGraph)
Constructor.
static CallSiteToIdMap csToIdMap
Call site information.
Definition CallGraph.h:259
const CallICFGNode * getCallSite(CallSiteID id) const
Definition CallGraph.h:396
bool isReachableBetweenFunctions(const FunObjVar *srcFn, const FunObjVar *dstFn) const
Whether its reachable between two functions.
void view()
View the graph from the debugger.
void dump(const std::string &filename)
Dump the graph.
void getDirCallSitesInvokingCallee(const FunObjVar *callee, CallGraphEdge::CallInstSet &csSet)
void verifyCallGraph()
Issue a warning if the function which has indirect call sites can not be reached from program entry.
u32_t numOfResolvedIndCallEdge
Definition CallGraph.h:268
CallGraphEdge * getGraphEdge(CallGraphNode *src, CallGraphNode *dst, CallGraphEdge::CEDGEK kind, CallSiteID csId)
Get call graph edge via nodes.
Map< CallSitePair, CallSiteID > CallSiteToIdMap
Definition CallGraph.h:242
Set< const FunObjVar * > FunctionSet
Definition CallGraph.h:244
const CallGraphNode * getCallGraphNode(const std::string &name)
Get call graph node.
const FunObjVar * getCalledFunction() const
Definition ICFGNode.h:512
const FunObjVar * getCaller() const
Return callsite.
Definition ICFGNode.h:464
NodeID getDstID() const
NodeID getSrcID() const
get methods of the components
void addGNode(NodeID id, NodeType *node)
Add a Node.
bool hasIncomingEdge() const
Has incoming/outgoing edge set.
bool hasOutgoingEdge() const
iterator OutEdgeEnd()
GEdgeSetTy::iterator iterator
iterator OutEdgeBegin()
iterators
GEdgeSetTy::const_iterator const_iterator
iterator InEdgeBegin()
iterator InEdgeEnd()
static void WriteGraphToFile(SVF::OutStream &O, const std::string &GraphName, const GraphType &GT, bool simple=false)
static const Option< bool > DisableWarn
Definition Options.h:205
NodeID getId() const
Get ID.
Definition SVFValue.h:158
NodeID id
Node ID.
Definition SVFValue.h:203
virtual const std::string & getName() const
Definition SVFValue.h:184
void set(unsigned Idx)
iterator begin() const
bool isProgEntryFunction(const FunObjVar *)
Program entry function e.g. main.
Definition SVFUtil.cpp:442
bool isExtCall(const FunObjVar *fun)
Definition SVFUtil.cpp:437
void writeWrnMsg(const std::string &msg)
Writes a message run through wrnMsg.
Definition SVFUtil.cpp:68
const ValVar * getForkedFun(const CallICFGNode *inst)
Return thread fork function.
Definition SVFUtil.h:331
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
unsigned CallSiteID
Definition GeneralType.h:58
u32_t NodeID
Definition GeneralType.h:56
void ViewGraph(const GraphType &G, const std::string &name, bool ShortNames=false, GraphProgram::Name Program=GraphProgram::DOT)
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
DOTGraphTraits(bool isSimple=false)
static std::string getNodeAttributes(CallGraphNode *node, CallGraph *)
static std::string getGraphName(CallGraph *)
Return name of the graph.
static std::string getNodeLabel(CallGraphNode *node, CallGraph *)
Return function name;.
static std::string getEdgeSourceLabel(NodeType *, EdgeIter EI)
static std::string getEdgeAttributes(CallGraphNode *, EdgeIter EI, CallGraph *)
NodeType::iterator ChildIteratorType