[External] : Re: How to return a optional string from native methods?

Jorn Vernee jorn.vernee at oracle.com
Sat Sep 30 13:19:39 UTC 2023


Note that if you just want to dispose of the string after the 
conversion, you can also do:

     private static String readUtf8String(MemorySegment segment) {
         final MemorySegment string = segment.reinterpret(Long.MAX_VALUE);
         String result = string.getUtf8String(0);
         NativeMethods.disposeString(segment);
         return result;
     }

This behaves more or less equivalently to your version. There's no need 
to create a new Arena. MemorySegment::reinterpret doesn't copy the 
underlying data, but MemorySegment::getUtf8String does.

Jorn

On 30/09/2023 05:18, tison wrote:
> Thank you! Finally I work out a string conversion utility:
>
>   private static String readUtf8String(MemorySegment segment) {
>         try (Arena arena = Arena.ofConfined()) {
>             final MemorySegment string = 
> segment.reinterpret(Long.MAX_VALUE, arena, NativeMethods::disposeString);
>             return string.getUtf8String(0);
>         }
>     }
>
> #[no_mangle]
> pub unsafe extern "C" fn dispose_string(s: *mut c_char) {
>     drop(CString::from_raw(s))
> }
>
> Hopefully, this doesn't create too much overhead (I'm not sure the 
> overhead of creating Arena), and no other leaks. It likes 
> JNIEnv#NewString and IIUC JNIEnv#NewString also copy bytes.
>
> > Native functions can just return NULL/0, and the linker will 
> automatically translate that into MemorySegment.NULL.
>
> Good to know!
>
> > So, there is no FFM-specific API for the native code to interact with 
> the Java side
>
> OK. Perhaps I should return an error union in this case and decode & 
> throw exceptions on the Java side.
>
> Best,
> tison.
>
>
> Jorn Vernee <jorn.vernee at oracle.com> 于2023年9月29日周五 23:42写道:
>
>     > And, if we think of error handling, how do we properly return a
>     NULL from native methods, and how do we throw exceptions from
>     native methods?
>     > MemorySegment.NULL seems an Object without overriding equals,
>     and JNIEnv's throw counterpart is missing or I don't find it
>     (perhaps upcall helps but it's still a bit away from real-world
>     usage).
>
>     Native functions can just return NULL/0, and the linker will
>     automatically translate that into MemorySegment.NULL.
>
>     As for throwing a Java exception from native code, similar to what
>     is possible with JNI Throw(New) [1], this is not supported. The
>     Linker operates on the general assumption that the native code it
>     is interacting with has no knowledge about Java at all. So, there
>     is no FFM-specific API for the native code to interact with the
>     Java side.
>
>     Jorn
>
>     [1]:
>     https://docs.oracle.com/en/java/javase/21/docs/specs/jni/functions.html#throw
>     On 29/09/2023 16:08, tison wrote:
>>     Here is my native method written in Rust:
>>
>>     #[no_mangle]
>>     pub unsafe extern "C" fn datafusion_version() -> *const c_char {
>>     CString::new(datafusion::DATAFUSION_VERSION).unwrap().into_raw()
>>     }
>>
>>     And below is my attempt to bind it with FFM APIs:
>>
>>         static MethodHandle createMethodHandle(String name,
>>     FunctionDescriptor descriptor) {
>>             final MemorySegment fp =
>>     LOADER.lookup.find(name).orElseThrow();
>>             return LOADER.linker.downcallHandle(fp, descriptor);
>>         }
>>
>>         private static final MethodHandle
>>     datafusionVersionMethodHandle =
>>     NativeLoader.createMethodHandle(datafusionVersionMethodName,
>>     datafusionVersionMethodDesc);
>>
>>         @SneakyThrows
>>         public static String datafusionVersion() {
>>             final MemorySegment version = (MemorySegment)
>>     datafusionVersionMethodHandle.invokeExact();
>>             return version.getUtf8String(0);
>>         }
>>
>>     I noticed that the returned MemorySegment is always with length 0
>>     and thus any access with return OutOfBoundException.
>>
>>     In the demos from JEP, all memory segments are allocated from the
>>     Java side and the native code only move or modify those allocated
>>     memory segments instead of allocate/shrink memory.
>>
>>     I wonder what is the formal method to allocate a string from
>>     native methods and pass back to the Java world.
>>
>>     And, if we think of error handling, how do we properly return a
>>     NULL from native methods, and how do we throw exceptions from
>>     native methods?
>>
>>     MemorySegment.NULL seems an Object without overriding equals, and
>>     JNIEnv's throw counterpart is missing or I don't find it (perhaps
>>     upcall helps but it's still a bit away from real-world usage).
>>
>>     Best,
>>     tison.
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/panama-dev/attachments/20230930/af505c46/attachment-0001.htm>


More information about the panama-dev mailing list