mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-30 23:31:12 +00:00
68 lines
1.8 KiB
Rust
68 lines
1.8 KiB
Rust
use std::ffi::CString;
|
|
use std::mem::MaybeUninit;
|
|
use std::os::raw::c_char;
|
|
use std::time::SystemTime;
|
|
use RocCallResult::*;
|
|
|
|
extern "C" {
|
|
#[link_name = "closure_1_exposed"]
|
|
fn closure(output: *mut u8) -> ();
|
|
|
|
#[link_name = "closure_1_size"]
|
|
fn closure_size() -> i64;
|
|
}
|
|
|
|
#[no_mangle]
|
|
pub fn rust_main() -> isize {
|
|
println!("Running Roc closure");
|
|
let start_time = SystemTime::now();
|
|
|
|
let size = unsafe { closure_size() } as usize;
|
|
let roc_closure = unsafe {
|
|
let mut output: MaybeUninit<RocCallResult<(fn(i64) -> i64, i64)>> = MaybeUninit::uninit();
|
|
|
|
closure(output.as_mut_ptr() as _);
|
|
|
|
match output.assume_init().into() {
|
|
Ok((function_pointer, closure_data)) => move || function_pointer(closure_data),
|
|
Err(msg) => panic!("Roc failed with message: {}", msg),
|
|
}
|
|
};
|
|
let answer = roc_closure();
|
|
let end_time = SystemTime::now();
|
|
let duration = end_time.duration_since(start_time).unwrap();
|
|
|
|
println!(
|
|
"Roc closure took {:.4} ms to compute this answer: {:?}",
|
|
duration.as_secs_f64() * 1000.0,
|
|
// truncate the answer, so stdout is not swamped
|
|
answer
|
|
);
|
|
|
|
// Exit code
|
|
0
|
|
}
|
|
|
|
#[repr(u64)]
|
|
pub enum RocCallResult<T> {
|
|
Success(T),
|
|
Failure(*mut c_char),
|
|
}
|
|
|
|
impl<T: Sized> Into<Result<T, String>> for RocCallResult<T> {
|
|
fn into(self) -> Result<T, String> {
|
|
match self {
|
|
Success(value) => Ok(value),
|
|
Failure(failure) => Err({
|
|
let raw = unsafe { CString::from_raw(failure) };
|
|
|
|
let result = format!("{:?}", raw);
|
|
|
|
// make sure rust does not try to free the Roc string
|
|
std::mem::forget(raw);
|
|
|
|
result
|
|
}),
|
|
}
|
|
}
|
|
}
|