/*
* 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 java.util.HashMap;
import java.util.HashSet;
import java.util.Set;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.Method;
import edu.umd.cs.findbugs.BugInstance;
import edu.umd.cs.findbugs.BugReporter;
import edu.umd.cs.findbugs.BytecodeScanningDetector;
import edu.umd.cs.findbugs.MethodAnnotation;
public class FindUnsyncGet extends BytecodeScanningDetector {
String prevClassName = " none ";
private BugReporter bugReporter;
static final int doNotConsider = ACC_PRIVATE | ACC_STATIC | ACC_NATIVE;
// Maps of property names to get and set methods
private HashMap<String, MethodAnnotation> getMethods = new HashMap<String, MethodAnnotation>();
private HashMap<String, MethodAnnotation> setMethods = new HashMap<String, MethodAnnotation>();
public FindUnsyncGet(BugReporter bugReporter) {
this.bugReporter = bugReporter;
}
@Override
public void report() {
// Find the set of properties for which we have both
// unsynchronized get and synchronized set methods
Set<String> commonProperties = new HashSet<String>(getMethods.keySet());
commonProperties.retainAll(setMethods.keySet());
// Report method pairs
for (String propName : commonProperties) {
MethodAnnotation getMethod = getMethods.get(propName);
MethodAnnotation setMethod = setMethods.get(propName);
bugReporter.reportBug(new BugInstance(this, "UG_SYNC_SET_UNSYNC_GET", NORMAL_PRIORITY).addClass(prevClassName)
.addMethod(getMethod).addMethod(setMethod));
}
getMethods.clear();
setMethods.clear();
}
@Override
public void visit(JavaClass obj) {
report();
prevClassName = getDottedClassName();
}
@Override
public void visit(Method obj) {
int flags = obj.getAccessFlags();
if ((flags & doNotConsider) != 0)
return;
String name = obj.getName();
boolean isSynchronized = (flags & ACC_SYNCHRONIZED) != 0;
/*
* String sig = obj.getSignature(); char firstArg = sig.charAt(1); char
* returnValue = sig.charAt(1 + sig.indexOf(')')); boolean firstArgIsRef
* = (firstArg == 'L') || (firstArg == '['); boolean returnValueIsRef =
* (returnValue == 'L') || (returnValue == '[');
*
* System.out.println(className + "." + name + " " + firstArgIsRef + " "
* + returnValueIsRef + " " + isSynchronized + " " + isNative );
*/
if (name.startsWith("get") && !isSynchronized
// && returnValueIsRef
) {
getMethods.put(name.substring(3), MethodAnnotation.fromVisitedMethod(this));
} else if (name.startsWith("set") && isSynchronized
// && firstArgIsRef
) {
setMethods.put(name.substring(3), MethodAnnotation.fromVisitedMethod(this));
}
}
}