diff options
Diffstat (limited to 'src/guile/guile.rs')
-rw-r--r-- | src/guile/guile.rs | 76 |
1 files changed, 36 insertions, 40 deletions
diff --git a/src/guile/guile.rs b/src/guile/guile.rs index 4a0e25b..36df62a 100644 --- a/src/guile/guile.rs +++ b/src/guile/guile.rs @@ -2,34 +2,20 @@ use std::ffi::{CStr, CString}; use std::os::raw::{c_char, c_int, c_void}; #[link(name = "guile-3.0")] -#[allow(dead_code)] extern "C" { fn scm_init_guile(); - fn scm_c_define_gsubr( - name: *const c_char, - req: c_int, - opt: c_int, - rst: c_int, - func: extern "C" fn() -> *mut c_void, - ); fn scm_c_eval_string(expr: *const c_char) -> *mut c_void; fn scm_to_locale_string(scm_obj: *mut c_void) -> *const c_char; - fn scm_c_write(scm_obj: *mut c_void, port: *mut c_void); - fn scm_open_output_string() -> *mut c_void; - fn scm_get_output_string(port: *mut c_void) -> *mut c_void; fn scm_is_string(scm_obj: *mut c_void) -> c_int; - fn scm_is_number(scm_obj: *mut c_void) -> c_int; fn scm_object_to_string(scm_obj: *mut c_void, printer: *mut c_void) -> *mut c_void; } #[link(name = "guiledefs")] extern "C" { pub static scm_undefined: *mut c_void; -} - -extern "C" fn my_rust_function() -> *mut c_void { - println!("Hello from Rust!"); - unsafe { scm_undefined } + pub static scm_unspecified: *mut c_void; + fn defs_scm_is_eq(x: *mut c_void, y: *mut c_void) -> c_int; + fn defs_scm_define_string(name: *const c_char, value: *const c_char); } /// Convert a scm object into a string using Guile-builtins @@ -38,16 +24,14 @@ fn string_from_scm(scm_obj: *mut c_void) -> Result<String, ()> { if scm_obj.is_null() { return Err(()); } - if scm_is_string(scm_obj) != 0 { + if defs_scm_is_eq(scm_obj, scm_unspecified) != 0 { + return Ok(String::new()); + } else if scm_is_string(scm_obj) != 0 { let res_str = CStr::from_ptr(scm_to_locale_string(scm_obj)) .to_string_lossy() .into_owned(); return Ok(res_str); } else { - let port = scm_open_output_string(); - if port.is_null() { - return Err(()); - } let res = scm_object_to_string(scm_obj, scm_undefined); if res.is_null() { return Err(()); @@ -60,27 +44,39 @@ fn string_from_scm(scm_obj: *mut c_void) -> Result<String, ()> { } } -pub fn smp_guile_init() {} +#[derive(Clone)] +pub struct Guile {} -pub fn testguile() { - unsafe { - scm_init_guile(); +impl Guile { + pub fn new() -> Self { + unsafe { + scm_init_guile(); + } + Guile {} + } - let func_name = CString::new("my-rust-function").unwrap(); - scm_c_define_gsubr( - func_name.as_ptr(), - 0, // required arguments - 0, // optional arguments - 0, // rest arguments - my_rust_function, - ); + pub fn evaluate_expression(&self, expr: &str) -> Result<String, ()> { + unsafe { + let c_expr = CString::new(expr).map_err(|_| ())?; + let result = scm_c_eval_string(c_expr.as_ptr()); + string_from_scm(result) + } + } - let expr = CString::new("(+ 2 3)").unwrap(); - let result = scm_c_eval_string(expr.as_ptr()); - println!("{:?}", string_from_scm(result)); + pub fn define_string(&self, name: &str, value: &str) { + let c_name = CString::new(name).expect("CString::new failed"); + let c_value = CString::new(value).expect("CString::new failed"); + unsafe { + defs_scm_define_string(c_name.as_ptr(), c_value.as_ptr()); + } + } +} - let expr = CString::new("(my-rust-function)").unwrap(); - let result = scm_c_eval_string(expr.as_ptr()); - println!("{:?}", string_from_scm(result)); +impl Drop for Guile { + fn drop(&mut self) { + if let Err(e) = self.evaluate_expression("(exit)") { + panic!("Error while exiting {:#?}", e); + } } } + |