Compare commits

...

4 Commits

Author SHA1 Message Date
ee1ab4191c Fix file path formatting bug in Injector.java 2021-04-02 08:07:13 +02:00
Gabriel Tofvesson
58982c8f99 Update README.md 2021-02-05 21:56:45 +01:00
Gabriel Tofvesson
531a5ffe0c Update test files 2021-02-05 21:29:17 +01:00
Gabriel Tofvesson
dd0748a52c Implement injected assertions 2021-02-05 21:28:47 +01:00
5 changed files with 109 additions and 2 deletions

View File

@ -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

View File

@ -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

View File

@ -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 {

View File

@ -22,6 +22,8 @@ public class MergeInject extends MergeTest implements Runnable {
Directives.callSuper();
s = "Hello";
number = 10;
assert false : "Test";
}

View File

@ -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();