/* * FindBugs - Find bugs in Java programs * 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.detect; import org.apache.bcel.Constants; import org.apache.bcel.generic.ConstantPoolGen; import org.apache.bcel.generic.Instruction; import org.apache.bcel.generic.InstructionHandle; import org.apache.bcel.generic.InvokeInstruction; import edu.umd.cs.findbugs.ba.BasicBlock; import edu.umd.cs.findbugs.ba.DataflowAnalysisException; import edu.umd.cs.findbugs.ba.Location; import edu.umd.cs.findbugs.ba.ResourceValue; import edu.umd.cs.findbugs.ba.ResourceValueFrame; import edu.umd.cs.findbugs.ba.ResourceValueFrameModelingVisitor; /** * A visitor to model the effect of instructions on the status of the resource * (in this case, Streams). */ public class StreamFrameModelingVisitor extends ResourceValueFrameModelingVisitor { private StreamResourceTracker resourceTracker; private Stream stream; private Location location; public StreamFrameModelingVisitor(ConstantPoolGen cpg, StreamResourceTracker resourceTracker, Stream stream) { super(cpg); this.resourceTracker = resourceTracker; this.stream = stream; } @Override public void transferInstruction(InstructionHandle handle, BasicBlock basicBlock) throws DataflowAnalysisException { // Record what Location we are analyzing this.location = new Location(handle, basicBlock); final Instruction ins = handle.getInstruction(); final ResourceValueFrame frame = getFrame(); int status = -1; boolean created = false; // Is a resource created, opened, or closed by this instruction? Location creationPoint = stream.getLocation(); if (handle == creationPoint.getHandle() && basicBlock == creationPoint.getBasicBlock()) { // Resource creation if (stream.isOpenOnCreation()) { status = ResourceValueFrame.OPEN; stream.setOpenLocation(location); resourceTracker.addStreamOpenLocation(location, stream); } else { status = ResourceValueFrame.CREATED; } created = true; } else if (resourceTracker.isResourceOpen(basicBlock, handle, cpg, stream, frame)) { // Resource opened status = ResourceValueFrame.OPEN; stream.setOpenLocation(location); resourceTracker.addStreamOpenLocation(location, stream); } else if (resourceTracker.isResourceClose(basicBlock, handle, cpg, stream, frame)) { // Resource closed status = ResourceValueFrame.CLOSED; } // Model use of instance values in frame slots analyzeInstruction(ins); // If needed, update frame status if (status != -1) { frame.setStatus(status); if (created) frame.setValue(frame.getNumSlots() - 1, ResourceValue.instance()); } } @Override protected boolean instanceEscapes(InvokeInstruction inv, int instanceArgNum) { ConstantPoolGen cpg = getCPG(); String className = inv.getClassName(cpg); // System.out.print("[Passed as arg="+instanceArgNum+" at " + inv + // "]"); boolean escapes = (inv.getOpcode() == Constants.INVOKESTATIC || instanceArgNum != 0); String methodName = inv.getMethodName(cpg); String methodSig = inv.getSignature(cpg); if (inv.getOpcode() == Constants.INVOKEVIRTUAL && (methodName.equals("load") || methodName.equals("loadFromXml") || methodName.equals("store") || methodName .equals("save")) && className.equals("java.util.Properties")) escapes = false; if (inv.getOpcode() == Constants.INVOKEVIRTUAL && (methodName.equals("load") || methodName.equals("store")) && className.equals("java.security.KeyStore")) escapes = false; if (inv.getOpcode() == Constants.INVOKEVIRTUAL && "getChannel".equals(methodName) && "()Ljava/nio/channels/FileChannel;".equals(methodSig)) escapes = true; if (FindOpenStream.DEBUG && escapes) { System.out.println("ESCAPE at " + location + " at call to " + className + "." + methodName + ":" + methodSig); } // Record the fact that this might be a stream escape if (stream.getOpenLocation() != null) resourceTracker.addStreamEscape(stream, location); return escapes; } } // vim:ts=3