net.bytebuddy.instrumentation.method.bytecode.stack.member.MethodInvocation Maven / Gradle / Ivy
package net.bytebuddy.instrumentation.method.bytecode.stack.member;
import net.bytebuddy.instrumentation.Instrumentation;
import net.bytebuddy.instrumentation.method.MethodDescription;
import net.bytebuddy.instrumentation.method.bytecode.stack.StackManipulation;
import net.bytebuddy.instrumentation.type.TypeDescription;
import net.bytebuddy.jar.asm.MethodVisitor;
import net.bytebuddy.jar.asm.Opcodes;
/**
* A builder for a method invocation.
*/
public enum MethodInvocation {
/**
* A virtual method invocation.
*/
VIRTUAL(Opcodes.INVOKEVIRTUAL),
/**
* An interface-typed virtual method invocation.
*/
INTERFACE(Opcodes.INVOKEINTERFACE),
/**
* A static method invocation.
*/
STATIC(Opcodes.INVOKESTATIC),
/**
* A specialized virtual method invocation.
*/
SPECIAL(Opcodes.INVOKESPECIAL);
/**
* The opcode for invoking a method.
*/
private final int invocationOpcode;
/**
* Creates a new type of method invocation.
*
* @param callOpcode The opcode for invoking a method.
*/
private MethodInvocation(int callOpcode) {
this.invocationOpcode = callOpcode;
}
/**
* Creates a method invocation with an implicitly determined invocation type.
*
* @param methodDescription The method to be invoked.
* @return A stack manipulation with implicitly determined invocation type.
*/
public static WithImplicitInvocationTargetType invoke(MethodDescription methodDescription) {
if (methodDescription.isTypeInitializer()) {
throw new IllegalArgumentException("The type initializer cannot be invoked explicitly");
} else if (methodDescription.isStatic()) { // Check this property first, private static methods must use INVOKESTATIC
return STATIC.new Invocation(methodDescription);
} else if (methodDescription.isPrivate() || methodDescription.isConstructor() || methodDescription.isDefaultMethod()) {
return SPECIAL.new Invocation(methodDescription);
} else if (methodDescription.getDeclaringType().isInterface()) { // Check this property last, default methods must be called by INVOKESPECIAL
return INTERFACE.new Invocation(methodDescription);
} else {
return VIRTUAL.new Invocation(methodDescription);
}
}
/**
* Represents a method invocation where the invocation type (static, virtual, special, interface) is derived
* from the given method's description.
*/
public static interface WithImplicitInvocationTargetType extends StackManipulation {
/**
* Transforms this method invocation into a virtual (or interface) method invocation on the given type. If the
* represented method cannot be dispatched on the given invocation target type using virtual invocation,
* an exception is thrown.
*
* @param invocationTarget The type on which the method is to be invoked virtually on.
* @return A stack manipulation representing this method invocation.
*/
StackManipulation virtual(TypeDescription invocationTarget);
/**
* Transforms this method invocation into a special invocation on the given type. If the represented method
* cannot be dispatched on the given invocation target type using special invocation, an exception is thrown.
*
* @param invocationTarget The type on which the method is to be invoked specially on.
* @return A stack manipulation representing this method invocation.
*/
StackManipulation special(TypeDescription invocationTarget);
/**
* Returns the invocation type that was determined implicitly for the given method.
*
* @return The method invocation type that was determined implicitly for the given method.
*/
MethodInvocation getImplicitInvocationType();
}
/**
* An implementation of a method invoking stack manipulation.
*/
private class Invocation implements WithImplicitInvocationTargetType {
/**
* The method to be invoked.
*/
private final TypeDescription typeDescription;
/**
* The type on which this method is to be invoked.
*/
private final MethodDescription methodDescription;
/**
* The operand stack size implication of applying this invocation.
*/
private final Size size;
/**
* Creates an invocation of a given method on its declaring type as an invocation target.
*
* @param methodDescription The method to be invoked.
*/
private Invocation(MethodDescription methodDescription) {
this(methodDescription, methodDescription.getDeclaringType());
}
/**
* Creates an invocation of a given method on a given invocation target type.
*
* @param methodDescription The method to be invoked.
* @param typeDescription The type on which this method is to be invoked.
*/
private Invocation(MethodDescription methodDescription, TypeDescription typeDescription) {
this.typeDescription = typeDescription;
this.methodDescription = methodDescription;
int parameterSize = methodDescription.getStackSize();
int returnValueSize = methodDescription.getReturnType().getStackSize().getSize();
size = new Size(returnValueSize - parameterSize, Math.max(0, returnValueSize - parameterSize));
}
@Override
public boolean isValid() {
return true;
}
@Override
public Size apply(MethodVisitor methodVisitor, Instrumentation.Context instrumentationContext) {
methodVisitor.visitMethodInsn(invocationOpcode,
typeDescription.getInternalName(),
methodDescription.getInternalName(),
methodDescription.getDescriptor(),
typeDescription.isInterface());
return size;
}
@Override
public StackManipulation virtual(TypeDescription invocationTarget) {
if (methodDescription.isPrivate() || methodDescription.isConstructor() || methodDescription.isStatic()) {
return Illegal.INSTANCE;
}
if (invocationTarget.isInterface()) {
return INTERFACE.new Invocation(methodDescription, invocationTarget);
} else {
return VIRTUAL.new Invocation(methodDescription, invocationTarget);
}
}
@Override
public StackManipulation special(TypeDescription invocationTarget) {
if (!methodDescription.isSpecializableFor(invocationTarget)) {
return Illegal.INSTANCE;
}
return SPECIAL.new Invocation(methodDescription, invocationTarget);
}
@Override
public MethodInvocation getImplicitInvocationType() {
return MethodInvocation.this;
}
@Override
public boolean equals(Object other) {
if (this == other) return true;
if (other == null || getClass() != other.getClass()) return false;
Invocation that = (Invocation) other;
return MethodInvocation.this.equals(((Invocation) other).getImplicitInvocationType())
&& methodDescription.getInternalName().equals(that.methodDescription.getInternalName())
&& methodDescription.getReturnType().equals(((Invocation) other).methodDescription.getReturnType())
&& methodDescription.getParameterTypes().equals(((Invocation) other).methodDescription.getParameterTypes())
&& typeDescription.equals(that.typeDescription);
}
@Override
public int hashCode() {
int result = typeDescription.hashCode();
result = 31 * result + MethodInvocation.this.hashCode();
result = 31 * result + methodDescription.getInternalName().hashCode();
result = 31 * result + methodDescription.getParameterTypes().hashCode();
result = 31 * result + methodDescription.getReturnType().hashCode();
return result;
}
@Override
public String toString() {
return "MethodInvocation.Invocation{" +
"typeDescription=" + typeDescription +
", methodDescription=" + methodDescription +
", size=" + size +
'}';
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy