2015-05-03 20:57:38 -06:00
|
|
|
package java.lang.invoke;
|
2015-08-05 15:55:52 -06:00
|
|
|
|
2015-08-06 13:24:06 -06:00
|
|
|
import avian.Classes;
|
support AOT-compilation of Java 8 lambda expressions
These expressions are tricky because they rely on invokedynamic, which
normally implies runtime code generation. However, since lambdas
don't actually use the "dynamicness" of invokedynamic, we can convert
them into static calls to synthetic classes at compile time.
Since I had already written code to synthesize such classes in Java
and I didn't want to rewrite it in C++, I needed to add support for
running Java code to the bootimage generator. And since the primary
VM used by the generator is purpose-built to generate AOT-compiled
code for a specific target architecture and is not capable of
generating or running JIT-compiled code for the host architecture, I
added support for loading a second, independent, host-specific VM for
running Java code.
The rest of the patch handles the fact that each method compilation
might cause new, synthetic classes to be created, so we need to make
sure those classes and their methods are included in the final heap
and code images. This required breaking some giant code blocks out of
makeCodeImage into their own methods, which makes the diff look
scarier than it really is.
2015-09-12 20:15:46 -06:00
|
|
|
import avian.SystemClassLoader;
|
2015-08-06 13:24:06 -06:00
|
|
|
|
2015-08-05 15:55:52 -06:00
|
|
|
public class MethodHandle {
|
2015-08-06 13:24:06 -06:00
|
|
|
static final int REF_invokeStatic = 6;
|
|
|
|
static final int REF_invokeSpecial = 7;
|
|
|
|
|
|
|
|
final int kind;
|
2015-08-05 15:55:52 -06:00
|
|
|
private final ClassLoader loader;
|
|
|
|
final avian.VMMethod method;
|
|
|
|
private volatile MethodType type;
|
|
|
|
|
2015-08-06 13:24:06 -06:00
|
|
|
MethodHandle(int kind, ClassLoader loader, avian.VMMethod method) {
|
|
|
|
this.kind = kind;
|
2015-08-05 15:55:52 -06:00
|
|
|
this.loader = loader;
|
|
|
|
this.method = method;
|
|
|
|
}
|
2015-08-06 13:24:06 -06:00
|
|
|
|
support AOT-compilation of Java 8 lambda expressions
These expressions are tricky because they rely on invokedynamic, which
normally implies runtime code generation. However, since lambdas
don't actually use the "dynamicness" of invokedynamic, we can convert
them into static calls to synthetic classes at compile time.
Since I had already written code to synthesize such classes in Java
and I didn't want to rewrite it in C++, I needed to add support for
running Java code to the bootimage generator. And since the primary
VM used by the generator is purpose-built to generate AOT-compiled
code for a specific target architecture and is not capable of
generating or running JIT-compiled code for the host architecture, I
added support for loading a second, independent, host-specific VM for
running Java code.
The rest of the patch handles the fact that each method compilation
might cause new, synthetic classes to be created, so we need to make
sure those classes and their methods are included in the final heap
and code images. This required breaking some giant code blocks out of
makeCodeImage into their own methods, which makes the diff look
scarier than it really is.
2015-09-12 20:15:46 -06:00
|
|
|
MethodHandle(String class_,
|
|
|
|
String name,
|
|
|
|
String spec,
|
|
|
|
int kind)
|
|
|
|
{
|
|
|
|
this.kind = kind;
|
|
|
|
this.loader = SystemClassLoader.appLoader();
|
|
|
|
try {
|
|
|
|
this.method = Classes.findMethod(this.loader, class_, name, spec);
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 15:55:52 -06:00
|
|
|
public String toString() {
|
2015-08-06 13:24:06 -06:00
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
if (method.class_ != null) {
|
|
|
|
sb.append(Classes.makeString(method.class_.name, 0,
|
|
|
|
method.class_.name.length - 1));
|
|
|
|
sb.append(".");
|
|
|
|
}
|
|
|
|
sb.append(Classes.makeString(method.name, 0,
|
|
|
|
method.name.length - 1));
|
|
|
|
sb.append(Classes.makeString(method.spec, 0,
|
|
|
|
method.spec.length - 1));
|
|
|
|
return sb.toString();
|
|
|
|
}
|
2015-08-05 15:55:52 -06:00
|
|
|
|
|
|
|
public MethodType type() {
|
|
|
|
if (type == null) {
|
|
|
|
type = new MethodType(loader, method.spec);
|
|
|
|
}
|
|
|
|
return type;
|
|
|
|
}
|
2015-05-03 20:57:38 -06:00
|
|
|
}
|