/* * Copyright (C) 2009-2015 The Project Lombok Authors. * * 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 lombok.patcher.scripts; import java.util.ArrayList; import java.util.List; import lombok.patcher.Hook; import lombok.patcher.MethodLogistics; import lombok.patcher.TargetMatcher; import lombok.patcher.TransplantMapper; import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.ClassWriter; import org.objectweb.asm.Label; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Opcodes; /** * Will wrap any invocation to a given method in another given method by setting a symbol for the duration * of the method, properly guarded with a try/finally block. See {@link lombok.patcher.Symbols} for how to * query symbols states. */ public class SetSymbolDuringMethodCallScript extends MethodLevelPatchScript { private final Hook callToWrap; private final String symbol; private final boolean report; SetSymbolDuringMethodCallScript(List<TargetMatcher> matchers, Hook callToWrap, String symbol, boolean report) { super(matchers); if (callToWrap == null) throw new NullPointerException("callToWrap"); if (symbol == null) throw new NullPointerException("symbol"); this.callToWrap = callToWrap; this.symbol = symbol; this.report = report; } @Override protected MethodPatcher createPatcher(ClassWriter writer, final String classSpec, TransplantMapper transplantMapper) { final List<WrapperMethodDescriptor> descriptors = new ArrayList<WrapperMethodDescriptor>(); final MethodPatcher patcher = new MethodPatcher(writer, transplantMapper, new MethodPatcherFactory() { public MethodVisitor createMethodVisitor(String name, String desc, MethodVisitor parent, MethodLogistics logistics) { return new WrapWithSymbol(name, parent, classSpec, descriptors); } }) { @Override public void visitEnd() { for (WrapperMethodDescriptor wmd : descriptors) { makeWrapperMethod(this, wmd); } super.visitEnd(); } }; return patcher; } private void makeWrapperMethod(ClassVisitor cv, WrapperMethodDescriptor wmd) { MethodVisitor mv = cv.visitMethod(Opcodes.ACC_SYNTHETIC | Opcodes.ACC_PRIVATE | Opcodes.ACC_STATIC, wmd.getWrapperName(), wmd.getWrapperDescriptor(), null, null); MethodLogistics logistics = new MethodLogistics(Opcodes.ACC_STATIC, wmd.getWrapperDescriptor()); mv.visitCode(); Label start = new Label(); Label end = new Label(); Label handler = new Label(); mv.visitTryCatchBlock(start, end, handler, null); mv.visitLabel(start); mv.visitLdcInsn(symbol); mv.visitMethodInsn(Opcodes.INVOKESTATIC, "lombok/patcher/Symbols", "push", "(Ljava/lang/String;)V", false); for (int i = 0; i < logistics.getParamCount(); i++) { logistics.generateLoadOpcodeForParam(i, mv); } mv.visitMethodInsn(wmd.getOpcode(), wmd.getOwner(), wmd.getName(), wmd.getTargetDescriptor(), wmd.isItf()); mv.visitLabel(end); mv.visitMethodInsn(Opcodes.INVOKESTATIC, "lombok/patcher/Symbols", "pop", "()V", false); logistics.generateReturnOpcode(mv); mv.visitLabel(handler); mv.visitFrame(Opcodes.F_FULL, 0, null, 1, new Object[] {"java/lang/Throwable"}); mv.visitMethodInsn(Opcodes.INVOKESTATIC, "lombok/patcher/Symbols", "pop", "()V", false); mv.visitInsn(Opcodes.ATHROW); mv.visitMaxs(Math.max(1, logistics.getParamCount()), logistics.getParamCount()); mv.visitEnd(); } private class WrapWithSymbol extends MethodVisitor { private final String selfMethodName; private final String selfTypeName; private final List<WrapperMethodDescriptor> descriptors; public WrapWithSymbol(String selfMethodName, MethodVisitor mv, String selfTypeName, List<WrapperMethodDescriptor> descriptors) { super(Opcodes.ASM4, mv); this.selfMethodName = selfMethodName; this.selfTypeName = selfTypeName; this.descriptors = descriptors; } @Override public void visitMethodInsn(int opcode, String owner, String name, String desc, boolean itf) { boolean addOwner; if (opcode == Opcodes.INVOKEINTERFACE || opcode == Opcodes.INVOKEVIRTUAL) addOwner = true; else if (opcode == Opcodes.INVOKESTATIC) addOwner = false; else { super.visitMethodInsn(opcode, owner, name, desc, itf); return; } if (!callToWrap.getClassSpec().equals(owner) || !callToWrap.getMethodName().equals(name) || !callToWrap.getMethodDescriptor().equals(desc)) { super.visitMethodInsn(opcode, owner, name, desc, itf); return; } String fixedDesc; if (addOwner) { fixedDesc = "(L" + callToWrap.getClassSpec() + ";" + desc.substring(1); } else { fixedDesc = desc; } WrapperMethodDescriptor wmd = new WrapperMethodDescriptor(descriptors.size(), opcode, owner, name, fixedDesc, desc, itf); if (report) System.out.println("Changing method " + selfTypeName + "::" + selfMethodName + " wrapping call to " + owner + "::" + name + " to set symbol " + symbol); super.visitMethodInsn(Opcodes.INVOKESTATIC, selfTypeName, wmd.getWrapperName(), fixedDesc, false); descriptors.add(wmd); } } }