org.spongepowered.asm.mixin.injection.callback.CallbackInfo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of sponge-mixin Show documentation
Show all versions of sponge-mixin Show documentation
Fabric Mixin is a trait/mixin and bytecode weaving framework for Java using ASM.
The newest version!
/*
* This file is part of Mixin, licensed under the MIT License (MIT).
*
* Copyright (c) SpongePowered
* Copyright (c) contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package org.spongepowered.asm.mixin.injection.callback;
import org.objectweb.asm.Type;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.util.Constants;
/**
* CallbackInfo instances are passed to callbacks in order to provide
* information and handling opportunities to the callback to interact with the
* callback itself. For example by allowing the callback to be "cancelled" and
* return from a method prematurely.
*/
public class CallbackInfo implements Cancellable {
/**
* Method name being injected into, this is useful if a single callback is
* injecting into multiple methods.
*/
private final String name;
/**
* True if this callback is cancellable
*/
private final boolean cancellable;
/**
* True if this callback has been cancelled
*/
private boolean cancelled;
/**
* This ctor is always called by injected code
*
* @param name calling method name
* @param cancellable true if the callback can be cancelled
*/
public CallbackInfo(String name, boolean cancellable) {
this.name = name;
this.cancellable = cancellable;
}
/**
* Get the ID of the injector which defined this callback. This defaults to
* the method name but can be overridden by specifying the {@link Inject#id}
* parameter on the injector
*
* @return the injector ID
*/
public String getId() {
return this.name;
}
/* (non-Javadoc)
* @see java.lang.Object#toString()
*/
@Override
public String toString() {
return String.format("CallbackInfo[TYPE=%s,NAME=%s,CANCELLABLE=%s]", this.getClass().getSimpleName(), this.name, this.cancellable);
}
@Override
public final boolean isCancellable() {
return this.cancellable;
}
@Override
public final boolean isCancelled() {
return this.cancelled;
}
/* (non-Javadoc)
* @see org.spongepowered.asm.mixin.injection.callback.Cancellable#cancel()
*/
@Override
public void cancel() throws CancellationException {
if (!this.cancellable) {
throw new CancellationException(String.format("The call %s is not cancellable.", this.name));
}
this.cancelled = true;
}
// Methods below this point used by the CallbackInjector
static String getCallInfoClassName() {
return CallbackInfo.class.getName();
}
/**
* Gets the {@link CallbackInfo} class name to use for the specified return
* type. Currently returns {@link CallbackInfo} for void types and
* {@link CallbackInfoReturnable} for non-void types.
*
* @param returnType return type of the target method
* @return CallbackInfo class name to use
*/
public static String getCallInfoClassName(Type returnType) {
return (returnType.equals(Type.VOID_TYPE) ? CallbackInfo.class.getName() : CallbackInfoReturnable.class.getName()).replace('.', '/');
}
static String getConstructorDescriptor(Type returnType) {
if (returnType.equals(Type.VOID_TYPE)) {
return CallbackInfo.getConstructorDescriptor();
}
if (returnType.getSort() == Type.OBJECT || returnType.getSort() == Type.ARRAY) {
return String.format("(%sZ%s)V", Constants.STRING_DESC, Constants.OBJECT_DESC);
}
return String.format("(%sZ%s)V", Constants.STRING_DESC, returnType.getDescriptor());
}
static String getConstructorDescriptor() {
return String.format("(%sZ)V", Constants.STRING_DESC);
}
static String getIsCancelledMethodName() {
return "isCancelled";
}
static String getIsCancelledMethodSig() {
return "()Z";
}
}