From 7d9b40bf6ae8c98111fb087262ae6d516bf224cb Mon Sep 17 00:00:00 2001
From: LlamaLad7
Date: Sun, 21 Dec 2025 23:23:24 +0000
Subject: [PATCH 1/5] Fix: Transform field refs to inner classes
---
.../spongepowered/asm/mixin/transformer/MixinTargetContext.java | 2 ++
1 file changed, 2 insertions(+)
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinTargetContext.java b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinTargetContext.java
index e41a3e2b..e66c1979 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinTargetContext.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinTargetContext.java
@@ -637,6 +637,8 @@ private void transformFieldRef(MethodNode method, Iterator ite
if (field != null && field.isRenamed() && field.getOriginalName().equals(fieldRef.getName()) && field.isStatic()) {
fieldRef.setName(field.getName());
}
+ } else if (this.innerClasses.containsKey(fieldRef.getOwner())) {
+ fieldRef.setOwner(this.innerClasses.get(fieldRef.getOwner()));
} else {
if (ClassInfo.isMixin(fieldRef.getOwner())) {
ClassInfo fieldOwner = ClassInfo.forName(fieldRef.getOwner());
From c4f8e0d3270dcaf9cadff3fe44fc4ecb49b9901f Mon Sep 17 00:00:00 2001
From: LlamaLad7
Date: Mon, 22 Dec 2025 16:14:28 +0000
Subject: [PATCH 2/5] Refactor: Don't special case synthetic inner classes.
This was a very strange exception to have in the first place, and the default path works fine.
Notably, you are also allowed to have enums inside Mixins, in which case those would be given different names for each target class and therefore any switch tables referencing them are *not* safe to share.
---
.../mixin/transformer/MixinCoprocessor.java | 9 +-
.../MixinCoprocessorSyntheticInner.java | 104 ------------------
.../asm/mixin/transformer/MixinInfo.java | 33 +-----
.../asm/mixin/transformer/MixinProcessor.java | 1 -
4 files changed, 9 insertions(+), 138 deletions(-)
delete mode 100644 src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessorSyntheticInner.java
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessor.java b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessor.java
index 38354043..e044d2e0 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessor.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessor.java
@@ -35,11 +35,10 @@
* mixins.
*
* Such tasks as {@link MixinCoprocessorAccessor making accessor mixins
- * loadable (and transforming the accessors therein)}, {@link
- * MixinCoprocessorSyntheticInner exposing synthetic inner classes to all
- * consumers} and {@link MixinCoprocessorNestHost applying nest member
- * attributes to nest hosts which may themselves not be mixin targets} are
- * handled by different coprocessors.
+ * loadable (and transforming the accessors therein)}, and {@link
+ * MixinCoprocessorNestHost applying nest member attributes to nest hosts
+ * which may themselves not be mixin targets} are handled by different
+ * coprocessors.
*
* These classes were previously encapsulated in a single companion class
* called MixinPostProcessor, but the mixture of responsibilities of
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessorSyntheticInner.java b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessorSyntheticInner.java
deleted file mode 100644
index ee1f3030..00000000
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinCoprocessorSyntheticInner.java
+++ /dev/null
@@ -1,104 +0,0 @@
-/*
- * This file is part of Mixin, licensed under the MIT License (MIT).
- *
- * Copyright (c) SpongePowered
- * Copyright (c) contributors
- *
- * 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 org.spongepowered.asm.mixin.transformer;
-
-import java.util.HashSet;
-import java.util.Set;
-
-import org.objectweb.asm.Opcodes;
-import org.objectweb.asm.tree.ClassNode;
-import org.objectweb.asm.tree.FieldNode;
-import org.objectweb.asm.tree.MethodNode;
-
-/**
- * Coprocessor which handles synthetic inner classes in mixins. Since synthetic
- * inner classes are usually just switch tables, it is reasonably safe to share
- * them amongst all mixin targets. Therefore the class is simply transformed so
- * that all members are public.
- *
- * Nest based-access control could potentially be used where a mixin has a
- * single target, and Java 11 or higher is in use, but at the moment it's more
- * straightforward to just make everything public.
- */
-class MixinCoprocessorSyntheticInner extends MixinCoprocessor {
-
- /**
- * Synthetic inner classes in mixins
- */
- private final Set syntheticInnerClasses = new HashSet();
-
- MixinCoprocessorSyntheticInner() {
- }
-
- @Override
- String getName() {
- return "syntheticinner";
- }
-
- @Override
- public void onInit(MixinInfo mixin) {
- for (String innerClass : mixin.getSyntheticInnerClasses()) {
- this.registerSyntheticInner(innerClass.replace('/', '.'));
- }
- }
-
- void registerSyntheticInner(String className) {
- this.syntheticInnerClasses.add(className);
- }
-
- @Override
- public boolean couldTransform(String className) {
- return this.syntheticInnerClasses.contains(className);
- }
-
- /**
- * "Pass through" a synthetic inner class. Transforms package-private
- * members in the class into public so that they are accessible from their
- * new home in the target class
- */
- @Override
- ProcessResult process(String className, ClassNode classNode) {
- if (!this.syntheticInnerClasses.contains(className)) {
- return ProcessResult.NONE;
- }
-
- classNode.access |= Opcodes.ACC_PUBLIC;
-
- for (FieldNode field : classNode.fields) {
- if ((field.access & (Opcodes.ACC_PRIVATE | Opcodes.ACC_PROTECTED)) == 0) {
- field.access |= Opcodes.ACC_PUBLIC;
- }
- }
-
- for (MethodNode method : classNode.methods) {
- if ((method.access & (Opcodes.ACC_PRIVATE | Opcodes.ACC_PROTECTED)) == 0) {
- method.access |= Opcodes.ACC_PUBLIC;
- }
- }
-
- return ProcessResult.PASSTHROUGH_TRANSFORMED;
- }
-
-}
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinInfo.java b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinInfo.java
index 1740b096..a26ac474 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinInfo.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinInfo.java
@@ -224,14 +224,9 @@ class State {
* Interfaces soft-implemented using {@link Implements}
*/
protected final List softImplements = new ArrayList();
-
- /**
- * Synthetic inner classes
- */
- protected final Set syntheticInnerClasses = new HashSet();
/**
- * Non-synthetic inner classes
+ * Inner classes
*/
protected final Set innerClasses = new HashSet();
@@ -286,10 +281,6 @@ List extends InterfaceInfo> getSoftImplements() {
return this.softImplements;
}
- Set getSyntheticInnerClasses() {
- return this.syntheticInnerClasses;
- }
-
Set getInnerClasses() {
return this.innerClasses;
}
@@ -420,20 +411,13 @@ void readImplementations(SubType type) {
}
/**
- * Read inner class definitions for the class and locate any synthetic
- * inner classes so that we can add them to the passthrough set in our
- * parent config.
+ * Read inner class definitions for the class and locate any of our inner classes.
*/
void readInnerClasses() {
for (InnerClassNode inner : this.validationClassNode.innerClasses) {
if ((inner.outerName != null && inner.outerName.equals(this.classInfo.getName()))
|| inner.name.startsWith(this.validationClassNode.name + "$")) {
- boolean isStatic = (inner.access & Opcodes.ACC_STATIC) != 0;
- boolean isSynthetic = (inner.access & Opcodes.ACC_SYNTHETIC) != 0;
-
- if (isStatic && isSynthetic) {
- this.syntheticInnerClasses.add(inner.name);
- } else if (!ClassInfo.isMixin(inner.name)) {
+ if (!ClassInfo.isMixin(inner.name)) {
this.innerClasses.add(inner.name);
}
}
@@ -466,7 +450,7 @@ class Reloaded extends State {
*/
@Override
protected void validateChanges(SubType type, List targetClasses) {
- if (!this.syntheticInnerClasses.equals(this.previous.syntheticInnerClasses)) {
+ if (!this.innerClasses.equals(this.previous.innerClasses)) {
throw new MixinReloadException(MixinInfo.this, "Cannot change inner classes");
}
if (!this.interfaces.equals(this.previous.interfaces)) {
@@ -1237,16 +1221,9 @@ public List getTargetClasses() {
List getSoftImplements() {
return Collections.unmodifiableList(this.getState().getSoftImplements());
}
-
- /**
- * Get the synthetic inner classes for this mixin
- */
- Set getSyntheticInnerClasses() {
- return Collections.unmodifiableSet(this.getState().getSyntheticInnerClasses());
- }
/**
- * Get the user-defined inner classes for this mixin
+ * Get the inner classes for this mixin
*/
Set getInnerClasses() {
return Collections.unmodifiableSet(this.getState().getInnerClasses());
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinProcessor.java b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinProcessor.java
index 82ef8821..1a2985a5 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/MixinProcessor.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/MixinProcessor.java
@@ -226,7 +226,6 @@ public String getErrorMessage(IMixinInfo mixin, IMixinConfig config, Phase phase
this.hotSwapper = hotSwapper;
this.coprocessors.add(new MixinCoprocessorPassthrough());
- this.coprocessors.add(new MixinCoprocessorSyntheticInner());
this.coprocessors.add(new MixinCoprocessorAccessor(this.sessionId));
this.coprocessors.add(nestHostCoprocessor);
From 92dff773330821cbcbefe40e2b7503f025bc5958 Mon Sep 17 00:00:00 2001
From: LlamaLad7
Date: Mon, 22 Dec 2025 16:22:16 +0000
Subject: [PATCH 3/5] Improvement: Support nested inner classes in Mixins.
---
.../transformer/InnerClassGenerator.java | 51 +++++++++++--------
1 file changed, 30 insertions(+), 21 deletions(-)
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
index 4a5a56b9..f285a0ad 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
@@ -29,6 +29,7 @@
import java.util.Map;
import java.util.UUID;
+import org.objectweb.asm.tree.InnerClassNode;
import org.spongepowered.asm.logging.ILogger;
import org.objectweb.asm.AnnotationVisitor;
import org.objectweb.asm.ClassVisitor;
@@ -57,7 +58,7 @@ final class InnerClassGenerator implements IClassGenerator {
* Information about an inner class instance. Implements {@link Remapper} so
* that it can participate in the remapping process.
*/
- static class InnerClassInfo extends Remapper implements ISyntheticClassInfo {
+ class InnerClassInfo extends Remapper implements ISyntheticClassInfo {
/**
* Mixin which provides this class
@@ -155,10 +156,21 @@ String getNestHostName() {
void accept(final ClassVisitor classVisitor) throws ClassNotFoundException, IOException {
ClassNode classNode = MixinService.getService().getBytecodeProvider().getClassNode(this.originalName);
+ this.readInnerClasses(classNode);
classNode.accept(classVisitor);
this.loadCounter++;
}
+ private void readInnerClasses(ClassNode classNode) {
+ // Read possibly nested inner classes
+ for (InnerClassNode inner : classNode.innerClasses) {
+ if ((inner.outerName != null && this.findRemappedName(inner.outerName) != null)
+ || inner.name.startsWith(this.mixin.getClassRef() + "$")) {
+ InnerClassGenerator.this.registerInnerClass(this.owner, this.targetClassInfo, inner.name);
+ }
+ }
+ }
+
/**
* Used to remap fields which have been renamed while being applied to
* the target class or by an implementation of IRemapper.
@@ -198,10 +210,9 @@ public String mapMethodName(String owner, String name, String desc) {
*/
@Override
public String map(String key) {
- if (this.originalName.equals(key)) {
- return this.name;
- } else if (this.ownerName.equals(key)) {
- return this.targetClassInfo.getName();
+ String remappedName = this.findRemappedName(key);
+ if (remappedName != null) {
+ return remappedName;
}
return key;
}
@@ -213,6 +224,15 @@ public String map(String key) {
public String toString() {
return this.name;
}
+
+ private String findRemappedName(String originalName) {
+ if (this.ownerName.equals(originalName)) {
+ return this.targetClassInfo.getName();
+ }
+ return InnerClassGenerator.this.innerClassNames.get(
+ InnerClassGenerator.innerClassCoordinate(this.owner, this.targetClassInfo, originalName)
+ );
+ }
}
@@ -252,21 +272,6 @@ public void visitSource(String source, String debug) {
av.visitEnd();
}
- /* (non-Javadoc)
- * @see org.objectweb.asm.commons.RemappingClassAdapter
- * #visitInnerClass(java.lang.String, java.lang.String,
- * java.lang.String, int)
- */
- @Override
- public void visitInnerClass(String name, String outerName, String innerName, int access) {
- if (name.startsWith(this.info.getOriginalName() + "$")) {
- throw new InvalidMixinException(this.info.getOwner(), "Found unsupported nested inner class " + name + " in "
- + this.info.getOriginalName());
- }
-
- super.visitInnerClass(name, outerName, innerName, access);
- }
-
}
/**
@@ -338,7 +343,7 @@ public String getName() {
* @param innerClassName Original inner class name
*/
void registerInnerClass(MixinInfo owner, ClassInfo targetClass, String innerClassName) {
- String coordinate = String.format("%s:%s:%s", owner, innerClassName, targetClass.getName());
+ String coordinate = InnerClassGenerator.innerClassCoordinate(owner, targetClass, innerClassName);
String uniqueName = this.innerClassNames.get(coordinate);
if (uniqueName != null) {
return;
@@ -457,4 +462,8 @@ private static ClassVisitor createRemappingAdapter(ClassVisitor cv, Remapper rem
return InnerClassGenerator.clRemapper.getConstructor(ClassVisitor.class, Remapper.class).newInstance(cv, remapper);
}
+ private static String innerClassCoordinate(MixinInfo owner, ClassInfo targetClass, String innerClassName) {
+ return String.format("%s:%s:%s", owner, innerClassName, targetClass.getName());
+ }
+
}
From b779e291a05a9edf273c85cfef1b37de6a359399 Mon Sep 17 00:00:00 2001
From: LlamaLad7
Date: Tue, 23 Dec 2025 16:30:25 +0000
Subject: [PATCH 4/5] Fix: Don't use MixinInfo#toString for inner class
coordinate generation.
---
.../asm/mixin/transformer/InnerClassGenerator.java | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
index f285a0ad..5f2b5d3b 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
@@ -463,7 +463,7 @@ private static ClassVisitor createRemappingAdapter(ClassVisitor cv, Remapper rem
}
private static String innerClassCoordinate(MixinInfo owner, ClassInfo targetClass, String innerClassName) {
- return String.format("%s:%s:%s", owner, innerClassName, targetClass.getName());
+ return String.format("%s:%s:%s", owner.getClassRef(), innerClassName, targetClass.getName());
}
}
From ffd5c7e7616cc4446fdd3a56d072ca497ce4be50 Mon Sep 17 00:00:00 2001
From: LlamaLad7
Date: Sun, 28 Dec 2025 14:13:49 +0000
Subject: [PATCH 5/5] Improvement: Use deterministic UUIDs for generated inner
class names.
---
.../asm/mixin/transformer/InnerClassGenerator.java | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
index 5f2b5d3b..d70c0537 100644
--- a/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
+++ b/src/main/java/org/spongepowered/asm/mixin/transformer/InnerClassGenerator.java
@@ -25,6 +25,7 @@
package org.spongepowered.asm.mixin.transformer;
import java.io.IOException;
+import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
@@ -426,7 +427,8 @@ private static String getUniqueReference(String originalName, ClassInfo targetCl
if (name.matches("^[0-9]+$")) {
name = "Anonymous";
}
- return String.format("%s$%s$%s", targetClass, name, UUID.randomUUID().toString().replace("-", ""));
+ UUID uuid = UUID.nameUUIDFromBytes(originalName.getBytes(StandardCharsets.UTF_8));
+ return String.format("%s$%s$%s", targetClass, name, uuid.toString().replace("-", ""));
}
/**