java.lang.VerifyError: Expecting a stackmap frame occuring with ASM generated byte code

java.lang.VerifyError: Expecting a stackmap frame occuring with ASM generated byte code

我使用 ASM 5.0main.class 中生成了 java 字节代码,如下所示。

这是我生成的代码:

javap -c _main.class输出

  public jcalc.lang.CalcObject call();
    Code:
       0: ldc           #9                  // String _main/max
       2: invokestatic  #28                 // Method jcalc/lang/Binding.getBindingFromMain:(Ljava/lang/String;)Ljcalc/lang/Binding;
       5: ldc           #20                 // String A
       7: invokevirtual #38                 // Method jcalc/lang/Binding.refVariable:(Ljava/lang/String;)Ljcalc/lang/CalcObject;
      10: checkcast     #40                 // class jcalc/lang/CalcNumber
      13: ldc           #9                  // String _main/max
      15: invokestatic  #28                 // Method jcalc/lang/Binding.getBindingFromMain:(Ljava/lang/String;)Ljcalc/lang/Binding;
      18: ldc           #22                 // String B
      20: invokevirtual #38                 // Method jcalc/lang/Binding.refVariable:(Ljava/lang/String;)Ljcalc/lang/CalcObject;
      23: checkcast     #40                 // class jcalc/lang/CalcNumber
      26: invokevirtual #44                 // Method jcalc/lang/CalcNumber.operatorGreaterThan:(Ljcalc/lang/CalcNumber;)Ljcalc/lang/CalcBoolean;
      29: dup
      30: astore_3
      31: instanceof    #46                 // class jcalc/lang/CalcBoolean
      34: iconst_0
      35: if_icmpeq     38
      38: aload_3
      39: invokevirtual #52                 // Method jcalc/lang/CalcObject.isNonNull:()I
      42: istore_2
      43: aload_3
      44: if_icmpeq     61
      47: ldc           #9                  // String _main/max
      49: invokestatic  #28                 // Method jcalc/lang/Binding.getBindingFromMain:(Ljava/lang/String;)Ljcalc/lang/Binding;
      52: ldc           #20                 // String A
      54: invokevirtual #38                 // Method jcalc/lang/Binding.refVariable:(Ljava/lang/String;)Ljcalc/lang/CalcObject;
      57: astore_1
      58: goto          72
      61: ldc           #9                  // String _main/max
      63: invokestatic  #28                 // Method jcalc/lang/Binding.getBindingFromMain:(Ljava/lang/String;)Ljcalc/lang/Binding;
      66: ldc           #22                 // String B
      68: invokevirtual #38                 // Method jcalc/lang/Binding.refVariable:(Ljava/lang/String;)Ljcalc/lang/CalcObject;
      71: astore_1
      72: aload_1
      73: areturn

但是执行的时候,

Caused by: java.lang.VerifyError: Expecting a stackmap frame at branch target 38
Exception Details:
  Location:
    _main/max.call()Ljcalc/lang/CalcObject; @35: if_icmpeq
  Reason:
    Expected stackmap frame at this location.
  Bytecode:
    0x0000000: 1209 b800 1c12 14b6 0026 c000 2812 09b8
    0x0000010: 001c 1216 b600 26c0 0028 b600 2c59 4ec1
    0x0000020: 002e 039f 0003 2db6 0034 3d2d 9f00 1112
    0x0000030: 09b8 001c 1214 b600 264c a700 0e12 09b8
    0x0000040: 001c 1216 b600 264c 2bb0               

    at java.lang.Class.forName0(Native Method)
    at java.lang.Class.forName(Class.java:264)
    at jcalc.lang.Binding.defineFunction(Binding.java:87)
    at _main.call(Unknown Source)
    ... 7 more

我知道无论条件是否满足,分支操作都不能改变堆栈大小。

但是第35行的if_icmpeq操作只是跳转到下一个操作,所以不会影响堆栈。

我正在使用 Java 1.6。 谁能详细解释一下指令验证规则或在哪里可以找到文档。

非常感谢。

我解决了我的问题。

当 load/store 指令出现 TypeCast 问题时,ASM 无法生成正确的堆栈映射帧。

所以下次你看到一条消息时

java.lang.VerifyError: Expecting a stackmap frame at branch target 45

检查 load/store 指令的类型。