rlox/interpreter/src/function.rs

104 lines
2.4 KiB
Rust
Raw Normal View History

2023-01-25 19:01:13 +01:00
use std::fmt::{Debug, Display};
use rlox2_frontend::parser::Stmt;
2023-01-25 19:01:13 +01:00
use super::environment::{Environment, Scope};
use super::interpret::EvalResult;
use super::{Runtime, Value};
2023-01-25 19:01:13 +01:00
#[derive(Debug, Clone)]
pub struct LoxFunction {
name: String,
closure: Scope,
param_names: Vec<String>,
body: Box<Stmt>,
}
impl LoxFunction {
pub fn new(name: impl Into<String>, closure: Scope, param_names: Vec<String>, body: Stmt) -> Self {
2023-01-25 19:01:13 +01:00
let name = name.into();
let body = Box::new(body);
LoxFunction {
name,
closure,
2023-01-25 19:01:13 +01:00
param_names,
body,
}
}
pub fn arity(&self) -> usize {
self.param_names.len()
}
pub fn name(&self) -> &str {
&self.name
}
pub fn closure(&self) -> &Scope {
&self.closure
}
pub fn param_names(&self) -> impl Iterator<Item = &String> {
self.param_names.iter()
}
pub fn body(&self) -> &Stmt {
&self.body
}
}
impl Display for LoxFunction {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "<fun {}>", self.name)
}
}
/*====================================================================================================================*/
pub type ExternFunClosure = fn(Vec<Value>, &mut Environment) -> EvalResult<Value>;
#[derive(Clone)]
pub struct LoxExternFunction {
name: String,
closure: ExternFunClosure,
}
impl LoxExternFunction {
pub fn new(name: impl Into<String>, closure: ExternFunClosure) -> Self {
let name = name.into();
LoxExternFunction { name, closure }
}
pub fn register(self, env: &mut Runtime) {
2023-01-25 19:01:13 +01:00
let name = self.name.clone();
let fun = Value::extern_function(self);
env.define_global(name, fun);
2023-01-25 19:01:13 +01:00
}
pub fn call(&self, args: Vec<Value>, env: &mut Environment) -> EvalResult<Value> {
(self.closure)(args, env)
}
}
impl Display for LoxExternFunction {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "<extern fun {}>", self.name)
}
}
impl Debug for LoxExternFunction {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("LoxExternFunction")
.field("name", &self.name)
.field("closure", &"<closure>")
.finish()
}
}
impl PartialEq for LoxExternFunction {
fn eq(&self, other: &Self) -> bool {
self.name == other.name
}
}