All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.evosuite.runtime.instrumentation.MethodCallReplacementClassAdapter Maven / Gradle / Ivy

There is a newer version: 1.0.6
Show newest version
/**
 * 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(); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy