extern crate dlopen; #[macro_use] extern crate dlopen_derive; extern crate libc; use dlopen::wrapper::{Container, WrapperApi}; use libc::{c_char, c_int}; use std::ffi::CStr; mod commons; use commons::{example_lib_path, SomeData}; #[derive(WrapperApi)] struct Api<'a> { rust_fun_print_something: fn(), rust_fun_add_one: fn(arg: i32) -> i32, c_fun_print_something_else: unsafe extern "C" fn(), c_fun_add_two: unsafe extern "C" fn(arg: c_int) -> c_int, rust_i32: &'a i32, rust_i32_mut: &'a mut i32, #[dlopen_name = "rust_i32_mut"] rust_i32_ptr: *const i32, c_int: &'a c_int, c_struct: &'a SomeData, rust_str: &'a &'static str, c_const_char_ptr: *const c_char, } //those methods won't be generated impl<'a> Api<'a> { fn rust_i32_ptr(&self) -> *const i32 { self.rust_i32_ptr } fn c_const_str(&self) -> &CStr { unsafe { CStr::from_ptr(self.c_const_char_ptr) } } } #[test] fn open_play_close_wrapper_api() { let lib_path = example_lib_path(); let mut cont: Container = unsafe { Container::load(lib_path) }.expect("Could not open library or load symbols"); cont.rust_fun_print_something(); //should not crash assert_eq!(cont.rust_fun_add_one(5), 6); unsafe { cont.c_fun_print_something_else() }; //should not crash assert_eq!(unsafe { cont.c_fun_add_two(2) }, 4); assert_eq!(43, *cont.rust_i32()); assert_eq!(42, *cont.rust_i32_mut_mut()); *cont.rust_i32_mut_mut() = 55; //should not crash assert_eq!(55, unsafe { *cont.rust_i32_ptr() }); //the same with C assert_eq!(45, *cont.c_int()); //now static c struct assert_eq!(1, cont.c_struct().first); assert_eq!(2, cont.c_struct().second); //let's play with strings assert_eq!("Hello!", *cont.rust_str()); let converted = cont.c_const_str().to_str().unwrap(); assert_eq!(converted, "Hi!"); }