Transform class literals to Java 5 format

In Java 1.4 and earlier, the compiler creates a synthetic method for
invoking Class.forName() and a static field for caching the Class<?>
instance.

This will not interact well with the static scrambling transformer. The
obfuscator strips some synthetic flags. Furthermore, the field and
method need to be moved together with the code that uses them for
decompilers like Fernflower to recognise them and convert them back to
class literals.

If Fernflower misses one, the decompiled Class.forName() helper does not
compile cleanly (as the bytecode is missing a CHECKCAST).

Rather than complicating the future static scrambling transformer, it's
easier to convert these to use LDC and upgrade the .class file version
to Java 5.
pull/66/head
Graham 4 years ago
parent bfcb373ade
commit a82d2e3cef
  1. 2
      deob/src/main/java/dev/openrs2/deob/Deobfuscator.kt
  2. 112
      deob/src/main/java/dev/openrs2/deob/transform/ClassLiteralTransformer.kt

@ -12,6 +12,7 @@ import dev.openrs2.deob.remap.PrefixRemapper.create
import dev.openrs2.deob.transform.BitShiftTransformer
import dev.openrs2.deob.transform.BitwiseOpTransformer
import dev.openrs2.deob.transform.CanvasTransformer
import dev.openrs2.deob.transform.ClassLiteralTransformer
import dev.openrs2.deob.transform.CounterTransformer
import dev.openrs2.deob.transform.DummyArgTransformer
import dev.openrs2.deob.transform.DummyLocalTransformer
@ -162,6 +163,7 @@ class Deobfuscator(private val input: Path, private val output: Path) {
CounterTransformer(),
ResetTransformer(),
FinalTransformer(),
ClassLiteralTransformer(),
VisibilityTransformer(),
OverrideTransformer()
)

@ -0,0 +1,112 @@
package dev.openrs2.deob.transform
import com.github.michaelbull.logging.InlineLogger
import dev.openrs2.asm.ClassVersionUtils
import dev.openrs2.asm.InsnMatcher
import dev.openrs2.asm.MemberRef
import dev.openrs2.asm.classpath.ClassPath
import dev.openrs2.asm.classpath.Library
import dev.openrs2.asm.toInternalClassName
import dev.openrs2.asm.transform.Transformer
import org.objectweb.asm.Opcodes
import org.objectweb.asm.Type
import org.objectweb.asm.tree.ClassNode
import org.objectweb.asm.tree.FieldInsnNode
import org.objectweb.asm.tree.LdcInsnNode
import org.objectweb.asm.tree.MethodInsnNode
import org.objectweb.asm.tree.MethodNode
class ClassLiteralTransformer : Transformer() {
private val classForNameMethods = mutableListOf<MemberRef>()
private var classLiterals = 0
override fun preTransform(classPath: ClassPath) {
classForNameMethods.clear()
classLiterals = 0
for (library in classPath.libraries) {
for (clazz in library) {
clazz.methods.removeIf { method ->
if (method.desc != "(Ljava/lang/String;)Ljava/lang/Class;") {
return@removeIf false
}
if (method.access and Opcodes.ACC_STATIC == 0) {
return@removeIf false
}
val match = CLASS_FOR_NAME_MATCHER.match(method).singleOrNull() ?: return@removeIf false
val invokestatic = match[1] as MethodInsnNode
if (
invokestatic.owner != "java/lang/Class" ||
invokestatic.name != "forName" ||
invokestatic.desc != "(Ljava/lang/String;)Ljava/lang/Class;"
) {
return@removeIf false
}
classForNameMethods.add(MemberRef(clazz, method))
return@removeIf true
}
}
}
logger.info { "Identified Class::forName methods $classForNameMethods" }
}
override fun transformCode(classPath: ClassPath, library: Library, clazz: ClassNode, method: MethodNode): Boolean {
for (match in CLASS_LITERAL_MATCHER.match(method)) {
val getstatic1 = MemberRef(match[0] as FieldInsnNode)
val putstatic = MemberRef(match[5] as FieldInsnNode)
val getstatic2 = MemberRef(match[7] as FieldInsnNode)
val invokestatic = MemberRef(match[3] as MethodInsnNode)
if (getstatic1 != putstatic || putstatic != getstatic2) {
continue
}
if (getstatic1.owner != clazz.name) {
continue
}
if (invokestatic.owner != clazz.name) {
continue
}
if (!classForNameMethods.contains(invokestatic)) {
continue
}
for ((i, insn) in match.withIndex()) {
if (i == 2) {
val ldc = insn as LdcInsnNode
ldc.cst = Type.getObjectType((ldc.cst as String).toInternalClassName())
} else {
method.instructions.remove(insn)
}
}
clazz.version = ClassVersionUtils.maxVersion(clazz.version, Opcodes.V1_5)
clazz.fields.removeIf { it.name == getstatic1.name && it.desc == getstatic1.desc }
classLiterals++
}
return false
}
override fun postTransform(classPath: ClassPath) {
logger.info { "Updated $classLiterals class literals to Java 5 style" }
}
companion object {
private val logger = InlineLogger()
private val CLASS_FOR_NAME_MATCHER = InsnMatcher.compile(
"^ALOAD INVOKESTATIC ARETURN ASTORE NEW DUP INVOKESPECIAL ALOAD INVOKEVIRTUAL ATHROW$"
)
private val CLASS_LITERAL_MATCHER = InsnMatcher.compile(
"GETSTATIC IFNONNULL LDC INVOKESTATIC DUP PUTSTATIC GOTO GETSTATIC"
)
}
}
Loading…
Cancel
Save