Make method injection target resolution more intelligent
This commit is contained in:
parent
5729537864
commit
092976ee5c
@ -33,8 +33,6 @@ public class Combine {
|
|||||||
public void inject(MethodNode node, GraftSource source) {
|
public void inject(MethodNode node, GraftSource source) {
|
||||||
final AsmAnnotation<Inject> annotation = source.getMethodInjectAnnotation(node);
|
final AsmAnnotation<Inject> annotation = source.getMethodInjectAnnotation(node);
|
||||||
|
|
||||||
final boolean acceptReturn = annotation.getEntry("acceptOriginalReturn");
|
|
||||||
|
|
||||||
switch ((InPlaceInjection)annotation.getEnumEntry("value")) {
|
switch ((InPlaceInjection)annotation.getEnumEntry("value")) {
|
||||||
case INSERT: // Explicitly insert a *new* method
|
case INSERT: // Explicitly insert a *new* method
|
||||||
insert(node, source);
|
insert(node, source);
|
||||||
@ -51,7 +49,7 @@ public class Combine {
|
|||||||
finishGrafting(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);
|
||||||
break;
|
break;
|
||||||
case BEFORE: // Inject a method's instructions before the original instructions in a given method
|
case BEFORE: // Inject a method's instructions before the original instructions in a given method
|
||||||
prepend(node, source);
|
prepend(node, source);
|
||||||
@ -65,22 +63,22 @@ public class Combine {
|
|||||||
* return value from the original node as the "argument" to the grafted node.
|
* return value from the original node as the "argument" to the grafted node.
|
||||||
* @param extension Node to extend method with
|
* @param extension Node to extend method with
|
||||||
* @param source The {@link GraftSource} from which the method node will be adapted
|
* @param source The {@link GraftSource} from which the method node will be adapted
|
||||||
* @param acceptReturn Whether or not the grafted method should "receive" the original method's return value as an "argument"
|
|
||||||
*/
|
*/
|
||||||
public void append(MethodNode extension, GraftSource source, boolean acceptReturn) {
|
public void append(MethodNode extension, GraftSource source) {
|
||||||
if (initiateGrafting(extension, source))
|
if (initiateGrafting(extension, source))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
final MethodNode target = resolveMethod(extension, source, true);
|
final MethodResolution resolution = resolveMethod(extension, source, true);
|
||||||
|
boolean acceptReturn = resolution.acceptReturn;
|
||||||
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 ;)
|
||||||
final MethodSignature msig = new MethodSignature(target.desc);
|
final MethodSignature msig = new MethodSignature(resolution.node.desc);
|
||||||
final MethodSignature xsig = new MethodSignature(extension.desc);
|
final MethodSignature xsig = new MethodSignature(extension.desc);
|
||||||
|
|
||||||
// Get total argument count, including implicit "this" argument
|
// Get total argument count, including implicit "this" argument
|
||||||
final int graftArgCount = xsig.getArgCount() + (isStatic(extension) ? 0 : 1);
|
final int graftArgCount = xsig.getArgCount() + (isStatic(extension) ? 0 : 1);
|
||||||
final int targetArgCount = msig.getArgCount() + (isStatic(target) ? 0 : 1);
|
final int targetArgCount = msig.getArgCount() + (isStatic(resolution.node) ? 0 : 1);
|
||||||
|
|
||||||
// If graft method cares about the return value of the original method, i.e. accepts it as an extra "argument"
|
// If graft method cares about the return value of the original method, i.e. accepts it as an extra "argument"
|
||||||
if (acceptReturn && !msig.getRet().isVoidType()) {
|
if (acceptReturn && !msig.getRet().isVoidType()) {
|
||||||
@ -92,49 +90,49 @@ public class Combine {
|
|||||||
.get();
|
.get();
|
||||||
|
|
||||||
// Inject return variable
|
// Inject return variable
|
||||||
adjustArgument(target, retVar, true, true);
|
adjustArgument(resolution.node, retVar, true, true);
|
||||||
|
|
||||||
// Handle retvar specially
|
// Handle retvar specially
|
||||||
extension.localVariables.remove(retVar);
|
extension.localVariables.remove(retVar);
|
||||||
|
|
||||||
// Make space in the original frames for the return var
|
// Make space in the original frames for the return var
|
||||||
// This isn't an optimal solution, but it works for now
|
// This isn't an optimal solution, but it works for now
|
||||||
adjustFramesForRetVar(target.instructions, targetArgCount);
|
adjustFramesForRetVar(resolution.node.instructions, targetArgCount);
|
||||||
|
|
||||||
// Replace return instructions with GOTOs to the last instruction in the list
|
// Replace return instructions with GOTOs to the last instruction in the list
|
||||||
// Return values are stored in retVar
|
// Return values are stored in retVar
|
||||||
storeAndGotoFromReturn(target, target.instructions, retVar.index, xsig);
|
storeAndGotoFromReturn(resolution.node, resolution.node.instructions, retVar.index, xsig);
|
||||||
} else {
|
} else {
|
||||||
// If we don't care about the return value from the original, we can replace returns with pops
|
// If we don't care about the return value from the original, we can replace returns with pops
|
||||||
popAndGotoFromReturn(target, target.instructions, xsig);
|
popAndGotoFromReturn(resolution.node, resolution.node.instructions, xsig);
|
||||||
}
|
}
|
||||||
|
|
||||||
List<LocalVariableNode> extVars = getVarsOver(extension.localVariables, xsig.getArgCount());
|
List<LocalVariableNode> extVars = getVarsOver(extension.localVariables, xsig.getArgCount());
|
||||||
|
|
||||||
// Add extension vars to target
|
// Add extension vars to target
|
||||||
target.localVariables.addAll(extVars);
|
resolution.node.localVariables.addAll(extVars);
|
||||||
|
|
||||||
// Add extension instructions to instruction list
|
// Add extension instructions to instruction list
|
||||||
target.instructions.add(extension.instructions);
|
resolution.node.instructions.add(extension.instructions);
|
||||||
|
|
||||||
// Make sure we extend the scope of the original method arguments
|
// Make sure we extend the scope of the original method arguments
|
||||||
for (int i = 0; i < targetArgCount; ++i)
|
for (int i = 0; i < targetArgCount; ++i)
|
||||||
adjustArgument(target, getVarAt(target.localVariables, i), false, false);
|
adjustArgument(resolution.node, getVarAt(resolution.node.localVariables, i), false, false);
|
||||||
|
|
||||||
// Recompute maximum variable count
|
// Recompute maximum variable count
|
||||||
target.maxLocals = Math.max(
|
resolution.node.maxLocals = Math.max(
|
||||||
Math.max(
|
Math.max(
|
||||||
targetArgCount + 1,
|
targetArgCount + 1,
|
||||||
graftArgCount + 1
|
graftArgCount + 1
|
||||||
),
|
),
|
||||||
target.localVariables
|
resolution.node.localVariables
|
||||||
.stream()
|
.stream()
|
||||||
.map(it -> it.index)
|
.map(it -> it.index)
|
||||||
.max(Comparator.comparingInt(a -> a)).orElse(0) + 1
|
.max(Comparator.comparingInt(a -> a)).orElse(0) + 1
|
||||||
);
|
);
|
||||||
|
|
||||||
// Recompute maximum stack size
|
// Recompute maximum stack size
|
||||||
target.maxStack = Math.max(target.maxStack, extension.maxStack);
|
resolution.node.maxStack = Math.max(resolution.node.maxStack, extension.maxStack);
|
||||||
|
|
||||||
finishGrafting(extension, source);
|
finishGrafting(extension, source);
|
||||||
}
|
}
|
||||||
@ -143,7 +141,7 @@ public class Combine {
|
|||||||
if (initiateGrafting(extension, source))
|
if (initiateGrafting(extension, source))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
final MethodNode target = resolveMethod(extension, source, false);
|
final MethodNode target = resolveMethod(extension, source, false).node;
|
||||||
adaptMethod(extension, source);
|
adaptMethod(extension, source);
|
||||||
|
|
||||||
MethodSignature sig = new MethodSignature(extension.desc);
|
MethodSignature sig = new MethodSignature(extension.desc);
|
||||||
@ -316,6 +314,10 @@ public class Combine {
|
|||||||
return target.name;
|
return target.name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ClassNode getClassNode() {
|
||||||
|
return target;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Prepares a {@link MethodNode} for grafting on to a given method and into the targeted {@link ClassNode}
|
* Prepares a {@link MethodNode} for grafting on to a given method and into the targeted {@link ClassNode}
|
||||||
* @param node Node to adapt
|
* @param node Node to adapt
|
||||||
@ -846,7 +848,7 @@ public class Combine {
|
|||||||
return null; // Nothing was found
|
return null; // Nothing was found
|
||||||
}
|
}
|
||||||
|
|
||||||
protected MethodNode resolveMethod(MethodNode inject, GraftSource source, boolean allowAcceptRet) {
|
protected MethodResolution resolveMethod(MethodNode inject, GraftSource source, boolean allowAcceptRet) {
|
||||||
AsmAnnotation<Inject> annot = AsmAnnotation.getAnnotation(Inject.class, inject.visibleAnnotations);
|
AsmAnnotation<Inject> annot = AsmAnnotation.getAnnotation(Inject.class, inject.visibleAnnotations);
|
||||||
if (!allowAcceptRet && (Boolean)annot.getEntry("acceptOriginalReturn"))
|
if (!allowAcceptRet && (Boolean)annot.getEntry("acceptOriginalReturn"))
|
||||||
throw new MethodNodeResolutionException(String.format(
|
throw new MethodNodeResolutionException(String.format(
|
||||||
@ -876,10 +878,6 @@ public class Combine {
|
|||||||
inject.desc
|
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 we accept original return value, target with not contain final argument
|
||||||
if (acceptRet) {
|
if (acceptRet) {
|
||||||
if (!mSig.getRet().equals(mSig.getArg(mSig.getArgCount() - 1)))
|
if (!mSig.getRet().equals(mSig.getArg(mSig.getArgCount() - 1)))
|
||||||
@ -892,19 +890,36 @@ public class Combine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final String findSig = sig;
|
final String findSig = sig;
|
||||||
|
final List<MethodNode> cand = candidates;
|
||||||
candidates = candidates.stream().filter(it -> it.desc.equals(findSig)).collect(Collectors.toList());
|
candidates = candidates.stream().filter(it -> it.desc.equals(findSig)).collect(Collectors.toList());
|
||||||
|
|
||||||
// We have no candidates
|
// We have no candidates
|
||||||
if (candidates.isEmpty())
|
if (candidates.isEmpty()) {
|
||||||
|
// If no candidates were found for the explicitly declared signature,
|
||||||
|
// check if accepting original return value was implied
|
||||||
|
if (!acceptRet &&
|
||||||
|
allowAcceptRet &&
|
||||||
|
mSig.getArgCount() > 0 &&
|
||||||
|
mSig.getRet().equals(mSig.getArg(mSig.getArgCount() - 1))) {
|
||||||
|
// Search for method without the implied return value argument
|
||||||
|
final String fSig = mSig.withoutLastArg().toString();
|
||||||
|
candidates = cand.stream().filter(it -> it.desc.equals(fSig)).collect(Collectors.toList());
|
||||||
|
|
||||||
|
// Do we have a match?
|
||||||
|
if (candidates.size() == 1)
|
||||||
|
return new MethodResolution(candidates.get(0), true);
|
||||||
|
}
|
||||||
|
|
||||||
throw new MethodNodeResolutionException(String.format(
|
throw new MethodNodeResolutionException(String.format(
|
||||||
"Cannot find and target candidates for method %s%s",
|
"Cannot find and target candidates for method %s%s",
|
||||||
inject.name,
|
inject.name,
|
||||||
inject.desc
|
inject.desc
|
||||||
));
|
));
|
||||||
|
}
|
||||||
|
|
||||||
// If we have a candidate, it will have a specific name and signature
|
// If we have a candidate, it will have a specific name and signature
|
||||||
// Therefore there cannot be more than one candidate by JVM convention
|
// Therefore there cannot be more than one candidate by JVM convention
|
||||||
return candidates.get(0);
|
return new MethodResolution(candidates.get(0), acceptRet && allowAcceptRet);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static boolean isStatic(MethodNode node) {
|
protected static boolean isStatic(MethodNode node) {
|
||||||
@ -937,4 +952,14 @@ public class Combine {
|
|||||||
return Objects.hash(source, node);
|
return Objects.hash(source, node);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class MethodResolution {
|
||||||
|
public final MethodNode node;
|
||||||
|
public final boolean acceptReturn;
|
||||||
|
|
||||||
|
public MethodResolution(MethodNode node, boolean acceptReturn) {
|
||||||
|
this.node = node;
|
||||||
|
this.acceptReturn = acceptReturn;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -57,8 +57,12 @@ public final class GraftSource {
|
|||||||
public String getMethodTarget(MethodNode node) {
|
public String getMethodTarget(MethodNode node) {
|
||||||
if (methodAnnotations.containsKey(node)) {
|
if (methodAnnotations.containsKey(node)) {
|
||||||
String target = getInjectionDirective(methodAnnotations.get(node)).getEntry("target");
|
String target = getInjectionDirective(methodAnnotations.get(node)).getEntry("target");
|
||||||
if (target != null && target.length() != 0)
|
if (target != null && target.length() != 0) {
|
||||||
return target;
|
if (target.indexOf('(') != -1)
|
||||||
|
return target;
|
||||||
|
else
|
||||||
|
return target + node.desc;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return node.name + node.desc;
|
return node.name + node.desc;
|
||||||
|
@ -25,7 +25,7 @@ public class MergeInject extends MergeTest implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Inject(value = BEFORE, target = "stackTest()I")
|
@Inject(value = BEFORE, target = "stackTest")
|
||||||
public int beforeStackTest() {
|
public int beforeStackTest() {
|
||||||
System.out.println("This is before stack test");
|
System.out.println("This is before stack test");
|
||||||
if (ThreadLocalRandom.current().nextBoolean()) {
|
if (ThreadLocalRandom.current().nextBoolean()) {
|
||||||
@ -42,7 +42,7 @@ public class MergeInject extends MergeTest implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Inject(value = AFTER, acceptOriginalReturn = true)
|
@Inject(AFTER)
|
||||||
public int stackTest(int arg) {
|
public int stackTest(int arg) {
|
||||||
Runnable r = () -> {
|
Runnable r = () -> {
|
||||||
System.out.println(arg / 15);
|
System.out.println(arg / 15);
|
||||||
@ -53,7 +53,7 @@ public class MergeInject extends MergeTest implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Inject(value = AFTER, acceptOriginalReturn = true)
|
@Inject(AFTER)
|
||||||
public String test(String retVal){
|
public String test(String retVal){
|
||||||
|
|
||||||
System.out.println(retVal + "Cringe");
|
System.out.println(retVal + "Cringe");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user