update at 2022-02-23 09:39:30 by ehlxr
This commit is contained in:
parent
767b834be1
commit
61b8d5338b
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,2 +1,3 @@
|
|||||||
mylib/target
|
mylib/target
|
||||||
**/*.class
|
**/*.class
|
||||||
|
**/*.h
|
12
Makefile
Normal file
12
Makefile
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
java_run: lib
|
||||||
|
cd java_src && \
|
||||||
|
javac me/ehlxr/HelloWorld.java && \
|
||||||
|
java -Djava.library.path=../mylib/target/debug/ me.ehlxr.HelloWorld
|
||||||
|
.PHONY: lib
|
||||||
|
|
||||||
|
javah:
|
||||||
|
cd java_src && \
|
||||||
|
javah me.ehlxr.HelloWorld
|
||||||
|
|
||||||
|
lib:
|
||||||
|
cd mylib && cargo build
|
@ -1,15 +1,51 @@
|
|||||||
package me.ehlxr;
|
package me.ehlxr;
|
||||||
|
|
||||||
public class HelloWorld {
|
class HelloWorld {
|
||||||
|
private static native String hello(String input);
|
||||||
|
private static native byte[] helloByte(byte[] input);
|
||||||
|
private static native void factAndCallMeBack(int n, HelloWorld callback);
|
||||||
|
|
||||||
|
private static native long counterNew(HelloWorld callback);
|
||||||
|
private static native void counterIncrement(long counter_ptr);
|
||||||
|
private static native void counterDestroy(long counter_ptr);
|
||||||
|
|
||||||
|
private static native void asyncComputation(HelloWorld callback);
|
||||||
|
|
||||||
static {
|
static {
|
||||||
// Linux: export LD_LIBRARY_PATH=/Users/ehlxr/Desktop/jni_rs/mylib/target/debug
|
// Linux: export LD_LIBRARY_PATH=/Users/ehlxr/Desktop/jni_rs/mylib/target/debug
|
||||||
// Mac: export JAVA_LIBRARY_PATH=/Users/ehlxr/Desktop/jni_rs/mylib/target/debug
|
// Mac: export JAVA_LIBRARY_PATH=/Users/ehlxr/Desktop/jni_rs/mylib/target/debug
|
||||||
System.loadLibrary("mylib");
|
System.loadLibrary("mylib");
|
||||||
}
|
}
|
||||||
private static native String hello(String input);
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
String output = HelloWorld.hello("Java");
|
String output = HelloWorld.hello("Java");
|
||||||
System.out.println(output);
|
System.out.println(output);
|
||||||
|
byte[] outputByte = HelloWorld.helloByte("byte".getBytes());
|
||||||
|
System.out.println(outputByte);
|
||||||
|
|
||||||
|
HelloWorld.factAndCallMeBack(6, new HelloWorld());
|
||||||
|
|
||||||
|
long counter_ptr = counterNew(new HelloWorld());
|
||||||
|
|
||||||
|
for (int i = 0; i < 5; i++) {
|
||||||
|
counterIncrement(counter_ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
counterDestroy(counter_ptr);
|
||||||
|
|
||||||
|
System.out.println("Invoking asyncComputation (thread id = " + Thread.currentThread().getId() + ")");
|
||||||
|
asyncComputation(new HelloWorld());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void factCallback(int res) {
|
||||||
|
System.out.println("factCallback: res = " + res);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void counterCallback(int count) {
|
||||||
|
System.out.println("counterCallback: count = " + count);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void asyncCallback(int progress) {
|
||||||
|
System.out.println("asyncCallback: thread id = " + Thread.currentThread().getId() + ", progress = " + progress + "%");
|
||||||
}
|
}
|
||||||
}
|
}
|
129
mylib/src/lib.rs
129
mylib/src/lib.rs
@ -1,7 +1,11 @@
|
|||||||
use jni::objects::{JClass, JString};
|
|
||||||
use jni::sys::jstring;
|
|
||||||
use jni::JNIEnv;
|
use jni::JNIEnv;
|
||||||
|
|
||||||
|
use jni::objects::{GlobalRef, JClass, JObject, JString};
|
||||||
|
|
||||||
|
use jni::sys::{jbyteArray, jint, jlong, jstring};
|
||||||
|
|
||||||
|
use std::{sync::mpsc, thread, time::Duration};
|
||||||
|
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "system" fn Java_me_ehlxr_HelloWorld_hello(
|
pub extern "system" fn Java_me_ehlxr_HelloWorld_hello(
|
||||||
env: JNIEnv,
|
env: JNIEnv,
|
||||||
@ -14,8 +18,125 @@ pub extern "system" fn Java_me_ehlxr_HelloWorld_hello(
|
|||||||
.into();
|
.into();
|
||||||
|
|
||||||
let output = env
|
let output = env
|
||||||
.new_string(format!("Hello, {}! from Rust...111", input))
|
.new_string(format!("Hello, {}! from Rust..", input))
|
||||||
.expect("Couldn't create java string!");
|
.expect("Couldn't create java string!");
|
||||||
|
|
||||||
output.into_inner()
|
output.into_inner()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub extern "system" fn Java_me_ehlxr_HelloWorld_helloByte(
|
||||||
|
env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
input: jbyteArray,
|
||||||
|
) -> jbyteArray {
|
||||||
|
let _input = env.convert_byte_array(input).unwrap();
|
||||||
|
|
||||||
|
let buf = [1; 2000];
|
||||||
|
let output = env.byte_array_from_slice(&buf).unwrap();
|
||||||
|
output
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub extern "system" fn Java_me_ehlxr_HelloWorld_factAndCallMeBack(
|
||||||
|
env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
n: jint,
|
||||||
|
callback: JObject,
|
||||||
|
) {
|
||||||
|
let i = n as i32;
|
||||||
|
let res: jint = (2..i + 1).product();
|
||||||
|
|
||||||
|
env.call_method(callback, "factCallback", "(I)V", &[res.into()])
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Counter {
|
||||||
|
count: i32,
|
||||||
|
callback: GlobalRef,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Counter {
|
||||||
|
pub fn new(callback: GlobalRef) -> Counter {
|
||||||
|
Counter {
|
||||||
|
count: 0,
|
||||||
|
callback: callback,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn increment(&mut self, env: JNIEnv) {
|
||||||
|
self.count = self.count + 1;
|
||||||
|
env.call_method(
|
||||||
|
&self.callback,
|
||||||
|
"counterCallback",
|
||||||
|
"(I)V",
|
||||||
|
&[self.count.into()],
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern "system" fn Java_me_ehlxr_HelloWorld_counterNew(
|
||||||
|
env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
callback: JObject,
|
||||||
|
) -> jlong {
|
||||||
|
let global_ref = env.new_global_ref(callback).unwrap();
|
||||||
|
let counter = Counter::new(global_ref);
|
||||||
|
|
||||||
|
Box::into_raw(Box::new(counter)) as jlong
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern "system" fn Java_me_ehlxr_HelloWorld_counterIncrement(
|
||||||
|
env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
counter_ptr: jlong,
|
||||||
|
) {
|
||||||
|
let counter = &mut *(counter_ptr as *mut Counter);
|
||||||
|
|
||||||
|
counter.increment(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern "system" fn Java_me_ehlxr_HelloWorld_counterDestroy(
|
||||||
|
_env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
counter_ptr: jlong,
|
||||||
|
) {
|
||||||
|
let _boxed_counter = Box::from_raw(counter_ptr as *mut Counter);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub extern "system" fn Java_me_ehlxr_HelloWorld_asyncComputation(
|
||||||
|
env: JNIEnv,
|
||||||
|
_class: JClass,
|
||||||
|
callback: JObject,
|
||||||
|
) {
|
||||||
|
let jvm = env.get_java_vm().unwrap();
|
||||||
|
|
||||||
|
let callback = env.new_global_ref(callback).unwrap();
|
||||||
|
|
||||||
|
let (tx, rx) = mpsc::channel();
|
||||||
|
|
||||||
|
let _ = thread::spawn(move || {
|
||||||
|
tx.send(()).unwrap();
|
||||||
|
|
||||||
|
let env = jvm.attach_current_thread().unwrap();
|
||||||
|
|
||||||
|
for i in 0..11 {
|
||||||
|
let progress = (i * 10) as jint;
|
||||||
|
env.call_method(&callback, "asyncCallback", "(I)V", &[progress.into()])
|
||||||
|
.unwrap();
|
||||||
|
thread::sleep(Duration::from_millis(100));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
rx.recv().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
#[test]
|
||||||
|
fn it_works() {}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user