/* * 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(); } }