aboutsummaryrefslogtreecommitdiff
path: root/rust/src/core.rs
diff options
context:
space:
mode:
authorJoel Martin <github@martintribe.org>2014-10-27 21:42:46 -0500
committerJoel Martin <github@martintribe.org>2015-01-06 21:59:00 -0600
commit06fef9b51830a2aa696f3e108c56f306967ce465 (patch)
treed3568abe85505a7c825ad96ffabd44adc7b4d456 /rust/src/core.rs
parentb12d98e4e3424be6b1e851c9450e9532910996ec (diff)
downloadmal-06fef9b51830a2aa696f3e108c56f306967ce465.tar.gz
mal-06fef9b51830a2aa696f3e108c56f306967ce465.zip
rust: add atom support. Fix vector params.
Diffstat (limited to 'rust/src/core.rs')
-rw-r--r--rust/src/core.rs60
1 files changed, 59 insertions, 1 deletions
diff --git a/rust/src/core.rs b/rust/src/core.rs
index 5eafa4c..b4abd2f 100644
--- a/rust/src/core.rs
+++ b/rust/src/core.rs
@@ -5,7 +5,7 @@ use std::collections::HashMap;
use std::io::File;
use types::{MalVal,MalRet,err_val,err_str,err_string,
- Nil,Int,Strn,List,Vector,Hash_Map,
+ Nil,Int,Strn,List,Vector,Hash_Map,Atom,
_nil,_true,_false,_int,string,list,func};
use types;
use readline;
@@ -382,6 +382,59 @@ pub fn map(a:Vec<MalVal>) -> MalRet {
}
+// Atom funcions
+fn deref(a:Vec<MalVal>) -> MalRet {
+ if a.len() != 1 {
+ return err_str("Wrong arity to deref call");
+ }
+ match *a[0].clone() {
+ Atom(ref val) => {
+ let val_cell = val.borrow();
+ Ok(val_cell.clone())
+ },
+ _ => err_str("deref called on non-atom"),
+ }
+}
+
+fn reset_bang(a:Vec<MalVal>) -> MalRet {
+ if a.len() != 2 {
+ return err_str("Wrong arity to map call");
+ }
+ let a1 = a[1].clone();
+ match *a[0].clone() {
+ Atom(ref val) => {
+ let mut val_cell = val.borrow_mut();
+ let atm_mv = val_cell.deref_mut();
+ *atm_mv = a1.clone();
+ Ok(a1)
+ },
+ _ => err_str("reset! called on non-atom"),
+ }
+}
+
+fn swap_bang(a:Vec<MalVal>) -> MalRet {
+ if a.len() < 2 {
+ return err_str("Wrong arity to swap_q call");
+ }
+ let f = a[1].clone();
+ match *a[0].clone() {
+ Atom(ref val) => {
+ let mut val_cell = val.borrow_mut();
+ let atm_mv = val_cell.deref_mut();
+ let mut args = a.slice(2,a.len()).to_vec();
+ args.insert(0, atm_mv.clone());
+ match f.apply(args) {
+ Ok(new_mv) => {
+ *atm_mv = new_mv.clone();
+ Ok(new_mv)
+ }
+ Err(e) => Err(e),
+ }
+ },
+ _ => err_str("swap! called on non-atom"),
+ }
+}
+
pub fn ns() -> HashMap<String,MalVal> {
let mut ns: HashMap<String,MalVal> = HashMap::new();;
@@ -435,5 +488,10 @@ pub fn ns() -> HashMap<String,MalVal> {
ns.insert("apply".to_string(), func(apply));
ns.insert("map".to_string(), func(map));
+ ns.insert("atom".to_string(), func(types::atom));
+ ns.insert("deref".to_string(), func(deref));
+ ns.insert("reset!".to_string(), func(reset_bang));
+ ns.insert("swap!".to_string(), func(swap_bang));
+
return ns;
}