diff --git a/build.gradle b/build.gradle index 8c426029a..34ca305ab 100644 --- a/build.gradle +++ b/build.gradle @@ -24,7 +24,8 @@ def brokenConfigurations = [ "commonDecompilerRuntimeClasspath", "fernflowerRuntimeClasspath", "cfrRuntimeClasspath", - "vineflowerRuntimeClasspath" + "vineflowerRuntimeClasspath", + "jadxRuntimeClasspath", ] configurations.configureEach { @@ -109,6 +110,11 @@ sourceSets { srcDir("src/decompilers/vineflower") } } + jadx { + java { + srcDir("src/decompilers/jadx") + } + } } dependencies { @@ -147,14 +153,24 @@ dependencies { vineflowerCompileOnly runtimeLibs.vineflower vineflowerCompileOnly libs.fabric.mapping.io + jadxCompileOnly dependencies.create(runtimeLibs.jadx.core.get()) { + exclude group: 'com.android.tools.build', module: 'aapt2-proto' + } + jadxCompileOnly dependencies.create(runtimeLibs.jadx.java.get()) { + exclude group: 'io.github.skylot', module: 'raung-disasm' + } + jadxCompileOnly libs.fabric.mapping.io + fernflowerApi sourceSets.commonDecompiler.output cfrApi sourceSets.commonDecompiler.output vineflowerApi sourceSets.commonDecompiler.output + jadxApi sourceSets.commonDecompiler.output implementation sourceSets.commonDecompiler.output implementation sourceSets.fernflower.output implementation sourceSets.cfr.output implementation sourceSets.vineflower.output + implementation sourceSets.jadx.output // source code remapping implementation libs.fabric.mercury @@ -198,6 +214,7 @@ jar { from sourceSets.cfr.output.classesDirs from sourceSets.fernflower.output.classesDirs from sourceSets.vineflower.output.classesDirs + from sourceSets.jadx.output.classesDirs } base { diff --git a/gradle/runtime.libs.versions.toml b/gradle/runtime.libs.versions.toml index cf2a01c7c..52d9c2103 100644 --- a/gradle/runtime.libs.versions.toml +++ b/gradle/runtime.libs.versions.toml @@ -3,6 +3,7 @@ fernflower = "2.0.0" cfr = "0.2.2" vineflower = "1.9.3" +jadx = "1.4.7" # Runtime depedencies mixin-compile-extensions = "0.6.0" @@ -16,6 +17,9 @@ native-support = "1.0.1" fernflower = { module = "net.fabricmc:fabric-fernflower", version.ref = "fernflower" } cfr = { module = "net.fabricmc:cfr", version.ref = "cfr" } vineflower = { module = "org.vineflower:vineflower", version.ref = "vineflower" } +jadx-core = { module = "io.github.skylot:jadx-core", version.ref = "jadx" } +jadx-java = { module = "io.github.skylot:jadx-java-input", version.ref = "jadx" } + # Runtime depedencies mixin-compile-extensions = { module = "net.fabricmc:fabric-mixin-compile-extensions", version.ref = "mixin-compile-extensions" } diff --git a/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/JarEntryWriter.java b/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/JarEntryWriter.java new file mode 100644 index 000000000..6e5f5f46e --- /dev/null +++ b/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/JarEntryWriter.java @@ -0,0 +1,70 @@ +/* + * This file is part of fabric-loom, licensed under the MIT License (MIT). + * + * Copyright (c) 2024 FabricMC + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ + +package net.fabricmc.loom.decompilers.jadx; + +import java.io.IOException; +import java.util.Date; +import java.util.HashSet; +import java.util.Set; +import java.util.jar.JarEntry; +import java.util.jar.JarOutputStream; + +public class JarEntryWriter { + private final Set addedDirectories = new HashSet<>(); + private final JarOutputStream outputStream; + + JarEntryWriter(JarOutputStream outputStream) { + this.outputStream = outputStream; + } + + synchronized void write(String filename, byte[] data) throws IOException { + String[] path = filename.split("/"); + String pathPart = ""; + + for (int i = 0; i < path.length - 1; i++) { + pathPart += path[i] + "/"; + + if (addedDirectories.add(pathPart)) { + JarEntry entry = new JarEntry(pathPart); + entry.setTime(new Date().getTime()); + + try { + outputStream.putNextEntry(entry); + outputStream.closeEntry(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + } + + JarEntry entry = new JarEntry(filename); + entry.setTime(new Date().getTime()); + entry.setSize(data.length); + + outputStream.putNextEntry(entry); + outputStream.write(data); + outputStream.closeEntry(); + } +} diff --git a/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/LoomJadxDecompiler.java b/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/LoomJadxDecompiler.java new file mode 100644 index 000000000..40bedb13c --- /dev/null +++ b/src/decompilers/jadx/net/fabricmc/loom/decompilers/jadx/LoomJadxDecompiler.java @@ -0,0 +1,203 @@ +/* + * This file is part of fabric-loom, licensed under the MIT License (MIT). + * + * Copyright (c) 2024 FabricMC + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ + +package net.fabricmc.loom.decompilers.jadx; + +import java.io.BufferedReader; +import java.io.File; +import java.io.IOException; +import java.io.UncheckedIOException; +import java.io.Writer; +import java.nio.charset.StandardCharsets; +import java.nio.file.Files; +import java.util.List; +import java.util.Locale; +import java.util.Map; +import java.util.jar.Attributes; +import java.util.jar.JarOutputStream; +import java.util.jar.Manifest; + +import jadx.api.CommentsLevel; +import jadx.api.ICodeInfo; +import jadx.api.JadxArgs; +import jadx.api.JadxDecompiler; +import jadx.api.JavaClass; +import jadx.api.JavaField; +import jadx.api.JavaMethod; +import jadx.api.impl.NoOpCodeCache; +import jadx.core.codegen.TypeGen; +import jadx.core.dex.attributes.AFlag; +import jadx.core.dex.attributes.AType; +import jadx.core.dex.info.MethodInfo; +import jadx.core.dex.nodes.ClassNode; +import jadx.plugins.input.java.JavaInputPlugin; + +import net.fabricmc.loom.decompilers.LoomInternalDecompiler; +import net.fabricmc.mappingio.MappingReader; +import net.fabricmc.mappingio.adapter.MappingSourceNsSwitch; +import net.fabricmc.mappingio.tree.MappingTree; +import net.fabricmc.mappingio.tree.MappingTree.ClassMapping; +import net.fabricmc.mappingio.tree.MappingTree.FieldMapping; +import net.fabricmc.mappingio.tree.MappingTree.MethodMapping; +import net.fabricmc.mappingio.tree.MemoryMappingTree; + +public final class LoomJadxDecompiler implements LoomInternalDecompiler { + private static JadxArgs getJadxArgs() { + JadxArgs jadxArgs = new JadxArgs(); + jadxArgs.setCodeCache(NoOpCodeCache.INSTANCE); + jadxArgs.setShowInconsistentCode(true); + // jadxArgs.setInlineAnonymousClasses(false); + // jadxArgs.setInlineMethods(false); + jadxArgs.setSkipResources(true); + jadxArgs.setRenameValid(false); + jadxArgs.setRespectBytecodeAccModifiers(true); + jadxArgs.setCommentsLevel(CommentsLevel.WARN); + + return jadxArgs; + } + + @Override + public void decompile(LoomInternalDecompiler.Context context) { + System.out.println(System.getProperty("java.io.tmpdir")); + JadxArgs jadxArgs = getJadxArgs(); + jadxArgs.setThreadsCount(context.numberOfThreads()); + + Manifest manifest = new Manifest(); + manifest.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0"); + + try (JadxDecompiler jadx = new JadxDecompiler(jadxArgs); + JarOutputStream jarOutputStream = new JarOutputStream(Files.newOutputStream(context.sourcesDestination()), manifest); + Writer lineMapWriter = Files.newBufferedWriter(context.linemapDestination(), StandardCharsets.UTF_8)) { + jadx.addCustomLoad(JavaInputPlugin.loadClassFiles(List.of(context.compiledJar()))); + jadx.load(); + + MappingTree tree = readMappings(context.javaDocs().toFile()); + JarEntryWriter jarEntryWriter = new JarEntryWriter(jarOutputStream); + + for (JavaClass cls : jadx.getClasses()) { + if (cls.getClassNode().contains(AFlag.DONT_GENERATE)) { + continue; + } + + String clsName = internalNameOf(cls.getClassNode()); + + // Add Javadocs + addJavadocs(cls, tree.getClass(clsName)); + + // Decompile + ICodeInfo codeInfo = cls.getCodeInfo(); + + if (codeInfo == null) { + context.logger().error("Code not generated for class " + cls.getFullName()); + continue; + } + + if (codeInfo == ICodeInfo.EMPTY) { + continue; + } + + // Write to JAR + String filename = clsName + ".java"; + + try { + byte[] code = codeInfo.getCodeStr().getBytes(StandardCharsets.UTF_8); + jarEntryWriter.write(filename, code); + } catch (IOException e) { + throw new RuntimeException("Unable to create archive: " + filename, e); + } + + // Write line map + int maxLine = 0; + int maxLineDest = 0; + StringBuilder builder = new StringBuilder(); + + for (Map.Entry mappingEntry : cls.getCodeInfo().getCodeMetadata().getLineMapping().entrySet()) { + final int src = mappingEntry.getKey(); + final int dst = mappingEntry.getValue(); + + maxLine = Math.max(maxLine, src); + maxLineDest = Math.max(maxLineDest, dst); + + builder.append("\t").append(src).append("\t").append(dst).append("\n"); + } + + lineMapWriter.write(String.format(Locale.ENGLISH, "%s\t%d\t%d\n", clsName, maxLine, maxLineDest)); + lineMapWriter.write(builder.toString()); + lineMapWriter.write("\n"); + } + } catch (IOException e) { + throw new UncheckedIOException("Failed to decompile", e); + } + } + + private MappingTree readMappings(File input) { + try (BufferedReader reader = Files.newBufferedReader(input.toPath())) { + MemoryMappingTree mappingTree = new MemoryMappingTree(); + MappingSourceNsSwitch nsSwitch = new MappingSourceNsSwitch(mappingTree, "named"); + MappingReader.read(reader, nsSwitch); + + return mappingTree; + } catch (IOException e) { + throw new RuntimeException("Failed to read mappings", e); + } + } + + private void addJavadocs(JavaClass cls, ClassMapping clsMapping) { + String comment; + + if (clsMapping == null) { + return; + } + + if ((comment = emptyToNull(clsMapping.getComment())) != null) { + cls.getClassNode().addAttr(AType.CODE_COMMENTS, comment); + } + + for (JavaField fld : cls.getFields()) { + FieldMapping fldMapping = clsMapping.getField(fld.getFieldNode().getName(), TypeGen.signature(fld.getType())); + + if (fldMapping != null && (comment = emptyToNull(fldMapping.getComment())) != null) { + fld.getFieldNode().addAttr(AType.CODE_COMMENTS, comment); + } + } + + for (JavaMethod mth : cls.getMethods()) { + MethodInfo mthInfo = mth.getMethodNode().getMethodInfo(); + String mthName = mthInfo.getName(); + MethodMapping mthMapping = clsMapping.getMethod(mthName, mthInfo.getShortId().substring(mthName.length())); + + if (mthMapping != null && (comment = emptyToNull(mthMapping.getComment())) != null) { + mth.getMethodNode().addAttr(AType.CODE_COMMENTS, comment); + } + } + } + + private String internalNameOf(ClassNode cls) { + return cls.getClassInfo().makeRawFullName().replace('.', '/'); + } + + public static String emptyToNull(String string) { + return string == null || string.isEmpty() ? null : string; + } +} diff --git a/src/main/java/net/fabricmc/loom/decompilers/DecompilerConfiguration.java b/src/main/java/net/fabricmc/loom/decompilers/DecompilerConfiguration.java index 618629555..6c2b69c97 100644 --- a/src/main/java/net/fabricmc/loom/decompilers/DecompilerConfiguration.java +++ b/src/main/java/net/fabricmc/loom/decompilers/DecompilerConfiguration.java @@ -34,6 +34,8 @@ import org.gradle.api.NamedDomainObjectProvider; import org.gradle.api.Project; import org.gradle.api.artifacts.Configuration; +import org.gradle.api.artifacts.ExternalModuleDependency; +import org.gradle.api.artifacts.dsl.DependencyFactory; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -42,6 +44,7 @@ import net.fabricmc.loom.api.decompilers.LoomDecompiler; import net.fabricmc.loom.decompilers.cfr.LoomCFRDecompiler; import net.fabricmc.loom.decompilers.fernflower.FabricFernFlowerDecompiler; +import net.fabricmc.loom.decompilers.jadx.LoomJadxDecompiler; import net.fabricmc.loom.decompilers.vineflower.VineflowerDecompiler; import net.fabricmc.loom.util.LoomVersions; import net.fabricmc.loom.util.ZipUtils; @@ -55,16 +58,45 @@ public void run() { var fernflowerConfiguration = createConfiguration("fernflower", LoomVersions.FERNFLOWER); var cfrConfiguration = createConfiguration("cfr", LoomVersions.CFR); var vineflowerConfiguration = createConfiguration("vineflower", LoomVersions.VINEFLOWER); + var jadxConfiguration = createJadxConfiguration(); registerDecompiler(getProject(), "fernFlower", BuiltinFernflower.class, fernflowerConfiguration); registerDecompiler(getProject(), "cfr", BuiltinCfr.class, cfrConfiguration); registerDecompiler(getProject(), "vineflower", BuiltinVineflower.class, vineflowerConfiguration); + registerDecompiler(getProject(), "jadx", BuiltinJadx.class, jadxConfiguration); } - private NamedDomainObjectProvider createConfiguration(String name, LoomVersions version) { + private NamedDomainObjectProvider createConfiguration(String name, LoomVersions... versions) { final String configurationName = name + "DecompilerClasspath"; NamedDomainObjectProvider configuration = getProject().getConfigurations().register(configurationName); - getProject().getDependencies().add(configurationName, version.mavenNotation()); + + for (LoomVersions version : versions) { + getProject().getDependencies().add(configurationName, version.mavenNotation()); + } + + return configuration; + } + + private NamedDomainObjectProvider createJadxConfiguration() { + final String configurationName = "jadxDecompilerClasspath"; + final NamedDomainObjectProvider configuration = getProject().getConfigurations().register(configurationName); + final DependencyFactory dependencyFactory = getProject().getDependencyFactory(); + + final ExternalModuleDependency jadxCore = dependencyFactory.create(LoomVersions.JADX_CORE.mavenNotation()); + jadxCore.exclude(Map.of( + "group", "com.android.tools.build", + "module", "aapt2-proto" + )); + + final ExternalModuleDependency jadxJava = dependencyFactory.create(LoomVersions.JADX_JAVA.mavenNotation()); + jadxJava.exclude(Map.of( + "group", "io.github.skylot", + "module", "raung-disasm" + )); + + getProject().getDependencies().add(configurationName, jadxCore); + getProject().getDependencies().add(configurationName, jadxJava); + return configuration; } @@ -77,7 +109,7 @@ private void registerDecompiler(Project project, String name, Class