org.evosuite.runtime.instrumentation.MethodCallReplacementClassAdapter Maven / Gradle / Ivy
/**
* Copyright (C) 2010-2016 Gordon Fraser, Andrea Arcuri and EvoSuite
* contributors
*
* This file is part of EvoSuite.
*
* EvoSuite 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 3.0 of the License, or
* (at your option) any later version.
*
* EvoSuite 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 Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with EvoSuite. If not, see .
*/
/**
*
*/
package org.evosuite.runtime.instrumentation;
import java.io.ObjectStreamClass;
import java.io.Serializable;
import java.util.Arrays;
import org.evosuite.runtime.RuntimeSettings;
import org.evosuite.runtime.annotation.EvoSuiteExclude;
import org.evosuite.runtime.mock.MockList;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.FieldVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
import org.objectweb.asm.Type;
import org.objectweb.asm.commons.GeneratorAdapter;
import org.objectweb.asm.commons.Method;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* MethodCallReplacementClassAdapter class.
*
* @author fraser
*/
public class MethodCallReplacementClassAdapter extends ClassVisitor {
private final String className;
private String superClassName;
private boolean definesHashCode = false;
private boolean isInterface = false;
private boolean definesUid = false;
private boolean canChangeSignature = true;
/**
* Constructor for MethodCallReplacementClassAdapter.
*
* @param cv a {@link org.objectweb.asm.ClassVisitor} object.
* @param className a {@link java.lang.String} object.
*/
public MethodCallReplacementClassAdapter(ClassVisitor cv, String className) {
this(cv,className,true);
}
public MethodCallReplacementClassAdapter(ClassVisitor cv, String className, boolean canAddMethods) {
super(Opcodes.ASM5, cv);
this.className = className;
this.superClassName = null;
this.canChangeSignature = canAddMethods;
}
/* (non-Javadoc)
* @see org.objectweb.asm.ClassVisitor#visitMethod(int, java.lang.String, java.lang.String, java.lang.String, java.lang.String[])
*/
/** {@inheritDoc} */
@Override
public MethodVisitor visitMethod(int access, String name, String desc,
String signature, String[] exceptions) {
if(name.equals("hashCode"))
definesHashCode = true;
MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);
if(name.equals("")) {
mv = new RegisterObjectForDeterministicHashCodeVisitor(mv, access, name, desc);
}
return new MethodCallReplacementMethodAdapter(mv, className, superClassName, name, access, desc);
}
@Override
public FieldVisitor visitField(int access, String name, String desc,
String signature, Object value) {
if(name.equals("serialVersionUID")) {
definesUid = true;
}
return super.visitField(access, name, desc, signature, value);
}
@Override
public void visit(int version, int access, String name, String signature,
String superName, String[] interfaces) {
String superNameWithDots = superName.replace('/', '.');
superClassName = superNameWithDots;
if((access & Opcodes.ACC_INTERFACE) == Opcodes.ACC_INTERFACE)
isInterface = true;
else {
/*
FIXME: this should be moved in its own adapter, because it is not executed if we do
only reset of static state and no mocking
*/
boolean found = false;
String instrumentedInterface = InstrumentedClass.class.getCanonicalName().replace('.', '/');
for(String interf : interfaces) {
if(interf.equals(instrumentedInterface))
found = true;
}
if(!found) {
logger.info("Adding mock interface to class "+name);
String[] mockedInterfaces = Arrays.copyOf(interfaces, interfaces.length + 1);
mockedInterfaces[interfaces.length] = InstrumentedClass.class.getCanonicalName().replace('.', '/');
interfaces = mockedInterfaces;
}
}
if(MockList.shouldBeMocked(superNameWithDots)) {
/*
* TODO: likely need to suppress the change of superclass if !canChangeSignature
*/
Class> mockSuperClass = MockList.getMockClass(superNameWithDots);
String mockSuperClassName = mockSuperClass.getCanonicalName().replace('.', '/');
super.visit(version, access, name, signature, mockSuperClassName, interfaces);
} else {
super.visit(version, access, name, signature, superName, interfaces);
}
}
private static final Logger logger = LoggerFactory.getLogger(MethodCallReplacementClassAdapter.class);
@Override
public void visitEnd() {
if(canChangeSignature && !definesHashCode && !isInterface && RuntimeSettings.mockJVMNonDeterminism) {
logger.info("No hashCode defined for: "+className+", superclass = "+superClassName);
if(superClassName.equals("java.lang.Object")) { //TODO: why only if superclass is Object??? unclear
Method hashCodeMethod = Method.getMethod("int hashCode()");
GeneratorAdapter mg = new GeneratorAdapter(Opcodes.ACC_PUBLIC, hashCodeMethod, null, null, this);
mg.loadThis();
mg.visitAnnotation(Type.getDescriptor(EvoSuiteExclude.class), true);
mg.invokeStatic(Type.getType(org.evosuite.runtime.System.class), Method.getMethod("int identityHashCode(Object)"));
mg.returnValue();
mg.endMethod();
}
}
/*
* If the class is serializable, then doing any change (adding hashCode, static reset, etc)
* will change the serialVersionUID if it is not defined in the class.
* Hence, if it is not defined, we have to define it to
* avoid problems in serialising the class, as reading Master will not do instrumentation.
* The serialVersionUID HAS to be the same as the un-instrumented class
*/
if(!definesUid && !isInterface && RuntimeSettings.applyUIDTransformation) {
ClassLoader threadCL = Thread.currentThread().getContextClassLoader();
try {
ClassLoader evoCL = MethodCallReplacementClassAdapter.class.getClassLoader();
Thread.currentThread().setContextClassLoader(evoCL);
Class> clazz = Class.forName(className.replace('/', '.'), false, evoCL);
if(Serializable.class.isAssignableFrom(clazz)) {
ObjectStreamClass c = ObjectStreamClass.lookup(clazz);
long serialID = c.getSerialVersionUID();
logger.info("Adding serialId to class "+className);
visitField(Opcodes.ACC_PRIVATE | Opcodes.ACC_STATIC | Opcodes.ACC_FINAL, "serialVersionUID", "J", null, serialID);
}
} catch(ClassNotFoundException e) {
logger.warn("Failed to add serialId to class "+className+": "+e.getMessage());
} catch (NoClassDefFoundError e) {
logger.warn("Failed to add serialId to class "+className+": "+e.getMessage());
} finally {
Thread.currentThread().setContextClassLoader(threadCL);
}
}
super.visitEnd();
}
}