Improve method resolution

This commit is contained in:
Gabriel Tofvesson 2021-01-28 18:21:36 +01:00
parent 169e30fc03
commit 1a7383278d
3 changed files with 103 additions and 23 deletions

View File

@ -43,7 +43,12 @@ public class Combine {
replace(node, source, true);
break;
case INJECT: // Insert method by either replacing an existing method or inserting a new method
if (initiateGrafting(node, source))
return;
insertOrReplace(node, source);
finishGrafting(node, source);
break;
case AFTER: // Inject a method's instructions after the original instructions in a given method
append(node, source, acceptReturn);
@ -66,10 +71,7 @@ public class Combine {
if (initiateGrafting(extension, source))
return;
final MethodNode target = checkMethodExists(
source.getMethodTargetName(extension),
source.getMethodTargetSignature(extension)
);
final MethodNode target = resolveMethod(extension, source, true);
adaptMethod(extension, source);
// Get the method signatures so we know what we're working with local-variable-wise ;)
@ -141,10 +143,7 @@ public class Combine {
if (initiateGrafting(extension, source))
return;
final MethodNode target = checkMethodExists(
source.getMethodTargetName(extension),
source.getMethodTargetSignature(extension)
);
final MethodNode target = resolveMethod(extension, source, false);
adaptMethod(extension, source);
MethodSignature sig = new MethodSignature(extension.desc);
@ -165,7 +164,7 @@ public class Combine {
if (initiateGrafting(inject, source))
return;
final MethodNode remove = checkMethodExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject));
final MethodNode remove = checkMethodExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject, false));
ensureMatchingSignatures(remove, inject, Opcodes.ACC_STATIC);
if (preserveOriginalAccess)
copySignatures(remove, inject, Opcodes.ACC_PUBLIC | Opcodes.ACC_PROTECTED | Opcodes.ACC_PRIVATE);
@ -179,14 +178,14 @@ public class Combine {
if (initiateGrafting(inject, source))
return;
checkMethodNotExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject));
checkMethodNotExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject, false));
insertOrReplace(inject, source);
finishGrafting(inject, source);
}
protected void insertOrReplace(MethodNode inject, GraftSource source) {
MethodNode replace = findMethodNode(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject));
MethodNode replace = findMethodNode(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject, false));
if (replace != null)
this.target.methods.remove(replace);
@ -325,7 +324,7 @@ public class Combine {
protected void adaptMethod(MethodNode node, GraftSource source) {
// Adapt instructions
for (AbstractInsnNode insn = node.instructions.getFirst(); insn != null; insn = insn.getNext()) {
if (insn instanceof MethodInsnNode) adaptMethodInsn((MethodInsnNode) insn, source, node);
if (insn instanceof MethodInsnNode) insn = adaptMethodInsn((MethodInsnNode) insn, source, node);
else if (insn instanceof LdcInsnNode) adaptLdcInsn((LdcInsnNode) insn, source.getTypeName());
else if (insn instanceof FrameNode) adaptFrameNode((FrameNode) insn, source);
else if (insn instanceof FieldInsnNode) adaptFieldInsn((FieldInsnNode) insn, source);
@ -506,14 +505,15 @@ public class Combine {
* @param node Grafted method instruction node
* @param source The {@link GraftSource} from which the instruction node will be adapted
*/
protected void adaptMethodInsn(MethodInsnNode node, GraftSource source, MethodNode sourceMethod) {
protected AbstractInsnNode adaptMethodInsn(MethodInsnNode node, GraftSource source, MethodNode sourceMethod) {
if (node.owner.equals(source.getTypeName())) {
final MethodNode injected = source.getInjectedMethod(node.name, node.desc);
if (injected != null) {
//final MethodNode injected = source.getInjectedMethod(node.name, node.desc);
//if (injected != null) {
node.owner = this.target.name;
node.name = source.getMethodTargetName(injected);
node.desc = adaptMethodSignature(node.desc, source);
}
//node.name = source.getMethodTargetName(injected);
//node.desc = adaptMethodSignature(node.desc, source);
//}
return node;
} else if (node.owner.equals("dev/w1zzrd/asm/Directives")) { // ASM target directives
if (node.name.equals(Directives.directiveNameByTarget(DirectiveTarget.TargetType.CALL_SUPER))) {
// We're attempting to redirect a call to a superclass
@ -537,7 +537,9 @@ public class Combine {
continue;
}
return;
sourceMethod.instructions.remove(node);
return prev;
}
throw new RuntimeException(String.format("Could not locate a target for directive %s", node.name));
@ -572,8 +574,12 @@ public class Combine {
// This should just remove the extraneous RETURN instruction
insnList.remove(afterJump);
}
return jumpInsn;
}
}
return node;
}
/**
@ -840,6 +846,67 @@ public class Combine {
return null; // Nothing was found
}
protected MethodNode resolveMethod(MethodNode inject, GraftSource source, boolean allowAcceptRet) {
AsmAnnotation<Inject> annot = AsmAnnotation.getAnnotation(Inject.class, inject.visibleAnnotations);
if (!allowAcceptRet && (Boolean)annot.getEntry("acceptOriginalReturn"))
throw new MethodNodeResolutionException(String.format(
"Method %s marked as accepting original return, but injection strategy prohibits this!",
inject.name
));
boolean acceptRet = annot.getEntry("acceptOriginalReturn");
String sig = adaptMethodSignature(source.getMethodTarget(inject), source);
final MethodSignature mSig = new MethodSignature(sig);
final String targetName = source.getMethodTargetName(inject);
// Collect possible method candidates by name
List<MethodNode> candidates = this.target
.methods
.stream()
.filter(it -> it.name.equals(targetName))
.collect(Collectors.toList());
// No candidates match the base criteria
if (candidates.isEmpty())
throw new MethodNodeResolutionException(String.format(
"Cannot find and target candidates for method %s%s",
inject.name,
inject.desc
));
// We have a unique candidate
if (candidates.size() == 1)
return candidates.get(0);
// If we accept original return value, target with not contain final argument
if (acceptRet) {
if (!mSig.getRet().equals(mSig.getArg(mSig.getArgCount() - 1)))
throw new MethodNodeResolutionException(String.format(
"Return value must match final method argument when accepting return: %s%s",
inject.name,
inject.desc
));
sig = mSig.withoutLastArg().toString();
}
final String findSig = sig;
candidates = candidates.stream().filter(it -> it.desc.equals(findSig)).collect(Collectors.toList());
// We have no candidates
if (candidates.isEmpty())
throw new MethodNodeResolutionException(String.format(
"Cannot find and target candidates for method %s%s",
inject.name,
inject.desc
));
// If we have a candidate, it will have a specific name and signature
// Therefore there cannot be more than one candidate by JVM convention
return candidates.get(0);
}
protected static boolean isStatic(MethodNode node) {
return (node.access & Opcodes.ACC_STATIC) != 0;
}

View File

@ -67,19 +67,20 @@ public final class GraftSource {
public String getMethodTargetName(MethodNode node) {
String target = getMethodTarget(node);
if (target.contains("("))
if (target.indexOf("(") > 0)
return target.substring(0, target.indexOf('('));
return target;
return node.name;
}
public MethodSignature getMethodTargetSignature(MethodNode node) {
public MethodSignature getMethodTargetSignature(MethodNode node, boolean acceptOriginalReturn) {
String target = getMethodTarget(node);
if (target.contains("("))
return new MethodSignature(target.substring(target.indexOf('(')));
return new MethodSignature(node.desc);
MethodSignature sig = new MethodSignature(node.desc);
return acceptOriginalReturn ? sig.withoutLastArg() : sig;
}
public boolean isMethodInjected(String name, String desc) {

View File

@ -47,6 +47,11 @@ public class MethodSignature {
this.ret = ret;
}
private MethodSignature(TypeSignature ret, TypeSignature[] args) {
this.ret = ret;
this.args = args;
}
private static TypeSignature parseOneSignature(String sig, int startAt) {
final int len = sig.length();
switch (sig.charAt(startAt)) {
@ -102,6 +107,13 @@ public class MethodSignature {
args[idx] = sig;
}
public MethodSignature withoutLastArg() {
if (args.length == 0)
throw new IndexOutOfBoundsException();
return new MethodSignature(ret, Arrays.copyOf(args, args.length - 1));
}
public void setRet(TypeSignature sig) {
ret = sig;
}