How to wrap Method handle behind a class?

Maurizio Cimadamore maurizio.cimadamore at oracle.com
Wed Sep 27 09:42:40 UTC 2023


Glavo's suggestion is correct. If you want the method handle call to 
apply the maximum set of conversions, you should use `invokeWithArguments`.

The price to pay is that, the further you move away from `invokeExact` 
the slower the method call will go (because of the conversion code than 
needs to surround the code itself).

Also notice that, with your approach, the method handle wrapping the 
function is not static final (it is final, but not static). This means 
it is not a true JVM constant. This means your method handle call will 
probably not be inlined very well. Wrapping with a record instead of a 
plain class might be better because records fields are trusted. So if 
you wrap a method handle in a NativeMethod _record_ and then you stick 
the record instance in a static final field somewhere, that _should_ 
work as expected (waving hands furiously).

Anyway, of course all the above doesn't mean you shouldn't do what you 
are trying to do - if performance is not a concern for you then your 
approach is totally reasonable.

Cheers
Maurizio

On 27/09/2023 10:29, Glavo wrote:
> See MethodHandle::invokeWithArguments.
>
> Glavo
>
> On Wed, Sep 27, 2023 at 5:14 PM tison <wander4096 at gmail.com> wrote:
>
>     I'm using OpenJDK 21 and prototyping FFM APIs.
>
>     Instead of passing the MethodHandle returned
>     by linker.downcallHandle here and there, I'm trying to wrap it
>     into a NativeMethod class:
>
>     public final class NativeMethod {
>         private final String name;
>         private final FunctionDescriptor descriptor;
>         private final MethodHandle handle;
>         public Object invoke(Object... args) throws Throwable {
>             return this.handle.invoke(args);
>         }
>     }
>
>     But it seems the varargs pass through is not quit fluent:
>
>     java.lang.invoke.WrongMethodTypeException: cannot convert
>     MethodHandle(int)int to (Object[])Object
>
>     at
>     java.base/java.lang.invoke.MethodHandle.asTypeUncached(MethodHandle.java:903)
>     at
>     java.base/java.lang.invoke.MethodHandle.asType(MethodHandle.java:870)
>     at
>     java.base/java.lang.invoke.Invokers.checkGenericType(Invokers.java:541)
>     at
>     io.montumi.datafusion.core.NativeMethod.invoke(NativeMethod.java:22)
>
>     While if I public the handle and call invoke from the handle
>     directly, all the behavior is expected.
>
>     Best,
>     tison.
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/panama-dev/attachments/20230927/8e1d8460/attachment.htm>


More information about the panama-dev mailing list