soot.jimple.infoflow.memory.FlowDroidTimeoutWatcher Maven / Gradle / Ivy
package soot.jimple.infoflow.memory;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import soot.jimple.infoflow.memory.IMemoryBoundedSolver.IMemoryBoundedSolverStatusNotification;
import soot.jimple.infoflow.memory.reasons.TimeoutReason;
import soot.jimple.infoflow.results.InfoflowResults;
import soot.jimple.infoflow.util.ThreadUtils;
/**
* Class for enforcing timeouts on IFDS solvers
*
* @author Steven Arzt
*
*/
public class FlowDroidTimeoutWatcher implements IMemoryBoundedSolverStatusNotification {
/**
* Enumeration containing all states in which a solver can be
*
* @author Steven Arzt
*
*/
private enum SolverState {
/**
* The solver has not been started yet
*/
IDLE,
/**
* The solver is running
*/
RUNNING,
/**
* The solver has completed its work
*/
DONE
}
private final Logger logger = LoggerFactory.getLogger(getClass());
private final long timeout;
private final InfoflowResults results;
private final Map solvers = new ConcurrentHashMap<>();
private volatile boolean stopped = false;
private ISolversTerminatedCallback terminationCallback = null;
/**
* Creates a new instance of the {@link FlowDroidTimeoutWatcher} class
*
* @param timeout The timeout in seconds after which the solvers shall be
* stopped
*/
public FlowDroidTimeoutWatcher(long timeout) {
this.timeout = timeout;
this.results = null;
}
/**
* Creates a new instance of the {@link FlowDroidTimeoutWatcher} class
*
* @param timeout The timeout in seconds after which the solvers shall be
* stopped
* @param res The InfoflowResults object
*/
public FlowDroidTimeoutWatcher(long timeout, InfoflowResults res) {
this.timeout = timeout;
this.results = res;
}
/**
* Gets the timeout after which the IFDS solvers are aborted
*
* @return The timeout after which the IFDS solvers are aborted
*/
public long getTimeout() {
return this.timeout;
}
/**
* Adds a solver that shall be terminated when the timeout is reached
*
* @param solver A solver that shall be terminated when the timeout is reached
*/
public void addSolver(IMemoryBoundedSolver solver) {
this.solvers.put(solver, SolverState.IDLE);
solver.addStatusListener(this);
}
/**
* Starts the timeout watcher
*/
public void start() {
final long startTime = System.nanoTime();
logger.info("FlowDroid timeout watcher started");
this.stopped = false;
ThreadUtils.createGenericThread(new Runnable() {
@Override
public void run() {
// Sleep until we have reached the timeout
boolean allTerminated = isTerminated();
long timeoutNano = TimeUnit.SECONDS.toNanos(timeout);
while (!stopped && ((System.nanoTime() - startTime) < timeoutNano)) {
allTerminated = isTerminated();
if (allTerminated) {
break;
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// There's little we can do here
}
}
long timeElapsed = TimeUnit.NANOSECONDS.toSeconds(System.nanoTime() - startTime);
// If things have not stopped on their own account, we force
// them to
if (!stopped && !allTerminated) {
logger.warn("Timeout reached, stopping the solvers...");
if (results != null) {
results.addException("Timeout reached");
}
TimeoutReason reason = new TimeoutReason(timeElapsed, timeout);
for (IMemoryBoundedSolver solver : solvers.keySet()) {
solver.forceTerminate(reason);
}
if (terminationCallback != null) {
terminationCallback.onSolversTerminated();
}
}
logger.info("FlowDroid timeout watcher terminated");
}
private boolean isTerminated() {
// Check whether all solvers in our watchlist have finished
// their work
for (IMemoryBoundedSolver solver : solvers.keySet()) {
if (solvers.get(solver) != SolverState.DONE || !solver.isTerminated())
return false;
}
return true;
}
}, "FlowDroid Timeout Watcher", true).start();
}
/**
* Stops the timeout watcher so that it no longer interferes with solver
* execution
*/
public void stop() {
this.stopped = true;
}
/**
* Resets the internal state of the watcher so that it can be used again after
* being stopped
*/
public void reset() {
this.stopped = false;
for (IMemoryBoundedSolver solver : this.solvers.keySet())
this.solvers.put(solver, SolverState.IDLE);
}
@Override
public void notifySolverStarted(IMemoryBoundedSolver solver) {
solvers.put(solver, SolverState.RUNNING);
}
@Override
public void notifySolverTerminated(IMemoryBoundedSolver solver) {
solvers.put(solver, SolverState.DONE);
}
/**
* Sets a callback that shall be invoked when the solvers have been terminated
* due to a timeout
*
* @param terminationCallback The callback to invoke when the solvers have been
* terminated due to a timeout
*/
public void setTerminationCallback(ISolversTerminatedCallback terminationCallback) {
this.terminationCallback = terminationCallback;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy