Я использую Rust 1.0 beta и смог создать небольшой пример для вызова функций, написанных на Rust с Java. Я просто скомпилировал следующий код Rust в mylib.rs, используя rustc, который создает mylib.dll в Windows:
#![crate_type = "dylib"]
use std::any::Any;
#[no_mangle]
pub extern fn Java_tests_Test_hello(env: *const Any, jclass: *const Any) {
println!("hello from rust");
}
#[no_mangle]
pub extern fn Java_tests_Test_sum(env: *const Any, jclass: *const Any, a: i32, b: i32) -> i32 {
return a + b;
}
Затем я могу вызывать эти функции из тестов класса Java. Тест:
package tests;
import java.io.File;
public class Test {
public static native void hello();
public static native int sum(int a, int b);
public static void main(String[] args) {
File f = new File("mylib.dll");
System.load(f.getAbsolutePath());
Test.hello();
System.out.println(Test.sum(20, 22));
}
}
Запуск главной страницы Java печатает ожидаемый результат:
hello from rust
42
В методах Rust я объявил env
как указатель на тип Any
, но на самом деле это C-структура с указателями на функции, как описано в (пример кода 4-1), которые необходимы для обмена данными со средой выполнения Java.
Из этого ответа я понял, что такие структуры с указателями функций должны иметь аналог в коде Rust для вызова этих функций. Поэтому я попытался реализовать такую структуру, установив все значения полей в *mut Any
, за исключением поля GetVersion
:
#[repr(C)]
pub struct JavaEnv {
reserved0: *mut Any,
reserved1: *mut Any,
reserved2: *mut Any,
reserved3: *mut Any,
GetVersion: extern "C" fn(env: *mut JavaEnv) -> i32,
DefineClass: *mut Any,
FindClass: *mut Any,
…
Когда я вызываю следующую функцию из Java, которая должна вызывать функцию GetVersion, JVM аварийно завершает работу:
#[no_mangle]
pub extern fn Java_tests_Test_helloJre(jre: *mut JavaEnv, class: *const Any) {
unsafe {
let v = ((*jre).GetVersion)(jre);
println!("version: {:?}", v);
}
}
Как я могу правильно вызвать функцию GetVersion? Обратите внимание, что я действительно новичок в подобных материалах, поэтому, пожалуйста, не стесняйтесь редактировать этот вопрос.