字节伙伴堆栈操作:如何使用局部变量和if语句

时间:2017-06-21 19:11:22

标签: java byte-buddy

一般来说,问题是:

  • ByteBuddy如何以及何时生成局部变量表和堆栈映射帧?
  • 使用本地变量并在ByteBuddy Implementation API中生成if语句的正确方法是什么?

详细说明:

我使用bytebuddy来生成某些类的equals方法。为此,我使用net.bytebuddy.implementation.Implementation的自定义实现。理论上,我计划生成的字节码应该具有几乎以下语义:

@Override
public boolean equals(Object obj) {
    if (this == obj) {
        return true;
    }
    if (obj == null) {
        return false;
    }
    if (getClass() != obj.getClass()) {
        return false;
    }
    final T other = (T) obj;
    if (this.a != other.a) {
        return false;
    }
    if (!Objects.equals(this.b, other.b)) {
        return false;
    }
    return true;
}

上面的代码有一个局部变量和几个 if s。我还没有为他们两个找到官方StackManipulation,所以:

  • 对于使用本地人,我使用MethodVariableAccess.REFERENCE.loadFromMethodVariableAccess.REFERENCE.storeAt
  • 为了生成if语句,我使用StackManipulation s
  • 的自定义实现

像:

interface Branching extends StackManipulation {
    @Override
    default boolean isValid() {
        return true;
    }

    class Mark implements Branching {

        private final Label label;

        public Mark(Label label) {
            this.label = label;
        }

        @Override
        public final Size apply(MethodVisitor mv, Implementation.Context ctx) {
            mv.visitLabel(label);
            return new Size(0, 0);
        }
    }

    class IfNe implements Branching {

        private final Label label;

        public IfNe(Label label) {
            this.label = label;
        }

        @Override
        public final Size apply(MethodVisitor mv, Implementation.Context ctx) {
            mv.visitJumpInsn(Opcodes.IFNE, label);
            return new Size(-2, 0);
        }
    }
}

看起来我做错了,因为生成的字节码错过了局部变量表和堆栈映射帧。并且它当然没有通过验证,抱怨"在分支目标X"期待堆栈图框架。

更新:

我认为值得在这里添加一个例子。我刚才谈到的初始案例非常庞大,所以我重新编写了一个模块来演示一个问题。它非常大,但我无法想象如何缩小它:

package com.xxx.proba.bytebuddy;

import java.io.File;
import java.io.PrintStream;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import net.bytebuddy.ByteBuddy;
import net.bytebuddy.description.field.FieldDescription;
import net.bytebuddy.description.method.MethodDescription;
import net.bytebuddy.description.modifier.Visibility;
import net.bytebuddy.dynamic.DynamicType;
import net.bytebuddy.dynamic.scaffold.InstrumentedType;
import net.bytebuddy.implementation.Implementation;
import net.bytebuddy.implementation.bytecode.ByteCodeAppender;
import net.bytebuddy.implementation.bytecode.StackManipulation;
import net.bytebuddy.implementation.bytecode.collection.ArrayAccess;
import net.bytebuddy.implementation.bytecode.constant.IntegerConstant;
import net.bytebuddy.implementation.bytecode.constant.TextConstant;
import net.bytebuddy.implementation.bytecode.member.FieldAccess;
import net.bytebuddy.implementation.bytecode.member.MethodInvocation;
import net.bytebuddy.implementation.bytecode.member.MethodReturn;
import net.bytebuddy.implementation.bytecode.member.MethodVariableAccess;
import net.bytebuddy.jar.asm.Label;
import net.bytebuddy.jar.asm.MethodVisitor;
import net.bytebuddy.jar.asm.Opcodes;


/**
 * Assuming that I want to generate the class like this one
 * 
 * @author skapral
 */
class ExampleClass {
    public void main(String[] args) {
        if (args[0].equals("")) {
            System.out.println("a");
        } else {
            System.out.println("b");
        }
    }
}

public class Main {
    private final static Method EQUALS;
    private final static Method PRINTLN;
    private final static Field SYSTEM_OUT;

    static {
        try {
            EQUALS = Object.class.getMethod("equals", Object.class);
            PRINTLN = PrintStream.class.getMethod("println", String.class);
            SYSTEM_OUT = System.class.getField("out");
        } catch (Exception ex) {
            throw new RuntimeException(ex);
        }
    }

