cl-interpret: copy-capture closures
This commit is contained in:
@@ -1,7 +1,7 @@
|
||||
use crate::{
|
||||
Callable,
|
||||
convalue::ConValue,
|
||||
env::{Environment, Place},
|
||||
env::Environment,
|
||||
error::{Error, ErrorKind, IResult},
|
||||
function::collect_upvars::CollectUpvars,
|
||||
interpret::Interpret,
|
||||
@@ -11,11 +11,11 @@ use cl_ast::{Sym, ast_visitor::Visit};
|
||||
use std::{collections::HashMap, fmt::Display};
|
||||
|
||||
/// Represents an ad-hoc anonymous function
|
||||
/// which captures surrounding state by reference.
|
||||
/// which captures surrounding state by COPY
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct Closure {
|
||||
decl: cl_ast::Closure,
|
||||
lift: HashMap<Sym, Place>,
|
||||
lift: HashMap<Sym, Option<ConValue>>,
|
||||
}
|
||||
|
||||
impl Closure {
|
||||
@@ -24,7 +24,7 @@ impl Closure {
|
||||
|
||||
impl Closure {
|
||||
pub fn new(env: &mut Environment, decl: &cl_ast::Closure) -> Self {
|
||||
let lift = CollectUpvars::new(env).visit(decl).finish();
|
||||
let lift = CollectUpvars::new(env).visit(decl).finish_copied();
|
||||
Self { decl: decl.clone(), lift }
|
||||
}
|
||||
}
|
||||
@@ -42,8 +42,8 @@ impl Callable for Closure {
|
||||
let mut env = env.frame(Self::NAME);
|
||||
|
||||
// place lifts in scope
|
||||
for (name, place) in lift {
|
||||
env.insert(*name, Some(ConValue::Ref(*place)));
|
||||
for (name, value) in lift.clone() {
|
||||
env.insert(name, value);
|
||||
}
|
||||
|
||||
let mut env = env.frame("args");
|
||||
|
@@ -99,6 +99,11 @@ impl Environment {
|
||||
function.call(self, args)
|
||||
}
|
||||
|
||||
/// Gets all registered globals, bound or unbound.
|
||||
pub fn globals(&self) -> &HashMap<Sym, Option<ConValue>> {
|
||||
&self.global
|
||||
}
|
||||
|
||||
/// Adds builtins
|
||||
///
|
||||
/// # Panics
|
||||
|
@@ -7,12 +7,7 @@ use cl_ast::{
|
||||
use std::collections::{HashMap, HashSet};
|
||||
|
||||
pub fn collect_upvars(f: &Function, env: &Environment) -> super::Upvars {
|
||||
CollectUpvars::new(env)
|
||||
.visit(f)
|
||||
.finish()
|
||||
.into_iter()
|
||||
.map(|(k, v)| (k, env.get_id(v).cloned()))
|
||||
.collect()
|
||||
CollectUpvars::new(env).visit(f).finish_copied()
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
@@ -31,6 +26,14 @@ impl<'env> CollectUpvars<'env> {
|
||||
std::mem::take(&mut self.upvars)
|
||||
}
|
||||
|
||||
pub fn finish_copied(&mut self) -> super::Upvars {
|
||||
let Self { env, upvars, blacklist: _ } = self;
|
||||
std::mem::take(upvars)
|
||||
.into_iter()
|
||||
.map(|(k, v)| (k, env.get_id(v).cloned()))
|
||||
.collect()
|
||||
}
|
||||
|
||||
pub fn add_upvar(&mut self, name: &Sym) {
|
||||
let Self { env, upvars, blacklist } = self;
|
||||
if blacklist.contains(name) || upvars.contains_key(name) {
|
||||
|
Reference in New Issue
Block a user