From a132a2964ddd7f47aa48942acb7c21c092c2bc64 Mon Sep 17 00:00:00 2001 From: Haowei Wen Date: Sun, 3 Jul 2022 15:30:46 +0800 Subject: [PATCH 1/3] Add exception to the AGPL This exception allows launchers to bundle authlib-injector in their program. --- LICENSE | 14 ++++++++++++++ README.en.md | 10 ++++++++++ README.md | 10 ++++++++++ 3 files changed, 34 insertions(+) diff --git a/LICENSE b/LICENSE index be3f7b2..f509636 100644 --- a/LICENSE +++ b/LICENSE @@ -659,3 +659,17 @@ specific requirements. if any, to sign a "copyright disclaimer" for the program, if necessary. For more information on this, and how to apply and follow the GNU AGPL, see . + + +"AUTHLIB-INJECTOR" EXCEPTION TO THE AGPL + + As a special exception, using this work in the following ways does not cause +your program to be covered by the AGPL: + + a) Bundling the unaltered binary form of this work in your program without + statically or dynamically linking to it; or + + b) Interacting with this work through the provided inter-process + communication interface, such as the HTTP API; or + + c) Loading this work as a Java Agent into a Java Virtual Machine. diff --git a/README.en.md b/README.en.md index d913e07..3b0ae5b 100644 --- a/README.en.md +++ b/README.en.md @@ -111,3 +111,13 @@ Configure Minecraft server with the following JVM parameter: It's disabled by default if the authentication server does NOT send feature.usernameCheck option. Turning on this option will prevent players whose username contains special characters from joining the server. ``` + +## License +This work is licensed under the [GNU Affero General Public License v3.0](https://github.com/yushijinhun/authlib-injector/blob/develop/LICENSE) or later, with the "AUTHLIB-INJECTOR" exception. + +> **"AUTHLIB-INJECTOR" EXCEPTION TO THE AGPL** +> +> As a special exception, using this work in the following ways does not cause your program to be covered by the AGPL: +> 1. Bundling the unaltered binary form of this work in your program without statically or dynamically linking to it; or +> 2. Interacting with this work through the provided inter-process communication interface, such as the HTTP API; or +> 3. Loading this work as a Java Agent into a Java Virtual Machine. diff --git a/README.md b/README.md index 0fdc9be..6334003 100644 --- a/README.md +++ b/README.md @@ -121,3 +121,13 @@ gradle ## 捐助 BMCLAPI 为 authlib-injector 提供了[下载镜像站](https://github.com/yushijinhun/authlib-injector/wiki/%E8%8E%B7%E5%8F%96-authlib-injector#bmclapi-%E9%95%9C%E5%83%8F)。如果您想要支持 authlib-injector 的开发,您可以[捐助 BMCLAPI](https://bmclapidoc.bangbang93.com/)。 + +## 许可 +本程序使用 [GNU Affero General Public License v3.0 or later](https://github.com/yushijinhun/authlib-injector/blob/develop/LICENSE) 许可,并附有以下例外: + +> **AGPL 的例外情况:** +> +> 作为特例,如果您的程序通过以下方式利用本作品,则相应的行为不会导致您的作品被 AGPL 协议涵盖。 +> 1. 您的程序通过打包的方式包含本作品未经修改的二进制形式,而没有静态或动态地链接到本作品;或 +> 2. 您的程序通过本作品提供的进程间通信接口(如 HTTP API)进行交互;或 +> 3. 您的程序将本作品作为 Java Agent 加载进 Java 虚拟机。 From 4862db0572b40d1a7ba2d2787ae1eb621be4372e Mon Sep 17 00:00:00 2001 From: Haowei Wen Date: Sat, 9 Jul 2022 23:27:21 +0800 Subject: [PATCH 2/3] fix setting feature.username_check doesn't disable BungeeCordAllowedCharactersTransformer --- src/main/java/moe/yushi/authlibinjector/AuthlibInjector.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/moe/yushi/authlibinjector/AuthlibInjector.java b/src/main/java/moe/yushi/authlibinjector/AuthlibInjector.java index 97c30e3..76024ce 100644 --- a/src/main/java/moe/yushi/authlibinjector/AuthlibInjector.java +++ b/src/main/java/moe/yushi/authlibinjector/AuthlibInjector.java @@ -276,7 +276,6 @@ private static ClassTransformer createTransformer(APIMetadata config) { transformer.units.add(new ConstantURLTransformUnit(urlProcessor)); transformer.units.add(new CitizensTransformer()); transformer.units.add(new ConcatenateURLTransformUnit()); - transformer.units.add(new BungeeCordAllowedCharactersTransformer()); boolean usernameCheckDefault = Boolean.TRUE.equals(config.getMeta().get("feature.username_check")); if (Config.usernameCheck.isEnabled(usernameCheckDefault)) { @@ -284,6 +283,7 @@ private static ClassTransformer createTransformer(APIMetadata config) { } else { transformer.units.add(new UsernameCharacterCheckTransformer()); transformer.units.add(new PaperUsernameCheckTransformer()); + transformer.units.add(new BungeeCordAllowedCharactersTransformer()); } transformer.units.add(new SkinWhitelistTransformUnit()); From 5059a55789883394ec211d22b653d54e29a59cb7 Mon Sep 17 00:00:00 2001 From: Haowei Wen Date: Sun, 10 Jul 2022 04:25:24 +0800 Subject: [PATCH 3/3] fix class version below 50 is not supported --- .../CallbackMetafactoryTransformer.java | 71 -------- .../transform/CallbackSupport.java | 155 +++++++++++++++++- .../transform/ClassTransformer.java | 139 ++++++++++------ .../transform/ClassVersionException.java | 23 --- .../transform/ClassVersionTransformUnit.java | 60 ------- .../transform/TransformContext.java | 16 +- .../support/AuthlibLogInterceptor.java | 5 +- .../support/ConcatenateURLTransformUnit.java | 5 +- .../support/MC52974_1710Workaround.java | 7 +- .../support/MainArgumentsTransformer.java | 5 +- .../support/SkinWhitelistTransformUnit.java | 6 +- .../support/YggdrasilKeyTransformUnit.java | 18 +- 12 files changed, 269 insertions(+), 241 deletions(-) delete mode 100644 src/main/java/moe/yushi/authlibinjector/transform/CallbackMetafactoryTransformer.java delete mode 100644 src/main/java/moe/yushi/authlibinjector/transform/ClassVersionException.java delete mode 100644 src/main/java/moe/yushi/authlibinjector/transform/ClassVersionTransformUnit.java diff --git a/src/main/java/moe/yushi/authlibinjector/transform/CallbackMetafactoryTransformer.java b/src/main/java/moe/yushi/authlibinjector/transform/CallbackMetafactoryTransformer.java deleted file mode 100644 index 6107155..0000000 --- a/src/main/java/moe/yushi/authlibinjector/transform/CallbackMetafactoryTransformer.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright (C) 2021 Haowei Wen and contributors - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU Affero General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU Affero General Public License for more details. - * - * You should have received a copy of the GNU Affero General Public License - * along with this program. If not, see . - */ -package moe.yushi.authlibinjector.transform; - -import static org.objectweb.asm.Opcodes.ACC_PRIVATE; -import static org.objectweb.asm.Opcodes.ACC_STATIC; -import static org.objectweb.asm.Opcodes.ACC_SYNTHETIC; -import static org.objectweb.asm.Opcodes.ALOAD; -import static org.objectweb.asm.Opcodes.ARETURN; -import static org.objectweb.asm.Opcodes.ASM9; -import static org.objectweb.asm.Opcodes.DUP; -import static org.objectweb.asm.Opcodes.INVOKESPECIAL; -import static org.objectweb.asm.Opcodes.INVOKESTATIC; -import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL; -import static org.objectweb.asm.Opcodes.NEW; -import java.util.Optional; -import org.objectweb.asm.ClassVisitor; -import org.objectweb.asm.MethodVisitor; - -class CallbackMetafactoryTransformer implements TransformUnit { - - @Override - public Optional transform(ClassLoader classLoader, String className, ClassVisitor writer, TransformContext context) { - return Optional.of(new ClassVisitor(ASM9, writer) { - @Override - public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) { - super.visit(version, access, name, signature, superName, interfaces); - - MethodVisitor mv = super.visitMethod(ACC_PRIVATE | ACC_STATIC | ACC_SYNTHETIC, - CallbackSupport.METAFACTORY_NAME, - CallbackSupport.METAFACTORY_SIGNATURE, - null, null); - mv.visitCode(); - mv.visitTypeInsn(NEW, "java/lang/invoke/ConstantCallSite"); - mv.visitInsn(DUP); - mv.visitVarInsn(ALOAD, 0); - mv.visitMethodInsn(INVOKESTATIC, "java/lang/ClassLoader", "getSystemClassLoader", "()Ljava/lang/ClassLoader;", false); - mv.visitVarInsn(ALOAD, 3); - mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/ClassLoader", "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;", false); - mv.visitVarInsn(ALOAD, 1); - mv.visitVarInsn(ALOAD, 2); - mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/invoke/MethodHandles$Lookup", "findStatic", "(Ljava/lang/Class;Ljava/lang/String;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/MethodHandle;", false); - mv.visitMethodInsn(INVOKESPECIAL, "java/lang/invoke/ConstantCallSite", "", "(Ljava/lang/invoke/MethodHandle;)V", false); - mv.visitInsn(ARETURN); - mv.visitMaxs(-1, -1); - mv.visitEnd(); - - context.markModified(); - } - }); - } - - @Override - public String toString() { - return "Callback Metafactory Transformer"; - } -} diff --git a/src/main/java/moe/yushi/authlibinjector/transform/CallbackSupport.java b/src/main/java/moe/yushi/authlibinjector/transform/CallbackSupport.java index 917ca34..f618378 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/CallbackSupport.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/CallbackSupport.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -16,17 +16,42 @@ */ package moe.yushi.authlibinjector.transform; +import static org.objectweb.asm.Opcodes.AASTORE; +import static org.objectweb.asm.Opcodes.ACC_PRIVATE; +import static org.objectweb.asm.Opcodes.ACC_STATIC; +import static org.objectweb.asm.Opcodes.ACC_SYNTHETIC; +import static org.objectweb.asm.Opcodes.ALOAD; +import static org.objectweb.asm.Opcodes.ANEWARRAY; +import static org.objectweb.asm.Opcodes.ARETURN; +import static org.objectweb.asm.Opcodes.DLOAD; +import static org.objectweb.asm.Opcodes.DRETURN; +import static org.objectweb.asm.Opcodes.DUP; +import static org.objectweb.asm.Opcodes.FLOAD; +import static org.objectweb.asm.Opcodes.FRETURN; +import static org.objectweb.asm.Opcodes.GETSTATIC; +import static org.objectweb.asm.Opcodes.H_INVOKESTATIC; +import static org.objectweb.asm.Opcodes.ILOAD; +import static org.objectweb.asm.Opcodes.INVOKESPECIAL; +import static org.objectweb.asm.Opcodes.INVOKESTATIC; +import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL; +import static org.objectweb.asm.Opcodes.IRETURN; +import static org.objectweb.asm.Opcodes.LLOAD; +import static org.objectweb.asm.Opcodes.LRETURN; +import static org.objectweb.asm.Opcodes.NEW; +import static org.objectweb.asm.Opcodes.RETURN; import java.lang.reflect.Method; import java.lang.reflect.Modifier; +import org.objectweb.asm.ClassVisitor; +import org.objectweb.asm.Handle; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Type; -public final class CallbackSupport { +final class CallbackSupport { private CallbackSupport() { } - static final String METAFACTORY_NAME = "__authlibinjector_metafactory"; - static final String METAFACTORY_SIGNATURE = "(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/String;)Ljava/lang/invoke/CallSite;"; + private static final String METAFACTORY_NAME = "__authlibinjector_metafactory"; + private static final String METAFACTORY_SIGNATURE = "(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/String;)Ljava/lang/invoke/CallSite;"; private static Method findCallbackMethod(Class owner, String methodName) { for (Method method : owner.getDeclaredMethods()) { @@ -40,11 +65,123 @@ private static Method findCallbackMethod(Class owner, String methodName) { throw new IllegalArgumentException("No such method: " + methodName); } - public static void invoke(TransformContext ctx, MethodVisitor mv, Class owner, String methodName) { - ctx.requireMinimumClassVersion(50); - ctx.upgradeClassVersion(51); - + static void callWithInvokeDynamic(MethodVisitor mv, Class owner, String methodName, TransformContext ctx) { String descriptor = Type.getMethodDescriptor(findCallbackMethod(owner, methodName)); - mv.visitInvokeDynamicInsn(methodName, descriptor, ctx.acquireCallbackMetafactory(), owner.getName()); + Handle callbackMetafactory = new Handle( + H_INVOKESTATIC, + ctx.getClassName().replace('.', '/'), + CallbackSupport.METAFACTORY_NAME, + CallbackSupport.METAFACTORY_SIGNATURE, + ctx.isInterface()); + mv.visitInvokeDynamicInsn(methodName, descriptor, callbackMetafactory, owner.getName()); + } + + static void callWithIntermediateMethod(MethodVisitor mv0, Class owner, String methodName, TransformContext ctx) { + Method callbackMethod = findCallbackMethod(owner, methodName); + String descriptor = Type.getMethodDescriptor(callbackMethod); + String intermediateMethod = "__authlibinjector_intermediate__" + owner.getName().replace('.', '_') + "__" + methodName; + mv0.visitMethodInsn(INVOKESTATIC, ctx.getClassName().replace('.', '/'), intermediateMethod, descriptor, ctx.isInterface()); + + ctx.addGeneratedMethod(intermediateMethod, cv -> { + int paramNum = callbackMethod.getParameterCount(); + Class[] paramTypes = callbackMethod.getParameterTypes(); + Class returnType = callbackMethod.getReturnType(); + + MethodVisitor mv = cv.visitMethod(ACC_PRIVATE | ACC_STATIC | ACC_SYNTHETIC, intermediateMethod, descriptor, null, null); + mv.visitCode(); + mv.visitMethodInsn(INVOKESTATIC, "java/lang/invoke/MethodHandles", "publicLookup", "()Ljava/lang/invoke/MethodHandles$Lookup;", false); + mv.visitMethodInsn(INVOKESTATIC, "java/lang/ClassLoader", "getSystemClassLoader", "()Ljava/lang/ClassLoader;", false); + mv.visitLdcInsn(owner.getName()); + mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/ClassLoader", "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;", false); + mv.visitLdcInsn(methodName); + pushType(mv, returnType); + mv.visitLdcInsn(paramNum); + mv.visitTypeInsn(ANEWARRAY, "java/lang/Class"); + for (int i = 0; i < paramNum; i++) { + mv.visitInsn(DUP); + mv.visitLdcInsn(i); + pushType(mv, paramTypes[i]); + mv.visitInsn(AASTORE); + } + mv.visitMethodInsn(INVOKESTATIC, "java/lang/invoke/MethodType", "methodType", "(Ljava/lang/Class;[Ljava/lang/Class;)Ljava/lang/invoke/MethodType;", false); + mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/invoke/MethodHandles$Lookup", "findStatic", "(Ljava/lang/Class;Ljava/lang/String;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/MethodHandle;", false); + for (int i = 0; i < paramNum; i++) { + Class type = paramTypes[i]; + if (type == boolean.class || type == byte.class || type == char.class || type == short.class || type == int.class) { + mv.visitVarInsn(ILOAD, i); + } else if (type == long.class) { + mv.visitVarInsn(LLOAD, i); + } else if (type == float.class) { + mv.visitVarInsn(FLOAD, i); + } else if (type == double.class) { + mv.visitVarInsn(DLOAD, i); + } else { + mv.visitVarInsn(ALOAD, i); + } + } + mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/invoke/MethodHandle", "invokeExact", descriptor, false); + if (returnType == void.class) { + mv.visitInsn(RETURN); + } else if (returnType == boolean.class || returnType == byte.class || returnType == char.class || returnType == short.class || returnType == int.class) { + mv.visitInsn(IRETURN); + } else if (returnType == long.class) { + mv.visitInsn(LRETURN); + } else if (returnType == float.class) { + mv.visitInsn(FRETURN); + } else if (returnType == double.class) { + mv.visitInsn(DRETURN); + } else { + mv.visitInsn(ARETURN); + } + mv.visitMaxs(-1, -1); + mv.visitEnd(); + }); + } + + private static void pushType(MethodVisitor mv, Class type) { + if (type.isPrimitive()) { + if (type == boolean.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Boolean", "TYPE", "Ljava/lang/Class;"); + } else if (type == byte.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Byte", "TYPE", "Ljava/lang/Class;"); + } else if (type == char.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Character", "TYPE", "Ljava/lang/Class;"); + } else if (type == short.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Short", "TYPE", "Ljava/lang/Class;"); + } else if (type == int.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Integer", "TYPE", "Ljava/lang/Class;"); + } else if (type == float.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Float", "TYPE", "Ljava/lang/Class;"); + } else if (type == long.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Long", "TYPE", "Ljava/lang/Class;"); + } else if (type == double.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Double", "TYPE", "Ljava/lang/Class;"); + } else if (type == void.class) { + mv.visitFieldInsn(GETSTATIC, "java/lang/Void", "TYPE", "Ljava/lang/Class;"); + } + } else { + mv.visitLdcInsn(Type.getType(type)); + } + } + + static void insertMetafactory(ClassVisitor visitor) { + MethodVisitor mv = visitor.visitMethod(ACC_PRIVATE | ACC_STATIC | ACC_SYNTHETIC, + CallbackSupport.METAFACTORY_NAME, + CallbackSupport.METAFACTORY_SIGNATURE, + null, null); + mv.visitCode(); + mv.visitTypeInsn(NEW, "java/lang/invoke/ConstantCallSite"); + mv.visitInsn(DUP); + mv.visitVarInsn(ALOAD, 0); + mv.visitMethodInsn(INVOKESTATIC, "java/lang/ClassLoader", "getSystemClassLoader", "()Ljava/lang/ClassLoader;", false); + mv.visitVarInsn(ALOAD, 3); + mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/ClassLoader", "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;", false); + mv.visitVarInsn(ALOAD, 1); + mv.visitVarInsn(ALOAD, 2); + mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/invoke/MethodHandles$Lookup", "findStatic", "(Ljava/lang/Class;Ljava/lang/String;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/MethodHandle;", false); + mv.visitMethodInsn(INVOKESPECIAL, "java/lang/invoke/ConstantCallSite", "", "(Ljava/lang/invoke/MethodHandle;)V", false); + mv.visitInsn(ARETURN); + mv.visitMaxs(-1, -1); + mv.visitEnd(); } } diff --git a/src/main/java/moe/yushi/authlibinjector/transform/ClassTransformer.java b/src/main/java/moe/yushi/authlibinjector/transform/ClassTransformer.java index f35ccd5..28f3827 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/ClassTransformer.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/ClassTransformer.java @@ -22,20 +22,24 @@ import static moe.yushi.authlibinjector.util.Logging.Level.INFO; import static moe.yushi.authlibinjector.util.Logging.Level.WARNING; import static org.objectweb.asm.Opcodes.ACC_INTERFACE; -import static org.objectweb.asm.Opcodes.H_INVOKESTATIC; +import static org.objectweb.asm.Opcodes.ASM9; import java.lang.instrument.ClassFileTransformer; import java.lang.instrument.IllegalClassFormatException; import java.security.ProtectionDomain; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; +import java.util.LinkedHashMap; import java.util.List; +import java.util.Map; +import java.util.Map.Entry; import java.util.Optional; import java.util.concurrent.CopyOnWriteArrayList; +import java.util.function.Consumer; import org.objectweb.asm.ClassReader; import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.ClassWriter; -import org.objectweb.asm.Handle; +import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.Config; public class ClassTransformer implements ClassFileTransformer { @@ -50,8 +54,6 @@ private class TransformHandle { private class TransformContextImpl implements TransformContext { public boolean modifiedMark; - public int minVersionMark = -1; - public int upgradedVersionMark = -1; public boolean callbackMetafactoryRequested = false; @Override @@ -60,33 +62,38 @@ public void markModified() { } @Override - public void requireMinimumClassVersion(int version) { - if (this.minVersionMark < version) { - this.minVersionMark = version; - } + public List getStringConstants() { + return TransformHandle.this.getStringConstants(); } @Override - public void upgradeClassVersion(int version) { - if (this.upgradedVersionMark < version) { - this.upgradedVersionMark = version; - } + public String getClassName() { + return className; } @Override - public Handle acquireCallbackMetafactory() { - this.callbackMetafactoryRequested = true; - return new Handle( - H_INVOKESTATIC, - className.replace('.', '/'), - CallbackSupport.METAFACTORY_NAME, - CallbackSupport.METAFACTORY_SIGNATURE, - TransformHandle.this.isInterface()); + public boolean isInterface() { + return TransformHandle.this.isInterface(); } @Override - public List getStringConstants() { - return TransformHandle.this.getStringConstants(); + public void invokeCallback(MethodVisitor mv, Class owner, String methodName) { + boolean useInvokeDynamic = (getClassVersion() & 0xffff) >= 50; + + if (useInvokeDynamic) { + addCallbackMetafactory = true; + CallbackSupport.callWithInvokeDynamic(mv, owner, methodName, this); + } else { + CallbackSupport.callWithIntermediateMethod(mv, owner, methodName, this); + } + } + + @Override + public void addGeneratedMethod(String name, Consumer generator) { + if (generatedMethods == null) { + generatedMethods = new LinkedHashMap<>(); + } + generatedMethods.put(name, generator); } } @@ -97,9 +104,8 @@ public List getStringConstants() { private List cachedConstants; private List appliedTransformers; - private int minVersion = -1; - private int upgradedVersion = -1; private boolean addCallbackMetafactory = false; + private Map> generatedMethods; public TransformHandle(ClassLoader classLoader, String className, byte[] classBuffer) { this.className = className; @@ -123,6 +129,11 @@ private List getStringConstants() { return cachedConstants; } + private int getClassVersion() { + ClassReader reader = getClassReader(); + return reader.readInt(reader.getItem(1) - 7); + } + public void accept(TransformUnit... units) { long t0 = System.nanoTime(); @@ -168,43 +179,77 @@ public void accept(TransformUnit... units) { appliedTransformers = new ArrayList<>(); appliedTransformers.add(units[i]); - if (ctx.minVersionMark > this.minVersion) { - this.minVersion = ctx.minVersionMark; - } - if (ctx.upgradedVersionMark > this.upgradedVersion) { - this.upgradedVersion = ctx.upgradedVersionMark; - } this.addCallbackMetafactory |= ctx.callbackMetafactoryRequested; modified = true; } if (modified) { - classBuffer = writer.toByteArray(); - cachedClassReader = null; - cachedConstants = null; + updateClassBuffer(writer.toByteArray()); } } - public Optional finish() { - if (appliedTransformers == null || appliedTransformers.isEmpty()) { - return Optional.empty(); + private void injectCallbackMetafactory() { + log(DEBUG, "Adding callback metafactory"); + + int classVersion = getClassVersion(); + int majorVersion = classVersion & 0xffff; + + int newVersion; + if (majorVersion < 51) { + newVersion = 51; + log(DEBUG, "Upgrading class version from " + classVersion + " to " + newVersion); } else { - if (addCallbackMetafactory) { - accept(new CallbackMetafactoryTransformer()); + newVersion = classVersion; + } + + ClassReader reader = getClassReader(); + ClassWriter writer = new ClassWriter(reader, ClassWriter.COMPUTE_MAXS); + ClassVisitor visitor = new ClassVisitor(ASM9, writer) { + @Override + public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) { + super.visit(newVersion, access, name, signature, superName, interfaces); + CallbackSupport.insertMetafactory(this); } - if (minVersion == -1 && upgradedVersion == -1) { - return Optional.of(classBuffer); - } else { - try { - accept(new ClassVersionTransformUnit(minVersion, upgradedVersion)); - return Optional.of(classBuffer); - } catch (ClassVersionException e) { - log(WARNING, "Skipping [" + className + "], " + e.getMessage()); - return Optional.empty(); + }; + reader.accept(visitor, 0); + updateClassBuffer(writer.toByteArray()); + } + + private void injectGeneratedMethods() { + ClassReader reader = getClassReader(); + ClassWriter writer = new ClassWriter(reader, ClassWriter.COMPUTE_MAXS); + ClassVisitor visitor = new ClassVisitor(ASM9, writer) { + @Override + public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) { + super.visit(version, access, name, signature, superName, interfaces); + for (Entry> el : generatedMethods.entrySet()) { + log(DEBUG, "Adding generated method [" + el.getKey() + "]"); + el.getValue().accept(this); } } + }; + reader.accept(visitor, 0); + updateClassBuffer(writer.toByteArray()); + } + + private void updateClassBuffer(byte[] buf) { + classBuffer = buf; + cachedClassReader = null; + cachedConstants = null; + } + + public Optional finish() { + if (appliedTransformers == null || appliedTransformers.isEmpty()) { + return Optional.empty(); + } + if (addCallbackMetafactory) { + injectCallbackMetafactory(); + } + if (generatedMethods != null) { + injectGeneratedMethods(); } + return Optional.of(classBuffer); } public List getAppliedTransformers() { diff --git a/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionException.java b/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionException.java deleted file mode 100644 index 85067d5..0000000 --- a/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionException.java +++ /dev/null @@ -1,23 +0,0 @@ -/* - * Copyright (C) 2020 Haowei Wen and contributors - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU Affero General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU Affero General Public License for more details. - * - * You should have received a copy of the GNU Affero General Public License - * along with this program. If not, see . - */ -package moe.yushi.authlibinjector.transform; - -class ClassVersionException extends RuntimeException { - public ClassVersionException(String message) { - super(message); - } -} diff --git a/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionTransformUnit.java b/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionTransformUnit.java deleted file mode 100644 index f916ba8..0000000 --- a/src/main/java/moe/yushi/authlibinjector/transform/ClassVersionTransformUnit.java +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright (C) 2021 Haowei Wen and contributors - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU Affero General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU Affero General Public License for more details. - * - * You should have received a copy of the GNU Affero General Public License - * along with this program. If not, see . - */ -package moe.yushi.authlibinjector.transform; - -import static moe.yushi.authlibinjector.util.Logging.log; -import static moe.yushi.authlibinjector.util.Logging.Level.DEBUG; -import static org.objectweb.asm.Opcodes.ASM9; -import java.util.Optional; -import org.objectweb.asm.ClassVisitor; - -class ClassVersionTransformUnit implements TransformUnit { - - private final int minVersion; - private final int upgradedVersion; - - public ClassVersionTransformUnit(int minVersion, int upgradedVersion) { - this.minVersion = minVersion; - this.upgradedVersion = upgradedVersion; - } - - @Override - public Optional transform(ClassLoader classLoader, String className, ClassVisitor writer, TransformContext context) { - return Optional.of(new ClassVisitor(ASM9, writer) { - @Override - public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) { - int major = version & 0xffff; - - if (minVersion != -1 && major < minVersion) { - throw new ClassVersionException("class version (" + major + ") is lower than required(" + minVersion + ")"); - } - - if (upgradedVersion != -1 && major < upgradedVersion) { - log(DEBUG,"Upgrading class version from " + major + " to " + upgradedVersion); - version = upgradedVersion; - context.markModified(); - } - super.visit(version, access, name, signature, superName, interfaces); - } - }); - } - - @Override - public String toString() { - return "Class File Version Transformer"; - } -} diff --git a/src/main/java/moe/yushi/authlibinjector/transform/TransformContext.java b/src/main/java/moe/yushi/authlibinjector/transform/TransformContext.java index 525b031..92c9709 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/TransformContext.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/TransformContext.java @@ -17,17 +17,21 @@ package moe.yushi.authlibinjector.transform; import java.util.List; -import org.objectweb.asm.Handle; +import java.util.function.Consumer; +import org.objectweb.asm.ClassVisitor; +import org.objectweb.asm.MethodVisitor; public interface TransformContext { - void markModified(); - - void requireMinimumClassVersion(int version); + String getClassName(); - void upgradeClassVersion(int version); + boolean isInterface(); - Handle acquireCallbackMetafactory(); + void markModified(); List getStringConstants(); + + void invokeCallback(MethodVisitor mv, Class owner, String methodName); + + void addGeneratedMethod(String name, Consumer generator); } diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/AuthlibLogInterceptor.java b/src/main/java/moe/yushi/authlibinjector/transform/support/AuthlibLogInterceptor.java index 673b691..a09836f 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/AuthlibLogInterceptor.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/AuthlibLogInterceptor.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -37,7 +37,6 @@ import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Type; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; @@ -232,7 +231,7 @@ public void visitCode() { super.visitCode(); super.visitLdcInsn(Type.getType("L" + className.replace('.', '/') + ";")); super.visitMethodInsn(INVOKEVIRTUAL, "java/lang/Class", "getClassLoader", "()Ljava/lang/ClassLoader;", false); - CallbackSupport.invoke(ctx, mv, AuthlibLogInterceptor.class, "onClassLoading"); + ctx.invokeCallback(mv, AuthlibLogInterceptor.class, "onClassLoading"); ctx.markModified(); } }; diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/ConcatenateURLTransformUnit.java b/src/main/java/moe/yushi/authlibinjector/transform/support/ConcatenateURLTransformUnit.java index 09070ef..d26ed01 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/ConcatenateURLTransformUnit.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/ConcatenateURLTransformUnit.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -25,7 +25,6 @@ import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; @@ -59,7 +58,7 @@ public MethodVisitor visitMethod(int access, String name, String descriptor, Str mv.visitCode(); mv.visitVarInsn(ALOAD, 0); mv.visitVarInsn(ALOAD, 1); - CallbackSupport.invoke(ctx, mv, ConcatenateURLTransformUnit.class, "concatenateURL"); + ctx.invokeCallback(mv, ConcatenateURLTransformUnit.class, "concatenateURL"); mv.visitInsn(ARETURN); mv.visitMaxs(-1, -1); mv.visitEnd(); diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/MC52974_1710Workaround.java b/src/main/java/moe/yushi/authlibinjector/transform/support/MC52974_1710Workaround.java index 0b6dd92..dea9a01 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/MC52974_1710Workaround.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/MC52974_1710Workaround.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -33,7 +33,6 @@ import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.AuthlibInjector; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; import moe.yushi.authlibinjector.util.WeakIdentityHashMap; @@ -120,7 +119,7 @@ public void visitInsn(int opcode) { if (opcode == ARETURN) { ctx.markModified(); super.visitInsn(DUP); - CallbackSupport.invoke(ctx, mv, MC52974_1710Workaround.class, "markGameProfile"); + ctx.invokeCallback(mv, MC52974_1710Workaround.class, "markGameProfile"); } super.visitInsn(opcode); } @@ -163,7 +162,7 @@ public void visitFieldInsn(int opcode, String owner, String name, String descrip super.visitMethodInsn(INVOKESTATIC, "net/minecraft/server/MinecraftServer", "func_71276_C", "()Lnet/minecraft/server/MinecraftServer;", false); } super.visitLdcInsn(isNotchName ? 1 : 0); - CallbackSupport.invoke(ctx, mv, MC52974_1710Workaround.class, "accessGameProfile"); + ctx.invokeCallback(mv, MC52974_1710Workaround.class, "accessGameProfile"); super.visitTypeInsn(CHECKCAST, "com/mojang/authlib/GameProfile"); } } diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/MainArgumentsTransformer.java b/src/main/java/moe/yushi/authlibinjector/transform/support/MainArgumentsTransformer.java index 15dd9b2..72dc24f 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/MainArgumentsTransformer.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/MainArgumentsTransformer.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -31,7 +31,6 @@ import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; @@ -51,7 +50,7 @@ public void visitCode() { ctx.markModified(); super.visitVarInsn(ALOAD, 0); - CallbackSupport.invoke(ctx, mv, MainArgumentsTransformer.class, "processMainArguments"); + ctx.invokeCallback(mv, MainArgumentsTransformer.class, "processMainArguments"); super.visitVarInsn(ASTORE, 0); } }; diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/SkinWhitelistTransformUnit.java b/src/main/java/moe/yushi/authlibinjector/transform/support/SkinWhitelistTransformUnit.java index 566a1c8..4de0009 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/SkinWhitelistTransformUnit.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/SkinWhitelistTransformUnit.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 Haowei Wen and contributors + * Copyright (C) 2022 Haowei Wen and contributors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by @@ -27,7 +27,6 @@ import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; @@ -64,6 +63,7 @@ public static List getWhitelistedDomains() { @CallbackMethod public static boolean isWhitelistedDomain(String url) { + System.out.println(url); String domain; try { domain = new URI(url).getHost(); @@ -103,7 +103,7 @@ public MethodVisitor visitMethod(int access, String name, String desc, String si MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions); mv.visitCode(); mv.visitVarInsn(ALOAD, 0); - CallbackSupport.invoke(ctx, mv, SkinWhitelistTransformUnit.class, "isWhitelistedDomain"); + ctx.invokeCallback(mv, SkinWhitelistTransformUnit.class, "isWhitelistedDomain"); mv.visitInsn(IRETURN); mv.visitMaxs(-1, -1); mv.visitEnd(); diff --git a/src/main/java/moe/yushi/authlibinjector/transform/support/YggdrasilKeyTransformUnit.java b/src/main/java/moe/yushi/authlibinjector/transform/support/YggdrasilKeyTransformUnit.java index 2625beb..86b376b 100644 --- a/src/main/java/moe/yushi/authlibinjector/transform/support/YggdrasilKeyTransformUnit.java +++ b/src/main/java/moe/yushi/authlibinjector/transform/support/YggdrasilKeyTransformUnit.java @@ -16,10 +16,11 @@ */ package moe.yushi.authlibinjector.transform.support; +import static java.lang.invoke.MethodHandles.publicLookup; +import static java.lang.invoke.MethodType.methodType; import static moe.yushi.authlibinjector.util.Logging.Level.DEBUG; import static org.objectweb.asm.Opcodes.ALOAD; import static org.objectweb.asm.Opcodes.ASM9; -import static org.objectweb.asm.Opcodes.H_INVOKEVIRTUAL; import static org.objectweb.asm.Opcodes.INVOKEVIRTUAL; import static org.objectweb.asm.Opcodes.IRETURN; import java.lang.invoke.MethodHandle; @@ -32,10 +33,8 @@ import java.util.Optional; import java.util.concurrent.CopyOnWriteArrayList; import org.objectweb.asm.ClassVisitor; -import org.objectweb.asm.Handle; import org.objectweb.asm.MethodVisitor; import moe.yushi.authlibinjector.transform.CallbackMethod; -import moe.yushi.authlibinjector.transform.CallbackSupport; import moe.yushi.authlibinjector.transform.TransformContext; import moe.yushi.authlibinjector.transform.TransformUnit; import moe.yushi.authlibinjector.util.Logging; @@ -46,12 +45,14 @@ public class YggdrasilKeyTransformUnit implements TransformUnit { public static final List PUBLIC_KEYS = new CopyOnWriteArrayList<>(); @CallbackMethod - public static boolean verifyPropertySignature(Object property, PublicKey mojangKey, MethodHandle verifyAction) throws Throwable { - if ((boolean) verifyAction.invoke(property, mojangKey)) { + public static boolean verifyPropertySignature(Object property, PublicKey mojangKey) throws Throwable { + MethodHandle verifyAction = publicLookup().bind(property, "isSignatureValid", methodType(boolean.class, PublicKey.class)); + + if ((boolean) verifyAction.invokeExact(mojangKey)) { return true; } for (PublicKey customKey : PUBLIC_KEYS) { - if ((boolean) verifyAction.invoke(property, customKey)) { + if ((boolean) verifyAction.invokeExact(customKey)) { return true; } } @@ -101,8 +102,7 @@ public void visitMethodInsn(int opcode, String owner, String name, String descri && "isSignatureValid".equals(name) && "(Ljava/security/PublicKey;)Z".equals(descriptor)) { ctx.markModified(); - super.visitLdcInsn(new Handle(H_INVOKEVIRTUAL, owner, name, descriptor, isInterface)); - CallbackSupport.invoke(ctx, this, YggdrasilKeyTransformUnit.class, "verifyPropertySignature"); + ctx.invokeCallback(this, YggdrasilKeyTransformUnit.class, "verifyPropertySignature"); } else { super.visitMethodInsn(opcode, owner, name, descriptor, isInterface); } @@ -126,7 +126,7 @@ public MethodVisitor visitMethod(int access, String name, String desc, String si mv.visitMethodInsn(INVOKEVIRTUAL, "com/mojang/authlib/properties/Property", "getValue", "()Ljava/lang/String;", false); mv.visitVarInsn(ALOAD, 1); mv.visitMethodInsn(INVOKEVIRTUAL, "com/mojang/authlib/properties/Property", "getSignature", "()Ljava/lang/String;", false); - CallbackSupport.invoke(ctx, mv, YggdrasilKeyTransformUnit.class, "verifyPropertySignatureNew"); + ctx.invokeCallback(mv, YggdrasilKeyTransformUnit.class, "verifyPropertySignatureNew"); mv.visitInsn(IRETURN); mv.visitMaxs(-1, -1); mv.visitEnd();