Implement interface injection
This commit is contained in:
parent
bbeb87aa19
commit
4d267e491b
@ -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);
|
||||
|
@ -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;
|
||||
}
|
||||
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
22
src/dev/w1zzrd/asm/exception/SignatureCheckException.java
Normal file
22
src/dev/w1zzrd/asm/exception/SignatureCheckException.java
Normal 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);
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user