org.mutabilitydetector.checkers.EscapedThisReferenceChecker Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of MutabilityDetector Show documentation
Show all versions of MutabilityDetector Show documentation
Lightweight analysis tool for detecting mutability in Java
classes.
package org.mutabilitydetector.checkers;
/*
* #%L
* MutabilityDetector
* %%
* Copyright (C) 2008 - 2014 Graham Allan
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* #L%
*/
import java.util.ArrayList;
import java.util.List;
import org.mutabilitydetector.MutabilityReason;
import org.mutabilitydetector.asmoverride.AsmCompatibility;
import org.mutabilitydetector.checkers.util.StackPushingOpcodes;
import org.mutabilitydetector.locations.CodeLocation.ClassLocation;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
import org.objectweb.asm.Type;
import org.objectweb.asm.tree.AbstractInsnNode;
import org.objectweb.asm.tree.FieldInsnNode;
import org.objectweb.asm.tree.MethodInsnNode;
import org.objectweb.asm.tree.MethodNode;
import org.objectweb.asm.tree.VarInsnNode;
public final class EscapedThisReferenceChecker extends AsmMutabilityChecker {
@Override
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
super.visitMethod(access, name, desc, signature, exceptions);
return MethodIs.aConstructor(name)
? new ThisEscapingFromConstructorVistor(access, name, desc, signature, exceptions)
: null;
}
private final class ThisEscapingFromConstructorVistor extends MethodNode {
private final List methodCalls = new ArrayList();
private final List fieldAssignmentsInConstructor = new ArrayList();
private final StackPushingOpcodes stackPushingOpcodes = new StackPushingOpcodes();
public ThisEscapingFromConstructorVistor(int access,
String name,
String desc,
String signature,
String[] exceptions) {
super(AsmCompatibility.AsmApiVersion, access, name, desc, signature, exceptions);
}
@Override
public void visitMethodInsn(int opcode, String owner, String methodName, String methodDesc, boolean isInterface) {
super.visitMethodInsn(opcode, owner, methodName, methodDesc, isInterface);
if (MethodIs.aConstructor(methodName) && owner.equals("java/lang/Object")) { return; }
methodCalls.add((MethodInsnNode) instructions.getLast());
}
@Override
public void visitFieldInsn(int opcode, String owner, String fieldName, String fieldDesc) {
super.visitFieldInsn(opcode, owner, fieldName, fieldDesc);
if (opcode == Opcodes.PUTSTATIC || opcode == Opcodes.PUTFIELD) {
fieldAssignmentsInConstructor.add((FieldInsnNode) instructions.getLast());
}
}
@Override
public void visitEnd() {
super.visitEnd();
checkForPassingThisReferenceAsParameter();
checkForSettingFieldToThisReference();
}
private void checkForSettingFieldToThisReference() {
if (fieldAssignmentsInConstructor.isEmpty()) { return; }
for (FieldInsnNode fieldInstruction : fieldAssignmentsInConstructor) {
checkFieldAssignment(fieldInstruction);
}
}
private void checkFieldAssignment(FieldInsnNode assignment) {
AbstractInsnNode previous = assignment.getPrevious();
if (stackPushingOpcodes.includes(previous.getOpcode())) {
checkForThisReferenceBeingPutOnStack(previous);
}
}
private void checkForPassingThisReferenceAsParameter() {
if (methodCalls.isEmpty()) { return; }
for (MethodInsnNode methodInsnNode : methodCalls) {
checkMethodCall(methodInsnNode);
}
}
private void checkMethodCall(MethodInsnNode methodInsnNode) {
AbstractInsnNode previous = methodInsnNode.getPrevious();
Type[] argumentTypes = Type.getArgumentTypes(methodInsnNode.desc);
int numberOfArguments = argumentTypes.length;
for (int i = numberOfArguments - 1; i >= 0; i--) {
if (instructionPushesSomethingElseOnTheStack(previous)) {
i = i + 1;
}
checkForThisReferenceBeingPutOnStack(previous);
previous = previous.getPrevious();
}
}
private boolean instructionPushesSomethingElseOnTheStack(AbstractInsnNode previous) {
switch(previous.getOpcode()) {
case Opcodes.DUP:
case Opcodes.NEW:
return true;
default:
return false;
}
}
private void checkForThisReferenceBeingPutOnStack(AbstractInsnNode previous) {
if (previous instanceof VarInsnNode) {
VarInsnNode varInstruction = (VarInsnNode) previous;
if (varInstruction.var == 0) {
thisReferencesEscapes();
}
}
}
private void thisReferencesEscapes() {
setResult("The 'this' reference is passed outwith the constructor.",
ClassLocation.fromInternalName(ownerClass),
MutabilityReason.ESCAPED_THIS_REFERENCE);
}
}
}