mirror of
https://github.com/corda/corda.git
synced 2025-01-05 20:54:13 +00:00
2465459079
The two big pieces here are basic invokedynamic support and a working version of LambdaMetaFactory.metafactory. The latter works by dynamically building a synthetic class with three methods: a static factory method, a constructor for the factory method to call, and a method to satisfy the requested interface which defers to the specified MethodHandle. This work relies heavily on Avian's specific MethodType and MethodHandle implementations, which provide extra, non-standard features to make code generation easier. That means we'll probably need to use Avian's versions of java.lang.invoke.* even when building with the OpenJDK or Android class libraries.
136 lines
4.0 KiB
Java
136 lines
4.0 KiB
Java
/* Copyright (c) 2008-2015, Avian Contributors
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
for any purpose with or without fee is hereby granted, provided
|
|
that the above copyright notice and this permission notice appear
|
|
in all copies.
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
details. */
|
|
|
|
package avian;
|
|
|
|
import static avian.Stream.write1;
|
|
import static avian.Stream.write2;
|
|
import static avian.Stream.write4;
|
|
|
|
import avian.ConstantPool.PoolEntry;
|
|
|
|
import java.util.List;
|
|
import java.io.OutputStream;
|
|
import java.io.IOException;
|
|
|
|
public class Assembler {
|
|
public static final int ACC_PUBLIC = 1 << 0;
|
|
public static final int ACC_STATIC = 1 << 3;
|
|
|
|
public static final int aaload = 0x32;
|
|
public static final int aastore = 0x53;
|
|
public static final int aload = 0x19;
|
|
public static final int aload_0 = 0x2a;
|
|
public static final int aload_1 = 0x2b;
|
|
public static final int astore_0 = 0x4b;
|
|
public static final int anewarray = 0xbd;
|
|
public static final int areturn = 0xb0;
|
|
public static final int dload = 0x18;
|
|
public static final int dreturn = 0xaf;
|
|
public static final int dup = 0x59;
|
|
public static final int fload = 0x17;
|
|
public static final int freturn = 0xae;
|
|
public static final int getfield = 0xb4;
|
|
public static final int goto_ = 0xa7;
|
|
public static final int iload = 0x15;
|
|
public static final int invokeinterface = 0xb9;
|
|
public static final int invokespecial = 0xb7;
|
|
public static final int invokestatic = 0xb8;
|
|
public static final int invokevirtual = 0xb6;
|
|
public static final int ireturn = 0xac;
|
|
public static final int jsr = 0xa8;
|
|
public static final int ldc_w = 0x13;
|
|
public static final int lload = 0x16;
|
|
public static final int lreturn = 0xad;
|
|
public static final int new_ = 0xbb;
|
|
public static final int pop = 0x57;
|
|
public static final int putfield = 0xb5;
|
|
public static final int ret = 0xa9;
|
|
public static final int return_ = 0xb1;
|
|
|
|
public static void writeClass(OutputStream out,
|
|
List<PoolEntry> pool,
|
|
int name,
|
|
int super_,
|
|
int[] interfaces,
|
|
FieldData[] fields,
|
|
MethodData[] methods)
|
|
throws IOException
|
|
{
|
|
int codeAttributeName = ConstantPool.addUtf8(pool, "Code");
|
|
|
|
write4(out, 0xCAFEBABE);
|
|
write2(out, 0); // minor version
|
|
write2(out, 50); // major version
|
|
|
|
write2(out, pool.size() + 1);
|
|
for (PoolEntry e: pool) {
|
|
e.writeTo(out);
|
|
}
|
|
|
|
write2(out, ACC_PUBLIC); // flags
|
|
write2(out, name + 1);
|
|
write2(out, super_ + 1);
|
|
|
|
write2(out, interfaces.length);
|
|
for (int i: interfaces) {
|
|
write2(out, i + 1);
|
|
}
|
|
|
|
write2(out, fields.length);
|
|
for (FieldData f: fields) {
|
|
write2(out, f.flags);
|
|
write2(out, f.nameIndex + 1);
|
|
write2(out, f.specIndex + 1);
|
|
write2(out, 0); // attribute count
|
|
}
|
|
|
|
write2(out, methods.length);
|
|
for (MethodData m: methods) {
|
|
write2(out, m.flags);
|
|
write2(out, m.nameIndex + 1);
|
|
write2(out, m.specIndex + 1);
|
|
|
|
write2(out, 1); // attribute count
|
|
write2(out, codeAttributeName + 1);
|
|
write4(out, m.code.length);
|
|
out.write(m.code);
|
|
}
|
|
|
|
write2(out, 0); // attribute count
|
|
}
|
|
|
|
public static class MethodData {
|
|
public final int flags;
|
|
public final int nameIndex;
|
|
public final int specIndex;
|
|
public final byte[] code;
|
|
|
|
public MethodData(int flags, int nameIndex, int specIndex, byte[] code) {
|
|
this.flags = flags;
|
|
this.nameIndex = nameIndex;
|
|
this.specIndex = specIndex;
|
|
this.code = code;
|
|
}
|
|
}
|
|
|
|
public static class FieldData {
|
|
public final int flags;
|
|
public final int nameIndex;
|
|
public final int specIndex;
|
|
|
|
public FieldData(int flags, int nameIndex, int specIndex) {
|
|
this.flags = flags;
|
|
this.nameIndex = nameIndex;
|
|
this.specIndex = specIndex;
|
|
}
|
|
}
|
|
}
|