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); replace(node, source, true);
break; break;
case INJECT: // Insert method by either replacing an existing method or inserting a new method case INJECT: // Insert method by either replacing an existing method or inserting a new method
if (initiateGrafting(node, source))
return;
insertOrReplace(node, source); insertOrReplace(node, source);
finishGrafting(node, source);
break; break;
case AFTER: // Inject a method's instructions after the original instructions in a given method case AFTER: // Inject a method's instructions after the original instructions in a given method
append(node, source, acceptReturn); append(node, source, acceptReturn);
@ -66,10 +71,7 @@ public class Combine {
if (initiateGrafting(extension, source)) if (initiateGrafting(extension, source))
return; return;
final MethodNode target = checkMethodExists( final MethodNode target = resolveMethod(extension, source, true);
source.getMethodTargetName(extension),
source.getMethodTargetSignature(extension)
);
adaptMethod(extension, source); adaptMethod(extension, source);
// Get the method signatures so we know what we're working with local-variable-wise ;) // 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)) if (initiateGrafting(extension, source))
return; return;
final MethodNode target = checkMethodExists( final MethodNode target = resolveMethod(extension, source, false);
source.getMethodTargetName(extension),
source.getMethodTargetSignature(extension)
);
adaptMethod(extension, source); adaptMethod(extension, source);
MethodSignature sig = new MethodSignature(extension.desc); MethodSignature sig = new MethodSignature(extension.desc);
@ -165,7 +164,7 @@ public class Combine {
if (initiateGrafting(inject, source)) if (initiateGrafting(inject, source))
return; 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); ensureMatchingSignatures(remove, inject, Opcodes.ACC_STATIC);
if (preserveOriginalAccess) if (preserveOriginalAccess)
copySignatures(remove, inject, Opcodes.ACC_PUBLIC | Opcodes.ACC_PROTECTED | Opcodes.ACC_PRIVATE); copySignatures(remove, inject, Opcodes.ACC_PUBLIC | Opcodes.ACC_PROTECTED | Opcodes.ACC_PRIVATE);
@ -179,14 +178,14 @@ public class Combine {
if (initiateGrafting(inject, source)) if (initiateGrafting(inject, source))
return; return;
checkMethodNotExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject)); checkMethodNotExists(source.getMethodTargetName(inject), source.getMethodTargetSignature(inject, false));
insertOrReplace(inject, source); insertOrReplace(inject, source);
finishGrafting(inject, source); finishGrafting(inject, source);
} }
protected void insertOrReplace(MethodNode inject, GraftSource 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) if (replace != null)
this.target.methods.remove(replace); this.target.methods.remove(replace);
@ -325,7 +324,7 @@ public class Combine {
protected void adaptMethod(MethodNode node, GraftSource source) { protected void adaptMethod(MethodNode node, GraftSource source) {
// Adapt instructions // Adapt instructions
for (AbstractInsnNode insn = node.instructions.getFirst(); insn != null; insn = insn.getNext()) { 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 LdcInsnNode) adaptLdcInsn((LdcInsnNode) insn, source.getTypeName());
else if (insn instanceof FrameNode) adaptFrameNode((FrameNode) insn, source); else if (insn instanceof FrameNode) adaptFrameNode((FrameNode) insn, source);
else if (insn instanceof FieldInsnNode) adaptFieldInsn((FieldInsnNode) 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 node Grafted method instruction node
* @param source The {@link GraftSource} from which the instruction node will be adapted * @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())) { if (node.owner.equals(source.getTypeName())) {
final MethodNode injected = source.getInjectedMethod(node.name, node.desc); //final MethodNode injected = source.getInjectedMethod(node.name, node.desc);
if (injected != null) { //if (injected != null) {
node.owner = this.target.name; node.owner = this.target.name;
node.name = source.getMethodTargetName(injected); //node.name = source.getMethodTargetName(injected);
node.desc = adaptMethodSignature(node.desc, source); //node.desc = adaptMethodSignature(node.desc, source);
} //}
return node;
} else if (node.owner.equals("dev/w1zzrd/asm/Directives")) { // ASM target directives } else if (node.owner.equals("dev/w1zzrd/asm/Directives")) { // ASM target directives
if (node.name.equals(Directives.directiveNameByTarget(DirectiveTarget.TargetType.CALL_SUPER))) { if (node.name.equals(Directives.directiveNameByTarget(DirectiveTarget.TargetType.CALL_SUPER))) {
// We're attempting to redirect a call to a superclass // We're attempting to redirect a call to a superclass
@ -537,7 +537,9 @@ public class Combine {
continue; continue;
} }
return; sourceMethod.instructions.remove(node);
return prev;
} }
throw new RuntimeException(String.format("Could not locate a target for directive %s", node.name)); 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 // This should just remove the extraneous RETURN instruction
insnList.remove(afterJump); insnList.remove(afterJump);
} }
return jumpInsn;
} }
} }
return node;
} }
/** /**
@ -840,6 +846,67 @@ public class Combine {
return null; // Nothing was found 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) { protected static boolean isStatic(MethodNode node) {
return (node.access & Opcodes.ACC_STATIC) != 0; return (node.access & Opcodes.ACC_STATIC) != 0;
} }

View File

@ -67,19 +67,20 @@ public final class GraftSource {
public String getMethodTargetName(MethodNode node) { public String getMethodTargetName(MethodNode node) {
String target = getMethodTarget(node); String target = getMethodTarget(node);
if (target.contains("(")) if (target.indexOf("(") > 0)
return target.substring(0, target.indexOf('(')); 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); String target = getMethodTarget(node);
if (target.contains("(")) if (target.contains("("))
return new MethodSignature(target.substring(target.indexOf('('))); 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) { public boolean isMethodInjected(String name, String desc) {

View File

@ -47,6 +47,11 @@ public class MethodSignature {
this.ret = ret; this.ret = ret;
} }
private MethodSignature(TypeSignature ret, TypeSignature[] args) {
this.ret = ret;
this.args = args;
}
private static TypeSignature parseOneSignature(String sig, int startAt) { private static TypeSignature parseOneSignature(String sig, int startAt) {
final int len = sig.length(); final int len = sig.length();
switch (sig.charAt(startAt)) { switch (sig.charAt(startAt)) {
@ -102,6 +107,13 @@ public class MethodSignature {
args[idx] = sig; 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) { public void setRet(TypeSignature sig) {
ret = sig; ret = sig;
} }