Files
daisy/src/evaluate/evaluate.rs
2023-09-04 13:49:20 -07:00

108 lines
2.7 KiB
Rust

use crate::parser::Expression;
use crate::parser::Operator;
use crate::context::Context;
use crate::parser::LineLocation;
use crate::errors::DaisyError;
use super::operator::eval_operator;
use super::function::eval_function;
pub fn evaluate(
context: &mut Context,
t: &Expression
) -> Result<
Expression,
(LineLocation, DaisyError)
> {
// Keeps track of our position in the expression tree.
// For example, the coordinates [0, 2, 1] are interpreted as follows:
// Start at the root node,
// then move to that node's 0th child,
// then move to that node's 2nd child,
// then move to that node's 1st child.
//
let mut coords: Vec<usize> = Vec::with_capacity(16);
let mut root = t.clone();
// coords points to the *next* node we will move to.
coords.push(0);
// Repeats while we have coordinates to parse.
// Exits when we finish parsing the root node.
loop {
// Current position in the tree
let g = root.get_at_coords_mut(
&coords[0 .. coords.len() - 1]
).unwrap();
// "Move up" step.
// We move up if we're at a leaf or if we're out of children to move down to.
if {
g.is_quantity() ||
g.get_args().is_none() ||
(coords.len() != 0 && (*coords.last().unwrap() >= g.get_args().unwrap().len()))
} {
// If true, move to the next node
let mut move_up = true;
let new = match g {
Expression::Quantity(_, _) => None,
Expression::Tuple(_, _) => None,
Expression::Constant(_, c) => { Some(evaluate(context, &c.value()).unwrap()) },
Expression::Variable(l, s) => {
// Don't move up, re-evaluate
// This makes variables containing floating variables work properly
// (For example, try x = a + 2, a = 2, x. x should evaluate to 4.)
move_up = false;
if !context.is_varible(&s) {
return Err((*l, DaisyError::Undefined(s.clone())));
}
context.get_variable(&s)
},
Expression::Operator(_, Operator::Function(_), _) => { eval_function(g)? },
Expression::Operator(_, _, _) => { eval_operator(context, g)? },
};
if let Some(mut new) = new {
if let Expression::Constant(_,_) = g {
// Fix constant line location.
// Constant expansion does not change the location of a value,
// but other operations might.
new.set_linelocation(&g.get_linelocation());
}
*g = new;
} else {
/*
if let Expression::Quantity(_, _) = g {}
else {
let l = g.get_linelocation();
return Err((l, DaisyError::EvaluationError))
}
*/
// Always move up if we couldn't evaluate this node.
move_up = true;
}
if move_up {
// Move up the tree
coords.pop();
if coords.len() != 0 {
*coords.last_mut().unwrap() += 1;
} else { break; }
}
} else {
// Move down the tree
coords.push(0);
}
}
return Ok(root);
}