/* * Bytecode Analysis Framework * Copyright (C) 2003,2004 University of Maryland * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ package edu.umd.cs.findbugs.ba.vna; import java.util.HashMap; import org.apache.bcel.generic.InstructionHandle; import edu.umd.cs.findbugs.SystemProperties; import edu.umd.cs.findbugs.annotations.SuppressWarnings; import edu.umd.cs.findbugs.util.Strings; /** * A cache mapping instructions and input values to the output values they * produce. We must always produce the same output given identical * input, or else value number analysis will not terminate. * * @author David Hovemeyer * @see ValueNumberAnalysis */ public class ValueNumberCache { private static final boolean DEBUG = SystemProperties.getBoolean("vn.debug"); /** * An entry in the cache. * It represents an instruction with specific input values. */ public static class Entry { public final InstructionHandle handle; public final ValueNumber[] inputValueList; private int cachedHashCode; @SuppressWarnings("EI2") public Entry(InstructionHandle handle, ValueNumber[] inputValueList) { this.handle = handle; this.inputValueList = inputValueList; this.cachedHashCode = 0; } @Override public boolean equals(Object o) { if (!(o instanceof Entry)) return false; Entry other = (Entry) o; if (handle.getPosition() != other.handle.getPosition()) return false; ValueNumber[] myList = inputValueList; ValueNumber[] otherList = other.inputValueList; if (myList.length != otherList.length) return false; for (int i = 0; i < myList.length; ++i) if (!myList[i].equals(otherList[i])) return false; return true; } @Override public int hashCode() { if (cachedHashCode == 0) { int code = handle.getPosition(); for (ValueNumber aInputValueList : inputValueList) { code *= 101; ValueNumber valueNumber = aInputValueList; code += valueNumber.hashCode(); } cachedHashCode = code; } return cachedHashCode; } @Override public String toString() { StringBuilder buf = new StringBuilder(); buf.append(handle.toString()); for (ValueNumber aInputValueList : inputValueList) { buf.append(", "); buf.append(aInputValueList.toString()); } return buf.toString(); } } /** * Map of entries to output values. */ private HashMap<Entry, ValueNumber[]> entryToOutputMap = new HashMap<Entry, ValueNumber[]>(); /** * Look up cached output values for given entry. * * @param entry the entry * @return the list of output values, or null if there is no matching entry * in the cache */ public ValueNumber[] lookupOutputValues(Entry entry) { if (DEBUG) System.out.println("VN cache lookup: " + entry); ValueNumber[] result = entryToOutputMap.get(entry); if (DEBUG) System.out.println(" result ==> " + Strings.toString(result)); return result; } /** * Add output values for given entry. * Assumes that lookupOutputValues() has determined that the entry * is not in the cache. * * @param entry the entry * @param outputValueList the list of output values produced * by the entry's instruction and input values */ public void addOutputValues(Entry entry, ValueNumber[] outputValueList) { ValueNumber[] old = entryToOutputMap.put(entry, outputValueList); if (old != null) throw new IllegalStateException("overwriting output values for entry!"); } } // vim:ts=4