Static Value-Flow Analysis
Loading...
Searching...
No Matches
SCC.h
Go to the documentation of this file.
1//===- SCC.h -- SCC detection algorithm---------------------------------------//
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 * SCC.h
25 *
26 * Esko Nuutila and Eljas Soisalon-Soininen, "On finding the
27 * strongly connected components in a directed graph".
28 * Inf. Process. Letters, 49(1):9-14, 1994.
29 *
30 * The implementation is derived from the pseudo code in the following paper:
31 * Pereira and Berlin, "Wave Propagation and Deep Propagation for Pointer Analysis",
32 * CGO 2009, 126-135, 2009.
33 *
34 * And influenced by implementation from Open64 compiler
35 *
36 * Created on: Jul 12, 2013
37 * Author: yusui
38 */
39
40#ifndef SCC_H_
41#define SCC_H_
42
43#include "SVFIR/SVFValue.h" // for NodeBS
44#include <limits.h>
45#include <stack>
46#include <map>
47
48namespace SVF
49{
50
51class GNodeSCCInfo;
52
53template<class GraphType>
55{
56
57private:
60 typedef typename GTraits::NodeRef GNODE;
61 typedef typename GTraits::nodes_iterator node_iterator;
62 typedef typename GTraits::ChildIteratorType child_iterator;
63 typedef unsigned NodeID ;
64
65public:
66 typedef std::stack<NodeID> GNodeStack;
67
69 {
70 public:
72
73 inline bool visited(void) const
74 {
75 return _visited;
76 }
77 inline void visited(bool v)
78 {
79 _visited = v;
80 }
81 inline bool inSCC(void) const
82 {
83 return _inSCC;
84 }
85 inline void inSCC(bool v)
86 {
87 _inSCC = v;
88 }
89 inline NodeID rep(void)const
90 {
91 return _rep;
92 }
93 inline void rep(NodeID n)
94 {
95 _rep = n;
96 }
97 inline void addSubNodes(NodeID n)
98 {
100 }
101 inline NodeBS& subNodes()
102 {
103 return _subNodes;
104 }
105 inline const NodeBS& subNodes() const
106 {
107 return _subNodes;
108 }
109 private:
111 bool _inSCC;
114 };
115
118
120 : _graph(GT),
121 _I(0)
122 {}
123
124
125 // Return a handle to the stack of nodes in topological
126 // order. This will be used to seed the initial solution
127 // and improve efficiency.
129 {
130 return _T;
131 }
132
133 inline const GNodeStack& topoNodeStack() const
134 {
135 return _T;
136 }
137
142 {
145 while(!topoOrder.empty())
146 {
147 NodeID nodeID = topoOrder.top();
148 topoOrder.pop();
149 revTopoOrder.push(nodeID);
150 }
151 return revTopoOrder;
152 }
153
154 const inline GNODESCCInfoMap &GNodeSCCInfo() const
155 {
156 return _NodeSCCAuxInfo;
157 }
158
160 inline NodeID repNode(NodeID n) const
161 {
162 typename GNODESCCInfoMap::const_iterator it = _NodeSCCAuxInfo.find(n);
163 assert(it!=_NodeSCCAuxInfo.end() && "scc rep not found");
164 NodeID rep = it->second.rep();
165 return rep!= UINT_MAX ? rep : n ;
166 }
167
168
170 inline bool isInCycle(NodeID n) const
171 {
172 NodeID rep = repNode(n);
173 // multi-node cycle
174 if (subNodes(rep).count() > 1)
175 {
176 return true;
177 }
178 // self-cycle
179 else
180 {
181 child_iterator EI = GTraits::direct_child_begin(Node(rep));
182 child_iterator EE = GTraits::direct_child_end(Node(rep));
183 for (; EI != EE; ++EI)
184 {
185 NodeID w = Node_Index(*EI);
186 if(w==rep)
187 return true;
188 }
189 return false;
190 }
191 }
192
194 inline const NodeBS& subNodes(NodeID n) const
195 {
196 typename GNODESCCInfoMap::const_iterator it = _NodeSCCAuxInfo.find(n);
197 assert(it!=_NodeSCCAuxInfo.end() && "scc rep not found");
198 return it->second.subNodes();
199 }
200
202 inline const NodeBS &getRepNodes() const
203 {
204 return repNodes;
205 }
206
207 const inline GraphType & graph()
208 {
209 return _graph;
210 }
211private:
212
214
221
222 inline bool visited(NodeID n)
223 {
224 return _NodeSCCAuxInfo[n].visited();
225 }
226 inline bool inSCC(NodeID n)
227 {
228 return _NodeSCCAuxInfo[n].inSCC();
229 }
230
231 inline void setVisited(NodeID n,bool v)
232 {
233 _NodeSCCAuxInfo[n].visited(v);
234 }
235 inline void setInSCC(NodeID n,bool v)
236 {
237 _NodeSCCAuxInfo[n].inSCC(v);
238 }
239 inline void rep(NodeID n, NodeID r)
240 {
241 _NodeSCCAuxInfo[n].rep(r);
242 _NodeSCCAuxInfo[r].addSubNodes(n);
243 if (n != r)
244 {
245 _NodeSCCAuxInfo[n].subNodes().clear();
247 repNodes.set(r);
248 }
249 }
250
252 {
253 return _NodeSCCAuxInfo[n].rep();
254 }
255 inline bool isInSCC(NodeID n)
256 {
257 return _NodeSCCAuxInfo[n].inSCC();
258 }
259
260 inline GNODE Node(NodeID id) const
261 {
262 return GTraits::getNode(_graph, id);
263 }
264
265 inline NodeID Node_Index(GNODE node) const
266 {
267 return GTraits::getNodeID(node);
268 }
269
271 {
272 // SVFUtil::outs() << "visit GNODE: " << Node_Index(v)<< "\n";
273 _I += 1;
274 _D[v] = _I;
275 this->rep(v,v);
276 this->setVisited(v,true);
277
278 child_iterator EI = GTraits::direct_child_begin(Node(v));
279 child_iterator EE = GTraits::direct_child_end(Node(v));
280
281 for (; EI != EE; ++EI)
282 {
283 NodeID w = Node_Index(*EI);
284
285 if (!this->visited(w))
286 visit(w);
287 if (!this->inSCC(w))
288 {
289 NodeID rep;
290 rep = _D[this->rep(v)] < _D[this->rep(w)] ?
291 this->rep(v) : this->rep(w);
292 this->rep(v,rep);
293 }
294 }
295 if (this->rep(v) == v)
296 {
297 this->setInSCC(v,true);
298 while (!_SS.empty())
299 {
300 NodeID w = _SS.top();
301 if (_D[w] <= _D[v])
302 break;
303 else
304 {
305 _SS.pop();
306 this->setInSCC(w,true);
307 this->rep(w,v);
308 }
309 }
310 _T.push(v);
311 }
312 else
313 _SS.push(v);
314 }
315
316 void clear()
317 {
318 _NodeSCCAuxInfo.clear();
319 _I = 0;
320 _D.clear();
321 repNodes.clear();
322 while(!_SS.empty())
323 _SS.pop();
324 while(!_T.empty())
325 _T.pop();
326 }
327public:
328
329 void find(void)
330 {
331 // Visit each unvisited root node. A root node is defined
332 // to be a node that has no incoming copy/skew edges
333 clear();
334 node_iterator I = GTraits::nodes_begin(_graph);
335 node_iterator E = GTraits::nodes_end(_graph);
336 for (; I != E; ++I)
337 {
338 NodeID node = Node_Index(*I);
339 if (!this->visited(node))
340 {
341 // We skip any nodes that have a representative other than
342 // themselves. Such nodes occur as a result of merging
343 // nodes either through unifying an ACC or other node
344 // merging optimizations. Any such node should have no
345 // outgoing edges and therefore should no longer be a member
346 // of an SCC.
347 if (this->rep(node) == UINT_MAX || this->rep(node) == node)
348 visit(node);
349 else
350 this->visited(node);
351 }
352 }
353 }
354
355 void find(NodeSet &candidates)
356 {
357 // This function is reloaded to only visit candidate NODES
358 clear();
359 for (NodeID node : candidates)
360 {
361 if (!this->visited(node))
362 {
363 if (this->rep(node) == UINT_MAX || this->rep(node) == node)
364 visit(node);
365 else
366 this->visited(node);
367 }
368 }
369 }
370
371};
372
373} // End namespace SVF
374
375#endif /* SCC_H_ */
#define false
Definition cJSON.cpp:70
cJSON * n
Definition cJSON.cpp:2558
int count
Definition cJSON.h:216
const NodeBS & subNodes() const
Definition SCC.h:105
void addSubNodes(NodeID n)
Definition SCC.h:97
bool visited(void) const
Definition SCC.h:73
NodeID rep(void) const
Definition SCC.h:89
bool inSCC(void) const
Definition SCC.h:81
GNodeStack _T
Definition SCC.h:219
void rep(NodeID n, NodeID r)
Definition SCC.h:239
GNodeStack _SS
Definition SCC.h:218
void find(void)
Definition SCC.h:329
SCCDetection(const GraphType &GT)
Definition SCC.h:119
const GNodeStack & topoNodeStack() const
Definition SCC.h:133
void find(NodeSet &candidates)
Definition SCC.h:355
const GraphType & _graph
Definition SCC.h:215
NodeBS repNodes
Definition SCC.h:220
void clear()
Definition SCC.h:316
NodeID repNode(NodeID n) const
get the rep node if not found return itself
Definition SCC.h:160
unsigned NodeID
Definition SCC.h:63
GNODESCCInfoMap _NodeSCCAuxInfo
Definition SCC.h:213
Map< NodeID, GNodeSCCInfo > GNODESCCInfoMap
Definition SCC.h:116
NodeToNodeMap _D
Definition SCC.h:217
bool isInCycle(NodeID n) const
whether the node is in a cycle
Definition SCC.h:170
NodeID Node_Index(GNODE node) const
Definition SCC.h:265
bool inSCC(NodeID n)
Definition SCC.h:226
GNODE Node(NodeID id) const
Definition SCC.h:260
GTraits::NodeRef GNODE
Definition SCC.h:60
Map< NodeID, NodeID > NodeToNodeMap
Definition SCC.h:117
GNodeStack & topoNodeStack()
Definition SCC.h:128
std::stack< NodeID > GNodeStack
Definition SCC.h:66
SVF::GenericGraphTraits< GraphType > GTraits
Define the GTraits and node iterator for printing.
Definition SCC.h:59
NodeID _I
Definition SCC.h:216
FIFOWorkList< NodeID > revTopoNodeStack() const
Definition SCC.h:141
const NodeBS & getRepNodes() const
get all repNodeID
Definition SCC.h:202
bool visited(NodeID n)
Definition SCC.h:222
const NodeBS & subNodes(NodeID n) const
get all subnodes in one scc, if size is empty insert itself into the set
Definition SCC.h:194
void setInSCC(NodeID n, bool v)
Definition SCC.h:235
const GNODESCCInfoMap & GNodeSCCInfo() const
Definition SCC.h:154
GTraits::nodes_iterator node_iterator
Definition SCC.h:61
GTraits::ChildIteratorType child_iterator
Definition SCC.h:62
NodeID rep(NodeID n)
Definition SCC.h:251
void setVisited(NodeID n, bool v)
Definition SCC.h:231
const GraphType & graph()
Definition SCC.h:207
void visit(NodeID v)
Definition SCC.h:270
bool isInSCC(NodeID n)
Definition SCC.h:255
void set(unsigned Idx)
void reset(unsigned Idx)
for isBitcode
Definition BasicTypes.h:68
Set< NodeID > NodeSet
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
typename GraphType::UnknownGraphTypeError NodeRef
Definition GraphTraits.h:80