Change representation of variables

This commit is contained in:
greg 2016-12-29 12:21:14 -08:00
parent 29d9e50311
commit ed9d1312d1

View File

@ -9,41 +9,31 @@ type Reduction<T> = (T, Option<SideEffect>);
enum SideEffect { enum SideEffect {
Print(String), Print(String),
Bundle(Vec<SideEffect>), Bundle(Vec<SideEffect>),
AddBinding(String, Expression),
} }
struct Varmap { struct EnvFrame {
map: HashMap<String, Expression>, functions: HashMap<String, Function>,
variables: HashMap<String, Expression>,
} }
impl Varmap { impl EnvFrame {
fn new() -> Varmap { fn new() -> EnvFrame {
Varmap { map: HashMap::new() } EnvFrame {
functions: HashMap::new(),
variables: HashMap::new(),
} }
}
struct Funcmap {
map: HashMap<String, Function>,
}
impl Funcmap {
fn new() -> Funcmap {
let map = HashMap::new();
Funcmap { map: map }
} }
} }
pub struct Evaluator { pub struct Evaluator {
varmap: Varmap, frames: Vec<EnvFrame>,
funcmap: Funcmap,
frames: Vec<Varmap>,
} }
impl Evaluator { impl Evaluator {
pub fn new() -> Evaluator { pub fn new() -> Evaluator {
Evaluator { Evaluator {
varmap: Varmap::new(), frames: vec!(EnvFrame::new()),
funcmap: Funcmap::new(),
frames: Vec::new(),
} }
} }
@ -55,28 +45,36 @@ impl Evaluator {
fn add_binding(&mut self, var: String, value: Expression) { fn add_binding(&mut self, var: String, value: Expression) {
match self.frames.last_mut() { match self.frames.last_mut() {
Some(frame) => frame.map.insert(var, value), Some(frame) => frame.variables.insert(var, value),
None => self.varmap.map.insert(var, value), None => panic!("Evaluator should ensure that frames always has at least one element"),
}; };
} }
fn lookup_binding(&mut self, var: String) -> Option<Expression> { fn lookup_binding(&self, var: String) -> Option<Expression> {
for frame in self.frames.iter().rev() { for frame in self.frames.iter().rev() {
match frame.map.get(&var) { match frame.variables.get(&var) {
None => (), None => (),
Some(expr) => return Some(expr.clone()), Some(expr) => return Some(expr.clone()),
} }
} }
None
self.varmap.map.get(&var).map(|expr| expr.clone())
} }
fn add_function(&mut self, name: String, function: Function) { fn add_function(&mut self, name: String, function: Function) {
self.funcmap.map.insert(name, function); match self.frames.last_mut() {
Some(frame) => frame.functions.insert(name, function),
None => panic!("Evaluator should ensure that frames always has at least one element"),
};
} }
fn lookup_function(&self, name: String) -> Option<Function> { fn lookup_function(&self, name: String) -> Option<Function> {
self.funcmap.map.get(&name).map(|x| x.clone()) for frame in self.frames.iter().rev() {
match frame.functions.get(&name) {
None => (),
Some(function) => return Some(function.clone()),
}
}
None
} }
} }
@ -135,6 +133,9 @@ impl Evaluator {
self.perform_side_effect(side_effect); self.perform_side_effect(side_effect);
} }
} }
AddBinding(var, value) => {
self.add_binding(var, value);
}
} }
} }
@ -179,10 +180,10 @@ impl Evaluator {
if op == "=" { if op == "=" {
match left { match left {
Variable(var) => { Variable(var) => {
self.add_binding(var, right); let binding = SideEffect::AddBinding(var, right);
return (Null, None); //TODO variable binding should be an effect return (Null, Some(binding));
} }
_ => (), _ => return (Null, None)
} }
} }
@ -248,15 +249,6 @@ impl Evaluator {
_ => Null, _ => Null,
} }
} }
"=" => {
match (left, right) {
(Variable(var), right) => {
self.add_binding(var, right);
Null
}
_ => Null,
}
}
_ => Null, _ => Null,
} }
} }
@ -286,9 +278,9 @@ impl Evaluator {
return (Null, None); return (Null, None);
} }
let mut frame: Varmap = Varmap::new(); let mut frame = EnvFrame::new();
for (binding, expr) in function.prototype.parameters.iter().zip(arguments.iter()) { for (binding, expr) in function.prototype.parameters.iter().zip(arguments.iter()) {
frame.map.insert(binding.clone(), expr.clone()); frame.variables.insert(binding.clone(), expr.clone());
} }
self.frames.push(frame); self.frames.push(frame);