Compare commits
4 Commits
Author | SHA1 | Date | |
---|---|---|---|
ee1ab4191c | |||
![]() |
58982c8f99 | ||
![]() |
531a5ffe0c | ||
![]() |
dd0748a52c |
@ -35,6 +35,12 @@ methods and fields.
|
||||
|
||||
* Inject try-catch-finally
|
||||
|
||||
* Inject assertions
|
||||
|
||||
*A caveat regarding assert-statements: the compiler synthesizes a static final field named `$assertionsDisabled`, so if a target
|
||||
class declares a static field with this name and does not declare any assertions in its code, loading of the field may already
|
||||
be done in static initialization or field declaration, preventing assertions from functioning as intended for injected code.*
|
||||
|
||||
### TODO
|
||||
|
||||
* Better tests
|
||||
|
@ -19,6 +19,10 @@ import java.util.stream.Collectors;
|
||||
import static jdk.internal.org.objectweb.asm.ClassWriter.COMPUTE_MAXS;
|
||||
|
||||
public class Combine {
|
||||
public static final String VAR_ASSERT_NAME = "$assertionsDisabled";
|
||||
public static final int VAR_ASSERT_FLAGS = Opcodes.ACC_STATIC | Opcodes.ACC_SYNTHETIC | Opcodes.ACC_FINAL;
|
||||
|
||||
|
||||
private final ArrayList<DynamicSourceUnit> graftSources = new ArrayList<>();
|
||||
|
||||
private final ClassNode target;
|
||||
@ -323,6 +327,88 @@ public class Combine {
|
||||
return target;
|
||||
}
|
||||
|
||||
protected void ensureLoadClassAssertionState() {
|
||||
if (!hasDeclaredAssertionState())
|
||||
target.fields.add(new FieldNode(
|
||||
VAR_ASSERT_FLAGS,
|
||||
VAR_ASSERT_NAME,
|
||||
"Z",
|
||||
null,
|
||||
null
|
||||
));
|
||||
|
||||
// Check if state is loaded
|
||||
if (target.methods.stream().noneMatch(it -> it.name.equals("<clinit>"))) {
|
||||
MethodNode mnode = new MethodNode(Opcodes.ACC_STATIC, "<clinit>", "()V", null, null);
|
||||
injectAssertionLoad(mnode, true);
|
||||
target.methods.add(mnode);
|
||||
} else {
|
||||
MethodNode clinit = target.methods.stream().filter(it -> it.name.equals("<clinit>")).findAny().get();
|
||||
|
||||
for (AbstractInsnNode node = clinit.instructions.getFirst(); node != null; node = node.getNext())
|
||||
if (node instanceof FieldInsnNode &&
|
||||
node.getOpcode() == Opcodes.PUTSTATIC &&
|
||||
VAR_ASSERT_NAME.equals(((FieldInsnNode) node).name))
|
||||
return;
|
||||
|
||||
// Assertion state not loaded in the current clinit. Add it to the start of the clinit
|
||||
injectAssertionLoad(clinit, false);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* For internal use. Adds instructions to the given method to load assertion flag to a static final field
|
||||
* @param node Method to inject instructions into
|
||||
*/
|
||||
private void injectAssertionLoad(MethodNode node, boolean insertReturn) {
|
||||
if (node.instructions == null) {
|
||||
node.instructions = new InsnList();
|
||||
insertReturn = true;
|
||||
}
|
||||
|
||||
AbstractInsnNode current;
|
||||
if (node.instructions.getFirst() == null || !(node.instructions.getFirst() instanceof LabelNode))
|
||||
node.instructions.insert(current = new LabelNode());
|
||||
else
|
||||
current = node.instructions.getFirst();
|
||||
|
||||
node.instructions.insert(current, current = new LdcInsnNode(Type.getType("L"+target.name+";")));
|
||||
node.instructions.insert(current, current = new MethodInsnNode(
|
||||
Opcodes.INVOKEVIRTUAL,
|
||||
"java/lang/Class",
|
||||
"desiredAssertionStatus",
|
||||
"()Z",
|
||||
false
|
||||
));
|
||||
|
||||
final LabelNode jumpNE = new LabelNode();
|
||||
final LabelNode jumpGOTO = new LabelNode();
|
||||
|
||||
node.instructions.insert(current, current = new JumpInsnNode(Opcodes.IFNE, jumpNE));
|
||||
node.instructions.insert(current, current = new InsnNode(Opcodes.ICONST_1));
|
||||
node.instructions.insert(current, current = new JumpInsnNode(Opcodes.GOTO, jumpGOTO));
|
||||
node.instructions.insert(current, current = jumpNE);
|
||||
node.instructions.insert(current, current = new FrameNode(Opcodes.F_SAME, 0, new Object[0], 0, new Object[0]));
|
||||
node.instructions.insert(current, current = new InsnNode(Opcodes.ICONST_0));
|
||||
node.instructions.insert(current, current = jumpGOTO);
|
||||
node.instructions.insert(current, current = new FrameNode(
|
||||
Opcodes.F_SAME1,
|
||||
0,
|
||||
new Object[0],
|
||||
1,
|
||||
new Object[]{ Opcodes.INTEGER }
|
||||
));
|
||||
|
||||
node.instructions.insert(current, current = new FieldInsnNode(Opcodes.PUTSTATIC, target.name, VAR_ASSERT_NAME, "Z"));
|
||||
|
||||
if (insertReturn)
|
||||
node.instructions.insert(current, new InsnNode(Opcodes.RETURN));
|
||||
}
|
||||
|
||||
protected boolean hasDeclaredAssertionState() {
|
||||
return target.fields.stream().anyMatch(it -> VAR_ASSERT_NAME.equals(it.name));
|
||||
}
|
||||
|
||||
/**
|
||||
* Prepares a {@link MethodNode} for grafting on to a given method and into the targeted {@link ClassNode}
|
||||
* @param node Node to adapt
|
||||
@ -334,8 +420,14 @@ public class Combine {
|
||||
if (insn instanceof MethodInsnNode) insn = adaptMethodInsn((MethodInsnNode) insn, source, node);
|
||||
else if (insn instanceof LdcInsnNode) adaptLdcInsn((LdcInsnNode) insn, source.getTypeName());
|
||||
else if (insn instanceof FrameNode) adaptFrameNode((FrameNode) insn, source);
|
||||
else if (insn instanceof FieldInsnNode) adaptFieldInsn((FieldInsnNode) insn, source);
|
||||
else if (insn instanceof InvokeDynamicInsnNode) adaptInvokeDynamicInsn((InvokeDynamicInsnNode) insn, source);
|
||||
else if (insn instanceof FieldInsnNode) {
|
||||
adaptFieldInsn((FieldInsnNode) insn, source);
|
||||
|
||||
// If a method is trying to access the assertion state of the class, ensure the target class declares the state
|
||||
if (VAR_ASSERT_NAME.equals(((FieldInsnNode) insn).name) && insn.getOpcode() == Opcodes.GETSTATIC)
|
||||
ensureLoadClassAssertionState();
|
||||
}
|
||||
}
|
||||
|
||||
// Adapt variable types
|
||||
|
@ -9,6 +9,8 @@ import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.net.MalformedURLException;
|
||||
import java.net.URL;
|
||||
import java.net.URLDecoder;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.Enumeration;
|
||||
import java.util.Objects;
|
||||
|
||||
@ -25,7 +27,10 @@ public class Injector {
|
||||
public static void injectAll(ClassLoader loader, Combine merger) throws IOException {
|
||||
Enumeration<URL> resources = loader.getResources("");
|
||||
while (resources.hasMoreElements())
|
||||
injectDirectory(new File(resources.nextElement().getPath()), merger);
|
||||
injectDirectory(new File(URLDecoder.decode(
|
||||
resources.nextElement().getFile(),
|
||||
StandardCharsets.UTF_8.name())
|
||||
), merger);
|
||||
}
|
||||
|
||||
public static void injectAll(Combine merger) throws IOException {
|
||||
|
@ -22,6 +22,8 @@ public class MergeInject extends MergeTest implements Runnable {
|
||||
Directives.callSuper();
|
||||
s = "Hello";
|
||||
number = 10;
|
||||
|
||||
assert false : "Test";
|
||||
}
|
||||
|
||||
|
||||
|
@ -7,6 +7,8 @@ import java.io.IOException;
|
||||
|
||||
public class Test {
|
||||
public static void main(String... args) throws IOException {
|
||||
ClassLoader.getSystemClassLoader().setDefaultAssertionStatus(false);
|
||||
|
||||
// Load target class, inject all annotated classes and load compiled bytecode into JVM
|
||||
dumpFile(Injector.injectAll("MergeTest"), "MergeTest").compile();
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user