巴拿马项目:打通 JVM 与 Native 代码
作者:Denys Makogon
来源:denismakogon.github.io/openjdk/panama/2022/05/31/introduction-to-project-panama-part-1.html
准备
项目概述
外部函数和内存 API:JEP 424 Jextract 工具 Vector API:JEP 338
内存段及其地址:一组 API 类,用于处理本机内存和指向它的指针; 内存布局和描述符:用于模拟外部类型(结构、原语)和函数描述符的 API; 内存会话:管理一个或多个内存资源生命周期的抽象; 链接器和符号查找:一组用于执行向下和向上调用的 API 类; 段分配器:一种用于在内存会话中分配内存段的 API。
Hello World 程序
链接器
public static Linker getSystemLinker() {
return switch (CABI.current()) {
case Win64 - > Windowsx64Linker.getInstance();
case SysV - > SysVx64Linker.getInstance();
case LinuxAArch64 - > LinuxAArch64Linker.getInstance();
case MacOsAArch64 - > MacOsAArch64Linker.getInstance();
};
}
在 JDK 术语中,链接器是特定于平台的 C ABI 实现的一个实例。链接器提供一组方法来执行向下调用和向上调用,其中:
downcall 是从高级子系统发起的事件。在我们的例子中是 JVM 到较低级别的子系统,如操作系统内核或者一些 Java 代码调用一些本机代码。稍后将通过外部函数和内存 API 说明这一点。
upcall 例如一些本机代码调用一些 Java 代码。
虽然链接器就像电话一样,想打电话给谁,只需拨入正确的电话号码即可。符号查找方法就像通讯录,只需提供要打电话的人正确的信息即可。
要执行向下调用,需要提供调用的(本机)函数的描述符、通过符号查找分配的本机地址,以及用于创建调用本机函数的方法句柄对应的链接器。
从 Java 实现经典的 C 风格的 Hello World:
int printf(const char * __restrict, ...)
Java 中的 C 语言风格的“Hello World”
1. 找到 native 函数的地址
Linker linker = Linker.nativeLinker();
SymbolLookup linkerLookup = linker.defaultLookup();
SymbolLookup systemLookup = SymbolLookup.loaderLookup();
SymbolLookup symbolLookup = name ->
systemLookup.lookup(name).or(() -> linkerLookup.lookup(name));
Optional<MemorySegment> printfMemorySegment = symbolLookup.lookup("printf");
2. 构建正在调用的函数的描述符
FunctionDescriptor printfDescriptor = FunctionDescriptor.of(JAVA_INT, ADDRESS);
注意:从 Java 运行时的角度来看,C 指针背后的值类型无关紧要,因为 C 指针的内存布局不保存类型,而是平台固定的 32/64 位值。
一个描述符定义了一个返回值类型为 int 的函数,它的参数是一个指针。假设一个描述符几乎对应于它在 stdio.h 中的 C 定义,因为它定义了一个标准函数,而 printf 是一个可变参数函数。
通过值布局(Value Layout)在 Java 中对 C 类型建模
在 Java 中,值布局用于对与基本数据类型的值关联的内存布局建模,例如整数类型(有符号或无符号)和浮点类型。JAVA_INT 和 ADDRESS 都是对应的 C 类型的值布局。
JAVA_INT :
// ValueLayout.OfInt.class
OfInt JAVA_INT = new OfInt(ByteOrder.nativeOrder()).withBitAlignment(32);
// ValueLayout.OfAddress.class
OfAddress ADDRESS = new OfAddress(ByteOrder.nativeOrder())
.withBitAlignment(ValueLayout.ADDRESS_SIZE_BITS);
3. 从函数的本机内存地址构建方法句柄
MethodHandle printfMethodHandle = symbolLookup.lookup("printf").map(
addr - > linker.downcallHandle(addr, printfDescriptor)
).orElse(null);
downcall 是通过由本机函数地址及其 Java 版本的函数描述符形成的 MethodHandle调用本机函数。 upcall 是通过 MethodHandle 调用一些用 Java 编写的代码,该 MethodHandle 转换为本机内存段,然后可以将其作为函数指针传递给本机函数。
4. 分配本机内存
try (var memorySession = MemorySession.openConfined()) {
SegmentAllocator allocator = SegmentAllocator.newNativeArena(memorySession);
var cStringFromAllocator = allocator.allocateUtf8String("Hello World" + "\n");
var cStringFromSession = memorySession.allocateUtf8String("Hello World" + "\n");
}
MemorySegment cString = memorySession.allocateUtf8String(str + "\n");
private static int printf(String str, MemorySession memorySession) throws Throwable {
Objects.requireNonNull(printfMethodHandle);
var cString = memorySession.allocateUtf8String(str + "\n");
return (int) printfMethodHandle.invoke(cString);
}
public static void main(String[] args) throws Throwable {
var str = "Hello World";
try (var memorySession = MemorySession.openConfined()) {
System.out.println(printf(str, memorySession));
}
}
5. 小结
获取本机库及其对应的头文件。 在 Java 中构建函数描述符 ( FunctionDescriptor )。 查找函数符号的本机内存地址,并为其创建方法句柄。 创建一个相关的方法句柄并确认它已经正确创建(例如,如果本机库不在系统路径中,查找将失败并且返回一个方法句柄将为空)。 决定应用程序将如何分配内存段:通过段分配器或内存会话。确保内存分配技术在应用程序的整个代码库中保持一致。
代码清单
package com.java_devrel.samples.panama.part_1;
import java.lang.foreign.*;
import java.lang.invoke.MethodHandle;
import java.util.Objects;
import static java.lang.foreign.ValueLayout.ADDRESS;
import static java.lang.foreign.ValueLayout.JAVA_INT;
public class PrintfSimplified {
private static final Linker linker = Linker.nativeLinker();
private static final SymbolLookup linkerLookup = linker.defaultLookup();
private static final SymbolLookup systemLookup = SymbolLookup.loaderLookup();
private static final SymbolLookup symbolLookup = name - > systemLookup.lookup(name).or(() - > linkerLookup.lookup(name));
private static final FunctionDescriptor printfDescriptor = FunctionDescriptor.of(JAVA_INT.withBitAlignment(32), ADDRESS.withBitAlignment(64));
private static final MethodHandle printfMethodHandle = symbolLookup.lookup("printf").map(addr - > linker.downcallHandle(addr, printfDescriptor)).orElse(null);
private static int printf(String str, MemorySession memorySession) throws Throwable {
Objects.requireNonNull(printfMethodHandle);
var cString = memorySession.allocateUtf8String(str + "\n");
return (int) printfMethodHandle.invoke(cString);
}
public static void main(String[] args) throws Throwable {
var str = "hello world";
try (var memorySession = MemorySession.openConfined()) {
System.out.println(printf(str, memorySession));
}
}
}
推荐阅读
你好,我是程序猿DD,10年开发老司机、阿里云MVP、腾讯云TVP、出过书创过业、国企4年互联网6年。从普通开发到架构师、再到合伙人。一路过来,给我最深的感受就是一定要不断学习并关注前沿。只要你能坚持下来,多思考、少抱怨、勤动手,就很容易实现弯道超车!所以,不要问我现在干什么是否来得及。如果你看好一个事情,一定是坚持了才能看到希望,而不是看到希望才去坚持。相信我,只要坚持下来,你一定比现在更好!如果你还没什么方向,可以先关注我,这里会经常分享一些前沿资讯,帮你积累弯道超车的资本。