Comment resolveDynamic, add some commented out (failing) tests

This commit is contained in:
Andras Slemmer 2017-01-19 17:27:08 +00:00
parent 52d1ca9799
commit 3a81c7f997
3 changed files with 93 additions and 2 deletions

View File

@ -340,6 +340,7 @@ public class LambdaMetafactory {
String invokedName, String invokedName,
MethodType invokedType, MethodType invokedType,
Object... args) throws LambdaConversionException { Object... args) throws LambdaConversionException {
// See openjdk8/jdk/src/share/classes/java/lang/invoke/LambdaMetafactory.java
// Behaves as if the prototype is like this: // Behaves as if the prototype is like this:
// //
// CallSite altMetafactory( // CallSite altMetafactory(
@ -355,7 +356,6 @@ public class LambdaMetafactory {
// int bridgeCount, // IF flags has BRIDGES set // int bridgeCount, // IF flags has BRIDGES set
// MethodType... bridges // IF flags has BRIDGES set // MethodType... bridges // IF flags has BRIDGES set
// ) // )
MethodType methodType = (MethodType) args[0]; MethodType methodType = (MethodType) args[0];
MethodHandle methodImplementation = (MethodHandle) args[1]; MethodHandle methodImplementation = (MethodHandle) args[1];

View File

@ -6032,13 +6032,26 @@ GcJclass* getDeclaringClass(Thread* t, GcClass* c)
return 0; return 0;
} }
// Called when interpreting invokedynamic. `invocation` points to
// static data in the bootstrap method table, which in turn points to
// a bootstrap method and stores additional data to be passed to
// it. `resolveDynamic` will then call this bootstrap method after
// resolving the arguments as required. The called method is assumed
// to be a lambda `metafactory` or `altMetafactory`.
//
// Note that capture/bridging etc happens within the bootstrap method,
// this is just the code that dispatches to it.
//
// Returns the CallSite returned by the bootstrap method.
GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation) GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
{ {
PROTECT(t, invocation); PROTECT(t, invocation);
// Use the invocation's Class to get the bootstrap method table and get a classloader.
GcClass* c = invocation->class_(); GcClass* c = invocation->class_();
PROTECT(t, c); PROTECT(t, c);
// First element points to the bootstrap method. The rest are static data passed to the BSM.
GcCharArray* bootstrapArray = cast<GcCharArray>( GcCharArray* bootstrapArray = cast<GcCharArray>(
t, t,
cast<GcArray>(t, c->addendum()->bootstrapMethodTable()) cast<GcArray>(t, c->addendum()->bootstrapMethodTable())
@ -6046,6 +6059,7 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
PROTECT(t, bootstrapArray); PROTECT(t, bootstrapArray);
// Resolve the bootstrap method itself.
GcMethod* bootstrap = cast<GcMethod>(t, GcMethod* bootstrap = cast<GcMethod>(t,
resolve(t, resolve(t,
c->loader(), c->loader(),
@ -6055,22 +6069,29 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
GcNoSuchMethodError::Type)); GcNoSuchMethodError::Type));
PROTECT(t, bootstrap); PROTECT(t, bootstrap);
// Caller context info to be passed to the bootstrap method.
GcLookup* lookup GcLookup* lookup
= makeLookup(t, c, ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC); = makeLookup(t, c, ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC);
PROTECT(t, lookup); PROTECT(t, lookup);
// The name of the linked-to method.
GcByteArray* nameBytes = invocation->template_()->name(); GcByteArray* nameBytes = invocation->template_()->name();
GcString* name GcString* name
= t->m->classpath->makeString(t, nameBytes, 0, nameBytes->length() - 1); = t->m->classpath->makeString(t, nameBytes, 0, nameBytes->length() - 1);
PROTECT(t, name); PROTECT(t, name);
// This is the type of the linked-to method (e.g. lambda).
GcMethodType* type = makeMethodType( GcMethodType* type = makeMethodType(
t, c->loader(), invocation->template_()->spec(), 0, 0, 0); t, c->loader(), invocation->template_()->spec(), 0, 0, 0);
PROTECT(t, type); PROTECT(t, type);
// `array` stores either
// 1. All the arguments to be passed to the bootstrap method in the case of `metafactory`
// 2. The vararg object array to be passed to `altMetafactory`
GcArray* array = makeArray(t, bootstrap->parameterCount()); GcArray* array = makeArray(t, bootstrap->parameterCount());
PROTECT(t, array); PROTECT(t, array);
// These are common arguments to metafactory and altMetafactory
unsigned argument = 0; unsigned argument = 0;
array->setBodyElement(t, argument++, lookup); array->setBodyElement(t, argument++, lookup);
array->setBodyElement(t, argument++, name); array->setBodyElement(t, argument++, name);
@ -6079,16 +6100,28 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
THREAD_RUNTIME_ARRAY(t, char, specBuffer, bootstrap->spec()->length()); THREAD_RUNTIME_ARRAY(t, char, specBuffer, bootstrap->spec()->length());
const char* spec; const char* spec;
// `argArray` stores the final arguments to be passed to the bootstrap method.
// Later in this function we iterate through the method signature +
// bootstrap array and resolve the arguments as required into `array`.
//
// In the case of a `metafactory` call:
// `argArray = [caller, invokedName, invokedType, methodType, methodImplementation, instantiatedType]`
// `array = argArray`
//
// In the case of an `altMetafactory` call:
// `argArray = [caller, invokedName, invokedType, array]`
// `array = [methodType, methodImplementation, instantiatedType, flags, ...]`
GcArray* argArray = array; GcArray* argArray = array;
PROTECT(t, argArray); PROTECT(t, argArray);
// Check if the bootstrap method's signature matches that of an altMetafactory
if (::strcmp(reinterpret_cast<char*>(bootstrap->spec()->body().begin()), if (::strcmp(reinterpret_cast<char*>(bootstrap->spec()->body().begin()),
"(Ljava/lang/invoke/MethodHandles$Lookup;" "(Ljava/lang/invoke/MethodHandles$Lookup;"
"Ljava/lang/String;" "Ljava/lang/String;"
"Ljava/lang/invoke/MethodType;" "Ljava/lang/invoke/MethodType;"
"[Ljava/lang/Object;)" "[Ljava/lang/Object;)"
"Ljava/lang/invoke/CallSite;") == 0) { "Ljava/lang/invoke/CallSite;") == 0) {
// LambdaMetaFactory.altMetafactory // If so, create a new array to store the varargs in, and hardcode the BSM signature.
array = makeArray(t, bootstrapArray->length() - 1); array = makeArray(t, bootstrapArray->length() - 1);
spec = "(Ljava/lang/invoke/MethodHandles$Lookup;" spec = "(Ljava/lang/invoke/MethodHandles$Lookup;"
"Ljava/lang/String;" "Ljava/lang/String;"
@ -6103,6 +6136,9 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
"[Ljava/lang/invoke/MethodType;" "[Ljava/lang/invoke/MethodType;"
")Ljava/lang/invoke/CallSite;"; ")Ljava/lang/invoke/CallSite;";
} else if (bootstrap->parameterCount() == 2 + bootstrapArray->length()) { } else if (bootstrap->parameterCount() == 2 + bootstrapArray->length()) {
// We're calling the simpler `metafactory`. 2 + bootstrapArray->length() is the
// arguments to the bootstrap method (bootstrapArray->length() - 1), plus the 3 static
// arguments (lookup, name, type).
memcpy(RUNTIME_ARRAY_BODY(specBuffer), memcpy(RUNTIME_ARRAY_BODY(specBuffer),
bootstrap->spec()->body().begin(), bootstrap->spec()->body().begin(),
bootstrap->spec()->length()); bootstrap->spec()->length());
@ -6113,16 +6149,24 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
MethodSpecIterator it(t, spec); MethodSpecIterator it(t, spec);
// Skip over the already handled 3 arguments.
for (unsigned i = 0; i < argument; ++i) for (unsigned i = 0; i < argument; ++i)
it.next(); it.next();
// If we're calling altMetafactory then we reset the argument
// offset, because we are filling the vararg array instead of the
// final argument array.
if (argArray != array) { if (argArray != array) {
argument = 0; argument = 0;
} }
// `i` iterates through the bootstrap arguments (the +1 is because we skip
// the boostrap method's name), `it` iterates through the corresponding types
// in the method signature
unsigned i = 0; unsigned i = 0;
while (i + 1 < bootstrapArray->length() && it.hasNext()) { while (i + 1 < bootstrapArray->length() && it.hasNext()) {
const char* p = it.next(); const char* p = it.next();
switch (*p) { switch (*p) {
case 'L': { case 'L': {
const char* const methodType = "Ljava/lang/invoke/MethodType;"; const char* const methodType = "Ljava/lang/invoke/MethodType;";
@ -6198,10 +6242,12 @@ GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation)
? 0 ? 0
: makeMethodHandle(t, REF_invokeSpecial, c->loader(), bootstrap, 0); : makeMethodHandle(t, REF_invokeSpecial, c->loader(), bootstrap, 0);
// If we're calling altMetafactory we set the fourth argument to the vararg array.
if (argArray != array) { if (argArray != array) {
argArray->setBodyElement(t, 3, array); argArray->setBodyElement(t, 3, array);
} }
// Finally we make the bootstrap call.
return cast<GcCallSite>( return cast<GcCallSite>(
t, t->m->processor->invokeArray(t, bootstrap, handle, argArray)); t, t->m->processor->invokeArray(t, bootstrap, handle, argArray));
} }

View File

@ -41,6 +41,32 @@ public class InvokeDynamic {
} }
} }
private interface Foo extends java.io.Serializable {
void someFunction(Integer a, Integer b, String s);
}
private interface UnboxedSerializable extends java.io.Serializable {
int add(int a, int b);
}
private interface Unboxed {
int add(int a, int b);
}
private void requiresBridge(Number a, Object... rest) {
String s = "" + a;
for (Object r : rest) {
s += r;
}
}
private static Integer addBoxed(Integer a, Integer b) {
return a + b;
}
private interface Marker {
}
private void test() { private void test() {
{ int c = 2; { int c = 2;
Operation op = (a, b) -> ((a + b) * c) - foo; Operation op = (a, b) -> ((a + b) * c) - foo;
@ -62,5 +88,24 @@ public class InvokeDynamic {
expect(s.get().first == 42L); expect(s.get().first == 42L);
expect(s.get().second == 3.14D); expect(s.get().second == 3.14D);
} }
{ Foo s = this::requiresBridge;
s.someFunction(1, 2, "");
}
// This abort()s in machine.cpp
// { Foo s = (Foo & Marker) this::requiresBridge;
// s.someFunction(1, 2, "");
// }
// NPE
// { UnboxedSerializable s = InvokeDynamic::addBoxed;
// expect(s.add(1, 2) == 3);
// }
// NPE
// { Unboxed s = InvokeDynamic::addBoxed;
// expect(s.add(1, 2) == 3);
// }
} }
} }