Implement try-catch-finally injection
This commit is contained in:
parent
47723fccfb
commit
98b3b7cb3b
@ -11,8 +11,6 @@ import jdk.internal.org.objectweb.asm.Handle;
|
|||||||
import jdk.internal.org.objectweb.asm.Opcodes;
|
import jdk.internal.org.objectweb.asm.Opcodes;
|
||||||
import jdk.internal.org.objectweb.asm.Type;
|
import jdk.internal.org.objectweb.asm.Type;
|
||||||
import jdk.internal.org.objectweb.asm.tree.*;
|
import jdk.internal.org.objectweb.asm.tree.*;
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
import java.lang.reflect.InvocationTargetException;
|
import java.lang.reflect.InvocationTargetException;
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
@ -134,6 +132,10 @@ public class Combine {
|
|||||||
// Recompute maximum stack size
|
// Recompute maximum stack size
|
||||||
resolution.node.maxStack = Math.max(resolution.node.maxStack, extension.maxStack);
|
resolution.node.maxStack = Math.max(resolution.node.maxStack, extension.maxStack);
|
||||||
|
|
||||||
|
// Merge try-catch blocks
|
||||||
|
resolution.node.tryCatchBlocks.addAll(extension.tryCatchBlocks);
|
||||||
|
// Exception list not merged to maintain original signature
|
||||||
|
|
||||||
finishGrafting(extension, source);
|
finishGrafting(extension, source);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,6 +157,9 @@ public class Combine {
|
|||||||
for (int i = 0; i < sig.getArgCount(); ++i)
|
for (int i = 0; i < sig.getArgCount(); ++i)
|
||||||
adjustArgument(target, getVarAt(target.localVariables, i), true, false);
|
adjustArgument(target, getVarAt(target.localVariables, i), true, false);
|
||||||
|
|
||||||
|
target.tryCatchBlocks.addAll(extension.tryCatchBlocks);
|
||||||
|
// Exception list not merged to maintain original signature
|
||||||
|
|
||||||
finishGrafting(extension, source);
|
finishGrafting(extension, source);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -750,7 +755,7 @@ public class Combine {
|
|||||||
* @return The located method node
|
* @return The located method node
|
||||||
* @throws MethodNodeResolutionException If no method node matching the given description could be found
|
* @throws MethodNodeResolutionException If no method node matching the given description could be found
|
||||||
*/
|
*/
|
||||||
protected final @NotNull MethodNode checkMethodExists(String name, MethodSignature descriptor) {
|
protected final MethodNode checkMethodExists(String name, MethodSignature descriptor) {
|
||||||
final MethodNode target = findMethodNode(name, descriptor);
|
final MethodNode target = findMethodNode(name, descriptor);
|
||||||
|
|
||||||
if (target == null)
|
if (target == null)
|
||||||
@ -768,7 +773,7 @@ public class Combine {
|
|||||||
* @param desc Descriptor of the method node to find
|
* @param desc Descriptor of the method node to find
|
||||||
* @return A matching {@link MethodNode} if one exists, else null
|
* @return A matching {@link MethodNode} if one exists, else null
|
||||||
*/
|
*/
|
||||||
protected @Nullable MethodNode findMethodNode(String name, MethodSignature desc) {
|
protected MethodNode findMethodNode(String name, MethodSignature desc) {
|
||||||
return target.methods
|
return target.methods
|
||||||
.stream()
|
.stream()
|
||||||
.filter(it -> it.name.equals(name) && new MethodSignature(it.desc).equals(desc))
|
.filter(it -> it.name.equals(name) && new MethodSignature(it.desc).equals(desc))
|
||||||
@ -838,7 +843,7 @@ public class Combine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected static @Nullable LabelNode findLabelBeforeReturn(AbstractInsnNode start, INodeTraversal traverse) {
|
protected static LabelNode findLabelBeforeReturn(AbstractInsnNode start, INodeTraversal traverse) {
|
||||||
for (AbstractInsnNode cur = start; cur != null; cur = traverse.traverse(cur))
|
for (AbstractInsnNode cur = start; cur != null; cur = traverse.traverse(cur))
|
||||||
if (cur instanceof LabelNode) // Traversal hit label
|
if (cur instanceof LabelNode) // Traversal hit label
|
||||||
return (LabelNode) cur;
|
return (LabelNode) cur;
|
||||||
|
@ -6,8 +6,6 @@ import jdk.internal.org.objectweb.asm.tree.AnnotationNode;
|
|||||||
import jdk.internal.org.objectweb.asm.tree.ClassNode;
|
import jdk.internal.org.objectweb.asm.tree.ClassNode;
|
||||||
import jdk.internal.org.objectweb.asm.tree.FieldNode;
|
import jdk.internal.org.objectweb.asm.tree.FieldNode;
|
||||||
import jdk.internal.org.objectweb.asm.tree.MethodNode;
|
import jdk.internal.org.objectweb.asm.tree.MethodNode;
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@ -91,7 +89,7 @@ public final class GraftSource {
|
|||||||
return getInjectedMethod(name, desc) != null;
|
return getInjectedMethod(name, desc) != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public @Nullable MethodNode getInjectedMethod(String name, String desc) {
|
public MethodNode getInjectedMethod(String name, String desc) {
|
||||||
return methodAnnotations
|
return methodAnnotations
|
||||||
.entrySet()
|
.entrySet()
|
||||||
.stream()
|
.stream()
|
||||||
|
@ -7,9 +7,6 @@ import jdk.internal.org.objectweb.asm.Label;
|
|||||||
import jdk.internal.org.objectweb.asm.Opcodes;
|
import jdk.internal.org.objectweb.asm.Opcodes;
|
||||||
import jdk.internal.org.objectweb.asm.Type;
|
import jdk.internal.org.objectweb.asm.Type;
|
||||||
import jdk.internal.org.objectweb.asm.tree.*;
|
import jdk.internal.org.objectweb.asm.tree.*;
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
import sun.reflect.generics.reflectiveObjects.NotImplementedException;
|
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -192,7 +189,7 @@ public class FrameState {
|
|||||||
* @param node {@link LabelNode} to find jumps referencing
|
* @param node {@link LabelNode} to find jumps referencing
|
||||||
* @return A jump instruction node earlier in the instruction list or null if none could be found
|
* @return A jump instruction node earlier in the instruction list or null if none could be found
|
||||||
*/
|
*/
|
||||||
private static @Nullable JumpInsnNode findEarliestJump(LabelNode node) {
|
private static JumpInsnNode findEarliestJump(LabelNode node) {
|
||||||
JumpInsnNode jump = null;
|
JumpInsnNode jump = null;
|
||||||
|
|
||||||
// Traverse backward until we hit the beginning of the list
|
// Traverse backward until we hit the beginning of the list
|
||||||
@ -643,7 +640,7 @@ public class FrameState {
|
|||||||
private static List<String> getOpsByComplexity(
|
private static List<String> getOpsByComplexity(
|
||||||
boolean complexPush,
|
boolean complexPush,
|
||||||
boolean complexPop,
|
boolean complexPop,
|
||||||
@Nullable Predicate<Integer> insnP
|
Predicate<Integer> insnP
|
||||||
) {
|
) {
|
||||||
ArrayList<Integer> opcodes = new ArrayList<>();
|
ArrayList<Integer> opcodes = new ArrayList<>();
|
||||||
|
|
||||||
@ -687,9 +684,9 @@ public class FrameState {
|
|||||||
* @return Negative values for instructions previous to the current instruction, positive values for instructions
|
* @return Negative values for instructions previous to the current instruction, positive values for instructions
|
||||||
* after the current instruction. Null if instruction could not be found
|
* after the current instruction. Null if instruction could not be found
|
||||||
*/
|
*/
|
||||||
private static @Nullable Integer relativeIndexOf(
|
private static Integer relativeIndexOf(
|
||||||
@NotNull AbstractInsnNode current,
|
AbstractInsnNode current,
|
||||||
@NotNull AbstractInsnNode find
|
AbstractInsnNode find
|
||||||
) {
|
) {
|
||||||
// Check backward
|
// Check backward
|
||||||
int idx = 0;
|
int idx = 0;
|
||||||
|
@ -2,8 +2,6 @@ package dev.w1zzrd.asm.signature;
|
|||||||
|
|
||||||
import dev.w1zzrd.asm.exception.SignatureInstanceMismatchException;
|
import dev.w1zzrd.asm.exception.SignatureInstanceMismatchException;
|
||||||
import dev.w1zzrd.asm.exception.TypeSignatureParseException;
|
import dev.w1zzrd.asm.exception.TypeSignatureParseException;
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
@ -86,7 +84,7 @@ public class TypeSignature {
|
|||||||
* @param primitive Primitive type internal name (V, J or D for Top types)
|
* @param primitive Primitive type internal name (V, J or D for Top types)
|
||||||
* @param isTop Whether or not this is a Top type (only valid for 64-bit types J and D or as delimiter type V)
|
* @param isTop Whether or not this is a Top type (only valid for 64-bit types J and D or as delimiter type V)
|
||||||
*/
|
*/
|
||||||
public TypeSignature(@Nullable Character primitive, boolean isTop) {
|
public TypeSignature(Character primitive, boolean isTop) {
|
||||||
if (primitive != null) {
|
if (primitive != null) {
|
||||||
switch (Character.toUpperCase(primitive)) {
|
switch (Character.toUpperCase(primitive)) {
|
||||||
case 'J':
|
case 'J':
|
||||||
|
Loading…
x
Reference in New Issue
Block a user