From 130e21da7e3da600a72664763293095bf9e6cfbb Mon Sep 17 00:00:00 2001 From: Jason Penilla <11360596+jpenilla@users.noreply.github.com> Date: Thu, 4 Nov 2021 17:23:06 -0700 Subject: [PATCH] Deduplicate strings in ObfHelper (#6841) --- build.gradle.kts | 24 ++-- ...ktraces-in-log-messages-crash-report.patch | 123 +++++++++--------- patches/server/Implement-Mob-Goal-API.patch | 2 +- ...ocity-compression-and-cipher-natives.patch | 2 +- 4 files changed, 77 insertions(+), 74 deletions(-) diff --git a/build.gradle.kts b/build.gradle.kts index 5e325d0b6b..9a44dad124 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -56,14 +56,14 @@ paperweight { minecraftVersion.set(providers.gradleProperty("mcVersion")) serverProject.set(project(":Paper-Server")) + paramMappingsRepo.set("https://maven.quiltmc.org/repository/release/") + remapRepo.set("https://maven.fabricmc.net/") + decompileRepo.set("https://files.minecraftforge.net/maven/") + paper { spigotApiPatchDir.set(layout.projectDirectory.dir("patches/api")) spigotServerPatchDir.set(layout.projectDirectory.dir("patches/server")) - paramMappingsRepo.set("https://maven.quiltmc.org/repository/release/") - remapRepo.set("https://maven.fabricmc.net/") - decompileRepo.set("https://files.minecraftforge.net/maven/") - mappingsPatch.set(layout.projectDirectory.file("build-data/mappings-patch.tiny")) reobfMappingsPatch.set(layout.projectDirectory.file("build-data/reobf-mappings-patch.tiny")) @@ -86,18 +86,15 @@ paperweight { tasks.generateDevelopmentBundle { apiCoordinates.set("io.papermc.paper:paper-api") mojangApiCoordinates.set("io.papermc.paper:paper-mojangapi") - libraryRepositories.set( - listOf( - "https://repo.maven.apache.org/maven2/", - "https://libraries.minecraft.net/", - "https://papermc.io/repo/repository/maven-public/", - "https://maven.quiltmc.org/repository/release/", - ) + libraryRepositories.addAll( + "https://repo.maven.apache.org/maven2/", + "https://libraries.minecraft.net/", + "https://papermc.io/repo/repository/maven-public/", ) } publishing { - if (project.hasProperty("publishDevBundle")) { + if (project.providers.gradleProperty("publishDevBundle").forUseAtConfigurationTime().isPresent) { publications.create("devBundle") { artifact(tasks.generateDevelopmentBundle) { artifactId = "dev-bundle" @@ -109,9 +106,8 @@ publishing { allprojects { publishing { repositories { - maven { + maven("https://papermc.io/repo/repository/maven-snapshots/") { name = "paperSnapshots" - url = uri("https://papermc.io/repo/repository/maven-snapshots/") credentials(PasswordCredentials::class) } } diff --git a/patches/server/Deobfuscate-stacktraces-in-log-messages-crash-report.patch b/patches/server/Deobfuscate-stacktraces-in-log-messages-crash-report.patch index c2d3ee8c31..8233943c11 100644 --- a/patches/server/Deobfuscate-stacktraces-in-log-messages-crash-report.patch +++ b/patches/server/Deobfuscate-stacktraces-in-log-messages-crash-report.patch @@ -24,11 +24,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 repositories { maven("https://libraries.minecraft.net/") + // Paper start -+ maven("https://maven.quiltmc.org/repository/release/") { -+ mavenContent { -+ releasesOnly() -+ includeModule("org.quiltmc", "tiny-mappings-parser") -+ } ++ maven("https://maven.fabricmc.net/") { ++ mavenContent { includeModule("net.fabricmc", "mapping-io") } + } + // Paper end } @@ -38,11 +35,19 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 implementation("io.netty:netty-all:4.1.65.Final") // Paper -+ implementation("org.quiltmc:tiny-mappings-parser:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation ++ implementation("net.fabricmc:mapping-io:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation + testImplementation("junit:junit:4.13.1") testImplementation("org.hamcrest:hamcrest-library:1.3") } +@@ -0,0 +0,0 @@ relocation { + + relocate("org.jline:jline-terminal-jansi", "org.jline" to cb("jline")) + sequenceOf( ++ "net.fabricmc:mapping-io" to "net.fabricmc.mappingio", // Paper + "commons-codec:commons-codec" to "org.apache.commons.codec", + "commons-io:commons-io" to "org.apache.commons.io", + //"it.unimi.dsi:fastutil" to "it.unimi", // Paper - don't relocate fastutil @@ -0,0 +0,0 @@ tasks.shadowJar { transform(ModifiedLog4j2PluginsCacheFileTransformer::class.java) } @@ -168,20 +173,20 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 @@ -0,0 +0,0 @@ +package io.papermc.paper.util; + -+import com.google.common.base.Charsets; -+import com.google.common.collect.ImmutableMap; -+import com.google.common.collect.ImmutableSet; -+import com.mojang.datafixers.util.Pair; -+import java.io.BufferedReader; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; ++import java.nio.charset.StandardCharsets; ++import java.util.HashMap; ++import java.util.HashSet; +import java.util.Map; +import java.util.Set; -+import net.fabricmc.mapping.tree.ClassDef; -+import net.fabricmc.mapping.tree.MethodDef; -+import net.fabricmc.mapping.tree.TinyMappingFactory; -+import net.fabricmc.mapping.tree.TinyTree; ++import java.util.function.Function; ++import java.util.stream.Collectors; ++import net.fabricmc.mappingio.MappingReader; ++import net.fabricmc.mappingio.format.MappingFormat; ++import net.fabricmc.mappingio.tree.MappingTree; ++import net.fabricmc.mappingio.tree.MemoryMappingTree; +import org.checkerframework.checker.nullness.qual.NonNull; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.checkerframework.framework.qual.DefaultQualifier; @@ -199,8 +204,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + ObfHelper() { + final @Nullable Set maps = loadMappingsIfPresent(); + if (maps != null) { -+ this.mappingsByObfName = maps.stream().collect(ImmutableMap.toImmutableMap(ClassMapping::obfName, map -> map)); -+ this.mappingsByMojangName = maps.stream().collect(ImmutableMap.toImmutableMap(ClassMapping::mojangName, map -> map)); ++ this.mappingsByObfName = maps.stream().collect(Collectors.toUnmodifiableMap(ClassMapping::obfName, map -> map)); ++ this.mappingsByMojangName = maps.stream().collect(Collectors.toUnmodifiableMap(ClassMapping::mojangName, map -> map)); + } else { + this.mappingsByObfName = null; + this.mappingsByMojangName = null; @@ -260,34 +265,33 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + if (mappingsInputStream == null) { + return null; + } -+ final TinyTree tree = TinyMappingFactory.loadWithDetection(new BufferedReader(new InputStreamReader(mappingsInputStream, Charsets.UTF_8))); -+ final var builder = ImmutableSet.builder(); ++ final MemoryMappingTree tree = new MemoryMappingTree(); ++ MappingReader.read(new InputStreamReader(mappingsInputStream, StandardCharsets.UTF_8), MappingFormat.TINY_2, tree); ++ final Set classes = new HashSet<>(); + -+ for (final ClassDef classDef : tree.getClasses()) { -+ final String obfClassName = classDef.getName(SPIGOT_NAMESPACE).replace('/', '.'); -+ final var methodMappings = ImmutableMap., MethodMapping>builder(); ++ final StringPool pool = new StringPool(); ++ for (final MappingTree.ClassMapping cls : tree.getClasses()) { ++ final Map methods = new HashMap<>(); + -+ for (final MethodDef methodDef : classDef.getMethods()) { -+ final MethodMapping method = new MethodMapping( -+ methodDef.getName(SPIGOT_NAMESPACE), -+ methodDef.getName(MOJANG_PLUS_YARN_NAMESPACE), -+ methodDef.getDescriptor(SPIGOT_NAMESPACE) -+ ); -+ methodMappings.put( -+ new Pair<>(method.obfName(), method.descriptor()), -+ method ++ for (final MappingTree.MethodMapping methodMapping : cls.getMethods()) { ++ methods.put( ++ pool.string(methodKey( ++ methodMapping.getName(SPIGOT_NAMESPACE), ++ methodMapping.getDesc(SPIGOT_NAMESPACE) ++ )), ++ pool.string(methodMapping.getName(MOJANG_PLUS_YARN_NAMESPACE)) + ); + } + + final ClassMapping map = new ClassMapping( -+ obfClassName, -+ classDef.getName(MOJANG_PLUS_YARN_NAMESPACE).replace('/', '.'), -+ methodMappings.build() ++ cls.getName(SPIGOT_NAMESPACE).replace('/', '.'), ++ cls.getName(MOJANG_PLUS_YARN_NAMESPACE).replace('/', '.'), ++ Map.copyOf(methods) + ); -+ builder.add(map); ++ classes.add(map); + } + -+ return builder.build(); ++ return Set.copyOf(classes); + } catch (final IOException ex) { + System.err.println("Failed to load mappings for stacktrace deobfuscation."); + ex.printStackTrace(); @@ -295,16 +299,22 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + } + } + ++ public static String methodKey(final String obfName, final String obfDescriptor) { ++ return obfName + obfDescriptor; ++ } ++ ++ private static final class StringPool { ++ private final Map pool = new HashMap<>(); ++ ++ public String string(final String string) { ++ return this.pool.computeIfAbsent(string, Function.identity()); ++ } ++ } ++ + public record ClassMapping( + String obfName, + String mojangName, -+ Map, MethodMapping> methodMappings -+ ) {} -+ -+ public record MethodMapping( -+ String obfName, -+ String mojangName, -+ String descriptor ++ Map methodsByObf + ) {} +} diff --git a/src/main/java/io/papermc/paper/util/StacktraceDeobfuscator.java b/src/main/java/io/papermc/paper/util/StacktraceDeobfuscator.java @@ -316,7 +326,6 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 +package io.papermc.paper.util; + +import com.destroystokyo.paper.PaperConfig; -+import com.mojang.datafixers.util.Pair; +import it.unimi.dsi.fastutil.ints.IntArrayList; +import it.unimi.dsi.fastutil.ints.IntList; +import java.io.IOException; @@ -337,9 +346,9 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 +public enum StacktraceDeobfuscator { + INSTANCE; + -+ private final Map, Map, IntList>> lineMapCache = Collections.synchronizedMap(new LinkedHashMap<>(128, 0.75f, true) { ++ private final Map, Map> lineMapCache = Collections.synchronizedMap(new LinkedHashMap<>(128, 0.75f, true) { + @Override -+ protected boolean removeEldestEntry(final Map.Entry, Map, IntList>> eldest) { ++ protected boolean removeEldestEntry(final Map.Entry, Map> eldest) { + return this.size() > 127; + } + }); @@ -387,17 +396,15 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + } catch (final ClassNotFoundException ex) { + throw new RuntimeException(ex); + } -+ final @Nullable Pair nameDescriptorPair = this.determineMethodForLine(clazz, element.getLineNumber()); -+ final ObfHelper.@Nullable MethodMapping methodMapping = nameDescriptorPair == null -+ ? null -+ : classMapping.methodMappings().get(nameDescriptorPair); ++ final @Nullable String methodKey = this.determineMethodForLine(clazz, element.getLineNumber()); ++ final @Nullable String mappedMethodName = methodKey == null ? null : classMapping.methodsByObf().get(methodKey); + + result[i] = new StackTraceElement( + element.getClassLoaderName(), + element.getModuleName(), + element.getModuleVersion(), + classMapping.mojangName(), -+ methodMapping != null ? methodMapping.mojangName() : methodName, ++ mappedMethodName != null ? mappedMethodName : methodName, + sourceFileName(classMapping.mojangName()), + element.getLineNumber() + ); @@ -405,15 +412,15 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + return result; + } + -+ private @Nullable Pair determineMethodForLine(final Class clazz, final int lineNumber) { -+ final Map, IntList> lineMap = this.lineMapCache.computeIfAbsent(clazz, StacktraceDeobfuscator::buildLineMap); ++ private @Nullable String determineMethodForLine(final Class clazz, final int lineNumber) { ++ final Map lineMap = this.lineMapCache.computeIfAbsent(clazz, StacktraceDeobfuscator::buildLineMap); + for (final var entry : lineMap.entrySet()) { -+ final Pair pair = entry.getKey(); ++ final String methodKey = entry.getKey(); + final IntList lines = entry.getValue(); + for (int i = 0, linesSize = lines.size(); i < linesSize; i++) { + final int num = lines.getInt(i); + if (num == lineNumber) { -+ return pair; ++ return methodKey; + } + } + } @@ -429,8 +436,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + return rootClassName + ".java"; + } + -+ private static Map, IntList> buildLineMap(final Class key) { -+ final Map, IntList> lineMap = new HashMap<>(); ++ private static Map buildLineMap(final Class key) { ++ final Map lineMap = new HashMap<>(); + final class LineCollectingMethodVisitor extends MethodVisitor { + private final IntList lines = new IntArrayList(); + private final String name; @@ -451,7 +458,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 + @Override + public void visitEnd() { + super.visitEnd(); -+ lineMap.put(new Pair<>(this.name, this.descriptor), this.lines); ++ lineMap.put(ObfHelper.methodKey(this.name, this.descriptor), this.lines); + } + } + final ClassVisitor classVisitor = new ClassVisitor(Opcodes.ASM9) { diff --git a/patches/server/Implement-Mob-Goal-API.patch b/patches/server/Implement-Mob-Goal-API.patch index 465f0d0e40..fd8adab167 100644 --- a/patches/server/Implement-Mob-Goal-API.patch +++ b/patches/server/Implement-Mob-Goal-API.patch @@ -10,7 +10,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 +++ b/build.gradle.kts @@ -0,0 +0,0 @@ dependencies { - implementation("org.quiltmc:tiny-mappings-parser:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation + implementation("net.fabricmc:mapping-io:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation + testImplementation("io.github.classgraph:classgraph:4.8.47") // Paper - mob goal test testImplementation("junit:junit:4.13.1") diff --git a/patches/server/Use-Velocity-compression-and-cipher-natives.patch b/patches/server/Use-Velocity-compression-and-cipher-natives.patch index d1b239ce09..4a187b3f2c 100644 --- a/patches/server/Use-Velocity-compression-and-cipher-natives.patch +++ b/patches/server/Use-Velocity-compression-and-cipher-natives.patch @@ -11,7 +11,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000 @@ -0,0 +0,0 @@ dependencies { implementation("io.netty:netty-all:4.1.65.Final") // Paper - implementation("org.quiltmc:tiny-mappings-parser:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation + implementation("net.fabricmc:mapping-io:0.3.0") // Paper - needed to read mappings for stacktrace deobfuscation + implementation("com.velocitypowered:velocity-native:1.1.0-SNAPSHOT") // Paper testImplementation("io.github.classgraph:classgraph:4.8.47") // Paper - mob goal test