<html><head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  </head>
  <body>
    <p><br>
    </p>
    <div class="moz-cite-prefix">On 27/09/2023 10:59, tison wrote:<br>
    </div>
    <blockquote type="cite" cite="mid:CALL9TYJNdTjQqDhqgjvbinoxjXJ16pQFQQ6F__0RkOhDgOVOHg@mail.gmail.com">
      
      <div dir="ltr">
        <div class="gmail_default" style="font-family:arial,sans-serif">OT
          - perhaps I should start a new thread, but the first things
          come to me when I talk to the FFM APIs:</div>
        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
        </div>
        <div class="gmail_default" style="font-family:arial,sans-serif">1.
          Manually defining MemoryLayout can be clumsy. C# has a
          structure layout attribute for generating struct/class
          marshalling[1]</div>
        <div class="gmail_default" style="font-family:arial,sans-serif">2.
          I wonder if the new FFI calls can be JIT-ed. One of the major
          performance impact factor of JNI is that they treat the
          foreign method as black box and nothing optimize can be done
          =。=</div>
        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
        </div>
        <div class="gmail_default" style="font-family:arial,sans-serif">These
          are only rough ideas and I'm glad to know where I can start to
          learn the related work and previous discussions.</div>
      </div>
    </blockquote>
    <p>As Remi points out, our approach is to auto generate layouts and
      ancillary data structures, with a tool called jextract:</p>
    <p><a class="moz-txt-link-freetext" href="https://github.com/openjdk/jextract">https://github.com/openjdk/jextract</a></p>
    <p>You can download the jextract version compatible with 21 here:</p>
    <p><a class="moz-txt-link-freetext" href="https://jdk.java.net/">https://jdk.java.net/</a></p>
    <p>The current state of the FFM API is captured in a talk I gave few
      months ago, which could be a good starting point:</p>
    <p><a class="moz-txt-link-freetext" href="https://www.youtube.com/watch?v=kUFysMkMS00">https://www.youtube.com/watch?v=kUFysMkMS00</a><br>
    </p>
    <p>As to why we just didn't add a bunch of annotation... the answer
      is longer. That's kind of where we started:</p>
    <p><a class="moz-txt-link-freetext" href="https://cr.openjdk.org/~mcimadamore/panama/panama-binder-v3.html">https://cr.openjdk.org/~mcimadamore/panama/panama-binder-v3.html</a></p>
    <p>But we quickly realized that this approach was not "primitive"
      enough. E.g. trying to come up with an API to model C, while it
      might be handy for casual native interop, doesn't scale in at
      least two cases:</p>
    <p>* clients that want to manipulate off-heap memory but don't care
      about FFI (Netty, Lucene, ...)<br>
      * clients that want to build on top of our FFI support and define
      their own high-level interop stories (e.g. JNA, JNR, ...)</p>
    <p>This realization is captured in this talk I gave few years ago:</p>
    <p><a class="moz-txt-link-freetext" href="https://www.youtube.com/watch?v=r4dNRVWYaZI">https://www.youtube.com/watch?v=r4dNRVWYaZI</a></p>
    <p>There are of course a number of other technical reasons as to why
      annotations are not a great primitive fit. For instance, you'd
      need a reliable iteration order when scanning the methods of a
      class reflectively, which Java does NOT provide. Secondly, having
      to generate proxies on the fly for everything was particularly
      problematic, especially for startup, and especially for allowing
      Panama use from within the JDK itself.</p>
    <p>The FFM API you see today sits at a lower level than e.g. what C#
      provides. But the idea is that if a framework just wants
      annotations and wrapper interfaces, they should be able to do just
      that, and we shouldn't be in the way. This is, for example, what
      JPassport does:</p>
    <p><a class="moz-txt-link-freetext" href="https://github.com/boulder-on/JPassport">https://github.com/boulder-on/JPassport</a></p>
    <p>As FFM exits finalization, I hope that other popular frameworks
      such as JNA and JNR will switch from JNI to FFM, and deliver
      better performance for their users (such frameworks have to
      currently rely on libffi/JNI and, as a result, they can be slower
      compared to plain JNI).</p>
    <p>Note also that we're actively exploring solutions to make the
      mapping between _records_ and struct layouts easier:</p>
    <p><a class="moz-txt-link-freetext" href="https://github.com/openjdk/panama-foreign/pull/833">https://github.com/openjdk/panama-foreign/pull/833</a></p>
    <p>That work is still in progress, and will not make it 22 (as we
      are now focussing on finalizing the API), but it will appear at
      some point.<br>
    </p>
    <p><br>
    </p>
    <p>On your question re. foreign call being treated as blackbox -
      this is still the case in FFM - that is, FFM doesn't look into the
      assembly of the target function to try and inline that code
      elsewhere, or perform other optimizations. Such an approach seems
      fragile, unless one assumes that the target function code is
      available in some other form (e.g. LLVM bitcode). Otherwise, it's
      a bit like trying to reconstruct source code information from
      classfile bytecodes, which is one problem that project Babylon [1]
      aims to address). That said, we don't rule out other targeted
      optimizations (e.g. being able to elide some of the thread
      transition from Java to native for back to back native calls has
      long been on the agenda).</p>
    <p>Cheers<br>
      Maurizio</p>
    <p>[1] -
      <a class="moz-txt-link-freetext" href="https://mail.openjdk.org/pipermail/discuss/2023-September/006226.html">https://mail.openjdk.org/pipermail/discuss/2023-September/006226.html</a><br>
    </p>
    <blockquote type="cite" cite="mid:CALL9TYJNdTjQqDhqgjvbinoxjXJ16pQFQQ6F__0RkOhDgOVOHg@mail.gmail.com">
      <div dir="ltr">
        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
        </div>
        <div>
          <div dir="ltr" class="gmail_signature" data-smartmail="gmail_signature">
            <div dir="ltr">
              <div>
                <div dir="ltr">
                  <div>
                    <div dir="ltr">
                      <div>
                        <div dir="ltr">
                          <div><font face="arial, sans-serif">Best,</font></div>
                          <div><font face="arial, sans-serif">tison.</font></div>
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
          </div>
        </div>
        <br>
        <div>
          <div class="gmail_default" style="font-family:arial,sans-serif">[1] <a href="https://learn.microsoft.com/en-us/dotnet/standard/native-interop/type-marshalling" moz-do-not-send="true" class="moz-txt-link-freetext">https://learn.microsoft.com/en-us/dotnet/standard/native-interop/type-marshalling</a></div>
          <br>
        </div>
      </div>
      <br>
      <div class="gmail_quote">
        <div dir="ltr" class="gmail_attr">tison <<a href="mailto:wander4096@gmail.com" moz-do-not-send="true" class="moz-txt-link-freetext">wander4096@gmail.com</a>>
          于2023年9月27日周三 17:53写道:<br>
        </div>
        <blockquote class="gmail_quote" style="margin:0px 0px 0px
          0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
          <div dir="ltr">
            <div class="gmail_default" style="font-family:arial,sans-serif">Thanks for your quick
              response and suggestions! I realize that using static
              final MethodHandle and call invoke/invokeExact inside the
              wrapper class should be better. While the downside is I
              may have to duplicate some code like:</div>
            <div class="gmail_default" style="font-family:arial,sans-serif"><br>
            </div>
            <div class="gmail_default" style="font-family:arial,sans-serif">private static final
              MethodHandle xxxNativeMethod = ...;</div>
            <div class="gmail_default" style="font-family:arial,sans-serif">public static /*
              native */ ReturnType xxx(T1 t1, T2 t2, ...) {</div>
            <div class="gmail_default" style="font-family:arial,sans-serif">    // validate</div>
            <div class="gmail_default" style="font-family:arial,sans-serif">    return
              xxxNativeMethod.invoke(t1, t2, ...); // with return type
              coercion and exception handling</div>
            <div class="gmail_default" style="font-family:arial,sans-serif">}</div>
            <div class="gmail_default" style="font-family:arial,sans-serif"><br>
            </div>
            <div class="gmail_default" style="font-family:arial,sans-serif">I'll try to do it and
              share it openly when I make an MVP.</div>
            <div class="gmail_default" style="font-family:arial,sans-serif"><br>
            </div>
            <div class="gmail_default" style="font-family:arial,sans-serif">FWIW, in the past
              months, I built a Java binding for a Rust lib named
              OpenDAL[1] using JNI since it requires JDK 8
              compatibility. But this time I'd like to try out
              Datafusion with the new FFM APIs for fun :D</div>
            <div class="gmail_default" style="font-family:arial,sans-serif"><br>
            </div>
            <div>
              <div dir="ltr" class="gmail_signature">
                <div dir="ltr">
                  <div>
                    <div dir="ltr">
                      <div>
                        <div dir="ltr">
                          <div>
                            <div dir="ltr">
                              <div><font face="arial, sans-serif">Best,</font></div>
                              <div><font face="arial, sans-serif">tison.</font></div>
                              <div><font face="arial, sans-serif"><br>
                                </font></div>
                              <div><font face="arial, sans-serif">[<span class="gmail_default" style="font-family:arial,sans-serif">1] </span></font><a href="https://github.com/apache/incubator-opendal/tree/main/bindings/java" moz-do-not-send="true" class="moz-txt-link-freetext">https://github.com/apache/incubator-opendal/tree/main/bindings/java</a></div>
                            </div>
                          </div>
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
            <br>
          </div>
          <br>
          <div class="gmail_quote">
            <div dir="ltr" class="gmail_attr">Maurizio Cimadamore <<a href="mailto:maurizio.cimadamore@oracle.com" moz-do-not-send="true" class="moz-txt-link-freetext">maurizio.cimadamore@oracle.com</a>>
              于2023年9月27日周三 17:43写道:<br>
            </div>
            <blockquote class="gmail_quote" style="margin:0px 0px 0px
              0.8ex;border-left:1px solid
              rgb(204,204,204);padding-left:1ex">
              <div>
                <p>Glavo's suggestion is correct. If you want the method
                  handle call to apply the maximum set of conversions,
                  you should use `invokeWithArguments`.</p>
                <p>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).</p>
                <p>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).<br>
                </p>
                <p>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.</p>
                <p>Cheers<br>
                  Maurizio<br>
                </p>
                <div>On 27/09/2023 10:29, Glavo wrote:<br>
                </div>
                <blockquote type="cite">
                  <div dir="ltr">
                    <div dir="ltr">See
                      MethodHandle::invokeWithArguments.</div>
                    <div dir="ltr"><br>
                    </div>
                    <div>Glavo</div>
                  </div>
                  <br>
                  <div class="gmail_quote">
                    <div dir="ltr" class="gmail_attr">On Wed, Sep 27,
                      2023 at 5:14 PM tison <<a href="mailto:wander4096@gmail.com" target="_blank" moz-do-not-send="true" class="moz-txt-link-freetext">wander4096@gmail.com</a>>
                      wrote:<br>
                    </div>
                    <blockquote class="gmail_quote" style="margin:0px
                      0px 0px 0.8ex;border-left:1px solid
                      rgb(204,204,204);padding-left:1ex">
                      <div dir="ltr">
                        <div class="gmail_default" style="font-family:arial,sans-serif">I'm using
                          OpenJDK 21 and prototyping FFM APIs.</div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">Instead
                          of passing the MethodHandle returned
                          by linker.downcallHandle here and there, I'm
                          trying to wrap it into a NativeMethod class:</div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">public
                          final class NativeMethod {<br>
                              private final String name;<br>
                              private final FunctionDescriptor
                          descriptor;<br>
                              private final MethodHandle handle;<br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">   
                          public Object invoke(Object... args) throws
                          Throwable {<br>
                                  return this.handle.invoke(args);<br>
                              }<br>
                          }<br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">But it
                          seems the varargs pass through is not quit
                          fluent:</div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">java.lang.invoke.WrongMethodTypeException:
                          cannot convert MethodHandle(int)int to
                          (Object[])Object<br>
                          <br>
                          at
