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.analysis.AsmAnnotation;
|
||||||
import dev.w1zzrd.asm.exception.MethodNodeResolutionException;
|
import dev.w1zzrd.asm.exception.MethodNodeResolutionException;
|
||||||
|
import dev.w1zzrd.asm.exception.SignatureCheckException;
|
||||||
import dev.w1zzrd.asm.exception.SignatureInstanceMismatchException;
|
import dev.w1zzrd.asm.exception.SignatureInstanceMismatchException;
|
||||||
import dev.w1zzrd.asm.signature.MethodSignature;
|
import dev.w1zzrd.asm.signature.MethodSignature;
|
||||||
import dev.w1zzrd.asm.signature.TypeSignature;
|
import dev.w1zzrd.asm.signature.TypeSignature;
|
||||||
@ -30,7 +31,7 @@ public class Combine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void inject(MethodNode node, GraftSource source) {
|
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");
|
final boolean acceptReturn = annotation.getEntry("acceptOriginalReturn");
|
||||||
|
|
||||||
@ -195,6 +196,50 @@ public class Combine {
|
|||||||
this.target.methods.add(inject);
|
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) {
|
private boolean initiateGrafting(MethodNode node, GraftSource source) {
|
||||||
DynamicSourceUnit unit = new DynamicSourceUnit(source, node);
|
DynamicSourceUnit unit = new DynamicSourceUnit(source, node);
|
||||||
|
@ -132,18 +132,26 @@ public final class GraftSource {
|
|||||||
return methodAnnotations
|
return methodAnnotations
|
||||||
.keySet()
|
.keySet()
|
||||||
.stream()
|
.stream()
|
||||||
.sorted(Comparator.comparingInt(a -> getInjectAnnotation(a).getEntry("priority")))
|
.sorted(Comparator.comparingInt(a -> getMethodInjectAnnotation(a).getEntry("priority")))
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
public Set<FieldNode> getInjectFields() {
|
public List<FieldNode> getInjectFields() {
|
||||||
return fieldAnnotations.keySet();
|
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));
|
return getInjectionDirective(methodAnnotations.get(node));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public AsmAnnotation<Inject> getFieldInjectAnnotation(FieldNode node) {
|
||||||
|
return getInjectionDirective(fieldAnnotations.get(node));
|
||||||
|
}
|
||||||
|
|
||||||
private static boolean hasNoInjectionDirective(List<AsmAnnotation<?>> annotations) {
|
private static boolean hasNoInjectionDirective(List<AsmAnnotation<?>> annotations) {
|
||||||
return getInjectionDirective(annotations) == null;
|
return getInjectionDirective(annotations) == null;
|
||||||
}
|
}
|
||||||
|
@ -102,6 +102,18 @@ public final class AsmAnnotation<A extends Annotation> {
|
|||||||
return hasDefaultEntry(name) || hasExplicitEntry(name);
|
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) {
|
public static <T extends Annotation> AsmAnnotation<T> getAnnotation(AnnotationNode node) {
|
||||||
Class<T> cls;
|
Class<T> cls;
|
||||||
@ -116,6 +128,6 @@ public final class AsmAnnotation<A extends Annotation> {
|
|||||||
for (int i = 0; i < node.values.size(); i += 2)
|
for (int i = 0; i < node.values.size(); i += 2)
|
||||||
entries.put((String)node.values.get(i), node.values.get(i + 1));
|
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