Static Value-Flow Analysis
Loading...
Searching...
No Matches
GenericGraph.h
Go to the documentation of this file.
1//===- CenericGraph.h -- Generic graph ---------------------------------------//
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 * GenericGraph.h
25 *
26 * Created on: Mar 19, 2014
27 * Author: Yulei Sui
28 */
29
30#ifndef GENERICGRAPH_H_
31#define GENERICGRAPH_H_
32
33#include "SVFIR/SVFType.h"
34#include "Util/iterator.h"
35#include "Graphs/GraphTraits.h"
36#include "SVFIR/SVFValue.h"
37
38namespace SVF
39{
42template <typename, typename> class GenericGraphWriter;
43template <typename, typename> class GenericGraphReader;
45
49template<class NodeTy>
51{
52 friend class SVFIRWriter;
53 friend class SVFIRReader;
54
55public:
64private:
68
69public:
74
76 virtual ~GenericEdge()
77 {
78 }
79
81
82 inline NodeID getSrcID() const
83 {
84 return src->getId();
85 }
86 inline NodeID getDstID() const
87 {
88 return dst->getId();
89 }
90 inline GEdgeKind getEdgeKind() const
91 {
92 return (EdgeKindMask & edgeFlag);
93 }
95 {
96 return edgeFlag;
97 }
99 {
100 return src;
101 }
103 {
104 return dst;
105 }
107
109 // and duplicated elements in the set are not inserted (binary tree comparison)
111 typedef struct equalGEdge
112 {
114 {
115 if (lhs->edgeFlag != rhs->edgeFlag)
116 return lhs->edgeFlag < rhs->edgeFlag;
117 else if (lhs->getSrcID() != rhs->getSrcID())
118 return lhs->getSrcID() < rhs->getSrcID();
119 else
120 return lhs->getDstID() < rhs->getDstID();
121 }
122 } equalGEdge;
123
124 virtual inline bool operator==(const GenericEdge<NodeType>* rhs) const
125 {
126 return (rhs->edgeFlag == this->edgeFlag &&
127 rhs->getSrcID() == this->getSrcID() &&
128 rhs->getDstID() == this->getDstID());
129 }
131
132protected:
133 static constexpr unsigned char EdgeKindMaskBits = 8;
134 static constexpr u64_t EdgeKindMask = (~0ULL) >> (64 - EdgeKindMaskBits);
135};
136
137
138
142template<class NodeTy,class EdgeTy>
144{
145 friend class SVFIRWriter;
146 friend class SVFIRReader;
147
148public:
155 typedef typename GEdgeSetTy::iterator iterator;
156 typedef typename GEdgeSetTy::const_iterator const_iterator;
158
159private:
160
163
164public:
167 {
168
169 }
170
172 virtual ~GenericNode()
173 {
174 for (auto * edge : OutEdges)
175 delete edge;
176 }
177
180 inline const GEdgeSetTy& getOutEdges() const
181 {
182 return OutEdges;
183 }
184 inline const GEdgeSetTy& getInEdges() const
185 {
186 return InEdges;
187 }
189
191
192 inline bool hasIncomingEdge() const
193 {
194 return (InEdges.empty() == false);
195 }
196 inline bool hasOutgoingEdge() const
197 {
198 return (OutEdges.empty() == false);
199 }
201
203
205 {
206 return OutEdges.begin();
207 }
209 {
210 return OutEdges.end();
211 }
213 {
214 return InEdges.begin();
215 }
217 {
218 return InEdges.end();
219 }
221 {
222 return OutEdges.begin();
223 }
225 {
226 return OutEdges.end();
227 }
229 {
230 return InEdges.begin();
231 }
233 {
234 return InEdges.end();
235 }
237
239
241 {
242 return OutEdges.begin();
243 }
244 virtual inline iterator directOutEdgeEnd()
245 {
246 return OutEdges.end();
247 }
249 {
250 return InEdges.begin();
251 }
252 virtual inline iterator directInEdgeEnd()
253 {
254 return InEdges.end();
255 }
256
257 virtual inline const_iterator directOutEdgeBegin() const
258 {
259 return OutEdges.begin();
260 }
261 virtual inline const_iterator directOutEdgeEnd() const
262 {
263 return OutEdges.end();
264 }
265 virtual inline const_iterator directInEdgeBegin() const
266 {
267 return InEdges.begin();
268 }
269 virtual inline const_iterator directInEdgeEnd() const
270 {
271 return InEdges.end();
272 }
274
276
278 {
279 return InEdges.insert(inEdge).second;
280 }
282 {
283 return OutEdges.insert(outEdge).second;
284 }
286
290 {
291 iterator it = InEdges.find(edge);
292 assert(it != InEdges.end() && "can not find in edge in SVFG node");
293 InEdges.erase(it);
294 return 1;
295 }
297 {
298 iterator it = OutEdges.find(edge);
299 assert(it != OutEdges.end() && "can not find out edge in SVFG node");
300 OutEdges.erase(it);
301 return 1;
302 }
304
306
308 {
310 if (it != InEdges.end())
311 return *it;
312 else
313 return nullptr;
314 }
316 {
318 if (it != OutEdges.end())
319 return *it;
320 else
321 return nullptr;
322 }
324
325 static inline bool classof(const GenericNode<NodeTy, EdgeTy>*)
326 {
327 return true;
328 }
329
330 static inline bool classof(const SVFValue*)
331 {
332 return true;
333 }
334};
335
336/*
337 * Generic graph for program representation
338 * It is base class and needs to be instantiated
339 */
340template<class NodeTy, class EdgeTy>
342{
343 friend class SVFIRWriter;
344 friend class SVFIRReader;
345 friend class GenericGraphWriter<NodeTy, EdgeTy>;
346 friend class GenericGraphReader<NodeTy, EdgeTy>;
347
348public:
353
355
356 typedef typename IDToNodeMapTy::iterator iterator;
357 typedef typename IDToNodeMapTy::const_iterator const_iterator;
359
362
365 {
366 destroy();
367 }
368
370 void destroy()
371 {
372 for (auto &entry : IDToNodeMap)
373 delete entry.second;
374 }
376
378 {
379 return IDToNodeMap.begin();
380 }
381 inline iterator end()
382 {
383 return IDToNodeMap.end();
384 }
385 inline const_iterator begin() const
386 {
387 return IDToNodeMap.begin();
388 }
389 inline const_iterator end() const
390 {
391 return IDToNodeMap.end();
392 }
393 //}@
394
396 inline void addGNode(NodeID id, NodeType* node)
397 {
398 IDToNodeMap[id] = node;
399 nodeNum++;
400 }
401
403 inline NodeType* getGNode(NodeID id) const
404 {
405 const_iterator it = IDToNodeMap.find(id);
406 assert(it != IDToNodeMap.end() && "Node not found!");
407 return it->second;
408 }
409
411 inline bool hasGNode(NodeID id) const
412 {
413 const_iterator it = IDToNodeMap.find(id);
414 return it != IDToNodeMap.end();
415 }
416
418 inline void removeGNode(NodeType* node)
419 {
420 assert(node->hasIncomingEdge() == false
421 && node->hasOutgoingEdge() == false
422 && "node which have edges can't be deleted");
423 iterator it = IDToNodeMap.find(node->getId());
424 assert(it != IDToNodeMap.end() && "can not find the node");
425 IDToNodeMap.erase(it);
426 delete node;
427 }
428
430 inline u32_t getTotalNodeNum() const
431 {
432 return nodeNum;
433 }
434 inline u32_t getTotalEdgeNum() const
435 {
436 return edgeNum;
437 }
439 inline void incNodeNum()
440 {
441 nodeNum++;
442 }
443 inline void incEdgeNum()
444 {
445 edgeNum++;
446 }
447
448protected:
450
451public:
454};
455
456} // End namespace SVF
457
458/* !
459 * GenericGraphTraits specializations for generic graph algorithms.
460 * Provide graph traits for traversing from a node using standard graph traversals.
461 */
462namespace SVF
463{
464
465// mapped_iter - This is a simple iterator adapter that causes a function to
466// be applied whenever operator* is invoked on the iterator.
467
468template <typename ItTy, typename FuncTy,
469 typename FuncReturnTy =
470 decltype(std::declval<FuncTy>()(*std::declval<ItTy>()))>
472 : public iter_adaptor_base<
473 mapped_iter<ItTy, FuncTy>, ItTy,
474 typename std::iterator_traits<ItTy>::iterator_category,
475 typename std::remove_reference<FuncReturnTy>::type>
476{
477public:
479 : mapped_iter::iter_adaptor_base(std::move(U)), F(std::move(F)) {}
480
482 {
483 return this->I;
484 }
485
486 FuncReturnTy operator*() const
487 {
488 return F(*this->I);
489 }
490
491private:
492 FuncTy F;
493};
494
495// map_iter - Provide a convenient way to create mapped_iters, just like
496// make_pair is useful for creating pairs...
497template <class ItTy, class FuncTy>
499{
500 return mapped_iter<ItTy, FuncTy>(std::move(I), std::move(F));
501}
502
506template<class NodeTy,class EdgeTy> struct GenericGraphTraits<SVF::GenericNode<NodeTy,EdgeTy>* >
507{
510
511 static inline NodeType* edge_dest(const EdgeType* E)
512 {
513 return E->getDstNode();
514 }
515
516 // nodes_iterator/begin/end - Allow iteration over all nodes in the graph
518
520 {
521 return pagN;
522 }
523
525 {
526 return map_iter(N->OutEdgeBegin(), &edge_dest);
527 }
528 static inline ChildIteratorType child_end(const NodeType* N)
529 {
530 return map_iter(N->OutEdgeEnd(), &edge_dest);
531 }
533 {
534 return map_iter(N->directOutEdgeBegin(), &edge_dest);
535 }
537 {
538 return map_iter(N->directOutEdgeEnd(), &edge_dest);
539 }
540};
541
545template<class NodeTy,class EdgeTy>
546struct GenericGraphTraits<Inverse<SVF::GenericNode<NodeTy,EdgeTy>* > >
547{
550
551 static inline NodeType* edge_dest(const EdgeType* E)
552 {
553 return E->getSrcNode();
554 }
555
556 // nodes_iterator/begin/end - Allow iteration over all nodes in the graph
558
560 {
561 return G.Graph;
562 }
563
565 {
566 return map_iter(N->InEdgeBegin(), &edge_dest);
567 }
568 static inline ChildIteratorType child_end(const NodeType* N)
569 {
570 return map_iter(N->InEdgeEnd(), &edge_dest);
571 }
572
573 static inline unsigned getNodeID(const NodeType* N)
574 {
575 return N->getId();
576 }
577};
578
582template<class NodeTy,class EdgeTy> struct GenericGraphTraits<SVF::GenericGraph<NodeTy,EdgeTy>* > : public GenericGraphTraits<SVF::GenericNode<NodeTy,EdgeTy>* >
583{
587
589 {
590 return nullptr; // return null here, maybe later we could create a dummy node
591 }
592
593 typedef std::pair<SVF::NodeID, NodeType*> PairTy;
594 static inline NodeType* deref_val(PairTy P)
595 {
596 return P.second;
597 }
598
599 // nodes_iterator/begin/end - Allow iteration over all nodes in the graph
600 typedef mapped_iter<typename GenericGraphTy::iterator, decltype(&deref_val)> nodes_iterator;
601
603 {
604 return map_iter(G->begin(), &deref_val);
605 }
607 {
608 return map_iter(G->end(), &deref_val);
609 }
610
611 static unsigned graphSize(GenericGraphTy* G)
612 {
613 return G->getTotalNodeNum();
614 }
615
616 static inline unsigned getNodeID(NodeType* N)
617 {
618 return N->getId();
619 }
621 {
622 return G->getGNode(id);
623 }
624};
625
626} // End namespace llvm
627
628#endif /* GENERICGRAPH_H_ */
virtual bool operator==(const GenericEdge< NodeType > *rhs) const
NodeTy * src
source node
GEdgeKind getEdgeKindWithoutMask() const
NodeType * getSrcNode() const
virtual ~GenericEdge()
Destructor.
GenericEdge(NodeTy *s, NodeTy *d, GEdgeFlag k)
Constructor.
NodeType * getDstNode() const
static constexpr u64_t EdgeKindMask
friend class SVFIRReader
GEdgeFlag edgeFlag
edge kind
NodeTy * dst
destination node
GEdgeKind getEdgeKind() const
friend class SVFIRWriter
NodeID getDstID() const
NodeID getSrcID() const
get methods of the components
static constexpr unsigned char EdgeKindMaskBits
We use the lower 8 bits to denote edge kind.
NodeTy NodeType
Node type.
void addGNode(NodeID id, NodeType *node)
Add a Node.
iterator begin()
Iterators.
void removeGNode(NodeType *node)
Delete a node.
u32_t getTotalEdgeNum() const
u32_t edgeNum
total num of node
const_iterator end() const
const_iterator begin() const
u32_t nodeNum
total num of edge
virtual ~GenericGraph()
Destructor.
IDToNodeMapTy IDToNodeMap
node map
IDToNodeMapTy::const_iterator const_iterator
bool hasGNode(NodeID id) const
Has a node.
void incNodeNum()
Increase number of node/edge.
u32_t getTotalNodeNum() const
Get total number of node/edge.
friend class SVFIRReader
GenericGraph()
Constructor.
IDToNodeMapTy::iterator iterator
Node Iterators.
friend class SVFIRWriter
NodeType * getGNode(NodeID id) const
Get a node.
void destroy()
Release memory.
OrderedMap< NodeID, NodeType * > IDToNodeMapTy
NodeID to GenericNode map.
const_iterator InEdgeEnd() const
OrderedSet< EdgeType *, typename EdgeType::equalGEdge > GEdgeSetTy
Edge kind.
GEdgeSetTy OutEdges
all outgoing edge of this node
bool hasIncomingEdge() const
Has incoming/outgoing edge set.
bool hasOutgoingEdge() const
u32_t removeOutgoingEdge(EdgeType *edge)
GenericNode(NodeID i, GNodeK k, const SVFType *svfType=nullptr)
Constructor.
virtual iterator directInEdgeEnd()
iterator OutEdgeEnd()
EdgeType * hasIncomingEdge(EdgeType *edge) const
Find incoming and outgoing edges.
GEdgeSetTy InEdges
all incoming edge of this node
GEdgeSetTy::iterator iterator
const GEdgeSetTy & getOutEdges() const
virtual ~GenericNode()
Destructor.
virtual iterator directInEdgeBegin()
const_iterator OutEdgeBegin() const
virtual iterator directOutEdgeEnd()
const GEdgeSetTy & getInEdges() const
static bool classof(const SVFValue *)
const_iterator InEdgeBegin() const
virtual const_iterator directOutEdgeEnd() const
bool addIncomingEdge(EdgeType *inEdge)
Add incoming and outgoing edges.
u32_t removeIncomingEdge(EdgeType *edge)
friend class SVFIRReader
virtual iterator directOutEdgeBegin()
Iterators used for SCC detection, overwrite it in child class if necessary.
iterator OutEdgeBegin()
iterators
virtual const_iterator directInEdgeEnd() const
static bool classof(const GenericNode< NodeTy, EdgeTy > *)
virtual const_iterator directInEdgeBegin() const
GEdgeSetTy::const_iterator const_iterator
friend class SVFIRWriter
const_iterator OutEdgeEnd() const
virtual const_iterator directOutEdgeBegin() const
EdgeType * hasOutgoingEdge(EdgeType *edge) const
iterator InEdgeBegin()
bool addOutgoingEdge(EdgeType *outEdge)
iterator InEdgeEnd()
WrappedIteratorT I
Definition iterator.h:242
mapped_iter(ItTy U, FuncTy F)
FuncReturnTy operator*() const
for isBitcode
Definition BasicTypes.h:68
unsigned long long u64_t
Definition GeneralType.h:49
u32_t NodeID
Definition GeneralType.h:56
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
unsigned u32_t
Definition GeneralType.h:47
mapped_iter< ItTy, FuncTy > map_iter(ItTy I, FuncTy F)
signed long long s64_t
Definition GeneralType.h:50
Add the hash function for std::set (we also can overload operator< to implement this)
bool operator()(const GenericEdge< NodeType > *lhs, const GenericEdge< NodeType > *rhs) const
mapped_iter< typename SVF::GenericNode< NodeTy, EdgeTy >::iterator, decltype(&edge_dest)> ChildIteratorType
static NodeType * getNode(GenericGraphTy *G, SVF::NodeID id)
mapped_iter< typename GenericGraphTy::iterator, decltype(&deref_val)> nodes_iterator
static ChildIteratorType direct_child_begin(const NodeType *N)
static ChildIteratorType child_end(const NodeType *N)
static ChildIteratorType direct_child_end(const NodeType *N)
mapped_iter< typename SVF::GenericNode< NodeTy, EdgeTy >::iterator, decltype(&edge_dest)> ChildIteratorType
static ChildIteratorType child_begin(const NodeType *N)
const GraphType & Graph
Definition GraphTraits.h:99