Open-source multiplayer game server compatible with the RuneScape client
https://www.openrs2.org/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
77 lines
2.7 KiB
77 lines
2.7 KiB
package dev.openrs2.deob.transform
|
|
|
|
import com.github.michaelbull.logging.InlineLogger
|
|
import dev.openrs2.asm.InsnMatcher
|
|
import dev.openrs2.asm.classpath.ClassPath
|
|
import dev.openrs2.asm.classpath.Library
|
|
import dev.openrs2.asm.nextReal
|
|
import dev.openrs2.asm.transform.Transformer
|
|
import org.objectweb.asm.tree.ClassNode
|
|
import org.objectweb.asm.tree.MethodNode
|
|
import javax.inject.Singleton
|
|
|
|
/**
|
|
* A [Transformer] responsible for removing two kinds of redundant exception
|
|
* handler.
|
|
*
|
|
* - [ZKM](http://www.zelix.com/klassmaster/)'s
|
|
* [exception obfuscation](https://www.zelix.com/klassmaster/featuresExceptionObfuscation.html),
|
|
* which inserts exception handlers that catch [RuntimeException]s and
|
|
* immediately re-throw them. The exception handlers are inserted in
|
|
* locations where there is no Java source code equivalent, confusing
|
|
* decompilers.
|
|
*
|
|
* - Jagex inserts a try/catch block around every method that catches
|
|
* [RuntimeException]s, wraps them with a custom [RuntimeException]
|
|
* implementation and re-throws them. The wrapped exception's message
|
|
* contains the values of the method's arguments. While this is for debugging
|
|
* and not obfuscation, it is clearly automatically-generated and thus we
|
|
* remove these exception handlers too.
|
|
*/
|
|
@Singleton
|
|
class ExceptionTracingTransformer : Transformer() {
|
|
private var tracingTryCatches = 0
|
|
|
|
override fun preTransform(classPath: ClassPath) {
|
|
tracingTryCatches = 0
|
|
}
|
|
|
|
override fun transformCode(
|
|
classPath: ClassPath,
|
|
library: Library,
|
|
clazz: ClassNode,
|
|
method: MethodNode
|
|
): Boolean {
|
|
for (match in CATCH_MATCHER.match(method)) {
|
|
val foundTryCatch = method.tryCatchBlocks.removeIf { tryCatch ->
|
|
tryCatch.type == "java/lang/RuntimeException" && tryCatch.handler.nextReal === match[0]
|
|
}
|
|
|
|
if (foundTryCatch) {
|
|
match.forEach(method.instructions::remove)
|
|
tracingTryCatches++
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
override fun postTransform(classPath: ClassPath) {
|
|
logger.info { "Removed $tracingTryCatches tracing try/catch blocks" }
|
|
}
|
|
|
|
private companion object {
|
|
private val logger = InlineLogger()
|
|
private val CATCH_MATCHER = InsnMatcher.compile(
|
|
"""
|
|
(ASTORE ALOAD)?
|
|
(LDC INVOKESTATIC |
|
|
NEW DUP
|
|
(LDC INVOKESPECIAL | INVOKESPECIAL LDC INVOKEVIRTUAL)
|
|
((ILOAD | LLOAD | FLOAD | DLOAD | (ALOAD IFNULL LDC GOTO LDC) | BIPUSH | SIPUSH | LDC) INVOKEVIRTUAL)*
|
|
INVOKEVIRTUAL INVOKESTATIC
|
|
)?
|
|
ATHROW
|
|
"""
|
|
)
|
|
}
|
|
}
|
|
|