Implement try-catch-finally injection

This commit is contained in:
Gabriel Tofvesson 2021-02-05 14:58:42 +01:00
parent 47723fccfb
commit 98b3b7cb3b
4 changed files with 17 additions and 19 deletions

View File

@ -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;

View File

@ -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()

View File

@ -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;

View File

@ -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':