java.base/java.lang.invoke.MethodHandle.asTypeUncached(MethodHandle.java:903)<br>
                          at
                          java.base/java.lang.invoke.MethodHandle.asType(MethodHandle.java:870)<br>
                          at
                          java.base/java.lang.invoke.Invokers.checkGenericType(Invokers.java:541)<br>
                          at
                          io.montumi.datafusion.core.NativeMethod.invoke(NativeMethod.java:22)<br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div class="gmail_default" style="font-family:arial,sans-serif">While if
                          I public the handle and call invoke from the
                          handle directly, all the behavior is expected.</div>
                        <div class="gmail_default" style="font-family:arial,sans-serif"><br>
                        </div>
                        <div>
                          <div dir="ltr" class="gmail_signature">
                            <div dir="ltr">
                              <div>
                                <div dir="ltr">
                                  <div>
                                    <div dir="ltr">
                                      <div>
                                        <div dir="ltr">
                                          <div><font face="arial,
                                              sans-serif">Best,</font></div>
                                          <div><font face="arial,
                                              sans-serif">tison.</font></div>
                                        </div>
                                      </div>
                                    </div>
                                  </div>
                                </div>
                              </div>
                            </div>
                          </div>
                        </div>
                      </div>
                    </blockquote>
                  </div>
                </blockquote>
              </div>
            </blockquote>
          </div>
        </blockquote>
      </div>
    </blockquote>
  </body>
</html>