Fix bug in multi-method merge algorithm referencing retVal
This commit is contained in:
parent
518dfa80d8
commit
0b58fcf3a3
@ -1,7 +1,5 @@
|
|||||||
package dev.w1zzrd.asm;
|
package dev.w1zzrd.asm;
|
||||||
|
|
||||||
import com.sun.istack.internal.Nullable;
|
|
||||||
import com.sun.org.apache.bcel.internal.generic.GETSTATIC;
|
|
||||||
import jdk.internal.org.objectweb.asm.*;
|
import jdk.internal.org.objectweb.asm.*;
|
||||||
import jdk.internal.org.objectweb.asm.tree.*;
|
import jdk.internal.org.objectweb.asm.tree.*;
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
@ -25,7 +23,6 @@ public class Merger {
|
|||||||
private static final Pattern re_retTypes = Pattern.compile("((?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D|V)");
|
private static final Pattern re_retTypes = Pattern.compile("((?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D|V)");
|
||||||
|
|
||||||
protected final ClassNode targetNode;
|
protected final ClassNode targetNode;
|
||||||
protected final ArrayList<MethodSig> extended = new ArrayList<>();
|
|
||||||
|
|
||||||
|
|
||||||
public Merger(String targetClass) throws IOException {
|
public Merger(String targetClass) throws IOException {
|
||||||
@ -225,18 +222,48 @@ public class Merger {
|
|||||||
created = true;
|
created = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean isStaticMethod = (inject.access & Opcodes.ACC_STATIC) != 0;
|
boolean isStaticMethod = isStatic(inject);
|
||||||
boolean keepReturn = annotation.hasEntry("acceptOriginalReturn") && (Boolean)annotation.getEntry("acceptOriginalReturn");
|
boolean keepReturn = annotation.hasEntry("acceptOriginalReturn") && (Boolean)annotation.getEntry("acceptOriginalReturn");
|
||||||
boolean hasReturn = !signature.ret.equals("V");
|
boolean hasReturn = !signature.ret.equals("V");
|
||||||
|
|
||||||
// Received return value
|
// Received return value
|
||||||
LocalVariableNode retNode = keepReturn && hasReturn ? inject.localVariables.get((isStaticMethod ? 0 : 1) + signature.args.length) : null;
|
LocalVariableNode retNode = keepReturn && hasReturn ? inject.localVariables.get((isStaticMethod ? 0 : 1) + signature.args.length) : null;
|
||||||
|
Optional<LocalVariableNode> retNode_decl = adapt.get().localVariables.stream().filter(it -> it.name.startsWith(" ")).findFirst();
|
||||||
|
|
||||||
|
// Set the proper index and update maxLocals accordingly
|
||||||
|
// This is done here so that injected calls to modify this variable use the correct index
|
||||||
if (keepReturn && hasReturn) {
|
if (keepReturn && hasReturn) {
|
||||||
// Set the proper index and update maxLocals accordingly
|
if (retNode_decl.isPresent()) {
|
||||||
// This is done here so that injected calls to modify this variable use the correct index
|
// Extend scope of retNode variable
|
||||||
retNode.index = Math.max(retNode.index, adapt.get().localVariables.stream().mapToInt(it -> it.index).reduce(Math::max).orElse(0));
|
retNode_decl.get().end = retNode.end;
|
||||||
inject.maxLocals = retNode.index + 1;
|
|
||||||
|
// Remove unnecessary retNode
|
||||||
|
inject.localVariables.remove(retNode);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Insert retNode at the lowest index that isn't "this"
|
||||||
|
int lowestIndex = isStaticMethod ? 0 : 1;
|
||||||
|
int original = retNode.index;
|
||||||
|
retNode.index = lowestIndex;
|
||||||
|
|
||||||
|
toAdapt
|
||||||
|
.stream()
|
||||||
|
.filter(it -> it instanceof VarInsnNode && ((VarInsnNode) it).var >= retNode.index)
|
||||||
|
.forEach(it -> ++((VarInsnNode) it).var);
|
||||||
|
instr
|
||||||
|
.stream()
|
||||||
|
.filter(it -> it instanceof VarInsnNode && ((VarInsnNode) it).var >= retNode.index)
|
||||||
|
.forEach(it -> {
|
||||||
|
if (((VarInsnNode) it).var == original)
|
||||||
|
((VarInsnNode) it).var = retNode.index;
|
||||||
|
else ++((VarInsnNode) it).var;
|
||||||
|
});
|
||||||
|
|
||||||
|
inject.maxLocals = Math.max(inject.maxLocals, adapt.get().maxLocals);
|
||||||
|
|
||||||
|
// Mark variable as retVal holder in the most lazy way I could think of
|
||||||
|
retNode.name = ' ' + retNode.name;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If no goto instructions were added, just remove the added label
|
// If no goto instructions were added, just remove the added label
|
||||||
@ -282,6 +309,9 @@ public class Merger {
|
|||||||
inject.localVariables.add(origThis.get());
|
inject.localVariables.add(origThis.get());
|
||||||
inject.localVariables.remove(injectThis.get());
|
inject.localVariables.remove(injectThis.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Ensure all locals have a unique name
|
||||||
|
inject.localVariables.forEach(it -> makeLocalUnique(it, inject.localVariables));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,6 +343,7 @@ public class Merger {
|
|||||||
|
|
||||||
public byte[] toByteArray(int writerFlags) {
|
public byte[] toByteArray(int writerFlags) {
|
||||||
ClassWriter writer = new ClassWriter(writerFlags);
|
ClassWriter writer = new ClassWriter(writerFlags);
|
||||||
|
targetNode.methods.forEach(method -> method.localVariables.forEach(var -> var.name = var.name.replace(" ", "")));
|
||||||
targetNode.accept(writer);
|
targetNode.accept(writer);
|
||||||
|
|
||||||
return writer.toByteArray();
|
return writer.toByteArray();
|
||||||
@ -358,7 +389,6 @@ public class Merger {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
protected static SpecialCall getSpecialCall(MethodInsnNode node) {
|
protected static SpecialCall getSpecialCall(MethodInsnNode node) {
|
||||||
if (!node.owner.equals("dev/w1zzrd/asm/Merger")) return null;
|
if (!node.owner.equals("dev/w1zzrd/asm/Merger")) return null;
|
||||||
|
|
||||||
@ -375,7 +405,6 @@ public class Merger {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
protected static <T extends Annotation> AsmAnnotation<T> getAnnotation(Class<T> annotationType, ClassNode cNode) {
|
protected static <T extends Annotation> AsmAnnotation<T> getAnnotation(Class<T> annotationType, ClassNode cNode) {
|
||||||
if(cNode.visibleAnnotations == null)
|
if(cNode.visibleAnnotations == null)
|
||||||
return null;
|
return null;
|
||||||
@ -397,7 +426,6 @@ public class Merger {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
|
||||||
protected static <T extends Annotation> AsmAnnotation<T> getAnnotation(Class<T> annotationType, MethodNode cNode) {
|
protected static <T extends Annotation> AsmAnnotation<T> getAnnotation(Class<T> annotationType, MethodNode cNode) {
|
||||||
if(cNode.visibleAnnotations == null)
|
if(cNode.visibleAnnotations == null)
|
||||||
return null;
|
return null;
|
||||||
@ -488,7 +516,11 @@ public class Merger {
|
|||||||
return actualSignature;
|
return actualSignature;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
protected static void makeLocalUnique(LocalVariableNode node, List<LocalVariableNode> other) {
|
||||||
|
while (other.stream().anyMatch(it -> it != node && it.name.equals(node.name)))
|
||||||
|
node.name = '$'+node.name;
|
||||||
|
}
|
||||||
|
|
||||||
protected static MethodSig parseMethodSignature(String sig) {
|
protected static MethodSig parseMethodSignature(String sig) {
|
||||||
Matcher signatureMatcher = re_methodSignature.matcher(sig);
|
Matcher signatureMatcher = re_methodSignature.matcher(sig);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user