parent
c10134c1f6
commit
0823a0253f
@ -1,200 +0,0 @@ |
|||||||
package dev.openrs2.deob.transform; |
|
||||||
|
|
||||||
import java.util.ArrayList; |
|
||||||
import java.util.HashMap; |
|
||||||
import java.util.HashSet; |
|
||||||
import java.util.Map; |
|
||||||
import java.util.Set; |
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableSet; |
|
||||||
import dev.openrs2.asm.MemberRef; |
|
||||||
import dev.openrs2.asm.MethodNodeUtilsKt; |
|
||||||
import dev.openrs2.asm.classpath.ClassPath; |
|
||||||
import dev.openrs2.asm.classpath.Library; |
|
||||||
import dev.openrs2.asm.transform.Transformer; |
|
||||||
import dev.openrs2.common.collect.DisjointSet; |
|
||||||
import dev.openrs2.deob.ArgRef; |
|
||||||
import dev.openrs2.deob.analysis.ConstSourceInterpreter; |
|
||||||
import dev.openrs2.deob.remap.TypedRemapper; |
|
||||||
import org.objectweb.asm.Opcodes; |
|
||||||
import org.objectweb.asm.Type; |
|
||||||
import org.objectweb.asm.tree.AbstractInsnNode; |
|
||||||
import org.objectweb.asm.tree.ClassNode; |
|
||||||
import org.objectweb.asm.tree.MethodInsnNode; |
|
||||||
import org.objectweb.asm.tree.MethodNode; |
|
||||||
import org.objectweb.asm.tree.VarInsnNode; |
|
||||||
import org.objectweb.asm.tree.analysis.Analyzer; |
|
||||||
import org.objectweb.asm.tree.analysis.AnalyzerException; |
|
||||||
import org.slf4j.Logger; |
|
||||||
import org.slf4j.LoggerFactory; |
|
||||||
|
|
||||||
public final class UnusedArgTransformer extends Transformer { |
|
||||||
private static final Logger logger = LoggerFactory.getLogger(UnusedArgTransformer.class); |
|
||||||
|
|
||||||
private static final ImmutableSet<Integer> INT_SORTS = ImmutableSet.of(Type.BOOLEAN, Type.BYTE, Type.SHORT, Type.INT, Type.CHAR); |
|
||||||
|
|
||||||
private static Map<Integer, Integer> createLocalToArgMap(MethodNode method) { |
|
||||||
var type = Type.getType(method.desc); |
|
||||||
var argumentTypes = type.getArgumentTypes(); |
|
||||||
|
|
||||||
var map = new HashMap<Integer, Integer>(); |
|
||||||
var argIndex = 0; |
|
||||||
var localIndex = 0; |
|
||||||
|
|
||||||
if ((method.access & Opcodes.ACC_STATIC) == 0) { |
|
||||||
localIndex++; |
|
||||||
} |
|
||||||
|
|
||||||
for (var t : argumentTypes) { |
|
||||||
map.put(localIndex, argIndex++); |
|
||||||
localIndex += t.getSize(); |
|
||||||
} |
|
||||||
|
|
||||||
return map; |
|
||||||
} |
|
||||||
|
|
||||||
private final Set<ArgRef> retainedArgs = new HashSet<>(); |
|
||||||
private DisjointSet<MemberRef> inheritedMethodSets; |
|
||||||
private int deletedArgs; |
|
||||||
|
|
||||||
@Override |
|
||||||
protected void preTransform(ClassPath classPath) throws AnalyzerException { |
|
||||||
retainedArgs.clear(); |
|
||||||
inheritedMethodSets = classPath.createInheritedMethodSets(); |
|
||||||
deletedArgs = 0; |
|
||||||
|
|
||||||
for (var library : classPath.getLibraries()) { |
|
||||||
for (var clazz : library) { |
|
||||||
for (var method : clazz.methods) { |
|
||||||
if (MethodNodeUtilsKt.hasCode(method)) { |
|
||||||
populateRetainedArgs(classPath, clazz, method); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private void populateRetainedArgs(ClassPath classPath, ClassNode clazz, MethodNode method) throws AnalyzerException { |
|
||||||
var partition = inheritedMethodSets.get(new MemberRef(clazz, method)); |
|
||||||
var localToArgMap = createLocalToArgMap(method); |
|
||||||
|
|
||||||
var analyzer = new Analyzer<>(new ConstSourceInterpreter()); |
|
||||||
var frames = analyzer.analyze(clazz.name, method); |
|
||||||
|
|
||||||
for (var i = 0; i < frames.length; i++) { |
|
||||||
var frame = frames[i]; |
|
||||||
if (frame == null) { |
|
||||||
continue; |
|
||||||
} |
|
||||||
|
|
||||||
var stackSize = frame.getStackSize(); |
|
||||||
|
|
||||||
var insn = method.instructions.get(i); |
|
||||||
switch (insn.getOpcode()) { |
|
||||||
case Opcodes.ILOAD: |
|
||||||
var iload = (VarInsnNode) insn; |
|
||||||
var arg = localToArgMap.get(iload.var); |
|
||||||
if (arg != null) { |
|
||||||
retainedArgs.add(new ArgRef(partition, arg)); |
|
||||||
} |
|
||||||
break; |
|
||||||
case Opcodes.INVOKEVIRTUAL: |
|
||||||
case Opcodes.INVOKESPECIAL: |
|
||||||
case Opcodes.INVOKESTATIC: |
|
||||||
case Opcodes.INVOKEINTERFACE: |
|
||||||
var invoke = (MethodInsnNode) insn; |
|
||||||
var invokePartition = inheritedMethodSets.get(new MemberRef(invoke)); |
|
||||||
if (invokePartition == null || TypedRemapper.isMethodImmutable(classPath, invokePartition)) { |
|
||||||
continue; |
|
||||||
} |
|
||||||
|
|
||||||
var args = Type.getArgumentTypes(invoke.desc).length; |
|
||||||
for (var j = 0; j < args; j++) { |
|
||||||
var source = frame.getStack(stackSize - args + j); |
|
||||||
if (!source.isSingleSourceConstant()) { |
|
||||||
retainedArgs.add(new ArgRef(invokePartition, j)); |
|
||||||
} |
|
||||||
} |
|
||||||
break; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
protected boolean preTransformMethod(ClassPath classPath, Library library, ClassNode clazz, MethodNode method) throws AnalyzerException { |
|
||||||
/* delete unused int args from call sites */ |
|
||||||
if (MethodNodeUtilsKt.hasCode(method)) { |
|
||||||
var analyzer = new Analyzer<>(new ConstSourceInterpreter()); |
|
||||||
var frames = analyzer.analyze(clazz.name, method); |
|
||||||
var deadInsns = new ArrayList<AbstractInsnNode>(); |
|
||||||
|
|
||||||
for (var i = 0; i < frames.length; i++) { |
|
||||||
var frame = frames[i]; |
|
||||||
if (frame == null) { |
|
||||||
continue; |
|
||||||
} |
|
||||||
|
|
||||||
var stackSize = frame.getStackSize(); |
|
||||||
|
|
||||||
var insn = method.instructions.get(i); |
|
||||||
if (insn.getType() != AbstractInsnNode.METHOD_INSN) { |
|
||||||
continue; |
|
||||||
} |
|
||||||
|
|
||||||
var methodInsn = (MethodInsnNode) insn; |
|
||||||
var partition = inheritedMethodSets.get(new MemberRef(methodInsn)); |
|
||||||
if (partition == null || TypedRemapper.isMethodImmutable(classPath, partition)) { |
|
||||||
continue; |
|
||||||
} |
|
||||||
|
|
||||||
var type = Type.getType(methodInsn.desc); |
|
||||||
var argTypes = type.getArgumentTypes(); |
|
||||||
|
|
||||||
var newArgTypes = new ArrayList<Type>(); |
|
||||||
|
|
||||||
for (var j = 0; j < argTypes.length; j++) { |
|
||||||
var argType = argTypes[j]; |
|
||||||
if (INT_SORTS.contains(argType.getSort()) && !retainedArgs.contains(new ArgRef(partition, j))) { |
|
||||||
var source = frame.getStack(stackSize - argTypes.length + j).getSource(); |
|
||||||
deadInsns.add(source); |
|
||||||
} else { |
|
||||||
newArgTypes.add(argType); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
methodInsn.desc = Type.getMethodDescriptor(type.getReturnType(), newArgTypes.toArray(Type[]::new)); |
|
||||||
} |
|
||||||
|
|
||||||
deadInsns.forEach(method.instructions::remove); |
|
||||||
} |
|
||||||
|
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
protected boolean postTransformMethod(ClassPath classPath, Library library, ClassNode clazz, MethodNode method) { |
|
||||||
/* delete unused int args from the method itself */ |
|
||||||
var partition = inheritedMethodSets.get(new MemberRef(clazz, method)); |
|
||||||
if (TypedRemapper.isMethodImmutable(classPath, partition)) { |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
var type = Type.getType(method.desc); |
|
||||||
var argTypes = type.getArgumentTypes(); |
|
||||||
|
|
||||||
for (var i = argTypes.length - 1; i >= 0; i--) { |
|
||||||
var argType = argTypes[i]; |
|
||||||
if (INT_SORTS.contains(argType.getSort()) && !retainedArgs.contains(new ArgRef(partition, i))) { |
|
||||||
MethodNodeUtilsKt.removeArgument(method, i); |
|
||||||
deletedArgs++; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
protected void postTransform(ClassPath classPath) { |
|
||||||
logger.info("Removed {} dummy arguments", deletedArgs); |
|
||||||
} |
|
||||||
} |
|
@ -0,0 +1,196 @@ |
|||||||
|
package dev.openrs2.deob.transform |
||||||
|
|
||||||
|
import com.github.michaelbull.logging.InlineLogger |
||||||
|
import dev.openrs2.asm.MemberRef |
||||||
|
import dev.openrs2.asm.classpath.ClassPath |
||||||
|
import dev.openrs2.asm.classpath.Library |
||||||
|
import dev.openrs2.asm.hasCode |
||||||
|
import dev.openrs2.asm.removeArgument |
||||||
|
import dev.openrs2.asm.transform.Transformer |
||||||
|
import dev.openrs2.common.collect.DisjointSet |
||||||
|
import dev.openrs2.deob.ArgRef |
||||||
|
import dev.openrs2.deob.analysis.ConstSourceInterpreter |
||||||
|
import dev.openrs2.deob.remap.TypedRemapper |
||||||
|
import org.objectweb.asm.Opcodes |
||||||
|
import org.objectweb.asm.Type |
||||||
|
import org.objectweb.asm.tree.* |
||||||
|
import org.objectweb.asm.tree.analysis.Analyzer |
||||||
|
import org.objectweb.asm.tree.analysis.AnalyzerException |
||||||
|
|
||||||
|
class UnusedArgTransformer : Transformer() { |
||||||
|
private val retainedArgs = mutableSetOf<ArgRef>() |
||||||
|
private lateinit var inheritedMethodSets: DisjointSet<MemberRef> |
||||||
|
private var deletedArgs = 0 |
||||||
|
|
||||||
|
@Throws(AnalyzerException::class) |
||||||
|
override fun preTransform(classPath: ClassPath) { |
||||||
|
retainedArgs.clear() |
||||||
|
inheritedMethodSets = classPath.createInheritedMethodSets() |
||||||
|
deletedArgs = 0 |
||||||
|
|
||||||
|
for (library in classPath.libraries) { |
||||||
|
for (clazz in library) { |
||||||
|
for (method in clazz.methods) { |
||||||
|
if (method.hasCode()) { |
||||||
|
populateRetainedArgs(classPath, clazz, method) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Throws(AnalyzerException::class) |
||||||
|
private fun populateRetainedArgs(classPath: ClassPath, clazz: ClassNode, method: MethodNode) { |
||||||
|
val partition = inheritedMethodSets[MemberRef(clazz, method)]!! |
||||||
|
val localToArgMap = createLocalToArgMap(method) |
||||||
|
|
||||||
|
val analyzer = Analyzer(ConstSourceInterpreter()) |
||||||
|
val frames = analyzer.analyze(clazz.name, method) |
||||||
|
|
||||||
|
frame@ for ((i, frame) in frames.withIndex()) { |
||||||
|
if (frame == null) { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
val stackSize = frame.stackSize |
||||||
|
|
||||||
|
val insn = method.instructions[i] |
||||||
|
when (insn) { |
||||||
|
is VarInsnNode -> { |
||||||
|
if (insn.opcode != Opcodes.ILOAD) { |
||||||
|
continue@frame |
||||||
|
} |
||||||
|
|
||||||
|
val arg = localToArgMap[insn.`var`] |
||||||
|
if (arg != null) { |
||||||
|
retainedArgs.add(ArgRef(partition, arg)) |
||||||
|
} |
||||||
|
} |
||||||
|
is MethodInsnNode -> { |
||||||
|
val invokePartition = inheritedMethodSets[MemberRef(insn)] |
||||||
|
if (invokePartition == null || TypedRemapper.isMethodImmutable(classPath, invokePartition)) { |
||||||
|
continue@frame |
||||||
|
} |
||||||
|
|
||||||
|
val args = Type.getArgumentTypes(insn.desc).size |
||||||
|
for (j in 0 until args) { |
||||||
|
val source = frame.getStack(stackSize - args + j) |
||||||
|
if (!source.isSingleSourceConstant) { |
||||||
|
retainedArgs.add(ArgRef(invokePartition, j)) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Throws(AnalyzerException::class) |
||||||
|
override fun preTransformMethod( |
||||||
|
classPath: ClassPath, |
||||||
|
library: Library, |
||||||
|
clazz: ClassNode, |
||||||
|
method: MethodNode |
||||||
|
): Boolean { |
||||||
|
// delete unused int args from call sites |
||||||
|
if (!method.hasCode()) { |
||||||
|
return false |
||||||
|
} |
||||||
|
|
||||||
|
val analyzer = Analyzer(ConstSourceInterpreter()) |
||||||
|
val frames = analyzer.analyze(clazz.name, method) |
||||||
|
val deadInsns = mutableListOf<AbstractInsnNode>() |
||||||
|
|
||||||
|
for ((i, frame) in frames.withIndex()) { |
||||||
|
if (frame == null) { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
val stackSize = frame.stackSize |
||||||
|
|
||||||
|
val insn = method.instructions[i] |
||||||
|
if (insn !is MethodInsnNode) { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
val partition = inheritedMethodSets[MemberRef(insn)] |
||||||
|
if (partition == null || TypedRemapper.isMethodImmutable(classPath, partition)) { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
val type = Type.getType(insn.desc) |
||||||
|
val argTypes = type.argumentTypes |
||||||
|
val newArgTypes = mutableListOf<Type>() |
||||||
|
|
||||||
|
for ((j, argType) in argTypes.withIndex()) { |
||||||
|
if (INT_SORTS.contains(argType.sort) && !retainedArgs.contains(ArgRef(partition, j))) { |
||||||
|
val source = frame.getStack(stackSize - argTypes.size + j).source |
||||||
|
deadInsns.add(source) |
||||||
|
} else { |
||||||
|
newArgTypes.add(argType) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
insn.desc = Type.getMethodDescriptor(type.returnType, *newArgTypes.toTypedArray()) |
||||||
|
} |
||||||
|
|
||||||
|
deadInsns.forEach(method.instructions::remove) |
||||||
|
|
||||||
|
return false |
||||||
|
} |
||||||
|
|
||||||
|
override fun postTransformMethod( |
||||||
|
classPath: ClassPath, |
||||||
|
library: Library, |
||||||
|
clazz: ClassNode, |
||||||
|
method: MethodNode |
||||||
|
): Boolean { |
||||||
|
// delete unused int args from the method itself |
||||||
|
val partition = inheritedMethodSets[MemberRef(clazz, method)]!! |
||||||
|
if (TypedRemapper.isMethodImmutable(classPath, partition)) { |
||||||
|
return false |
||||||
|
} |
||||||
|
|
||||||
|
val argTypes = Type.getType(method.desc).argumentTypes |
||||||
|
for ((i, argType) in argTypes.withIndex().reversed()) { |
||||||
|
if (INT_SORTS.contains(argType.sort) && !retainedArgs.contains(ArgRef(partition, i))) { |
||||||
|
method.removeArgument(i) |
||||||
|
deletedArgs++ |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return false |
||||||
|
} |
||||||
|
|
||||||
|
override fun postTransform(classPath: ClassPath) { |
||||||
|
logger.info { "Removed $deletedArgs dummy arguments" } |
||||||
|
} |
||||||
|
|
||||||
|
companion object { |
||||||
|
private val logger = InlineLogger() |
||||||
|
private val INT_SORTS = setOf( |
||||||
|
Type.BOOLEAN, |
||||||
|
Type.BYTE, |
||||||
|
Type.SHORT, |
||||||
|
Type.INT, |
||||||
|
Type.CHAR |
||||||
|
) |
||||||
|
|
||||||
|
private fun createLocalToArgMap(method: MethodNode): Map<Int, Int> { |
||||||
|
val argTypes = Type.getType(method.desc).argumentTypes |
||||||
|
|
||||||
|
val map = mutableMapOf<Int, Int>() |
||||||
|
var localIndex = 0 |
||||||
|
|
||||||
|
if (method.access and Opcodes.ACC_STATIC == 0) { |
||||||
|
localIndex++ |
||||||
|
} |
||||||
|
|
||||||
|
for ((argIndex, t) in argTypes.withIndex()) { |
||||||
|
map[localIndex] = argIndex |
||||||
|
localIndex += t.size |
||||||
|
} |
||||||
|
|
||||||
|
return map |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue