/**
* Copyright 2010 JBoss Inc
*
* 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.
*/
package org.drools.core.util.asm;
import java.util.List;
import org.mvel2.asm.AnnotationVisitor;
import org.mvel2.asm.Attribute;
import org.mvel2.asm.ClassReader;
import org.mvel2.asm.ClassVisitor;
import org.mvel2.asm.FieldVisitor;
import org.mvel2.asm.MethodVisitor;
import org.mvel2.asm.util.TraceMethodVisitor;
/**
* The purpose of this utility it to check if 2 method implementations are equivalent, by comparing the bytecode.
* This essentual for node sharing where java semantics are involved.
* @author Michael Neale
*/
public class MethodComparator {
/**
* This actually does the comparing.
* Class1 and Class2 are class reader instances to the respective classes. method1 and method2 are looked up on the
* respective classes and their contents compared.
*
* This is a convenience method.
*/
public boolean equivalent(final String method1,
final ClassReader class1,
final String method2,
final ClassReader class2) {
final List list1 = getMethodBytecode( method1,
class1 );
final List list2 = getMethodBytecode( method2,
class2 );
return compareBytecode( list1,
list2 );
}
/**
* This will return a series of bytecode instructions which can be used to compare one method with another.
* debug info like local var declarations and line numbers are ignored, so the focus is on the content.
*/
public List getMethodBytecode(final String methodName,
final ClassReader classReader) {
final Tracer visit = new Tracer( methodName );
classReader.accept( visit,
ClassReader.SKIP_DEBUG );
final TraceMethodVisitor trace = visit.getTrace();
return trace.getText();
}
/**
* This will return a series of bytecode instructions which can be used to compare one method with another.
* debug info like local var declarations and line numbers are ignored, so the focus is on the content.
*/
public static List getMethodBytecode(final String methodName,
final byte[] bytes) {
final Tracer visit = new Tracer( methodName );
final ClassReader classReader = new ClassReader( bytes );
classReader.accept( visit,
ClassReader.SKIP_DEBUG );
final TraceMethodVisitor trace = visit.getTrace();
return trace.getText();
}
/**
* Compares 2 bytecode listings.
* Returns true if they are identical.
*/
public static boolean compareBytecode(final List b1,
final List b2) {
if ( b1.size() != b2.size() ) {
return false;
}
for ( int i = 0; i < b1.size(); i++ ) {
if ( !(b1.get( i ).equals( b2.get( i ) )) ) {
return false;
}
}
return true;
}
public static class Tracer
implements
ClassVisitor {
private TraceMethodVisitor trace;
private String methodName;
public Tracer(final String methodName) {
this.methodName = methodName;
}
public void visit(final int version,
final int access,
final String name,
final String signature,
final String superName,
final String[] interfaces) {
}
public AnnotationVisitor visitAnnotation(final String desc,
final boolean visible) {
return new DummyAnnotationVisitor();
}
public void visitAttribute(final Attribute attr) {
}
public void visitEnd() {
}
public FieldVisitor visitField(final int access,
final String name,
final String desc,
final String signature,
final Object value) {
return null;
}
public void visitInnerClass(final String name,
final String outerName,
final String innerName,
final int access) {
}
public MethodVisitor visitMethod(final int access,
final String name,
final String desc,
final String signature,
final String[] exceptions) {
if ( this.methodName.equals( name ) ) {
this.trace = new TraceMethodVisitor();
return this.trace;
}
return null;
}
public void visitOuterClass(final String owner,
final String name,
final String desc) {
}
public void visitSource(final String source,
final String debug) {
}
public TraceMethodVisitor getTrace() {
return this.trace;
}
}
static class DummyAnnotationVisitor
implements
AnnotationVisitor {
public void visit(final String name,
final Object value) {
}
public AnnotationVisitor visitAnnotation(final String name,
final String desc) {
return new DummyAnnotationVisitor();
}
public AnnotationVisitor visitArray(final String name) {
return new DummyAnnotationVisitor();
}
public void visitEnd() {
}
public void visitEnum(final String name,
final String desc,
final String value) {
}
}
}