/*
* ResolvingTransformerFactory.java - This file is part of the Jakstab project.
* Copyright 2007-2015 Johannes Kinder <jk@jakstab.org>
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, see <http://www.gnu.org/licenses/>.
*/
package org.jakstab.cfa;
import java.util.*;
import org.jakstab.Program;
import org.jakstab.analysis.AbstractState;
import org.jakstab.cfa.RTLLabel;
import org.jakstab.rtl.statements.*;
import org.jakstab.util.FastSet;
import org.jakstab.util.Logger;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.SetMultimap;
/**
* Abstract class for all resolving state transformer factories, that is, factories implementing
* the resolve-operator from "Kinder, Veith, Zuleger - An abstract interpretation-based
* framework for control flow reconstruction from binaries, VMCAI 2009".
*
* @author Johannes Kinder
*/
public abstract class ResolvingTransformerFactory implements StateTransformerFactory {
@SuppressWarnings("unused")
private static final Logger logger = Logger.getLogger(ResolvingTransformerFactory.class);
protected final Set<RTLLabel> unresolvedBranches = new FastSet<RTLLabel>();
protected boolean sound = true;
protected SetMultimap<RTLLabel,CFAEdge> outEdges = HashMultimap.create();
public boolean isSound() {
return sound;
}
public Set<RTLLabel> getUnresolvedBranches() {
return unresolvedBranches;
}
@Override
public Set<CFAEdge> getTransformers(final AbstractState a) {
RTLStatement stmt = Program.getProgram().getStatement((RTLLabel)a.getLocation());
Set<CFAEdge> transformers = stmt.accept(new DefaultStatementVisitor<Set<CFAEdge>>() {
@Override
protected Set<CFAEdge> visitDefault(RTLStatement stmt) {
return Collections.singleton(new CFAEdge(stmt.getLabel(), stmt.getNextLabel(), stmt));
}
@Override
public Set<CFAEdge> visit(RTLGoto stmt) {
// Call resolve function of subclass
return resolveGoto(a, stmt);
}
@Override
public Set<CFAEdge> visit(RTLHalt stmt) {
return Collections.emptySet();
}
});
saveNewEdges(transformers, (RTLLabel)a.getLocation());
return transformers;
}
protected void saveNewEdges(Set<CFAEdge> transformers, RTLLabel l) {
// Make sure we only add new edges. Edges are mutable so we cannot just implement
// hashCode and equals and add everything into a HashSet.
Set<CFAEdge> newEdges;
if (outEdges.containsKey(l)) {
newEdges = new FastSet<CFAEdge>();
for (CFAEdge edge : transformers) {
boolean found = false;
for (CFAEdge existingEdge : outEdges.get(l)) {
if (existingEdge.getTarget().equals(edge.getTarget())) {
found = true;
break;
}
}
if (!found) newEdges.add(edge);
}
} else {
newEdges = transformers;
}
outEdges.putAll(l, newEdges);
}
public Set<CFAEdge> getExistingOutEdges(RTLLabel l) {
return outEdges.get(l);
}
public Set<CFAEdge> getCFA() {
Set<CFAEdge> cfa = new HashSet<CFAEdge>();
for (CFAEdge edge : outEdges.values()) {
cfa.add(edge);
}
return cfa;
}
protected abstract Set<CFAEdge> resolveGoto(final AbstractState a, final RTLGoto stmt);
@Override
public Location getInitialLocation() {
return Program.getProgram().getStart();
}
}