    public static void main(String[] args) throws Exception {
        DynamicType.Unloaded<Object> unloaded = new ByteBuddy()
                .subclass(Object.class)
                .name("com.echelon.proba.bytebuddy.ExampleClassGenerated")
                .defineMethod("main", void.class, Visibility.PUBLIC)
                .withParameter(String[].class)
                .intercept(new Implementation() {
                    @Override
                    public ByteCodeAppender appender(Implementation.Target implementationTarget) {
                        return new ByteCodeAppender() {
                            @Override
                            public ByteCodeAppender.Size apply(MethodVisitor mv, Implementation.Context ctx, MethodDescription md) {
                                Label ifLabel = new Label();
                                Label elseLabel = new Label();

                                StackManipulation.Size size = new StackManipulation.Compound(
                                        MethodVariableAccess.REFERENCE.loadFrom(1),
                                        IntegerConstant.ZERO,
                                        ArrayAccess.REFERENCE.load(),
                                        new TextConstant(""),
                                        MethodInvocation.invoke(new MethodDescription.ForLoadedMethod(EQUALS)), 
                                        new IfEq(ifLabel),
                                        FieldAccess.forField(new FieldDescription.ForLoadedField(SYSTEM_OUT)).read(),
                                        new TextConstant("a"),
                                        MethodInvocation.invoke(new MethodDescription.ForLoadedMethod(PRINTLN)),
                                        new GoTo(elseLabel),
                                        new Mark(ifLabel),
                                        FieldAccess.forField(new FieldDescription.ForLoadedField(SYSTEM_OUT)).read(),
                                        new TextConstant("b"),
                                        MethodInvocation.invoke(new MethodDescription.ForLoadedMethod(PRINTLN)),
                                        new Mark(elseLabel),
                                        MethodReturn.VOID
                                ).apply(mv, ctx);
                                return new Size(size.getMaximalSize(), md.getStackSize());
                            }
                        };
                    }

                    @Override
                    public InstrumentedType prepare(InstrumentedType instrumentedType) {
                        return instrumentedType;
                    }
                })
                .make();
        unloaded.saveIn(new File("/tmp/aaa")); /* Preserve it for future investigation by javap */
        Object obj = unloaded.load(Main.class.getClassLoader()).getLoaded().newInstance();
        obj.getClass().getMethod("main", String[].class).invoke(obj, new String[] {"aaa"}); /* Trigger class loading and verification */
    }
}


class IfEq implements StackManipulation {
    private final Label label;

    public IfEq(Label label) {
        this.label = label;
    }

    @Override
    public boolean isValid() {
        return true;
    }

    @Override
    public StackManipulation.Size apply(MethodVisitor mv, Implementation.Context ctx) {
        mv.visitJumpInsn(Opcodes.IFEQ, label);
        return new StackManipulation.Size(-1, 0);
    }
}

class GoTo implements StackManipulation {
    private final Label label;

    public GoTo(Label label) {
        this.label = label;
    }

    @Override
    public boolean isValid() {
        return true;
    }

    @Override
    public StackManipulation.Size apply(MethodVisitor mv, Implementation.Context ctx) {
        mv.visitJumpInsn(Opcodes.GOTO, label);
        return new StackManipulation.Size(0, 0);
    }
}

class Mark implements StackManipulation {
    private final Label label;

    public Mark(Label label) {
        this.label = label;
    }

    @Override
    public boolean isValid() {
        return true;
    }

    @Override
    public StackManipulation.Size apply(MethodVisitor mv, Implementation.Context ctx) {
        mv.visitLabel(label);
        return new StackManipulation.Size(0, 0);
    }
}

在该示例中,在Main :: main方法中,我试图通过bytebuddy生成简单的ExampleClass。在尝试加载它并调用方法时,我得到了VerifyError。

  Location:
    com/echelon/proba/bytebuddy/ExampleClassGenerated.main([Ljava/lang/String;)V @8: ifeq
  Reason:
    Expected stackmap frame at this location.
  Bytecode:
    0x0000000: 2b03 3212 08b6 000c 9900 0eb2 0012 1214
    0x0000010: b600 1aa7 000b b200 1212 1cb6 001a b1  

    at java.lang.Class.getDeclaredConstructors0(Native Method)
    at java.lang.Class.privateGetDeclaredConstructors(Class.java:2671)
    at java.lang.Class.getConstructor0(Class.java:3075)
    at java.lang.Class.newInstance(Class.java:412)
    at com.xxx.proba.bytebuddy.Main.main(Main.java:103)

更新:总结一下:这个简单的AsmVisitorWrapper帮助了我:

public class EnableFramesComputing implements AsmVisitorWrapper {
    @Override
    public final int mergeWriter(int flags) {
        return flags | ClassWriter.COMPUTE_FRAMES;
    }

    @Override
    public final int mergeReader(int flags) {
        return flags | ClassWriter.COMPUTE_FRAMES;
    }

    @Override
    public final ClassVisitor wrap(TypeDescription td, ClassVisitor cv, Implementation.Context ctx, TypePool tp, FieldList<FieldDescription.InDefinedShape> fields, MethodList<?> methods, int wflags, int rflags) {
        return cv;
    }
}

可以通过调用DynamicType.Builder上的访问来实现它,例如:

DynamicType.Unloaded<Object> unloaded = new ByteBuddy()
            .subclass(Object.class)
            .visit(new EnableFramesComputing())
            ...

1 个答案:

答案 0 :(得分:2)

Byte Buddy打算成为一个高级字节代码操作库。如果你想创建低级字节代码,你最有可能直接使用ASM,这是一个很好的工具。

ASM通过设置COMPUTE_FRAMES标志来计算堆栈映射帧。您可以通过注册AsmVisitorWrapper来设置该标志,该Advice仅设置该标志而不注册包装器。

如果要创建自定义字节代码,您是否考虑了<![CDATA[]]>组件?它允许您在普通Java中编写代码,其中字节代码在运行时内联并映射到适当的参数。