schala/schala-lang/src/eval.rs

480 lines
15 KiB
Rust
Raw Normal View History

2017-10-13 03:05:18 -07:00
use std::rc::Rc;
2018-03-03 13:26:22 -08:00
use std::fmt::Write;
2018-05-12 02:27:54 -07:00
use std::io;
2018-03-03 13:26:22 -08:00
use itertools::Itertools;
2018-05-11 01:56:12 -07:00
use util::StateStack;
2018-05-11 01:30:02 -07:00
use ast_reducing::{ReducedAST, Stmt, Expr, Lit, Func};
2017-09-30 23:30:02 -07:00
pub struct State<'a> {
2018-05-11 01:56:12 -07:00
values: StateStack<'a, Rc<String>, ValueEntry>
2017-10-13 03:05:18 -07:00
}
2018-05-11 01:56:12 -07:00
2018-05-12 01:18:44 -07:00
macro_rules! builtin_binding {
($name:expr, $values:expr) => {
$values.insert(Rc::new(format!($name)), ValueEntry::Binding { constant: true, val: Expr::Func(Func::BuiltIn(Rc::new(format!($name)))) });
}
}
2018-05-11 01:56:12 -07:00
impl<'a> State<'a> {
pub fn new() -> State<'a> {
2018-05-12 01:18:44 -07:00
let mut values = StateStack::new(Some(format!("global")));
builtin_binding!("print", values);
builtin_binding!("println", values);
2018-05-12 02:27:54 -07:00
builtin_binding!("getline", values);
2018-05-12 01:18:44 -07:00
State { values }
2018-05-11 01:56:12 -07:00
}
2018-05-11 02:33:19 -07:00
pub fn debug_print(&self) -> String {
format!("Values: {:?}", self.values)
}
2018-05-11 01:56:12 -07:00
}
2018-05-11 01:57:29 -07:00
#[derive(Debug)]
enum ValueEntry {
Binding {
2018-05-11 02:24:38 -07:00
constant: bool,
2018-05-11 01:57:29 -07:00
val: /*FullyEvaluatedExpr*/ Expr,
}
}
type EvalResult<T> = Result<T, String>;
2017-10-01 19:09:55 -07:00
2018-05-11 01:56:12 -07:00
/*
2017-10-13 03:05:18 -07:00
#[derive(Debug, PartialEq, Clone)]
2017-10-01 19:09:55 -07:00
enum FullyEvaluatedExpr {
UnsignedInt(u64),
SignedInt(i64),
Float(f64),
Str(String),
Bool(bool),
2017-10-23 01:54:02 -07:00
FuncLit(Rc<String>),
2017-10-13 03:05:18 -07:00
Custom {
string_rep: Rc<String>,
2017-10-23 00:22:25 -07:00
},
Tuple(Vec<FullyEvaluatedExpr>),
2018-03-08 12:42:05 -08:00
List(Vec<FullyEvaluatedExpr>)
2017-10-01 19:09:55 -07:00
}
impl FullyEvaluatedExpr {
fn to_string(&self) -> String {
use self::FullyEvaluatedExpr::*;
match self {
&UnsignedInt(ref n) => format!("{}", n),
&SignedInt(ref n) => format!("{}", n),
&Float(ref f) => format!("{}", f),
&Str(ref s) => format!("\"{}\"", s),
&Bool(ref b) => format!("{}", b),
&Custom { ref string_rep } => format!("{}", string_rep),
2018-03-03 13:26:22 -08:00
&Tuple(ref items) => {
let mut buf = String::new();
write!(buf, "(").unwrap();
for term in items.iter().map(|e| Some(e)).intersperse(None) {
match term {
Some(e) => write!(buf, "{}", e.to_string()).unwrap(),
None => write!(buf, ", ").unwrap(),
};
}
write!(buf, ")").unwrap();
buf
},
&FuncLit(ref name) => format!("<function {}>", name),
2018-03-08 12:42:05 -08:00
&List(ref items) => {
let mut buf = String::new();
write!(buf, "[").unwrap();
for term in items.iter().map(|e| Some(e)).intersperse(None) {
match term {
Some(e) => write!(buf, "{}", e.to_string()).unwrap(),
None => write!(buf, ", ").unwrap()
}
}
write!(buf, "]").unwrap();
buf
}
}
}
}
*/
/*
pub fn new_with_parent(parent: &'a State<'a>) -> State<'a> {
State { parent_frame: Some(parent), values: HashMap::new() }
2017-09-30 23:30:02 -07:00
}
pub fn evaluate(&mut self, ast: AST) -> Vec<Result<String, String>> {
2017-10-01 19:29:05 -07:00
let mut acc = vec![];
2017-10-01 12:55:28 -07:00
for statement in ast.0 {
2017-10-01 19:09:55 -07:00
match self.eval_statement(statement) {
Ok(output) => {
if let Some(fully_evaluated) = output {
acc.push(Ok(fully_evaluated.to_string()));
2017-10-01 19:09:55 -07:00
}
},
Err(error) => {
acc.push(Err(format!("Eval error: {}", error)));
2017-10-01 19:09:55 -07:00
return acc;
},
2017-10-01 12:55:28 -07:00
}
}
2017-10-01 19:29:05 -07:00
acc
2017-10-01 12:55:28 -07:00
}
}
impl<'a> State<'a> {
fn eval_statement(&mut self, statement: Statement) -> EvalResult<Option<FullyEvaluatedExpr>> {
Ok(match statement {
Statement::ExpressionStatement(expr) => Some(self.eval_expr(expr)?),
Statement::Declaration(decl) => { self.eval_decl(decl)?; None }
})
2017-10-01 12:55:28 -07:00
}
2017-10-13 03:05:18 -07:00
fn eval_decl(&mut self, decl: Declaration) -> EvalResult<()> {
use self::Declaration::*;
use self::Variant::*;
match decl {
FuncDecl(signature, statements) => {
let name = signature.name;
2018-02-26 19:55:27 -08:00
let param_names: Vec<Rc<String>> = signature.params.iter().map(|fp| fp.0.clone()).collect();
2018-03-01 23:13:32 -08:00
self.insert(name, ValueEntry::Function { body: statements.clone(), param_names });
2017-10-13 03:05:18 -07:00
},
TypeDecl(_name, body) => {
for variant in body.0.iter() {
match variant {
2018-03-01 23:13:32 -08:00
&UnitStruct(ref name) => self.insert(name.clone(),
2017-10-13 03:05:18 -07:00
ValueEntry::Binding { val: FullyEvaluatedExpr::Custom { string_rep: name.clone() } }),
2017-10-23 01:54:02 -07:00
&TupleStruct(ref _name, ref _args) => unimplemented!(),
&Record(ref _name, ref _fields) => unimplemented!(),
2017-10-13 03:05:18 -07:00
};
}
},
2018-03-03 00:28:52 -08:00
Binding { name, expr, ..} => {
2018-02-26 18:18:42 -08:00
let val = self.eval_expr(expr)?;
2018-03-01 23:13:32 -08:00
self.insert(name.clone(), ValueEntry::Binding { val });
2018-02-26 18:18:42 -08:00
},
2017-10-13 03:05:18 -07:00
_ => return Err(format!("Declaration evaluation not yet implemented"))
}
Ok(())
2017-09-30 23:30:02 -07:00
}
2017-10-01 00:48:08 -07:00
2017-10-01 19:09:55 -07:00
fn eval_expr(&mut self, expr: Expression) -> EvalResult<FullyEvaluatedExpr> {
2017-10-01 12:55:28 -07:00
use self::ExpressionType::*;
2017-10-01 19:09:55 -07:00
use self::FullyEvaluatedExpr::*;
2017-10-01 12:55:28 -07:00
let expr_type = expr.0;
2017-10-01 19:09:55 -07:00
match expr_type {
IntLiteral(n) => Ok(UnsignedInt(n)),
FloatLiteral(f) => Ok(Float(f)),
StringLiteral(s) => Ok(Str(s.to_string())),
BoolLiteral(b) => Ok(Bool(b)),
2017-10-02 20:11:27 -07:00
PrefixExp(op, expr) => self.eval_prefix_exp(op, expr),
BinExp(op, lhs, rhs) => self.eval_binexp(op, lhs, rhs),
Value(name) => self.eval_value(name),
2017-10-23 00:22:25 -07:00
TupleLiteral(expressions) => {
let mut evals = Vec::new();
for expr in expressions {
match self.eval_expr(expr) {
Ok(fully_evaluated) => evals.push(fully_evaluated),
error => return error,
}
}
Ok(Tuple(evals))
}
2018-02-26 19:55:27 -08:00
Call { f, arguments } => {
let mut evaled_arguments = Vec::new();
for arg in arguments.into_iter() {
evaled_arguments.push(self.eval_expr(arg)?);
}
self.eval_application(*f, evaled_arguments)
},
2018-03-08 00:32:19 -08:00
Index { box indexee, indexers } => {
2018-03-07 21:46:21 -08:00
let evaled = self.eval_expr(indexee)?;
match evaled {
Tuple(mut exprs) => {
let len = indexers.len();
if len == 1 {
2018-03-08 00:32:19 -08:00
let idx = indexers.into_iter().nth(0).unwrap();
2018-03-07 21:46:21 -08:00
match self.eval_expr(idx)? {
UnsignedInt(n) if (n as usize) < exprs.len() => Ok(exprs.drain(n as usize..).next().unwrap()),
UnsignedInt(n) => Err(format!("Index {} out of range", n)),
other => Err(format!("{:?} is not an unsigned integer", other)),
}
} else {
Err(format!("Tuple index must be one integer"))
}
},
_ => Err(format!("Bad index expression"))
}
},
2018-03-08 12:42:05 -08:00
ListLiteral(items) => Ok(List(items.into_iter().map(|item| self.eval_expr(item)).collect::<Result<Vec<_>,_>>()?)),
2017-10-23 00:22:25 -07:00
x => Err(format!("Unimplemented thing {:?}", x)),
2017-10-01 19:09:55 -07:00
}
2017-10-01 12:55:28 -07:00
}
2017-10-02 20:11:27 -07:00
2018-02-26 19:55:27 -08:00
fn eval_application(&mut self, f: Expression, arguments: Vec<FullyEvaluatedExpr>) -> EvalResult<FullyEvaluatedExpr> {
2017-10-23 01:54:02 -07:00
use self::ExpressionType::*;
match f {
Expression(Value(ref identifier), _) if self.is_builtin(identifier) => self.eval_builtin(identifier, arguments),
Expression(Value(identifier), _) => {
2018-03-01 23:13:32 -08:00
match self.lookup(&identifier) {
2018-02-26 19:55:27 -08:00
Some(&ValueEntry::Function { ref body, ref param_names }) => {
if arguments.len() != param_names.len() {
return Err(format!("Wrong number of arguments for the function"));
}
2018-02-26 19:57:46 -08:00
let mut new_state = State::new_with_parent(self);
let sub_ast = body.clone();
2018-02-26 19:55:27 -08:00
for (param, val) in param_names.iter().zip(arguments.into_iter()) {
2018-03-01 23:13:32 -08:00
new_state.insert(param.clone(), ValueEntry::Binding { val });
2018-02-26 19:55:27 -08:00
}
2018-02-24 14:31:04 -08:00
let mut ret: Option<FullyEvaluatedExpr> = None;
for statement in sub_ast.into_iter() {
ret = new_state.eval_statement(statement)?;
}
Ok(ret.unwrap_or(FullyEvaluatedExpr::Custom { string_rep: Rc::new("()".to_string()) }))
2017-10-23 01:54:02 -07:00
},
_ => Err(format!("Function {} not found", identifier)),
}
},
x => Err(format!("Trying to apply {:?} which is not a function", x)),
}
}
fn is_builtin(&self, name: &Rc<String>) -> bool {
match &name.as_ref()[..] {
"print" | "println" => true,
_ => false
}
}
fn eval_builtin(&mut self, name: &Rc<String>, args: Vec<FullyEvaluatedExpr>) -> EvalResult<FullyEvaluatedExpr> {
use self::FullyEvaluatedExpr::*;
match &name.as_ref()[..] {
"print" => {
for arg in args {
print!("{}", arg.to_string());
}
Ok(Tuple(vec![]))
},
"println" => {
for arg in args {
println!("{}", arg.to_string());
}
Ok(Tuple(vec![]))
},
_ => unreachable!()
}
}
2017-10-13 03:05:18 -07:00
fn eval_value(&mut self, name: Rc<String>) -> EvalResult<FullyEvaluatedExpr> {
use self::ValueEntry::*;
2018-03-01 23:13:32 -08:00
match self.lookup(&name) {
2017-10-13 03:05:18 -07:00
None => return Err(format!("Value {} not found", *name)),
Some(lookup) => match lookup {
&Binding { ref val } => Ok(val.clone()),
&Function { .. } => Ok(FullyEvaluatedExpr::FuncLit(name.clone()))
2017-10-13 03:05:18 -07:00
}
}
}
2017-10-01 12:55:28 -07:00
}
*/
2018-05-09 02:27:57 -07:00
/* BELOW HERE NEW STUFF */
2018-05-11 00:25:43 -07:00
impl Expr {
fn to_repl(&self) -> String {
2018-05-11 01:07:18 -07:00
use self::Lit::*;
2018-05-11 23:23:54 -07:00
use self::Func::*;
2018-05-11 01:07:18 -07:00
match self {
Expr::Lit(ref l) => match l {
Nat(n) => format!("{}", n),
Int(i) => format!("{}", i),
Float(f) => format!("{}", f),
Bool(b) => format!("{}", b),
StringLit(s) => format!("{}", s),
},
2018-05-11 23:23:54 -07:00
Expr::Func(f) => match f {
BuiltIn(name) => format!("<built-in function {}>", name),
UserDefined { name: None, .. } => format!("<function>"),
UserDefined { name: Some(name), .. } => format!("<function {}>", name),
},
2018-05-11 01:07:18 -07:00
_ => format!("{:?}", self),
}
2018-05-11 00:25:43 -07:00
}
}
2018-05-09 02:27:57 -07:00
impl<'a> State<'a> {
2018-05-12 02:20:50 -07:00
pub fn evaluate(&mut self, ast: ReducedAST, repl: bool) -> Vec<Result<String, String>> {
2018-05-09 03:38:02 -07:00
use ast_reducing::*;
let mut acc = vec![];
for statement in ast.0 {
2018-05-11 00:25:43 -07:00
match self.statement(statement) {
Ok(Some(ref output)) if repl => acc.push(Ok(output.to_repl())),
Ok(_) => (),
2018-05-09 03:38:02 -07:00
Err(error) => {
acc.push(Err(format!("Eval error: {}", error)));
return acc;
},
}
}
acc
}
2018-05-11 01:57:29 -07:00
fn statement(&mut self, stmt: Stmt) -> EvalResult<Option<Expr>> {
2018-05-11 01:07:18 -07:00
match stmt {
2018-05-11 02:24:38 -07:00
Stmt::Binding { name, constant, expr } => {
let val = self.expression(expr)?;
self.values.insert(name.clone(), ValueEntry::Binding { constant, val });
2018-05-11 01:07:18 -07:00
Ok(None)
},
2018-05-11 01:30:02 -07:00
Stmt::Expr(expr) => Ok(Some(self.expression(expr)?)),
2018-05-11 01:07:18 -07:00
}
}
2018-05-11 01:57:29 -07:00
fn expression(&mut self, expr: Expr) -> EvalResult<Expr> {
2018-05-11 01:07:18 -07:00
use self::Expr::*;
match expr {
2018-05-11 01:30:02 -07:00
literal @ Lit(_) => Ok(literal),
Call { box f, args } => {
let f = match self.expression(f)? {
Func(f) => f,
other => return Err(format!("Tried to call {:?} which is not a function", other)),
};
self.apply_function(f, args)
},
2018-05-11 17:24:11 -07:00
Val(v) => self.value(v),
2018-05-11 23:23:54 -07:00
func @ Func(_) => Ok(func),
2018-05-12 03:51:42 -07:00
Assign { box val, box expr } => {
let name = match val {
Expr::Val(name) => name,
_ => return Err(format!("Trying to assign to a non-value")),
};
let constant = match self.values.lookup(&name) {
None => return Err(format!("Runtime error: {} is undefined", name)),
Some(ValueEntry::Binding { constant, .. }) => constant.clone(),
};
if constant {
return Err(format!("Runtime error: trying to update {}, a non-mutable binding", name));
}
let val = self.expression(expr)?;
self.values.insert(name.clone(), ValueEntry::Binding { constant: false, val });
Ok(Expr::Unit)
},
2018-05-11 23:23:54 -07:00
e => Err(format!("Expr {:?} eval not implemented", e))
2018-05-11 01:07:18 -07:00
}
2018-05-09 02:27:57 -07:00
}
2018-05-11 01:30:02 -07:00
2018-05-11 01:57:29 -07:00
fn apply_function(&mut self, f: Func, args: Vec<Expr>) -> EvalResult<Expr> {
2018-05-11 01:30:02 -07:00
match f {
Func::BuiltIn(sigil) => self.apply_builtin(sigil, args),
2018-05-12 00:59:50 -07:00
Func::UserDefined { params, body, name } => {
if params.len() != args.len() {
return Err(format!("Runtime error: calling a {}-argument function with {} args", params.len(), args.len()))
}
let mut func_state = State { values: self.values.new_frame(name.map(|n| format!("{}", n))) };
for (param, val) in params.into_iter().zip(args.into_iter()) {
func_state.values.insert(param, ValueEntry::Binding { constant: true, val });
}
// TODO figure out function return semantics
let mut ret = None;
for stmt in body {
ret = func_state.statement(stmt)?;
}
Ok(ret.unwrap_or(Expr::Unit))
2018-05-11 01:30:02 -07:00
}
}
}
2018-05-11 01:57:29 -07:00
fn apply_builtin(&mut self, name: Rc<String>, args: Vec<Expr>) -> EvalResult<Expr> {
2018-05-11 01:30:02 -07:00
use self::Expr::*;
use self::Lit::*;
let evaled_args: Result<Vec<Expr>, String> = args.into_iter().map(|arg| self.expression(arg)).collect();
let evaled_args = evaled_args?;
Ok(match (name.as_str(), evaled_args.as_slice()) {
2018-05-11 01:46:42 -07:00
/* binops */
2018-05-11 01:30:02 -07:00
("+", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l + r)),
2018-05-11 01:35:39 -07:00
("++", &[Lit(StringLit(ref s1)), Lit(StringLit(ref s2))]) => Lit(StringLit(Rc::new(format!("{}{}", s1, s2)))),
("-", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l - r)),
("*", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l * r)),
("/", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Float((l as f64)/ (r as f64))),
("//", &[Lit(Nat(l)), Lit(Nat(r))]) => if r == 0 {
2018-05-11 01:30:02 -07:00
return Err(format!("Runtime error: divide by zero"));
} else {
2018-05-11 01:35:39 -07:00
Lit(Nat(l / r))
2018-05-11 01:30:02 -07:00
},
2018-05-11 01:35:39 -07:00
("%", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l % r)),
("^", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l ^ r)),
("&", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l & r)),
("|", &[Lit(Nat(l)), Lit(Nat(r))]) => Lit(Nat(l | r)),
2018-05-11 01:46:42 -07:00
/* prefix ops */
("!", &[Lit(Bool(true))]) => Lit(Bool(false)),
("!", &[Lit(Bool(false))]) => Lit(Bool(true)),
("-", &[Lit(Nat(n))]) => Lit(Int(-1*(n as i64))),
("-", &[Lit(Int(n))]) => Lit(Int(-1*(n as i64))),
("+", &[Lit(Int(n))]) => Lit(Int(n)),
("+", &[Lit(Nat(n))]) => Lit(Nat(n)),
2018-05-12 01:18:44 -07:00
2018-05-12 03:51:42 -07:00
/* builtin functions */
2018-05-12 01:18:44 -07:00
("print", &[ref anything]) => {
print!("{}", anything.to_repl());
Expr::Unit
},
("println", &[ref anything]) => {
println!("{}", anything.to_repl());
Expr::Unit
},
2018-05-12 02:27:54 -07:00
("getline", &[]) => {
let mut buf = String::new();
io::stdin().read_line(&mut buf).expect("Error readling line in 'getline'");
Lit(StringLit(Rc::new(buf)))
},
2018-05-12 03:51:42 -07:00
(x, args) => return Err(format!("Runtime error: bad or unimplemented builtin {:?} | {:?}", x, args)),
2018-05-11 01:30:02 -07:00
})
}
2018-05-11 17:24:11 -07:00
fn value(&mut self, name: Rc<String>) -> EvalResult<Expr> {
use self::ValueEntry::*;
match self.values.lookup(&name) {
None => return Err(format!("Value {} not found", *name)),
Some(lookup) => match lookup {
2018-05-11 23:23:54 -07:00
Binding { val, .. } => Ok(
if let Expr::Func(Func::UserDefined { name: None, params, body }) = val {
Expr::Func(Func::UserDefined { name: Some(name.clone()), params: params.clone(), body: body.clone() }) //TODO here is unnecessary cloning
} else {
val.clone()
2018-05-12 03:51:42 -07:00
}
)
2018-05-11 17:24:11 -07:00
}
}
2018-05-11 23:23:54 -07:00
}
2018-05-09 02:27:57 -07:00
}
2018-05-12 02:18:34 -07:00
#[cfg(test)]
mod eval_tests {
use tokenizing::tokenize;
use parsing::parse;
use eval::State;
macro_rules! fresh_env {
($string:expr, $correct:expr) => {
let mut state = State::new();
2018-05-12 02:20:50 -07:00
let all_output = state.evaluate(parse(tokenize($string)).0.unwrap().reduce(), true);
2018-05-12 03:51:42 -07:00
let ref output = all_output.last().unwrap();
assert_eq!(**output, Ok($correct.to_string()));
2018-05-12 02:18:34 -07:00
}
}
#[test]
fn test_basic_eval() {
fresh_env!("1 + 2", "3");
2018-05-12 03:51:42 -07:00
fresh_env!("var a = 1; a = 2", "Unit");
fresh_env!("var a = 1; a = 2; a", "2");
2018-05-12 02:18:34 -07:00
}
}