-
CSR
-
Resolution: Approved
-
P3
-
None
-
source, binary, behavioral
-
medium
-
Some APIs have been removed/renamed. Clients will have to switch to alternatives. For var/method handles derived from memory layouts, clients will have to provide the additional base offset coordinate.
-
Java API
-
SE
Summary
This CSR refers to the finalization of the Foreign Function & Memory (FFM) API that first started previewing in Java 19.
Problem
Feedback and careful examination of the FFM API revealed the following issues:
- The FFM API offers ways to easily translate Java strings into native strings. But, the produced native strings are currently limited to the UTF-8 encoding.
- The MemoryLayout::sequenceLayout(MemoryLayout) factory method, which produces a sequence layout with a maximum element size (dervied from Long.MAX_VALUE) is a pitfall. If a user simply forgets to specify the size of the sequence, issues can occur later down the line. Such as: failure to allocate such a sequnce, or failure to link a function using a struct layout containing one of these sequences.
- While jextract can be used to automatically derive the layouts of native types, clients that don't use jextract are left figuring out what the layout of a native type is on their own.
- There are cases where the VarHandles and MethodHandles derived from memory layouts fall short, because a certain case can not be fully represented using memory layouts, for instance because it involves an array whose size is not statically known.
- The MemorySegment::segmentOffset method is no longer needed. The same thing can be trivially done by taking the difference between the base address of 2 segments.
- Calls to SegmentAllocator::allocateArray can be ambiguous. If this method is called with a single
long
argumentx
, which is intended to allocate an array containing the singlelong
valuex
, instead clients will get an array ofx
uninitializedlong
s. - The documentation of variadic functions mentions prototype-less functions, which are not variadic according to the C spec. This spec is also not as clear as could be.
- The Linker API is currently optional, which makes it impossible for the JDK to use it itself to implement access to native functions.
- The FMM API is currently in preview. We want to move it out of preview.
- The name of the isTrivial linker option is not descriptive enough of what the linker option does.
- It is currently possible to combine the isTrivial linker option with the captureCallState linker option, which constrains the implementation of future features.
- The Arena::allocate method is currently a default method. But, it is an important method that implements of Arena should override.
- When using an unsupported access mode and accessing a mis-aligned address using a memory access var handle, we currently throw an IllegalArgumentException, instead of an UnsupportedOperationException.
- There are several SegmentAllocator::allocateFrom methods that allocate an initialize the allocated memory. For those methods, we can skip memory zeroing of the allocated memory, since we know we'll be overwriting the memory right away any ways. But, the current API only allows initializing the allocated memory from a primitive value or a Java array.
- Executable jar files are allowed to specify Add-Opens and Add-Exports in their manifests, which function equivalently to the --add-opens and --add-exports command line flags of the java launcher. However, there is no equivalent manifest attribute for --enable-native-access.
Solution
The proposed solutions to each of these issues, in order, are:
- Expand the set of supported encodings to all encodings found in the
java.nio.charset.StandardCharsets
class. - Remove the MemoryLayout::sequenceLayout(MemoryLayout) factory method. Clients should instead use the MemoryLayout::sequenceLayout(long, MemoryLayout) factory method, and specify the element count that they want explicitly.
- A new API, Linker::canonicalLayouts(), is added. For native linkers, this API can be used to find the memory layouts of the most common primitive types of the C language.
- Change the var and method handles derived from memory layouts to accept an additional 'base offset' coordinate. When a handle is used, the value of this coordinate is added to the offset computed done by the handle. This effectively allows these handles to be composed with other offset computation code. This can help address cases where a particular memory access can not be fully represented using the memory layout and layout path APIs.
- Remove the MemorySegment::segmentOffset method.
- Rename the methods in SegmentAllocator that allocate and intialize a memory segment to 'allocateFrom'. This avoids the aforementioned ambiguity.
- Clarfiy the documentation of variadic functions, and drop the references to prototype-less functions.
- Make the linker a required API, but dropping the exception specification on the Linker::nativeLinker method stating that on unsupported platforms it throws an UnsupportedOperationException.
- Move the FFM API out of preview, by removing
@PreviewFeature
annotations, and updating@since
tags in javadoc. - Rename the Linker.Option.isTrivial API to 'critical'. The name name leans on the established meaning of 'critical' in the context of JNI. We have to keep the name somewhat vague to avoid making too many promises about what it does, as not every linker implementation is required to implement it.
- Disallow isTrivial from being combined with the Linker.Option.captureCallState option. This keeps the door open for future enhancements.
- Make the Arena::allocate method abstract.
- Update the implementation to throw an UnsupportedOperationException for unsupported access modes, even in the case of a mis-aligned access
- Add a new SegmentAllocator::allocateFrom overload that allows initializing the allocated memory from an arbitrary memory segment.
- Add an
Enable-Native-Access
jar manifest attribute that functions equivalently to the --enable-native-access command line flag of the java launcher.
Specification
A specdiff of the changes is available below:
- https://cr.openjdk.org/~jvernee/jep22_specdiff/v3/ (2023/08/11. Commit hash 141096b)
- https://cr.openjdk.org/~jvernee/jep22_specdiff/v4_inc/ (2023/09/11. Commit hash 0e702f0. Incremental)
- https://cr.openjdk.org/~jvernee/jep22_specdiff/v5_inc/ (2023/09/28. Commit hash 72650c4. Incremental)
Note that particular item #4 has a wide spread impact on the javadoc of the API.
. #15 does not reflect in any javadoc changes, so I'll describe the feature here:
The value of the Enable-Native-Access
manifest attribute is, at least for the time being, restricted to the 'ALL-UNNAMED', indicating that native access is enabled for unnamed modules. This is similar to how the Add-Opens/Add-Exports attributes only grant access to unnamed modules. When a value other than 'ALL-UNNAMED' is specified, the JVM will be abort during launch with an error message.
References
- https://github.com/openjdk/panama-foreign/pull/836
- https://github.com/openjdk/panama-foreign/pull/838
- https://github.com/openjdk/panama-foreign/pull/839
- https://github.com/openjdk/panama-foreign/pull/840
- https://github.com/openjdk/panama-foreign/pull/841
- https://github.com/openjdk/panama-foreign/pull/845
- https://github.com/openjdk/panama-foreign/pull/846
- https://github.com/openjdk/panama-foreign/pull/850
- https://github.com/openjdk/panama-foreign/pull/853
- https://github.com/openjdk/panama-foreign/pull/859
- https://github.com/openjdk/panama-foreign/pull/856
- no patch in panama-foreign
- https://github.com/openjdk/panama-foreign/pull/876
- https://github.com/openjdk/panama-foreign/pull/878
- https://github.com/openjdk/panama-foreign/pull/843
- csr of
-
JDK-8312522 Implementation of Foreign Function & Memory API
- Resolved
- relates to
-
JDK-8282192 Implementation of Foreign Function & Memory API (Preview)
- Closed
-
JDK-8295045 Implementation of Foreign Function and Memory API (Second Preview)
- Closed
-
JDK-8303240 Implementation of Foreign Function and Memory API (Third Preview)
- Closed