Complete FrameState analysis implementation
This commit is contained in:
parent
a0e5a3ef29
commit
4ddbf6821b
@ -195,7 +195,10 @@ public class Combine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected void adaptFrameNode(FrameNode node, MethodNode method, GraftSource source) {
|
protected void adaptFrameNode(FrameNode node, MethodNode method, GraftSource source) {
|
||||||
|
for (int i = 0; i < node.stack.size(); ++i)
|
||||||
|
if (node.stack.get(i) instanceof Type) {
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -7,11 +7,13 @@ import jdk.internal.org.objectweb.asm.Label;
|
|||||||
import jdk.internal.org.objectweb.asm.Opcodes;
|
import jdk.internal.org.objectweb.asm.Opcodes;
|
||||||
import jdk.internal.org.objectweb.asm.Type;
|
import jdk.internal.org.objectweb.asm.Type;
|
||||||
import jdk.internal.org.objectweb.asm.tree.*;
|
import jdk.internal.org.objectweb.asm.tree.*;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
import sun.reflect.generics.reflectiveObjects.NotImplementedException;
|
import sun.reflect.generics.reflectiveObjects.NotImplementedException;
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
import java.util.Stack;
|
import java.util.Stack;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
@ -69,11 +71,8 @@ public class FrameState {
|
|||||||
"??????????????????????????????????????????????MMMMMMMMIJFDLIIIIJJJJFFFFDDDDLLLL12345111$K$$$XXXKCVBNCVBNCVBNCVBNCVBNCVBNCVCVCVCVCVCV?IIIJJJFFFDDDIIIVBBNNIIIIIICCCCCC00?????IJFDL??XLXXXXXX?IILLLLLLXXLL??";
|
"??????????????????????????????????????????????MMMMMMMMIJFDLIIIIJJJJFFFFDDDDLLLL12345111$K$$$XXXKCVBNCVBNCVBNCVBNCVBNCVBNCVCVCVCVCVCV?IIIJJJFFFDDDIIIVBBNNIIIIIICCCCCC00?????IJFDL??XLXXXXXX?IILLLLLLXXLL??";
|
||||||
|
|
||||||
|
|
||||||
private final Stack<TypeSignature> stack = new Stack<>();
|
public static Stack<TypeSignature> getFrameStateAt(AbstractInsnNode targetNode, List<LocalVariableNode> locals) {
|
||||||
private final ArrayList<TypeSignature> locals = new ArrayList<>();
|
Stack<TypeSignature> stack = new Stack<>();
|
||||||
private final int stackSize;
|
|
||||||
|
|
||||||
private FrameState(AbstractInsnNode targetNode, List<TypeSignature> constants) {
|
|
||||||
AbstractInsnNode first = targetNode, tmp;
|
AbstractInsnNode first = targetNode, tmp;
|
||||||
|
|
||||||
// Computation is already O(n), no need to accept first instruction as an argument
|
// Computation is already O(n), no need to accept first instruction as an argument
|
||||||
@ -115,14 +114,9 @@ public class FrameState {
|
|||||||
// We now have a proposed set of instructions that might run if the instructions were to be called
|
// We now have a proposed set of instructions that might run if the instructions were to be called
|
||||||
// Next, we analyse the stack and locals throughout the execution of these instructions
|
// Next, we analyse the stack and locals throughout the execution of these instructions
|
||||||
while (!simulate.isEmpty()) {
|
while (!simulate.isEmpty()) {
|
||||||
if (simulate.peek() instanceof FrameNode)
|
updateFrameState(simulate.pop(), stack, locals);
|
||||||
stackSize = ((FrameNode) simulate.peek()).stack.size();
|
|
||||||
|
|
||||||
updateFrameState(simulate.pop(), stack, locals, constants);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
this.stackSize = stackSize;
|
|
||||||
|
|
||||||
// The stack and locals are now in the state they would be in after the target instruction is hit
|
// The stack and locals are now in the state they would be in after the target instruction is hit
|
||||||
// QED or something...
|
// QED or something...
|
||||||
|
|
||||||
@ -158,6 +152,39 @@ public class FrameState {
|
|||||||
* Also note that this kind of behaviour cannot be predicted due to the halting problem, so any program which
|
* Also note that this kind of behaviour cannot be predicted due to the halting problem, so any program which
|
||||||
* exhibits the aforementioned behaviour is inherently unpredictable.
|
* exhibits the aforementioned behaviour is inherently unpredictable.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
return stack;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a list of all local variables currently in scope at a given instruction
|
||||||
|
* @param insn Instruction to get local variable scope for
|
||||||
|
* @param allLocals All local variables in method
|
||||||
|
* @return A subset of all local variables such that accessing any value in said subset would not be an error
|
||||||
|
*/
|
||||||
|
public static List<LocalVariableNode> localsAt(AbstractInsnNode insn, List<LocalVariableNode> allLocals) {
|
||||||
|
ArrayList<LocalVariableNode> collect = new ArrayList<>();
|
||||||
|
|
||||||
|
for (LocalVariableNode vNode : allLocals) {
|
||||||
|
// Find relative index of start of variable scope
|
||||||
|
Integer start = relativeIndexOf(insn, vNode.start);
|
||||||
|
|
||||||
|
// If start of scope could not be found, or it begins after the given instruction, it's not in scope
|
||||||
|
if (start == null || start > 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// Find relative index of end of variable scope
|
||||||
|
Integer end = relativeIndexOf(insn, vNode.end);
|
||||||
|
|
||||||
|
// If end of scope could not be found, or it ends before the given instruction, it's not in scope
|
||||||
|
if (end == null || end < 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// Scope starts at (or before) given instruction and ends at (or after) given instruction
|
||||||
|
collect.add(vNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
return collect;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -182,33 +209,33 @@ public class FrameState {
|
|||||||
* @param instruction Instruction to "simulate"
|
* @param instruction Instruction to "simulate"
|
||||||
* @param stack Frame stack values
|
* @param stack Frame stack values
|
||||||
* @param locals Frame local variables
|
* @param locals Frame local variables
|
||||||
* @param constants Method constant pool types
|
|
||||||
*/
|
*/
|
||||||
private static void updateFrameState(
|
private static void updateFrameState(
|
||||||
AbstractInsnNode instruction,
|
AbstractInsnNode instruction,
|
||||||
Stack<TypeSignature> stack,
|
Stack<TypeSignature> stack,
|
||||||
ArrayList<TypeSignature> locals,
|
List<LocalVariableNode> locals
|
||||||
List<TypeSignature> constants
|
|
||||||
) {
|
) {
|
||||||
if (instruction instanceof FrameNode) {
|
if (instruction instanceof FrameNode) {
|
||||||
// Stack values are always updated at a FrameNode
|
// Stack values are always updated at a FrameNode
|
||||||
stack.clear();
|
stack.clear();
|
||||||
|
|
||||||
switch (instruction.getType()) {
|
// This is VERY different from getType() declared in AbstractInsnNode
|
||||||
|
switch (((FrameNode) instruction).type) {
|
||||||
case Opcodes.F_NEW:
|
case Opcodes.F_NEW:
|
||||||
case Opcodes.F_FULL:
|
case Opcodes.F_FULL:
|
||||||
|
case Opcodes.F_SAME: // This feels like undocumented behaviour
|
||||||
// Since this is a full frame, we start anew
|
// Since this is a full frame, we start anew
|
||||||
locals.clear();
|
//locals.clear();
|
||||||
|
|
||||||
// Ascertain stack types
|
// Ascertain stack types
|
||||||
appendTypes(((FrameNode) instruction).stack, stack, true);
|
appendTypes(((FrameNode) instruction).stack, stack, true);
|
||||||
|
|
||||||
// Ascertain local types
|
// Ascertain local types
|
||||||
appendTypes(((FrameNode) instruction).local, locals, false);
|
//appendTypes(((FrameNode) instruction).local, locals, false);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Opcodes.F_APPEND:
|
case Opcodes.F_APPEND:
|
||||||
appendTypes(((FrameNode) instruction).local, locals, false);
|
//appendTypes(((FrameNode) instruction).local, locals, false);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Opcodes.F_SAME1:
|
case Opcodes.F_SAME1:
|
||||||
@ -216,13 +243,15 @@ public class FrameState {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case Opcodes.F_CHOP:
|
case Opcodes.F_CHOP:
|
||||||
|
/*
|
||||||
List<Object> local = ((FrameNode) instruction).local;
|
List<Object> local = ((FrameNode) instruction).local;
|
||||||
if (local != null)
|
if (local != null)
|
||||||
while (local.size() > locals.size())
|
while (local.size() > locals.size())
|
||||||
locals.remove(locals.size() - 1);
|
locals.remove(locals.size() - 1);
|
||||||
|
*/
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else clobberStack(instruction, stack, locals, constants);
|
} else clobberStack(instruction, stack, locals);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -234,9 +263,16 @@ public class FrameState {
|
|||||||
private static void appendTypes(List<Object> types, List<TypeSignature> appendTo, boolean skipNulls) {
|
private static void appendTypes(List<Object> types, List<TypeSignature> appendTo, boolean skipNulls) {
|
||||||
if (types == null) return;
|
if (types == null) return;
|
||||||
|
|
||||||
for (Object o : types)
|
for (Object o : types) {
|
||||||
if (o == null && skipNulls) break;
|
if (o == null && skipNulls) break;
|
||||||
else appendTo.add(o == null ? null : parseFrameSignature(o));
|
else if (o == null) appendTo.add(null);
|
||||||
|
else {
|
||||||
|
TypeSignature sig = parseFrameSignature(o);
|
||||||
|
appendTo.add(sig);
|
||||||
|
if (sig.stackFrameElementWith() == 2)
|
||||||
|
appendTo.add(new TypeSignature(sig.getSig().charAt(0), true));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -278,8 +314,7 @@ public class FrameState {
|
|||||||
private static void clobberStack(
|
private static void clobberStack(
|
||||||
AbstractInsnNode insn,
|
AbstractInsnNode insn,
|
||||||
List<TypeSignature> stack,
|
List<TypeSignature> stack,
|
||||||
List<TypeSignature> locals,
|
List<LocalVariableNode> locals
|
||||||
List<TypeSignature> constants
|
|
||||||
) {
|
) {
|
||||||
// Look, before you go ahead and roast my code, just know that I have a "code first, think later" mentality,
|
// Look, before you go ahead and roast my code, just know that I have a "code first, think later" mentality,
|
||||||
// so this entire method was essentially throw together and structured this way before I realised what I was
|
// so this entire method was essentially throw together and structured this way before I realised what I was
|
||||||
@ -298,6 +333,7 @@ public class FrameState {
|
|||||||
// Complex argument and result
|
// Complex argument and result
|
||||||
// This behaviour is exhibited by 11 instructions in the JVM 8 spec
|
// This behaviour is exhibited by 11 instructions in the JVM 8 spec
|
||||||
int argCount = 0;
|
int argCount = 0;
|
||||||
|
MethodSignature msig = null;
|
||||||
switch (opcode) {
|
switch (opcode) {
|
||||||
case Opcodes.DUP2:
|
case Opcodes.DUP2:
|
||||||
case Opcodes.DUP2_X1:
|
case Opcodes.DUP2_X1:
|
||||||
@ -307,12 +343,16 @@ public class FrameState {
|
|||||||
stack.add(stack.size() - (opcode - 90), stack.get(stack.size() - 2));
|
stack.add(stack.size() - (opcode - 90), stack.get(stack.size() - 2));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Opcodes.INVOKEDYNAMIC:
|
||||||
|
msig = new MethodSignature(((InvokeDynamicInsnNode) insn).desc);
|
||||||
|
argCount = -1;
|
||||||
case Opcodes.INVOKEVIRTUAL:
|
case Opcodes.INVOKEVIRTUAL:
|
||||||
case Opcodes.INVOKESPECIAL:
|
case Opcodes.INVOKESPECIAL:
|
||||||
case Opcodes.INVOKEINTERFACE:
|
case Opcodes.INVOKEINTERFACE:
|
||||||
argCount = 1;
|
++argCount;
|
||||||
case Opcodes.INVOKESTATIC: {
|
case Opcodes.INVOKESTATIC:
|
||||||
MethodSignature msig = new MethodSignature(((MethodInsnNode)insn).desc);
|
if (msig == null)
|
||||||
|
msig = new MethodSignature(((MethodInsnNode)insn).desc);
|
||||||
argCount += msig.getArgCount();
|
argCount += msig.getArgCount();
|
||||||
for (int i = 0; i < argCount; ++i) {
|
for (int i = 0; i < argCount; ++i) {
|
||||||
// Longs and doubles pop 2 values from the stack
|
// Longs and doubles pop 2 values from the stack
|
||||||
@ -328,17 +368,12 @@ public class FrameState {
|
|||||||
stack.add(msig.getRet());
|
stack.add(msig.getRet());
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
|
|
||||||
case Opcodes.INVOKEDYNAMIC:
|
|
||||||
// TODO: Implement: this requires dynamic call-site resolution and injection
|
|
||||||
//InvokeDynamicInsnNode dyn = (InvokeDynamicInsnNode) insn;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 196: // WIDE
|
case 196: // WIDE
|
||||||
// WIDE instruction not expected in normal Java programs
|
// WIDE instruction not expected in normal Java programs
|
||||||
// TODO: Implement?
|
// TODO: Implement?
|
||||||
throw new NotImplementedException();
|
//throw new NotImplementedException();
|
||||||
|
break; // Ignore instruction, since it wraps the immediately following instruction
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (pushType == 'X') {
|
} else if (pushType == 'X') {
|
||||||
@ -380,9 +415,18 @@ public class FrameState {
|
|||||||
// type, so we just get the internal name of that field reflectively because I'm lazy
|
// type, so we just get the internal name of that field reflectively because I'm lazy
|
||||||
// TODO: Un-reflect-ify this because it can literally be solved with if-elses instead
|
// TODO: Un-reflect-ify this because it can literally be solved with if-elses instead
|
||||||
try {
|
try {
|
||||||
stack.add(new TypeSignature(
|
Class<?> cType = ((Class<?>)ldc.cst.getClass().getField("TYPE").get(null));
|
||||||
((Class<?>)ldc.cst.getClass().getField("TYPE").get(null)).getName()
|
char cLetter =
|
||||||
));
|
long.class.equals(cType) ? 'j' :
|
||||||
|
boolean.class.equals(cType) ? 'z' :
|
||||||
|
cType.getName().charAt(0);
|
||||||
|
|
||||||
|
stack.add(new TypeSignature(cLetter, false));
|
||||||
|
|
||||||
|
// For W pushes, add top value
|
||||||
|
if (long.class.equals(cType) || double.class.equals(cType))
|
||||||
|
stack.add(new TypeSignature(cLetter, true));
|
||||||
|
|
||||||
} catch (NoSuchFieldException | IllegalAccessException e) {
|
} catch (NoSuchFieldException | IllegalAccessException e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
@ -433,8 +477,8 @@ public class FrameState {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Trivial-ish argument and result
|
// Trivial-ish argument and result
|
||||||
trivialPop(insn, popType, stack, locals, constants);
|
trivialPop(insn, popType, stack, locals);
|
||||||
trivialPush(insn, pushType, stack, locals, constants);
|
trivialPush(insn, pushType, stack, locals);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -446,9 +490,8 @@ public class FrameState {
|
|||||||
* @param type Classification of push type
|
* @param type Classification of push type
|
||||||
* @param stack Simulated operand stand types
|
* @param stack Simulated operand stand types
|
||||||
* @param locals Simulated frame local types
|
* @param locals Simulated frame local types
|
||||||
* @param constants Method constant pool
|
|
||||||
*/
|
*/
|
||||||
private static void trivialPop(AbstractInsnNode insn, char type, List<TypeSignature> stack, List<TypeSignature> locals, List<TypeSignature> constants) {
|
private static void trivialPop(AbstractInsnNode insn, char type, List<TypeSignature> stack, List<LocalVariableNode> locals) {
|
||||||
// TODO: Fix type naming scheme; this is actually going to make me cry
|
// TODO: Fix type naming scheme; this is actually going to make me cry
|
||||||
// Yes, the fall-throughs are very intentional
|
// Yes, the fall-throughs are very intentional
|
||||||
switch (type) {
|
switch (type) {
|
||||||
@ -493,9 +536,8 @@ public class FrameState {
|
|||||||
* @param type Classification of push type
|
* @param type Classification of push type
|
||||||
* @param stack Simulated operand stand types
|
* @param stack Simulated operand stand types
|
||||||
* @param locals Simulated frame local types
|
* @param locals Simulated frame local types
|
||||||
* @param constants Method constant pool
|
|
||||||
*/
|
*/
|
||||||
private static void trivialPush(AbstractInsnNode insn, char type, List<TypeSignature> stack, List<TypeSignature> locals, List<TypeSignature> constants) {
|
private static void trivialPush(AbstractInsnNode insn, char type, List<TypeSignature> stack, List<LocalVariableNode> locals) {
|
||||||
// Pushing is a bit more tricky than popping because we have to resolve types (kind of)
|
// Pushing is a bit more tricky than popping because we have to resolve types (kind of)
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'I':
|
case 'I':
|
||||||
@ -525,7 +567,18 @@ public class FrameState {
|
|||||||
case 44: // ALOAD_2
|
case 44: // ALOAD_2
|
||||||
case 45: // ALOAD_3
|
case 45: // ALOAD_3
|
||||||
// Push a local variable to the stack
|
// Push a local variable to the stack
|
||||||
stack.add(locals.get(((VarInsnNode) insn).var));
|
Optional<LocalVariableNode> targetVar = localsAt(insn, locals)
|
||||||
|
.stream()
|
||||||
|
.filter(it -> it.index == ((VarInsnNode) insn).var)
|
||||||
|
.findFirst();
|
||||||
|
|
||||||
|
if (!targetVar.isPresent())
|
||||||
|
throw new StateAnalysisException(String.format(
|
||||||
|
"Attempt to access a local variable out of scope: Opcode = %s",
|
||||||
|
insn.getOpcode()
|
||||||
|
));
|
||||||
|
|
||||||
|
stack.add(new TypeSignature(targetVar.get().desc));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Opcodes.AALOAD:
|
case Opcodes.AALOAD:
|
||||||
@ -540,7 +593,7 @@ public class FrameState {
|
|||||||
case Opcodes.NEW:
|
case Opcodes.NEW:
|
||||||
// Allocate a new object (should really be marked as uninitialized, but meh)
|
// Allocate a new object (should really be marked as uninitialized, but meh)
|
||||||
// We'll burn that bridge when we get to it or something...
|
// We'll burn that bridge when we get to it or something...
|
||||||
stack.add(new TypeSignature(((TypeInsnNode) insn).desc));
|
stack.add(new TypeSignature(String.format("L%s;", ((TypeInsnNode) insn).desc)));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Opcodes.NEWARRAY:
|
case Opcodes.NEWARRAY:
|
||||||
@ -587,7 +640,11 @@ public class FrameState {
|
|||||||
* "Opcode<...>", please refer to the comments in {@link Opcodes} as well as the official JVM specification
|
* "Opcode<...>", please refer to the comments in {@link Opcodes} as well as the official JVM specification
|
||||||
* @see <a href="https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-6.html#jvms-6.5">JVM8 instructions spec</a>
|
* @see <a href="https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-6.html#jvms-6.5">JVM8 instructions spec</a>
|
||||||
*/
|
*/
|
||||||
private static List<String> getOpsByComplexity(boolean complexPush, boolean complexPop, @Nullable Predicate<Integer> insnP) {
|
private static List<String> getOpsByComplexity(
|
||||||
|
boolean complexPush,
|
||||||
|
boolean complexPop,
|
||||||
|
@Nullable Predicate<Integer> insnP
|
||||||
|
) {
|
||||||
ArrayList<Integer> opcodes = new ArrayList<>();
|
ArrayList<Integer> opcodes = new ArrayList<>();
|
||||||
|
|
||||||
for (int i = 0; i < FrameState.STACK_CLOBBER_PUSH.length(); ++i)
|
for (int i = 0; i < FrameState.STACK_CLOBBER_PUSH.length(); ++i)
|
||||||
@ -621,4 +678,40 @@ public class FrameState {
|
|||||||
}
|
}
|
||||||
}).collect(java.util.stream.Collectors.toList());
|
}).collect(java.util.stream.Collectors.toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find the index of a given instruction relative to a starting point
|
||||||
|
* @param current Starting point for search
|
||||||
|
* @param find Instruction to find
|
||||||
|
* @return Negative values for instructions previous to the current instruction, positive values for instructions
|
||||||
|
* after the current instruction. Null if instruction could not be found
|
||||||
|
*/
|
||||||
|
private static @Nullable Integer relativeIndexOf(
|
||||||
|
@NotNull AbstractInsnNode current,
|
||||||
|
@NotNull AbstractInsnNode find
|
||||||
|
) {
|
||||||
|
// Check backward
|
||||||
|
int idx = 0;
|
||||||
|
AbstractInsnNode check = current;
|
||||||
|
while (check != null) {
|
||||||
|
if (check == find)
|
||||||
|
return idx;
|
||||||
|
--idx;
|
||||||
|
check = check.getPrevious();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check forward
|
||||||
|
idx = 1;
|
||||||
|
check = current.getNext();
|
||||||
|
while (check != null) {
|
||||||
|
if (check == find)
|
||||||
|
return idx;
|
||||||
|
++idx;
|
||||||
|
check = check.getNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
// No match
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -88,20 +88,33 @@ public class TypeSignature {
|
|||||||
*/
|
*/
|
||||||
public TypeSignature(@Nullable Character primitive, boolean isTop) {
|
public TypeSignature(@Nullable Character primitive, boolean isTop) {
|
||||||
if (primitive != null) {
|
if (primitive != null) {
|
||||||
switch (primitive) {
|
switch (Character.toUpperCase(primitive)) {
|
||||||
case 'J':
|
case 'J':
|
||||||
case 'D':
|
case 'D':
|
||||||
case 'V':
|
case 'V':
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 'I':
|
||||||
|
case 'F':
|
||||||
|
case 'S':
|
||||||
|
case 'Z':
|
||||||
|
case 'C':
|
||||||
|
case 'B':
|
||||||
|
if (isTop)
|
||||||
|
throw new TypeSignatureParseException(String.format(
|
||||||
|
"Primitive type signature %c cannot have a Top value. To declare a Top delimiter, use 'V'",
|
||||||
|
primitive
|
||||||
|
));
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
throw new TypeSignatureParseException(String.format(
|
throw new TypeSignatureParseException(String.format(
|
||||||
"Primitive type signature %s cannot have a Top value. To declare a Top delimiter, use 'V'",
|
"Unknown primitive signature %c",
|
||||||
primitive.toString()
|
primitive
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.sig = primitive == null ? "V" : primitive.toString();
|
this.sig = primitive == null ? "V" : Character.toString(Character.toUpperCase(primitive));
|
||||||
modifier = TypeModifier.TOP.iff(isTop);
|
modifier = TypeModifier.TOP.iff(isTop);
|
||||||
dynamicRef = null;
|
dynamicRef = null;
|
||||||
this.arrayDepth = 0;
|
this.arrayDepth = 0;
|
||||||
|
@ -1,16 +1,24 @@
|
|||||||
import dev.w1zzrd.asm.Combine;
|
import dev.w1zzrd.asm.Combine;
|
||||||
import dev.w1zzrd.asm.GraftSource;
|
import dev.w1zzrd.asm.GraftSource;
|
||||||
import dev.w1zzrd.asm.Merger;
|
import dev.w1zzrd.asm.Merger;
|
||||||
|
import dev.w1zzrd.asm.analysis.FrameState;
|
||||||
|
import dev.w1zzrd.asm.signature.TypeSignature;
|
||||||
import jdk.internal.org.objectweb.asm.tree.ClassNode;
|
import jdk.internal.org.objectweb.asm.tree.ClassNode;
|
||||||
import jdk.internal.org.objectweb.asm.tree.MethodNode;
|
import jdk.internal.org.objectweb.asm.tree.MethodNode;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.Stack;
|
||||||
|
|
||||||
public class Test {
|
public class Test {
|
||||||
public static void main(String... args) throws IOException {
|
public static void main(String... args) throws IOException {
|
||||||
|
|
||||||
ClassNode target = Merger.getClassNode("MergeTest");
|
ClassNode target = Merger.getClassNode("MergeTest");
|
||||||
ClassNode inject = Merger.getClassNode("MergeInject");
|
ClassNode inject = Merger.getClassNode("MergeInject");
|
||||||
|
|
||||||
|
MethodNode stackTest = target.methods.stream().filter(it -> it.name.equals("stackTest")).findFirst().get();
|
||||||
|
|
||||||
|
Stack<TypeSignature> stack = FrameState.getFrameStateAt(stackTest.instructions.getLast().getPrevious().getPrevious().getPrevious().getPrevious().getPrevious(), stackTest.localVariables);
|
||||||
|
|
||||||
GraftSource source = new GraftSource(target);
|
GraftSource source = new GraftSource(target);
|
||||||
|
|
||||||
Combine combine = new Combine(target);
|
Combine combine = new Combine(target);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user