/* * Bytecode Analysis Framework * Copyright (C) 2005, 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.constant; import org.apache.bcel.generic.InstructionHandle; import org.apache.bcel.generic.MethodGen; import edu.umd.cs.findbugs.ba.BasicBlock; import edu.umd.cs.findbugs.ba.DataflowAnalysisException; import edu.umd.cs.findbugs.ba.DepthFirstSearch; import edu.umd.cs.findbugs.ba.Edge; import edu.umd.cs.findbugs.ba.FrameDataflowAnalysis; import edu.umd.cs.findbugs.ba.Location; /** * Dataflow analysis to find constant values. * * @see edu.umd.cs.findbugs.ba.constant.Constant * @author David Hovemeyer */ public class ConstantAnalysis extends FrameDataflowAnalysis<Constant, ConstantFrame> { private MethodGen methodGen; private ConstantFrameModelingVisitor visitor; public ConstantAnalysis(MethodGen methodGen, DepthFirstSearch dfs) { super(dfs); this.methodGen = methodGen; this.visitor = new ConstantFrameModelingVisitor(methodGen.getConstantPool()); } public ConstantFrame createFact() { return new ConstantFrame(methodGen.getMaxLocals()); } public void initEntryFact(ConstantFrame frame) { frame.setValid(); frame.clearStack(); int numSlots = frame.getNumSlots(); for (int i = 0; i < numSlots; ++i) { frame.setValue(i, Constant.NOT_CONSTANT); } } @Override public void transferInstruction(InstructionHandle handle, BasicBlock basicBlock, ConstantFrame frame) throws DataflowAnalysisException { visitor.setFrameAndLocation(frame, new Location(handle, basicBlock)); visitor.analyzeInstruction(handle.getInstruction()); } public void meetInto(ConstantFrame fact, Edge edge, ConstantFrame result) throws DataflowAnalysisException { if (fact.isValid()) { ConstantFrame tmpFact = null; if (edge.isExceptionEdge()) { tmpFact = modifyFrame(fact, tmpFact); tmpFact.clearStack(); tmpFact.pushValue(Constant.NOT_CONSTANT); } if (tmpFact != null) { fact = tmpFact; } } mergeInto(fact, result); } @Override protected void mergeValues(ConstantFrame otherFrame, ConstantFrame resultFrame, int slot) throws DataflowAnalysisException { Constant value = Constant.merge(resultFrame.getValue(slot), otherFrame.getValue(slot)); resultFrame.setValue(slot, value); } // /* // * Test driver. // */ // public static void main(String[] argv) throws Exception { // if (argv.length != 1) { // System.err.println("Usage: " + ConstantAnalysis.class.getName() + // " <class file>"); // System.exit(1); // } // // DataflowTestDriver<ConstantFrame, ConstantAnalysis> driver = // new DataflowTestDriver<ConstantFrame, ConstantAnalysis>() { // @Override // public Dataflow<ConstantFrame, ConstantAnalysis> createDataflow( // ClassContext classContext, // Method method) throws CFGBuilderException, DataflowAnalysisException { // return classContext.getConstantDataflow(method); // } // }; // // driver.execute(argv[0]); // } }