parent
0db3d979c9
commit
b6d96ac4e6
@ -1,241 +0,0 @@ |
||||
package dev.openrs2.deob.remap; |
||||
|
||||
import java.util.HashMap; |
||||
import java.util.List; |
||||
import java.util.Map; |
||||
|
||||
import com.google.common.base.Strings; |
||||
import com.google.common.collect.ImmutableSet; |
||||
import dev.openrs2.asm.MemberDesc; |
||||
import dev.openrs2.asm.MemberRef; |
||||
import dev.openrs2.asm.classpath.ClassMetadata; |
||||
import dev.openrs2.asm.classpath.ClassPath; |
||||
import dev.openrs2.common.StringUtilsKt; |
||||
import dev.openrs2.common.collect.DisjointSet; |
||||
import kotlin.text.StringsKt; |
||||
import org.objectweb.asm.Type; |
||||
import org.objectweb.asm.commons.Remapper; |
||||
import org.slf4j.Logger; |
||||
import org.slf4j.LoggerFactory; |
||||
|
||||
public final class TypedRemapper extends Remapper { |
||||
private static final Logger logger = LoggerFactory.getLogger(TypedRemapper.class); |
||||
|
||||
public static final ImmutableSet<String> EXCLUDED_CLASSES = ImmutableSet.of( |
||||
"client", |
||||
"jagex3/jagmisc/jagmisc", |
||||
"loader", |
||||
"unpack", |
||||
"unpackclass" |
||||
); |
||||
private static final ImmutableSet<String> EXCLUDED_METHODS = ImmutableSet.of( |
||||
"<clinit>", |
||||
"<init>", |
||||
"main", |
||||
"providesignlink", |
||||
"quit" |
||||
); |
||||
private static final ImmutableSet<String> EXCLUDED_FIELDS = ImmutableSet.of( |
||||
"cache" |
||||
); |
||||
private static final int MAX_OBFUSCATED_NAME_LEN = 2; |
||||
|
||||
public static TypedRemapper create(ClassPath classPath) { |
||||
var inheritedFieldSets = classPath.createInheritedFieldSets(); |
||||
var inheritedMethodSets = classPath.createInheritedMethodSets(); |
||||
|
||||
var classes = createClassMapping(classPath.getLibraryClasses()); |
||||
var fields = createFieldMapping(classPath, inheritedFieldSets, classes); |
||||
var methods = createMethodMapping(classPath, inheritedMethodSets); |
||||
|
||||
verifyMapping(classes); |
||||
verifyMemberMapping(fields); |
||||
verifyMemberMapping(methods); |
||||
|
||||
return new TypedRemapper(classes, fields, methods); |
||||
} |
||||
|
||||
private static void verifyMapping(Map<String, String> mapping) { |
||||
for (var entry : mapping.entrySet()) { |
||||
verifyMapping(entry.getKey(), entry.getValue()); |
||||
} |
||||
} |
||||
|
||||
private static void verifyMemberMapping(Map<MemberRef, String> mapping) { |
||||
for (var entry : mapping.entrySet()) { |
||||
verifyMapping(entry.getKey().getName(), entry.getValue()); |
||||
} |
||||
} |
||||
|
||||
private static void verifyMapping(String name, String mappedName) { |
||||
name = name.replaceAll("^(?:loader|unpacker)_", ""); |
||||
if (name.length() > MAX_OBFUSCATED_NAME_LEN && !name.equals(mappedName)) { |
||||
logger.warn("Remapping probably unobfuscated name {} to {}", name, mappedName); |
||||
} |
||||
} |
||||
|
||||
private static String generateName(Map<String, Integer> prefixes, String prefix) { |
||||
return prefix + prefixes.merge(prefix, 1, Integer::sum); |
||||
} |
||||
|
||||
private static Map<String, String> createClassMapping(List<ClassMetadata> classes) { |
||||
var mapping = new HashMap<String, String>(); |
||||
var prefixes = new HashMap<String, Integer>(); |
||||
|
||||
for (var clazz : classes) { |
||||
populateClassMapping(mapping, prefixes, clazz); |
||||
} |
||||
|
||||
return mapping; |
||||
} |
||||
|
||||
private static String populateClassMapping(Map<String, String> mapping, Map<String, Integer> prefixes, ClassMetadata clazz) { |
||||
var name = clazz.getName(); |
||||
if (mapping.containsKey(name) || EXCLUDED_CLASSES.contains(name) || clazz.getDependency()) { |
||||
return mapping.getOrDefault(name, name); |
||||
} |
||||
|
||||
var mappedName = name.substring(0, name.lastIndexOf('/') + 1); |
||||
|
||||
var superClass = clazz.getSuperClass(); |
||||
if (superClass != null && !superClass.getName().equals("java/lang/Object")) { |
||||
var superName = populateClassMapping(mapping, prefixes, superClass); |
||||
superName = superName.substring(superName.lastIndexOf('/') + 1); |
||||
|
||||
mappedName += generateName(prefixes, superName + "_Sub"); |
||||
} else if (clazz.getInterface()) { |
||||
mappedName += generateName(prefixes, "Interface"); |
||||
} else { |
||||
mappedName += generateName(prefixes, "Class"); |
||||
} |
||||
|
||||
mapping.put(name, mappedName); |
||||
return mappedName; |
||||
} |
||||
|
||||
private static Map<MemberRef, String> createFieldMapping(ClassPath classPath, DisjointSet<MemberRef> disjointSet, Map<String, String> classMapping) { |
||||
var mapping = new HashMap<MemberRef, String>(); |
||||
var prefixes = new HashMap<String, Integer>(); |
||||
|
||||
for (var partition : disjointSet) { |
||||
boolean skip = false; |
||||
|
||||
for (var field : partition) { |
||||
var clazz = classPath.get(field.getOwner()); |
||||
|
||||
if (EXCLUDED_FIELDS.contains(field.getName())) { |
||||
skip = true; |
||||
break; |
||||
} |
||||
|
||||
if (clazz.getDependency()) { |
||||
skip = true; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
if (skip) { |
||||
continue; |
||||
} |
||||
|
||||
var prefix = ""; |
||||
|
||||
var type = Type.getType(partition.iterator().next().getDesc()); |
||||
if (type.getSort() == Type.ARRAY) { |
||||
prefix = Strings.repeat("Array", type.getDimensions()); |
||||
type = type.getElementType(); |
||||
} |
||||
|
||||
switch (type.getSort()) { |
||||
case Type.BOOLEAN: |
||||
case Type.BYTE: |
||||
case Type.CHAR: |
||||
case Type.SHORT: |
||||
case Type.INT: |
||||
case Type.LONG: |
||||
case Type.FLOAT: |
||||
case Type.DOUBLE: |
||||
prefix = type.getClassName() + prefix; |
||||
break; |
||||
case Type.OBJECT: |
||||
var className = classMapping.getOrDefault(type.getInternalName(), type.getInternalName()); |
||||
className = className.substring(className.lastIndexOf('/') + 1); |
||||
prefix = className + prefix; |
||||
break; |
||||
default: |
||||
throw new IllegalArgumentException("Unknown field type " + type); |
||||
} |
||||
|
||||
prefix = StringUtilsKt.indefiniteArticle(prefix) + StringsKt.capitalize(prefix); |
||||
|
||||
var mappedName = generateName(prefixes, prefix); |
||||
for (var field : partition) { |
||||
mapping.put(field, mappedName); |
||||
} |
||||
} |
||||
|
||||
return mapping; |
||||
} |
||||
|
||||
public static boolean isMethodImmutable(ClassPath classPath, DisjointSet.Partition<MemberRef> partition) { |
||||
for (var method : partition) { |
||||
var clazz = classPath.get(method.getOwner()); |
||||
|
||||
if (EXCLUDED_METHODS.contains(method.getName())) { |
||||
return true; |
||||
} |
||||
|
||||
if (clazz.getDependency()) { |
||||
return true; |
||||
} |
||||
|
||||
if (clazz.isNative(new MemberDesc(method))) { |
||||
return true; |
||||
} |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
private static Map<MemberRef, String> createMethodMapping(ClassPath classPath, DisjointSet<MemberRef> disjointSet) { |
||||
var mapping = new HashMap<MemberRef, String>(); |
||||
var id = 0; |
||||
|
||||
for (var partition : disjointSet) { |
||||
if (isMethodImmutable(classPath, partition)) { |
||||
continue; |
||||
} |
||||
|
||||
var mappedName = "method" + (++id); |
||||
for (var method : partition) { |
||||
mapping.put(method, mappedName); |
||||
} |
||||
} |
||||
|
||||
return mapping; |
||||
} |
||||
|
||||
private final Map<String, String> classes; |
||||
private final Map<MemberRef, String> fields, methods; |
||||
|
||||
private TypedRemapper(Map<String, String> classes, Map<MemberRef, String> fields, Map<MemberRef, String> methods) { |
||||
this.classes = classes; |
||||
this.fields = fields; |
||||
this.methods = methods; |
||||
} |
||||
|
||||
@Override |
||||
public String map(String internalName) { |
||||
return classes.getOrDefault(internalName, internalName); |
||||
} |
||||
|
||||
@Override |
||||
public String mapFieldName(String owner, String name, String descriptor) { |
||||
return fields.getOrDefault(new MemberRef(owner, name, descriptor), name); |
||||
} |
||||
|
||||
@Override |
||||
public String mapMethodName(String owner, String name, String descriptor) { |
||||
return methods.getOrDefault(new MemberRef(owner, name, descriptor), name); |
||||
} |
||||
} |
@ -0,0 +1,229 @@ |
||||
package dev.openrs2.deob.remap |
||||
|
||||
import com.github.michaelbull.logging.InlineLogger |
||||
import com.google.common.base.Strings |
||||
import dev.openrs2.asm.MemberDesc |
||||
import dev.openrs2.asm.MemberRef |
||||
import dev.openrs2.asm.classpath.ClassMetadata |
||||
import dev.openrs2.asm.classpath.ClassPath |
||||
import dev.openrs2.common.collect.DisjointSet |
||||
import dev.openrs2.common.indefiniteArticle |
||||
import org.objectweb.asm.Type |
||||
import org.objectweb.asm.commons.Remapper |
||||
|
||||
class TypedRemapper private constructor( |
||||
private val classes: Map<String, String>, |
||||
private val fields: Map<MemberRef, String>, |
||||
private val methods: Map<MemberRef, String> |
||||
) : Remapper() { |
||||
override fun map(internalName: String): String { |
||||
return classes.getOrDefault(internalName, internalName) |
||||
} |
||||
|
||||
override fun mapFieldName(owner: String, name: String, descriptor: String): String { |
||||
return fields.getOrDefault(MemberRef(owner, name, descriptor), name) |
||||
} |
||||
|
||||
override fun mapMethodName(owner: String, name: String, descriptor: String): String { |
||||
return methods.getOrDefault(MemberRef(owner, name, descriptor), name) |
||||
} |
||||
|
||||
companion object { |
||||
private val logger = InlineLogger() |
||||
|
||||
val EXCLUDED_CLASSES = setOf( |
||||
"client", |
||||
"jagex3/jagmisc/jagmisc", |
||||
"loader", |
||||
"unpack", |
||||
"unpackclass" |
||||
) |
||||
private val EXCLUDED_METHODS = setOf( |
||||
"<clinit>", |
||||
"<init>", |
||||
"main", |
||||
"providesignlink", |
||||
"quit" |
||||
) |
||||
private val EXCLUDED_FIELDS = setOf( |
||||
"cache" |
||||
) |
||||
|
||||
private const val MAX_OBFUSCATED_NAME_LEN = 2 |
||||
|
||||
fun create(classPath: ClassPath): TypedRemapper { |
||||
val inheritedFieldSets = classPath.createInheritedFieldSets() |
||||
val inheritedMethodSets = classPath.createInheritedMethodSets() |
||||
|
||||
val classes = createClassMapping(classPath.libraryClasses) |
||||
val fields = createFieldMapping(classPath, inheritedFieldSets, classes) |
||||
val methods = createMethodMapping(classPath, inheritedMethodSets) |
||||
|
||||
verifyMapping(classes) |
||||
verifyMemberMapping(fields) |
||||
verifyMemberMapping(methods) |
||||
|
||||
return TypedRemapper(classes, fields, methods) |
||||
} |
||||
|
||||
private fun verifyMapping(mapping: Map<String, String>) { |
||||
for ((key, value) in mapping) { |
||||
verifyMapping(key, value) |
||||
} |
||||
} |
||||
|
||||
private fun verifyMemberMapping(mapping: Map<MemberRef, String>) { |
||||
for ((key, value) in mapping) { |
||||
verifyMapping(key.name, value) |
||||
} |
||||
} |
||||
|
||||
private fun verifyMapping(name: String, mappedName: String) { |
||||
val originalName = name.replace("^(?:loader|unpacker)_".toRegex(), "") |
||||
if (originalName.length > MAX_OBFUSCATED_NAME_LEN && originalName != mappedName) { |
||||
logger.warn { "Remapping probably unobfuscated name $originalName to $mappedName" } |
||||
} |
||||
} |
||||
|
||||
private fun generateName(prefixes: MutableMap<String, Int>, prefix: String): String { |
||||
return prefix + prefixes.merge(prefix, 1, Integer::sum) |
||||
} |
||||
|
||||
private fun createClassMapping(classes: List<ClassMetadata>): Map<String, String> { |
||||
val mapping = mutableMapOf<String, String>() |
||||
val prefixes = mutableMapOf<String, Int>() |
||||
for (clazz in classes) { |
||||
populateClassMapping(mapping, prefixes, clazz) |
||||
} |
||||
return mapping |
||||
} |
||||
|
||||
private fun populateClassMapping( |
||||
mapping: MutableMap<String, String>, |
||||
prefixes: MutableMap<String, Int>, |
||||
clazz: ClassMetadata |
||||
): String { |
||||
val name = clazz.name |
||||
if (mapping.containsKey(name) || EXCLUDED_CLASSES.contains(name) || clazz.dependency) { |
||||
return mapping.getOrDefault(name, name) |
||||
} |
||||
|
||||
var mappedName = name.substring(0, name.lastIndexOf('/') + 1) |
||||
|
||||
val superClass = clazz.superClass |
||||
if (superClass != null && superClass.name != "java/lang/Object") { |
||||
var superName = populateClassMapping(mapping, prefixes, superClass) |
||||
superName = superName.substring(superName.lastIndexOf('/') + 1) |
||||
mappedName += generateName(prefixes, superName + "_Sub") |
||||
} else if (clazz.`interface`) { |
||||
mappedName += generateName(prefixes, "Interface") |
||||
} else { |
||||
mappedName += generateName(prefixes, "Class") |
||||
} |
||||
|
||||
mapping[name] = mappedName |
||||
return mappedName |
||||
} |
||||
|
||||
private fun createFieldMapping( |
||||
classPath: ClassPath, |
||||
disjointSet: DisjointSet<MemberRef>, |
||||
classMapping: Map<String, String> |
||||
): Map<MemberRef, String> { |
||||
val mapping = mutableMapOf<MemberRef, String>() |
||||
val prefixes = mutableMapOf<String, Int>() |
||||
|
||||
for (partition in disjointSet) { |
||||
var skip = false |
||||
|
||||
for ((owner, name) in partition) { |
||||
val clazz = classPath[owner] |
||||
|
||||
if (EXCLUDED_FIELDS.contains(name)) { |
||||
skip = true |
||||
break |
||||
} |
||||
|
||||
if (clazz.dependency) { |
||||
skip = true |
||||
break |
||||
} |
||||
} |
||||
|
||||
if (skip) { |
||||
continue |
||||
} |
||||
|
||||
var prefix = "" |
||||
|
||||
var type = Type.getType(partition.iterator().next().desc) |
||||
if (type.sort == Type.ARRAY) { |
||||
prefix = Strings.repeat("Array", type.dimensions) |
||||
type = type.elementType |
||||
} |
||||
|
||||
when (type.sort) { |
||||
Type.BOOLEAN, Type.BYTE, Type.CHAR, Type.SHORT, Type.INT, Type.LONG, Type.FLOAT, Type.DOUBLE -> { |
||||
prefix = type.className + prefix |
||||
} |
||||
Type.OBJECT -> { |
||||
var className = classMapping.getOrDefault(type.internalName, type.internalName) |
||||
className = className.substring(className.lastIndexOf('/') + 1) |
||||
prefix = className + prefix |
||||
} |
||||
else -> throw IllegalArgumentException("Unknown field type $type") |
||||
} |
||||
|
||||
prefix = prefix.indefiniteArticle() + prefix.capitalize() |
||||
|
||||
val mappedName = generateName(prefixes, prefix) |
||||
for (field in partition) { |
||||
mapping[field] = mappedName |
||||
} |
||||
} |
||||
|
||||
return mapping |
||||
} |
||||
|
||||
fun isMethodImmutable(classPath: ClassPath, partition: DisjointSet.Partition<MemberRef>): Boolean { |
||||
for (method in partition) { |
||||
val clazz = classPath[method.owner] |
||||
|
||||
if (EXCLUDED_METHODS.contains(method.name)) { |
||||
return true |
||||
} |
||||
|
||||
if (clazz.dependency) { |
||||
return true |
||||
} |
||||
|
||||
if (clazz.isNative(MemberDesc(method))) { |
||||
return true |
||||
} |
||||
} |
||||
|
||||
return false |
||||
} |
||||
|
||||
private fun createMethodMapping( |
||||
classPath: ClassPath, |
||||
disjointSet: DisjointSet<MemberRef> |
||||
): Map<MemberRef, String> { |
||||
val mapping = mutableMapOf<MemberRef, String>() |
||||
var id = 0 |
||||
|
||||
for (partition in disjointSet) { |
||||
if (isMethodImmutable(classPath, partition)) { |
||||
continue |
||||
} |
||||
|
||||
val mappedName = "method" + ++id |
||||
for (method in partition) { |
||||
mapping[method] = mappedName |
||||
} |
||||
} |
||||
|
||||
return mapping |
||||
} |
||||
} |
||||
} |
Loading…
Reference in new issue