/* * This file is part of LanternServer, licensed under the MIT License (MIT). * * Copyright (c) LanternPowered <https://www.lanternpowered.org> * Copyright (c) SpongePowered <https://www.spongepowered.org> * 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.lanternpowered.server.event.filter.delegate; import static org.objectweb.asm.Opcodes.ACONST_NULL; import static org.objectweb.asm.Opcodes.ALOAD; import static org.objectweb.asm.Opcodes.ARETURN; import static org.objectweb.asm.Opcodes.ASTORE; import static org.objectweb.asm.Opcodes.DUP; import static org.objectweb.asm.Opcodes.GOTO; import static org.objectweb.asm.Opcodes.IFEQ; import static org.objectweb.asm.Opcodes.IFNE; import static org.objectweb.asm.Opcodes.INSTANCEOF; import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL; import org.objectweb.asm.Label; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Type; import org.spongepowered.api.event.cause.Cause; import org.spongepowered.api.event.filter.cause.Last; import java.lang.reflect.Parameter; public class LastCauseFilterSourceDelegate extends CauseFilterSourceDelegate { private final Last anno; public LastCauseFilterSourceDelegate(Last anno) { this.anno = anno; } @Override protected void insertCauseCall(MethodVisitor mv, Parameter param, Class<?> targetType) { mv.visitLdcInsn(Type.getType(targetType)); mv.visitMethodInsn(INVOKEVIRTUAL, Type.getInternalName(Cause.class), "last", "(Ljava/lang/Class;)Ljava/util/Optional;", false); } @Override protected void insertTransform(MethodVisitor mv, Parameter param, Class<?> targetType, int local) { mv.visitVarInsn(ALOAD, local); Label failure = new Label(); Label success = new Label(); mv.visitMethodInsn(INVOKEVIRTUAL, "java/util/Optional", "isPresent", "()Z", false); mv.visitJumpInsn(IFEQ, failure); mv.visitVarInsn(ALOAD, local); mv.visitMethodInsn(INVOKEVIRTUAL, "java/util/Optional", "get", "()Ljava/lang/Object;", false); mv.visitVarInsn(ASTORE, local); mv.visitVarInsn(ALOAD, local); mv.visitTypeInsn(INSTANCEOF, Type.getInternalName(targetType)); if (this.anno.typeFilter().length != 0) { mv.visitJumpInsn(IFEQ, failure); mv.visitVarInsn(ALOAD, local); // For each type we do an instance check and jump to either failure or success if matched for (int i = 0; i < this.anno.typeFilter().length; i++) { Class<?> filter = this.anno.typeFilter()[i]; if (i < this.anno.typeFilter().length - 1) { mv.visitInsn(DUP); } mv.visitTypeInsn(INSTANCEOF, Type.getInternalName(filter)); if (this.anno.inverse()) { mv.visitJumpInsn(IFNE, failure); } else { mv.visitJumpInsn(IFNE, success); } } if (this.anno.inverse()) { mv.visitJumpInsn(GOTO, success); } // If the annotation was not reversed then we fall into failure as no types were matched } else { mv.visitJumpInsn(IFNE, success); } mv.visitLabel(failure); mv.visitInsn(ACONST_NULL); mv.visitInsn(ARETURN); mv.visitLabel(success); } }