WALA/com.ibm.wala.util/src/com/ibm/wala/util/graph/Acyclic.java

178 lines
5.9 KiB
Java

/*******************************************************************************
* Copyright (c) 2007 IBM Corporation.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package com.ibm.wala.util.graph;
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;
import java.util.Stack;
import com.ibm.wala.util.collections.HashSetFactory;
import com.ibm.wala.util.intset.BasicNaturalRelation;
import com.ibm.wala.util.intset.IBinaryNaturalRelation;
import com.ibm.wala.util.intset.IntIterator;
import com.ibm.wala.util.intset.IntPair;
/**
* Utilities for dealing with acyclic subgraphs
*/
public class Acyclic {
/*
* prevent instantiation
*/
private Acyclic() {
}
/**
* This is slow. Fix it.
*/
public static <T> boolean isAcyclic(NumberedGraph<T> G, T root) {
IBinaryNaturalRelation r = computeBackEdges(G, root);
Iterator<IntPair> it = r.iterator();
return !it.hasNext();
}
public static final int THRESHOLD_FOR_NONRECURSIVE_DFS = 1000;
/**
* Compute a relation R s.t. (i,j) \in R iff (i,j) is a backedge according to a DFS of a numbered graph starting from some root.
*
* Not efficient. Recursive and uses hash sets.
*/
public static <T> IBinaryNaturalRelation computeBackEdges(NumberedGraph<T> G, T root) {
if (G == null) {
throw new IllegalArgumentException("G is null");
}
final BasicNaturalRelation result = new BasicNaturalRelation();
// for large methods (e.g. obfuscated library code as found in android libraries 'com.google.ads.ad.a([B[B)V')
// the recursive dfs can lead to a stack overflow error.
// for smaller methods the recursive solution seems to be faster, so we keep it.
if (G.getNumberOfNodes() <= THRESHOLD_FOR_NONRECURSIVE_DFS) {
final Set<T> visited = HashSetFactory.make();
final Set<T> onstack = HashSetFactory.make();
dfs(result, root, G, visited, onstack);
} else {
dfsNonRecursive(result, root, G);
}
return result;
}
private static <T> void dfs(BasicNaturalRelation result, T root, NumberedGraph<T> G, Set<T> visited, Set<T> onstack) {
visited.add(root);
onstack.add(root);
for (Iterator<? extends T> it = G.getSuccNodes(root); it.hasNext();) {
T dstNode = it.next();
if (onstack.contains(dstNode)) {
int src = G.getNumber(root);
int dst = G.getNumber(dstNode);
result.add(src, dst);
}
if (!visited.contains(dstNode)) {
dfs(result, dstNode, G, visited, onstack);
}
}
onstack.remove(root);
}
private static <T> void dfsNonRecursive(final BasicNaturalRelation result, final T root, final NumberedGraph<T> G) {
final Stack<T> stack = new Stack<T>();
final Set<T> stackSet = new HashSet<T>();
final Stack<Iterator<? extends T>> stackIt = new Stack<Iterator<? extends T>>();
final Set<T> finished = new HashSet<T>();
stack.push(root);
stackSet.add(root);
stackIt.push(G.getSuccNodes(root));
while (!stack.isEmpty()) {
final T current = stack.pop();
stackSet.remove(current);
final Iterator<? extends T> currentIt = stackIt.pop();
if (finished.contains(current)) { continue; }
boolean isFinished = true;
while (isFinished && currentIt.hasNext() ) {
final T succ = currentIt.next();
if (!finished.contains(succ)) {
if (succ == current || stackSet.contains(succ)) {
// found a backedge
final int src = G.getNumber(current);
final int dst = G.getNumber(succ);
result.add(src, dst);
} else {
stack.push(current);
stackSet.add(current);
stackIt.push(currentIt);
stack.push(succ);
stackSet.add(succ);
stackIt.push(G.getSuccNodes(succ));
isFinished = false;
}
}
}
if (isFinished) {
finished.add(current);
}
}
}
public static <T> boolean hasIncomingBackEdges(Path p, NumberedGraph<T> G, T root) {
/*
* TODO: pull out computeBackEdges, and pass in the backedge relation as a parameter to this call
*/
IBinaryNaturalRelation backedges = computeBackEdges(G, root);
for (int index = 0; index < p.size(); index++) {
int gn = p.get(index);
Iterator<? extends T> predIter = G.getPredNodes(G.getNode(gn));
while (predIter.hasNext()) {
if (backedges.contains(G.getNumber(predIter.next()), gn))
return true;
}
}
return false;
}
/**
* Compute a set of acyclic paths through a graph G from a node src to a node sink.
*
* This is not terribly efficient.
*
* @param max the max number of paths to return.
*/
public static <T> Collection<Path> computeAcyclicPaths(NumberedGraph<T> G, T root, T src, T sink, int max) {
Collection<Path> result = HashSetFactory.make();
EdgeFilteredNumberedGraph<T> acyclic = new EdgeFilteredNumberedGraph<T>(G, computeBackEdges(G, root));
Collection<Path> worklist = HashSetFactory.make();
Path sinkPath = Path.make(G.getNumber(sink));
worklist.add(sinkPath);
while (!worklist.isEmpty() && result.size() <= max) {
Path p = worklist.iterator().next();
worklist.remove(p);
int first = p.get(0);
if (first == G.getNumber(src)) {
result.add(p);
} else {
for (IntIterator it = acyclic.getPredNodeNumbers(acyclic.getNode(first)).intIterator(); it.hasNext();) {
worklist.add(Path.prepend(it.next(), p));
}
}
}
return result;
}
}