Implement interface injection

This commit is contained in:
Gabriel Tofvesson 2021-01-28 17:04:17 +01:00
parent bbeb87aa19
commit 4d267e491b
4 changed files with 93 additions and 6 deletions

View File

@ -2,6 +2,7 @@ package dev.w1zzrd.asm;
import dev.w1zzrd.asm.analysis.AsmAnnotation;
import dev.w1zzrd.asm.exception.MethodNodeResolutionException;
import dev.w1zzrd.asm.exception.SignatureCheckException;
import dev.w1zzrd.asm.exception.SignatureInstanceMismatchException;
import dev.w1zzrd.asm.signature.MethodSignature;
import dev.w1zzrd.asm.signature.TypeSignature;
@ -30,7 +31,7 @@ public class Combine {
}
public void inject(MethodNode node, GraftSource source) {
final AsmAnnotation<Inject> annotation = source.getInjectAnnotation(node);
final AsmAnnotation<Inject> annotation = source.getMethodInjectAnnotation(node);
final boolean acceptReturn = annotation.getEntry("acceptOriginalReturn");
@ -195,6 +196,50 @@ public class Combine {
this.target.methods.add(inject);
}
public void inject(FieldNode field, GraftSource source) {
if (field.desc.equals(source.getTypeName()))
field.desc = target.name;
// Remove existing field with same name
for (FieldNode node : target.fields)
if (node.name.equals(field.name)) {
target.fields.remove(node);
break;
}
target.fields.add(field);
}
public void setSuperClass(String superDesc) {
// Theoretically usable for redefining Object (under a new ClassLoader)
if (superDesc == null) {
target.superName = null;
return;
}
if (new TypeSignature(superDesc).isPrimitive())
throw new SignatureCheckException("Superclass cannot be primitive: "+superDesc);
target.superName = superDesc;
}
public String getSuperclass() {
return target.superName;
}
public TypeSignature[] getInterfaces() {
return target.interfaces.stream().map(TypeSignature::new).toArray(TypeSignature[]::new);
}
public boolean removeInterface(String interfaceDesc) {
return target.interfaces.remove(interfaceDesc);
}
public void addInterface(String interfaceDesc) {
if (!target.interfaces.contains(interfaceDesc))
target.interfaces.add(interfaceDesc);
}
private boolean initiateGrafting(MethodNode node, GraftSource source) {
DynamicSourceUnit unit = new DynamicSourceUnit(source, node);

View File

@ -132,18 +132,26 @@ public final class GraftSource {
return methodAnnotations
.keySet()
.stream()
.sorted(Comparator.comparingInt(a -> getInjectAnnotation(a).getEntry("priority")))
.sorted(Comparator.comparingInt(a -> getMethodInjectAnnotation(a).getEntry("priority")))
.collect(Collectors.toList());
}
public Set<FieldNode> getInjectFields() {
return fieldAnnotations.keySet();
public List<FieldNode> getInjectFields() {
return fieldAnnotations
.keySet()
.stream()
.sorted(Comparator.comparingInt(a -> getFieldInjectAnnotation(a).getEntry("priority")))
.collect(Collectors.toList());
}
public AsmAnnotation<Inject> getInjectAnnotation(MethodNode node) {
public AsmAnnotation<Inject> getMethodInjectAnnotation(MethodNode node) {
return getInjectionDirective(methodAnnotations.get(node));
}
public AsmAnnotation<Inject> getFieldInjectAnnotation(FieldNode node) {
return getInjectionDirective(fieldAnnotations.get(node));
}
private static boolean hasNoInjectionDirective(List<AsmAnnotation<?>> annotations) {
return getInjectionDirective(annotations) == null;
}

View File

@ -102,6 +102,18 @@ public final class AsmAnnotation<A extends Annotation> {
return hasDefaultEntry(name) || hasExplicitEntry(name);
}
public static <T extends Annotation> AsmAnnotation<T> getAnnotation(Class<T> find, Iterable<AnnotationNode> nodes) {
if (nodes == null)
return null;
for (AnnotationNode node : nodes) {
AsmAnnotation<T> annot = getAnnotation(node);
if (annot.getAnnotationType().equals(find))
return annot;
}
return null;
}
public static <T extends Annotation> AsmAnnotation<T> getAnnotation(AnnotationNode node) {
Class<T> cls;
@ -116,6 +128,6 @@ public final class AsmAnnotation<A extends Annotation> {
for (int i = 0; i < node.values.size(); i += 2)
entries.put((String)node.values.get(i), node.values.get(i + 1));
return new AsmAnnotation<T>(cls, entries);
return new AsmAnnotation<>(cls, entries);
}
}

View File

@ -0,0 +1,22 @@
package dev.w1zzrd.asm.exception;
public final class SignatureCheckException extends RuntimeException {
public SignatureCheckException() {
}
public SignatureCheckException(String message) {
super(message);
}
public SignatureCheckException(String message, Throwable cause) {
super(message, cause);
}
public SignatureCheckException(Throwable cause) {
super(cause);
}
public SignatureCheckException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
super(message, cause, enableSuppression, writableStackTrace);
}
}