diff --git a/src/dev/w1zzrd/asm/Combine.java b/src/dev/w1zzrd/asm/Combine.java index c5b51c9..c8af418 100644 --- a/src/dev/w1zzrd/asm/Combine.java +++ b/src/dev/w1zzrd/asm/Combine.java @@ -274,6 +274,10 @@ public class Combine { return null; } + public String getTargetName() { + return target.name; + } + /** * Prepares a {@link MethodNode} for grafting on to a given method and into the targeted {@link ClassNode} * @param node Node to adapt diff --git a/src/dev/w1zzrd/asm/Injector.java b/src/dev/w1zzrd/asm/Injector.java index 5d1004c..113afd4 100644 --- a/src/dev/w1zzrd/asm/Injector.java +++ b/src/dev/w1zzrd/asm/Injector.java @@ -1,6 +1,10 @@ package dev.w1zzrd.asm; +import dev.w1zzrd.asm.analysis.AsmAnnotation; +import jdk.internal.org.objectweb.asm.Type; +import jdk.internal.org.objectweb.asm.tree.AnnotationNode; import jdk.internal.org.objectweb.asm.tree.ClassNode; +import jdk.internal.org.objectweb.asm.tree.MethodNode; import java.io.File; import java.io.IOException; @@ -13,6 +17,7 @@ import java.util.Objects; * Simple class for automatically performing transformations */ public class Injector { + private static final String injectAnnotDesc = "L"+InjectClass.class.getName().replace('.', '/')+";"; /** * Attempt to inject all valid classes into the given merger from the given loader @@ -20,14 +25,14 @@ public class Injector { * @param merger Merger to inject resources into * @throws IOException If any resource could not be loaded properly */ - public static void injectAll(ClassLoader loader, Merger merger) throws IOException { + public static void injectAll(ClassLoader loader, Combine merger) throws IOException { Enumeration resources = loader.getResources(""); while (resources.hasMoreElements()) injectDirectory(new File(resources.nextElement().getPath()), merger); } // Inject all files in a given directory into the merger - private static void injectDirectory(File file, Merger merger) throws IOException { + private static void injectDirectory(File file, Combine merger) throws IOException { if (file.isDirectory()) for (File child : Objects.requireNonNull(file.listFiles())) injectDirectory(child, merger); @@ -35,7 +40,7 @@ public class Injector { } // Inject file into a given merger (if declared as such) - private static void injectFile(File file, Merger merger) throws IOException { + private static void injectFile(File file, Combine merger) throws IOException { URL url = null; try { url = file.toURI().toURL(); @@ -45,7 +50,21 @@ public class Injector { ClassNode cNode; assert url != null; - if(url.getPath().endsWith(".class") && merger.shouldInject(cNode = Merger.getClassNode(url))) - merger.inject(cNode); + if(url.getPath().endsWith(".class") && shouldInject(merger, cNode = Loader.getClassNode(url))) { + GraftSource source = new GraftSource(cNode); + for (MethodNode mNode : source.getInjectMethods()) + merger.inject(mNode, source); + } + } + + private static boolean shouldInject(Combine combine, ClassNode node) { + for (AnnotationNode annotNode : node.visibleAnnotations) + if (annotNode.desc.equals(injectAnnotDesc) && + ((Type)(AsmAnnotation.getAnnotation(annotNode).getEntry("value"))) + .getClassName() + .equals(combine.getTargetName())) + return true; + + return false; } } diff --git a/src/dev/w1zzrd/asm/Loader.java b/src/dev/w1zzrd/asm/Loader.java new file mode 100644 index 0000000..9b941dc --- /dev/null +++ b/src/dev/w1zzrd/asm/Loader.java @@ -0,0 +1,92 @@ +package dev.w1zzrd.asm; + +import jdk.internal.org.objectweb.asm.ClassReader; +import jdk.internal.org.objectweb.asm.tree.ClassNode; + +import java.io.IOException; +import java.io.InputStream; +import java.net.URL; +import java.util.Objects; + +public class Loader { + + /** + * Get a glass node from a given resource + * @param url Resource to load class node from + * @return Class node loaded from the resource + * @throws IOException If the resource cannot be loaded + */ + public static ClassNode getClassNode(URL url) throws IOException { + return readClass(getClassBytes(url)); + } + + /** + * Read class data to a class node + * @param data Bytecode to read + * @return Class node read + */ + public static ClassNode readClass(byte[] data) { + ClassNode node = new ClassNode(); + new ClassReader(data).accept(node, 0); + return node; + } + + /** + * Read a class node from a given class + * @param name Name of the class to get the class node from + * @return Loaded class node + * @throws IOException If the class data resource cannot be loaded + */ + public static ClassNode getClassNode(String name) throws IOException { + return readClass(getClassBytes(name)); + } + + /** + * Read a class node from a given class + * @param name Name of the class to get the class node from + * @param loader Loader to use when loading the class resource + * @return Loaded class node + * @throws IOException If the class data resource cannot be loaded + */ + public static ClassNode getClassNode(String name, ClassLoader loader) throws IOException { + return readClass(getClassBytes(name, loader)); + } + + /** + * Get class bytecode for a given class + * @param name Name of the class to get data for + * @return Bytecode for the requested class + * @throws IOException If the class data resource cannot be loaded + */ + public static byte[] getClassBytes(String name) throws IOException { + return getClassBytes(name, ClassLoader.getSystemClassLoader()); + } + + /** + * Get class bytecode for a given class + * @param name Name of the class to get data for + * @param loader Loader to use when loading the class resource + * @return Bytecode for the requested class + * @throws IOException If the class data resource cannot be loaded + */ + public static byte[] getClassBytes(String name, ClassLoader loader) throws IOException { + return getClassBytes(Objects.requireNonNull(loader.getResource(name.replace('.', '/') + ".class"))); + } + + /** + * Get class bytecode for a given class + * @param url Resource to load class data from + * @return Bytecode for the requested class resource + * @throws IOException If the class data resource cannot be loaded + */ + public static byte[] getClassBytes(URL url) throws IOException { + InputStream stream = url.openStream(); + byte[] classData = new byte[stream.available()]; + + int total = 0; + do total += stream.read(classData, total, classData.length - total); + while (total < classData.length); + + return classData; + } +} diff --git a/src/dev/w1zzrd/asm/Merger.java b/src/dev/w1zzrd/asm/Merger.java deleted file mode 100644 index a43603b..0000000 --- a/src/dev/w1zzrd/asm/Merger.java +++ /dev/null @@ -1,1023 +0,0 @@ -package dev.w1zzrd.asm; - -import dev.w1zzrd.asm.analysis.AsmAnnotation; -import jdk.internal.org.objectweb.asm.*; -import jdk.internal.org.objectweb.asm.tree.*; -import java.io.*; -import java.lang.annotation.Annotation; -import java.lang.reflect.Field; -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.net.URL; -import java.util.*; -import java.util.regex.Matcher; -import java.util.regex.Pattern; -import java.util.stream.Collectors; -import static dev.w1zzrd.asm.Merger.SpecialCall.FIELD; -import static dev.w1zzrd.asm.Merger.SpecialCall.SUPER; -import static jdk.internal.org.objectweb.asm.ClassWriter.COMPUTE_MAXS; - -/** - * Class data merger/transformer - */ -public class Merger { - - private static final Pattern re_methodSignature = Pattern.compile("((?:[a-zA-Z_$][a-zA-Z\\d_$]+)|(?:))\\(((?:(?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D)*)\\)((?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D|V)"); - private static final Pattern re_types = Pattern.compile("((?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D)"); - private static final Pattern re_retTypes = Pattern.compile("((?:\\[*L(?:[a-zA-Z_$][a-zA-Z\\d_$]*/)*[a-zA-Z_$][a-zA-Z\\d_$]*;)|Z|B|C|S|I|J|F|D|V)"); - - protected final ClassNode targetNode; - - private final HashMap overrideCount = new HashMap<>(); - - - /** - * Create a merger for the given target class - * @param targetClass Class to transform - * @throws IOException If a .class file cannot be found as a resource - */ - public Merger(String targetClass) throws IOException { - this(targetClass, ClassLoader.getSystemClassLoader()); - } - - /** - * Create a merger for the given target class from the given loader - * @param targetClass Class to transform - * @param loader Loader to get class data resource from - * @throws IOException If a .class file cannot be found as a resource - */ - public Merger(String targetClass, ClassLoader loader) throws IOException { - this(getClassNode(targetClass, loader)); - } - - /** - * Create a merger for the given bytecode - * @param data Data to transform - */ - public Merger(byte[] data) { - this(readClass(data)); - } - - /** - * Create a merger for the given ClassNode - * @param targetNode ClassNode to transform - */ - public Merger(ClassNode targetNode) { - this.targetNode = targetNode; - } - - - /** - * Name of the target class - * @return Class name - */ - public String getTargetName() { - return targetNode.name; - } - - /** - * Name of the superclass of the target class - * @return Superclass name - */ - public String getTargetSuperName() { return targetNode.superName; } - - /** - * Inject/override method into the target class - * @param inject MethodNode to inject - * @param injectOwner Name of the class that owns the method being injected (for example "net.example.InjectClass") - */ - public void inject(MethodNode inject, String injectOwner) { - transformInjection(inject, injectOwner.replace('.', '/')); - - targetNode - .methods - .stream() - .filter(it -> methodNodeEquals(it, inject)) - .findFirst() - .ifPresent(targetNode.methods::remove); - - targetNode.methods.add(inject); - } - - /** - * Inject/override a field into the target class - * @param inject Field to inject or override - */ - public void inject(FieldNode inject) { - targetNode - .fields - .stream() - .filter(it -> fieldNodeEquals(it, inject)) - .findFirst() - .ifPresent(targetNode.fields::remove); - - targetNode.fields.add(inject); - } - - /** - * Inject a class into the target class using the given loader - * @param className Name of the class to inject - * @param loader Loader to get the resource from - * @throws IOException If a .class file cannot be found as a resource - */ - public void inject(String className, ClassLoader loader) throws IOException { - inject(getClassNode(loader.getResource(className.replace('.', '/')+".class"))); - } - - /** - * Full name (including packages) of the class to inject - * @param className Name of the class - * @throws IOException If a .class file cannot be found as a resource - */ - public void inject(String className) throws IOException { - inject(className, ClassLoader.getSystemClassLoader()); - } - - /** - * Inject {@link ClassNode} into the target class - * @param inject ClassNode to inject - */ - public void inject(ClassNode inject) { - inject.methods.stream().filter(Merger::shouldInject).forEach(mNode -> inject(mNode, inject.name)); - inject.fields.stream().filter(Merger::shouldInject).forEach(this::inject); - - if (inject.visibleAnnotations != null && inject.interfaces != null) { - - AsmAnnotation injectAnnotation = getAnnotation(InjectClass.class, inject); - - // If there is not injectMethods annotation or there is an - // explicit request to not injectMethods interfaces, just return - if (injectAnnotation == null || - (injectAnnotation.hasEntry("injectInterfaces") && - !(Boolean)injectAnnotation.getEntry("injectInterfaces"))) - return; - - - if (targetNode.interfaces == null) - targetNode.interfaces = new ArrayList<>(); - - - inject.interfaces.stream().filter(it -> !targetNode.interfaces.contains(it)).forEach(targetNode.interfaces::add); - } - } - - /** - * Attempt to inject all annotated methods, fields superclasses and interfaces from the given class - * @param inject Class to inject into the target - * @throws IOException If a .class file cannot be found as a resource - */ - public void inject(Class inject) throws IOException { - inject(getClassNode(inject.getResource(inject.getSimpleName()+".class"))); - } - - /** - * Find a field in the target class - * @param fieldName Name of the field to find - * @return Signature of the found field - * @throws RuntimeException If no field could be found with the given name - */ - protected String resolveField(String fieldName) { - for(FieldNode fNode : targetNode.fields) - if (fNode.name.equals(fieldName)) - return fNode.desc; - - throw new RuntimeException(String.format("There is no field \"%s\" in %s", fieldName, getTargetName())); - } - - /** - * Transform instructions, signature, annotations and local variables of the given {@link MethodNode} - * @param inject MethodNode to inject - * @param injectOwner Type name of the owner (injection) class - */ - protected void transformInjection(MethodNode inject, String injectOwner) { - ArrayList instr = new ArrayList<>(); - - // Adapt instructions - for (int i = 0; i < inject.instructions.size(); ++i) { - AbstractInsnNode node = inject.instructions.get(i); - if (!(node instanceof LineNumberNode)) { - SpecialCall call = node instanceof MethodInsnNode ? getSpecialCall((MethodInsnNode) node) : null; - if (call != null) { - switch (call) { - case FIELD: { - // field access - AbstractInsnNode loadNode = instr.remove(instr.size() - 1); - - String constant = (String) ((LdcInsnNode) loadNode).cst; - - instr.add(new VarInsnNode(Opcodes.ALOAD, 0)); - instr.add(new FieldInsnNode(Opcodes.GETFIELD, getTargetName(), constant, resolveField(constant))); - break; - } - - case SUPER: { - // super call - AbstractInsnNode loadNode = instr.remove(instr.size() - 1); - - do { - node = inject.instructions.get(++i); - if (!(node instanceof MethodInsnNode && ((MethodInsnNode) node).name.equals(((LdcInsnNode)loadNode).cst) && ((MethodInsnNode) node).owner.equals(getTargetName()))) - instr.add(node); - else break; - } while(true); - - ((MethodInsnNode) node).owner = getTargetSuperName(); - instr.add(node); - - break; - } - } - } else { - // Attempt to fix injector ownership - for(Field f : node.getClass().getFields()) { - try { - f.setAccessible(true); - if (f.getName().equals("owner") && f.getType().equals(String.class) && - f.get(node).equals(injectOwner)) - f.set(node, getTargetName()); - } catch (IllegalAccessException e) { - e.printStackTrace(); - } - } - - if (node instanceof FrameNode) { - if (((FrameNode) node).local != null) - ((FrameNode) node).local = ((FrameNode) node).local.stream().map(it -> Objects.equals(it, injectOwner) ? getTargetName() : it).collect(Collectors.toList()); - - if (((FrameNode) node).stack != null) - ((FrameNode) node).stack = ((FrameNode) node).stack.stream().map(it -> Objects.equals(it, injectOwner) ? getTargetName() : it).collect(Collectors.toList()); - } - - instr.add(node); - } - } - } - - - MethodSig signature = getSignature(inject); - AsmAnnotation annotation = getAnnotation(Inject.class, inject); - if (annotation != null && annotation.hasEntry("value")) { - - InPlaceInjection injection = Objects.requireNonNull(annotation.getEntry("value")); - Optional adapt = targetNode.methods.stream().filter(it -> methodNodeEquals(it, inject)).findFirst(); - - if (injection != InPlaceInjection.REPLACE && adapt.isPresent()) { - ArrayList toAdapt = new ArrayList<>(); - adapt.get().instructions.iterator().forEachRemaining(toAdapt::add); - - switch (injection) { - case BEFORE: { // Inject method instructions before an existing method - LabelNode next; - boolean created = false; - if (toAdapt.size() > 0 && toAdapt.get(0) instanceof LabelNode) - next = (LabelNode)toAdapt.get(0); - else { - next = new LabelNode(); - toAdapt.add(0, next); - created = true; - } - - // If no goto instructions were added, just remove the added label - if (removeReturn(instr, next, true, null) && created) - toAdapt.remove(next); - else // A goto call was added. Make sure we inform the JVM of stack and locals with a frame - toAdapt.add(1, new FrameNode(Opcodes.F_SAME, -1, null, -1, null)); - - instr.addAll(toAdapt); - break; - } - - case AFTER: { // Inject method instructions after an existing method - LabelNode next; - boolean created = false; - if (toAdapt.size() > 0 && instr.get(0) instanceof LabelNode) - next = (LabelNode)instr.get(0); - else { - next = new LabelNode(); - instr.add(0, next); - created = true; - } - - boolean isStaticMethod = isStatic(inject); - boolean keepReturn = annotation.hasEntry("acceptOriginalReturn") && (Boolean)annotation.getEntry("acceptOriginalReturn"); - boolean hasReturn = !signature.ret.equals("V"); - - // Received return value - LocalVariableNode retNode = keepReturn && hasReturn ? inject.localVariables.get((isStaticMethod ? 0 : 1) + signature.args.length) : null; - Optional retNode_decl = adapt.get().localVariables.stream().filter(it -> it.name.startsWith(" ")).findFirst(); - - // Set the proper index and update maxLocals accordingly - // This is done here so that injected calls to modify this variable use the correct index - if (keepReturn && hasReturn) { - if (retNode_decl.isPresent()) { - // Extend scope of retNode variable - retNode_decl.get().end = retNode.end; - - // Remove unnecessary retNode - inject.localVariables.remove(retNode); - } - else { - // Insert retNode at the lowest index that isn't "this" - int lowestIndex = isStaticMethod ? 0 : 1; - int original = retNode.index; - retNode.index = lowestIndex; - - toAdapt - .stream() - .filter(it -> it instanceof VarInsnNode && ((VarInsnNode) it).var >= retNode.index) - .forEach(it -> ++((VarInsnNode) it).var); - instr - .stream() - .filter(it -> it instanceof VarInsnNode && ((VarInsnNode) it).var >= retNode.index) - .forEach(it -> { - if (((VarInsnNode) it).var == original) - ((VarInsnNode) it).var = retNode.index; - else ++((VarInsnNode) it).var; - }); - - inject.maxLocals = Math.max(inject.maxLocals, adapt.get().maxLocals); - - // Mark variable as retVal holder in the most lazy way I could think of - retNode.name = ' ' + retNode.name; - } - } - - // If no goto instructions were added, just remove the added label - boolean noGoto = removeReturn(toAdapt, next, !keepReturn && hasReturn, retNode); - if(noGoto) { - if(created) - instr.remove(next); - } - else // A goto call was added. Make sure we inform the JVM of stack and locals with a frame - instr.add(1, new FrameNode(Opcodes.F_SAME, -1, null, -1, null)); - - instr.addAll(0, toAdapt); - - if (keepReturn && hasReturn) { - // A little bit overkill, but I'm lazy - LabelNode first; - if (instr.get(0) instanceof LabelNode) - first = (LabelNode) instr.get(0); - else { - first = new LabelNode(); - instr.add(0, first); - } - - // Make the scope of received retVal span the entire method - retNode.start = first; - } - - break; - } - } - - // Add the original locals to the injection - inject.localVariables.addAll(adapt.get().localVariables.stream().filter(it -> !it.name.equals("this")).collect(Collectors.toList())); - Optional injectThis = inject.localVariables.stream().filter(it -> it.name.equals("this")).findFirst(); - Optional origThis = adapt.get().localVariables.stream().filter(it -> it.name.equals("this")).findFirst(); - - if (injectThis.isPresent() != origThis.isPresent()) - throw new RuntimeException("Method modifier mismatch! Cannot weave a static method and non-static method!"); - - // Update the scope of "this". It always has index 0 and spans the whole method - if (injectThis.isPresent()) { - origThis.get().end = injectThis.get().end; - inject.localVariables.add(origThis.get()); - inject.localVariables.remove(injectThis.get()); - } - - // Ensure all locals have a unique name - inject.localVariables.forEach(it -> makeLocalUnique(it, inject.localVariables)); - } - } - - // Collect instructions - inject.instructions = new InsnList(); - for(AbstractInsnNode node : instr) - inject.instructions.add(node); - - // Ensure local variables don't reference injection class - inject.localVariables.forEach(var -> { - if (var.desc.equals("L"+injectOwner+";")) - var.desc = "L"+getTargetName()+";"; - }); - - // Apply signature overrides - inject.name = signature.name; - inject.desc = '(' + signature.args_literal + ')' + signature.ret; - } - - /** - * Check if the given class is annotated to be injected into the targeted class - * @param inject ClassNode to check for annotations - * @return True if injection class is annotated with {@link InjectClass} and the value is the type of the targeted class - */ - public boolean shouldInject(ClassNode inject) { - AsmAnnotation injectAnnotation = getAnnotation(InjectClass.class, inject); - return injectAnnotation != null && - ((Type)injectAnnotation.getEntry("value")).getClassName().equals(getTargetName()); - } - - /** - * Compile target class data to a byte array - * @return Class data - */ - public byte[] toByteArray() { - return toByteArray(COMPUTE_MAXS); - } - - /** - * Compile target class data to a byte array - * @param writerFlags Flags to pass to the {@link ClassWriter} used to compile the target class - * @return Class data - */ - public byte[] toByteArray(int writerFlags) { - ClassWriter writer = new ClassWriter(writerFlags); - targetNode.methods.forEach(method -> method.localVariables.forEach(var -> var.name = var.name.replace(" ", ""))); - targetNode.accept(writer); - - return writer.toByteArray(); - } - - /** - * Compile target class data to byte array and load with system class loader - * @return Class loaded by the loader - */ - public Class compile() { - return compile(ClassLoader.getSystemClassLoader()); - } - - /** - * Compile target class data to byte array and load with the given class loader - * @param loader Loader to use when loading the class - * @return Class loaded by the loader - */ - public Class compile(ClassLoader loader) { - Method m = null; - try { - m = ClassLoader.class.getDeclaredMethod("defineClass", byte[].class, int.class, int.class); - } catch (NoSuchMethodException e) { - e.printStackTrace(); - } - - assert m != null; - m.setAccessible(true); - //ReflectCompat.setAccessible(m, true); - - byte[] data = toByteArray(); - - try { - return (Class) m.invoke(loader, data, 0, data.length); - } catch (IllegalAccessException | InvocationTargetException e) { - e.printStackTrace(); - } - return null; - } - - // To be used instead of referencing object constructs - - /** - * Reference a non-primitive field in the target class with the given name - * @param name Name of the field to get - * @return Nothing - */ - public static Object field(String name) { - throw new RuntimeException("Field not injected"); - } - - /** - * Inform injector that the next call to a given method should be addressed to the target class' superclass - * @param superMethodName Method name of the target superclass to invoke - */ - public static void superCall(String superMethodName){ - throw new RuntimeException("Super call not injected"); - } - - - /** - * Special transformer calls - */ - enum SpecialCall { - /** - * Indicates a call to {@link #field(String)} - */ - FIELD, - /** - * Indicates a call to {@link #superCall(String)} - */ - SUPER - } - - - /** - * Check if a call for an injector instruction has been made - * @param node Instruction node to check - * @return Call type to transform - */ - protected static SpecialCall getSpecialCall(MethodInsnNode node) { - if (!node.owner.equals("dev/w1zzrd/asm/Merger")) return null; - - switch (node.name) { - case "field": - return FIELD; - - case "superCall": - return SUPER; - - default: - return null; - } - } - - - /** - * Gets a simple representation of an annotation for a given ClassNode - * @param annotationType Type of the annotation to find - * @param cNode ClassNode to find annotation in - * @param Type of the annotation - * @return {@link AsmAnnotation} representing the annotation found or {@literal null} if no annotation of the requested type could not be found. - */ - protected static AsmAnnotation getAnnotation(Class annotationType, ClassNode cNode) { - if(cNode.visibleAnnotations == null) - return null; - - // Internal class name representation to look for - String targetAnnot = 'L' + annotationType.getTypeName().replace('.', '/') + ';'; - - // Check all annotations - for (AnnotationNode aNode : cNode.visibleAnnotations) - if (aNode.desc.equals(targetAnnot)) { - HashMap map = new HashMap<>(); - - // Collect annotation values - if (aNode.values != null) - NODE_LOOP: - for (int i = 1; i < aNode.values.size(); i+=2) { - String key = (String) aNode.values.get(i - 1); - Object toPut = aNode.values.get(i); - - // Attempt to parse non-primitive data type to its actual type - if (toPut instanceof String[] && ((String[]) toPut).length == 2) { - String enumType = ((String[])toPut)[0]; - String enumName = ((String[])toPut)[1]; - if (enumType.startsWith("L") && enumType.endsWith(";")) - try{ - Class type = Class.forName(enumType.substring(1, enumType.length()-1).replace('/', '.')); - Method m = Enum.class.getDeclaredMethod("name"); - Object[] values = (Object[]) type.getDeclaredMethod("values").invoke(null); - - for (Object value : values) - if (m.invoke(value).equals(enumName)) { - map.put(key, value); - continue NODE_LOOP; - } - - } catch (Throwable e) { - /* Just ignore */ - } - } - - // Default insertion policy - map.put(key, toPut); - } - - return new AsmAnnotation<>(annotationType, map); - } - - return null; - } - - /** - * Gets a simple representation of an annotation for a given MethodNode - * @param annotationType Type of the annotation to find - * @param mNode ClassNode to find annotation in - * @param Type of the annotation - * @return {@link AsmAnnotation} representing the annotation found or {@literal null} if no annotation of the requested type could not be found. - */ - protected static AsmAnnotation getAnnotation(Class annotationType, MethodNode mNode) { - if(mNode.visibleAnnotations == null) - return null; - - String targetAnnot = 'L' + annotationType.getTypeName().replace('.', '/') + ';'; - - // Internal class name representation to look for - for (AnnotationNode aNode : mNode.visibleAnnotations) - if (aNode.desc.equals(targetAnnot)) { - HashMap map = new HashMap<>(); - - // Collect annotation values - if (aNode.values != null) - NODE_LOOP: - for (int i = 1; i < aNode.values.size(); i+=2) { - String key = (String) aNode.values.get(i - 1); - Object toPut = aNode.values.get(i); - - // Attempt to parse non-primitive data type to its actual type - if (toPut instanceof String[] && ((String[]) toPut).length == 2) { - String enumType = ((String[])toPut)[0]; - String enumName = ((String[])toPut)[1]; - if (enumType.startsWith("L") && enumType.endsWith(";")) - try{ - Class type = Class.forName(enumType.substring(1, enumType.length()-1).replace('/', '.')); - Method m = Enum.class.getDeclaredMethod("name"); - Object[] values = (Object[]) type.getDeclaredMethod("values").invoke(null); - - for (Object value : values) - if (m.invoke(value).equals(enumName)) { - map.put(key, value); - continue NODE_LOOP; - } - - } catch (Throwable e) { - /* Just ignore */ - } - } - - // Default insertion policy - map.put(key, toPut); - } - - return new AsmAnnotation<>(annotationType, map); - } - - return null; - } - - /** - * Check semantic equality of two method nodes - * @param a First method node - * @param b Second method node - * @return True of the two method nodes represent the same method - */ - protected static boolean methodNodeEquals(MethodNode a, MethodNode b) { - return getSignature(a).equals(getSignature(b)) && (isStatic(a) == isStatic(b)); - } - - /** - * Check if method node is static - * @param node Node to check - * @return True if the node is declared as static - */ - protected static boolean isStatic(MethodNode node) { - return (node.access & Opcodes.ACC_STATIC) != 0; - } - - /** - * Parse the targeted or actual method signature of a method node - * @param node Node to parse signature of - * @return Parsed signature from {@link Inject#target()} (if it exists and is valid), otherwise the actual signature - */ - protected static MethodSig getSignature(MethodNode node) { - AsmAnnotation annotation = getAnnotation(Inject.class, node); - MethodSig actualSignature = Objects.requireNonNull(parseMethodSignature(node.name + node.desc)); - - // Attempt to parse a declared signature - if (annotation != null && annotation.hasEntry("target")) { - MethodSig overrideSignature = parseMethodSignature(annotation.getEntry("target")); - - if (overrideSignature != null) { - // Ensure the signatures are compatible - if (!Arrays.equals(overrideSignature.args, actualSignature.args)) { - if (overrideSignature.args.length + 1 != actualSignature.args.length) - throw new RuntimeException(String.format("Unreasonable signature declaration for method %s (actually %s)", overrideSignature.toString(), actualSignature.toString())); - - for (int i = 0; i < overrideSignature.args.length; ++i) - if (!overrideSignature.args[i].equals(actualSignature.args[i])) - throw new RuntimeException(String.format("Signature mismatch for method %s (actually %s)", overrideSignature.toString(), actualSignature.toString())); - - if (!actualSignature.args[overrideSignature.args.length].equals(overrideSignature.ret)) - throw new RuntimeException(String.format("Unreasonable additional argument declaration for method %s (actually %s)", overrideSignature.toString(), actualSignature.toString())); - } - - if (!overrideSignature.ret.equals(actualSignature.ret)) - throw new RuntimeException(String.format("Unreasonable return declaration for method %s (actually %s)", overrideSignature.toString(), actualSignature.toString())); - - // We have target signature override - // Use this instead of the actual method signature - return overrideSignature; - } - } - - // Parse implicit signature - return actualSignature; - } - - /** - * Change the variable name to ensure that it is unique - * @param node Node to change the name of (if necessary) - * @param other All the local variables in the relevant method node - */ - protected static void makeLocalUnique(LocalVariableNode node, List other) { - while (other.stream().anyMatch(it -> it != node && it.name.equals(node.name))) - node.name = '$'+node.name; - } - - /** - * Parse the signature of a method to a {@link MethodSig} - * @param sig String representation of the signature (e.g. {@code getMethodName(IIZ)Ljava/lang/String;} would represent {@code String getMethodName(int, int, boolean)}) - * @return Parsed signature - */ - protected static MethodSig parseMethodSignature(String sig) { - Matcher signatureMatcher = re_methodSignature.matcher(sig); - - if (sig.length() > 0 && signatureMatcher.matches()) { - String name = signatureMatcher.group(1); - String ret = signatureMatcher.group(3); - - // Match arguments - Matcher argMatcher = re_types.matcher(signatureMatcher.group(2)); - ArrayList args = new ArrayList<>(); - while (argMatcher.find()) - args.add(argMatcher.group(1)); - - return new MethodSig(name, ret, args.toArray(new String[args.size()])); - } - - return null; - } - - /** - * Data class for storing method signature and name - */ - protected static final class MethodSig { - public final String name; - public final String ret; - public final String[] args; - public final String args_literal; - - public MethodSig(String name, String ret, String[] args) { - this.name = name; - this.ret = ret; - this.args = args; - - StringBuilder builder = new StringBuilder(); - for (String s : args) - builder.append(s); - - args_literal = builder.toString(); - } - - @Override - public String toString() { - return name+'('+args_literal+')'+ret; - } - - @Override - public boolean equals(Object obj) { - return obj instanceof MethodSig && toString().equals(obj.toString()); - } - } - - - /** - * Resolve a type for a frame - * @param typeString String representation of the type to resolve - * @return Corresponding Opcode or String representing the type given - */ - protected static Object resolveFrameType(String typeString) { - Type sigType = Type.getType(typeString); - switch (sigType.getSort()) { - case 1: - case 2: - case 3: - case 4: - case 5: - return Opcodes.INTEGER; - case 6: - return Opcodes.FLOAT; - case 7: - return Opcodes.LONG; - case 8: - return Opcodes.DOUBLE; - case 9: - return sigType.getDescriptor(); - default: - return sigType.getInternalName(); - } - } - - /** - * Resolve bytecode instruction to use when storing a type to a variable - * @param typeString Type to store to a variable - * @return Bytecode instruction or -1 if the type is void - */ - protected static int resolveStoreInstr(String typeString) { - switch (typeString) { - case "Z": - case "I": - case "B": - case "C": - case "S": - return Opcodes.ISTORE; - case "J": - return Opcodes.LSTORE; - case "F": - return Opcodes.FSTORE; - case "D": - return Opcodes.DSTORE; - - // Void has no store type - case "V": - return -1; - - default: - return Opcodes.ASTORE; - } - } - - /** - * Check if two field nodes are semantically equivalent - * @param a First field node - * @param b Second field node - * @return True of the node have the same signature and name - */ - protected static boolean fieldNodeEquals(FieldNode a, FieldNode b) { - return a.name.equals(b.name) && Objects.equals(a.signature, b.signature); - } - - /** - * Check if the given method node should be injected into the target class - * @param node Node to check - * @return True if it should, else false - */ - protected static boolean shouldInject(MethodNode node) { - if (node.visibleAnnotations == null) return false; - - String targetDesc = 'L' + Inject.class.getTypeName().replace('.', '/') + ';'; - - for (AnnotationNode aNode : node.visibleAnnotations) - if (aNode.desc.equals(targetDesc)) - return true; - - return false; - } - - /** - * Check if the given field node should be injected into the target class - * @param node Node to check - * @return True if it should, else false - */ - protected static boolean shouldInject(FieldNode node) { - if (node.visibleAnnotations == null) return false; - - String targetDesc = 'L' + Inject.class.getTypeName().replace('.', '/') + ';'; - - for (AnnotationNode aNode : node.visibleAnnotations) - if (aNode.desc.equals(targetDesc)) - return true; - - return false; - } - - /** - * Replace/remove return instructions from a given set of instruction nodes - * @param instr Instructions to transform - * @param jumpReplace Label to jump to instead of returning - * @param popReturn Whether or not to simply pop the return value from the stack before jump - * @param storeNode Local variable to store the return value to if this has been requested - * @return True if any jump instructions were added, else false - */ - protected static boolean removeReturn(List instr, LabelNode jumpReplace, boolean popReturn, LocalVariableNode storeNode) { - ListIterator iter = instr.listIterator(); - JumpInsnNode finalJump = null; - int keepLabel = 0; - while (iter.hasNext()) { - AbstractInsnNode node = iter.next(); - if (node instanceof InsnNode && node.getOpcode() >= Opcodes.IRETURN && node.getOpcode() <= Opcodes.RETURN) { - iter.remove(); - - // If we're not keeping the return value and the return - // value is gotten from a method call, just pop the result of the call - if(popReturn && !removeRedundantLoad(iter)) { - if (node.getOpcode() == Opcodes.LRETURN || node.getOpcode() == Opcodes.DRETURN) - iter.add(new InsnNode(Opcodes.POP2)); - else if (node.getOpcode() != Opcodes.RETURN) - iter.add(new InsnNode(Opcodes.POP)); - } else { - iter.add(new VarInsnNode(resolveStoreInstr(storeNode.desc), storeNode.index)); - } - - iter.add(finalJump = new JumpInsnNode(Opcodes.GOTO, jumpReplace)); - ++keepLabel; - } - } - - if (finalJump != null) // This *should* always be true - instr.remove(finalJump); - - return keepLabel <= 1; - } - - /** - * Remove side-effect free load instructions - * @param iter Instructions to analyze - * @return True if the load was removed, else false - */ - protected static boolean removeRedundantLoad(ListIterator iter) { - boolean hasEffects = false; - int iterCount = 0; - while (iter.hasPrevious()) { - AbstractInsnNode node = iter.previous(); - ++iterCount; - - if (node instanceof MethodInsnNode) { - hasEffects = true; - break; - } - - if ((node instanceof FieldInsnNode && node.getOpcode() == Opcodes.GETSTATIC) || - (node instanceof InsnNode && (node.getOpcode() == Opcodes.LDC || - (node.getOpcode() >= Opcodes.ILOAD && node.getOpcode() <= Opcodes.ALOAD) || - (node.getOpcode() >= Opcodes.IALOAD && node.getOpcode() <= Opcodes.SALOAD)))) - break; - } - - for(int i = 0; i < iterCount; ++i) { - iter.next(); - if (!hasEffects) - iter.remove(); - } - - return hasEffects; - } - - /** - * Get a glass node from a given resource - * @param url Resource to load class node from - * @return Class node loaded from the resource - * @throws IOException If the resource cannot be loaded - */ - public static ClassNode getClassNode(URL url) throws IOException { - return readClass(getClassBytes(url)); - } - - /** - * Read class data to a class node - * @param data Bytecode to read - * @return Class node read - */ - public static ClassNode readClass(byte[] data) { - ClassNode node = new ClassNode(); - new ClassReader(data).accept(node, 0); - return node; - } - - /** - * Read a class node from a given class - * @param name Name of the class to get the class node from - * @return Loaded class node - * @throws IOException If the class data resource cannot be loaded - */ - public static ClassNode getClassNode(String name) throws IOException { - return readClass(getClassBytes(name)); - } - - /** - * Read a class node from a given class - * @param name Name of the class to get the class node from - * @param loader Loader to use when loading the class resource - * @return Loaded class node - * @throws IOException If the class data resource cannot be loaded - */ - public static ClassNode getClassNode(String name, ClassLoader loader) throws IOException { - return readClass(getClassBytes(name, loader)); - } - - /** - * Get class bytecode for a given class - * @param name Name of the class to get data for - * @return Bytecode for the requested class - * @throws IOException If the class data resource cannot be loaded - */ - public static byte[] getClassBytes(String name) throws IOException { - return getClassBytes(name, ClassLoader.getSystemClassLoader()); - } - - /** - * Get class bytecode for a given class - * @param name Name of the class to get data for - * @param loader Loader to use when loading the class resource - * @return Bytecode for the requested class - * @throws IOException If the class data resource cannot be loaded - */ - public static byte[] getClassBytes(String name, ClassLoader loader) throws IOException { - return getClassBytes(Objects.requireNonNull(loader.getResource(name.replace('.', '/') + ".class"))); - } - - /** - * Get class bytecode for a given class - * @param url Resource to load class data from - * @return Bytecode for the requested class resource - * @throws IOException If the class data resource cannot be loaded - */ - public static byte[] getClassBytes(URL url) throws IOException { - InputStream stream = url.openStream(); - byte[] classData = new byte[stream.available()]; - - int total = 0; - do total += stream.read(classData, total, classData.length - total); - while (total < classData.length); - - return classData; - } -} diff --git a/src/dev/w1zzrd/asm/ReflectCompat.java b/src/dev/w1zzrd/asm/reflect/ReflectCompat.java similarity index 99% rename from src/dev/w1zzrd/asm/ReflectCompat.java rename to src/dev/w1zzrd/asm/reflect/ReflectCompat.java index 944782d..be88850 100644 --- a/src/dev/w1zzrd/asm/ReflectCompat.java +++ b/src/dev/w1zzrd/asm/reflect/ReflectCompat.java @@ -1,4 +1,4 @@ -package dev.w1zzrd.asm; +package dev.w1zzrd.asm.reflect; import dev.w1zzrd.asm.reflect.BruteForceDummy; import sun.misc.Unsafe;