如何从 Java 的 JDI 中隐藏变量?
How to hide variables from Java's JDI?
我正在检测一些 类 并引入一些新的局部变量。现在,当用户在代码中放置断点并停止执行时,可以在 Intellij IDEA 的调试器中看到新引入的局部变量 window。我怎样才能隐藏它们?
更新:我将不得不以某种方式从检测代码中删除调试信息,但不确定该怎么做。
更新 2: 我正在使用 ASM 库进行检测。
public void visitCode() {
this.mv.visitLdcInsn(stringToPass);
this.mv.visitMethodInsn(Opcodes.INVOKESTATIC, "MyAgentClass", "loadData", "(Ljava/lang/String;)LDataClass;", false);
this.mv.visitVarInsn(Opcodes.ASTORE, this.getDataIndex());
}
public void visitMaxs(int maxStack, int maxLocals) {
if (this.myStartLabel != null && this.myEndLabel != null) {
this.mv.visitLocalVariable("__my__data__", "Ljava/lang/Object;", (String) null, this.myStartLabel, this.myEndLabel, this.getDataIndex());
}
super.visitMaxs(maxStack, maxLocals);
}
__my__data__
显示在 Intellij IDEA 中。
https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.7.13
根据 JVM 规范,您可以从 LocalVariableTable
中删除局部变量。 Javassist 在生成的代码中自动执行此操作,我在 运行:
期间看不到变量 secretCode
反编译结果class可以看出没有LocalVariableTable
个条目:
public void run();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=2, locals=3, args_size=1
0: ldc #39 // int -889275714
2: istore_1
3: iload_1
4: invokestatic #43 // Method org/example/App.test:(I)Ljava/lang/Integer;
7: astore_2
8: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
11: aload_2
12: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
15: new #2 // class SecreFoo
18: dup
19: invokespecial #3 // Method "<init>":()V
22: astore_1
23: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
26: aload_1
27: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
30: return
LineNumberTable:
line 14: 15
line 15: 23
line 16: 30
LocalVariableTable:
Start Length Slot Name Signature
0 31 0 this LSecreFoo;
23 8 1 fCopy LSecreFoo;
因此,当您检测 class 时,从 table 中删除所有局部变量(或者不添加它们)。
我正在检测一些 类 并引入一些新的局部变量。现在,当用户在代码中放置断点并停止执行时,可以在 Intellij IDEA 的调试器中看到新引入的局部变量 window。我怎样才能隐藏它们?
更新:我将不得不以某种方式从检测代码中删除调试信息,但不确定该怎么做。
更新 2: 我正在使用 ASM 库进行检测。
public void visitCode() {
this.mv.visitLdcInsn(stringToPass);
this.mv.visitMethodInsn(Opcodes.INVOKESTATIC, "MyAgentClass", "loadData", "(Ljava/lang/String;)LDataClass;", false);
this.mv.visitVarInsn(Opcodes.ASTORE, this.getDataIndex());
}
public void visitMaxs(int maxStack, int maxLocals) {
if (this.myStartLabel != null && this.myEndLabel != null) {
this.mv.visitLocalVariable("__my__data__", "Ljava/lang/Object;", (String) null, this.myStartLabel, this.myEndLabel, this.getDataIndex());
}
super.visitMaxs(maxStack, maxLocals);
}
__my__data__
显示在 Intellij IDEA 中。
https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.7.13
根据 JVM 规范,您可以从 LocalVariableTable
中删除局部变量。 Javassist 在生成的代码中自动执行此操作,我在 运行:
反编译结果class可以看出没有LocalVariableTable
个条目:
public void run();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=2, locals=3, args_size=1
0: ldc #39 // int -889275714
2: istore_1
3: iload_1
4: invokestatic #43 // Method org/example/App.test:(I)Ljava/lang/Integer;
7: astore_2
8: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
11: aload_2
12: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
15: new #2 // class SecreFoo
18: dup
19: invokespecial #3 // Method "<init>":()V
22: astore_1
23: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
26: aload_1
27: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
30: return
LineNumberTable:
line 14: 15
line 15: 23
line 16: 30
LocalVariableTable:
Start Length Slot Name Signature
0 31 0 this LSecreFoo;
23 8 1 fCopy LSecreFoo;
因此,当您检测 class 时,从 table 中删除所有局部变量(或者不添加它们)。