-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathClassInitHandler.java
More file actions
125 lines (104 loc) · 6.02 KB
/
ClassInitHandler.java
File metadata and controls
125 lines (104 loc) · 6.02 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package com.study.proxy.impl.handler;
import com.study.proxy.impl.util.ClassNameProvider;
import com.study.proxy.impl.util.MethodFilter;
import com.study.proxy.impl.util.TypeUtil;
import org.objectweb.asm.*;
import java.lang.reflect.Method;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import static org.objectweb.asm.Opcodes.*;
import static org.objectweb.asm.Opcodes.ANEWARRAY;
import static org.objectweb.asm.Opcodes.ICONST_0;
import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL;
import static org.objectweb.asm.Opcodes.PUTSTATIC;
public class ClassInitHandler implements Handler {
private final ClassWriter classWriter;
private final List<Method> methods;
private final Class<?> specifiedInterface;
private final IntConstInstructionGenerator intConstInstructionGenerator = new IntConstInstructionGenerator();
private final AtomicInteger methodCnt = new AtomicInteger();
public ClassInitHandler(ClassWriter classWriter, List<Method> methods, Class<?> specifiedInterface) {
this.classWriter = classWriter;
this.methods = methods;
this.specifiedInterface = specifiedInterface;
}
@Override
public void process() {
MethodVisitor methodVisitor = classWriter.visitMethod(ACC_STATIC, "<clinit>", "()V", null, null);
methodVisitor.visitCode();
// Put ClassLoader at local variable index 0, used by
// Class.forName(String, boolean, ClassLoader) calls
methodVisitor.visitLdcInsn(Type.getType(new ClassNameProvider().typeDescriptor()));
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, Type.getInternalName(Class.class), "getClassLoader", "()Ljava/lang/ClassLoader;", false);
methodVisitor.visitVarInsn(ASTORE, 0);
Label label0 = new Label();
Label label1 = new Label();
Label label2 = new Label();
methodVisitor.visitTryCatchBlock(label0, label1, label1, Type.getInternalName(NoSuchMethodException.class));
methodVisitor.visitTryCatchBlock(label0, label1, label2, Type.getInternalName(ClassNotFoundException.class));
methodVisitor.visitLabel(label0);
methodCnt.set(0);
methods.forEach(method -> generateCodeForFieldInitialization(method, methodVisitor));
methodVisitor.visitInsn(RETURN);
methodVisitor.visitLabel(label1);
methodVisitor.visitIntInsn(ASTORE, 1);
methodVisitor.visitTypeInsn(NEW, Type.getInternalName(NoSuchMethodError.class));
methodVisitor.visitInsn(DUP);
methodVisitor.visitVarInsn(ALOAD, 1);
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, Type.getInternalName(Throwable.class),
"getMessage", "()Ljava/lang/String;", false);
methodVisitor.visitMethodInsn(INVOKESPECIAL, Type.getInternalName(NoSuchMethodError.class),
"<init>", "(Ljava/lang/String;)V", false);
methodVisitor.visitInsn(ATHROW);
methodVisitor.visitLabel(label2);
methodVisitor.visitIntInsn(ASTORE, 1);
methodVisitor.visitTypeInsn(NEW, Type.getInternalName(NoClassDefFoundError.class));
methodVisitor.visitInsn(DUP);
methodVisitor.visitVarInsn(ALOAD, 1);
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, Type.getInternalName(Throwable.class), "getMessage",
"()Ljava/lang/String;", false);
methodVisitor.visitMethodInsn(INVOKESPECIAL, Type.getInternalName(NoClassDefFoundError.class),
"<init>", "(Ljava/lang/String;)V", false);
methodVisitor.visitInsn(ATHROW);
// Maxs computed by ClassWriter.COMPUTE_FRAMES, these arguments ignored
methodVisitor.visitMaxs(-1, -1);
methodVisitor.visitEnd();
}
private void generateCodeForFieldInitialization(Method method, MethodVisitor methodVisitor) {
methodVisitor.visitLdcInsn(toClassName(method));
methodVisitor.visitInsn(ICONST_0);
methodVisitor.visitVarInsn(ALOAD, 0);
methodVisitor.visitMethodInsn(INVOKESTATIC, Type.getInternalName(Class.class), "forName", "(Ljava/lang/String;ZLjava/lang/ClassLoader;)Ljava/lang/Class;", false);
methodVisitor.visitLdcInsn(method.getName());
Type[] types = Type.getArgumentTypes(method);
intConstInstructionGenerator.generate(methodVisitor, types.length);
methodVisitor.visitTypeInsn(ANEWARRAY, Type.getInternalName(Class.class));
int length = types.length;
// Construct an array with the parameter types mapping primitives to Wrapper types
for (int i = 0; i < length; i++) {
methodVisitor.visitInsn(DUP);
intConstInstructionGenerator.generate(methodVisitor, i);
Type type = types[i];
if (TypeUtil.forPrimitive(type)) {
methodVisitor.visitFieldInsn(GETSTATIC, TypeUtil.toWrapperType(type).getInternalName(), // something like "java/lang/Integer", "java/lang/Double"
"TYPE", Type.getDescriptor(Class.class)
);
} else {
methodVisitor.visitLdcInsn(type.getInternalName().replace('/', '.')); // TODO: is there a better way?
methodVisitor.visitInsn(ICONST_0); // false
methodVisitor.visitVarInsn(ALOAD, 0); // ClassLoader
methodVisitor.visitMethodInsn(INVOKESTATIC, Type.getInternalName(Class.class), "forName", "(Ljava/lang/String;ZLjava/lang/ClassLoader;)Ljava/lang/Class;", false);
}
methodVisitor.visitInsn(AASTORE);
}
methodVisitor.visitMethodInsn(INVOKEVIRTUAL, Type.getInternalName(Class.class), "getMethod", "(Ljava/lang/String;[Ljava/lang/Class;)Ljava/lang/reflect/Method;", false);
methodVisitor.visitFieldInsn(PUTSTATIC, new ClassNameProvider().internalName(), "m" + methodCnt.getAndIncrement(), Type.getDescriptor(Method.class));
}
private String toClassName(Method method) {
MethodFilter methodFilter = new MethodFilter();
if (methodFilter.notObjectMethod.test(method)) {
return Type.getType(specifiedInterface).getClassName();
}
return Type.getType(Object.class).getClassName();
}
}