1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
use std::any::Any;
use std::sync::{Arc, RwLock};
use std::error::Error as StdError;
use base::ast::{self, Expr, MutVisitor, SpannedExpr};
use base::error::Errors;
use base::fnv::FnvMap;
use base::pos;
use base::symbol::Symbol;
use thread::Thread;
pub type Error = Box<StdError + Send + Sync>;
pub trait Macro: ::mopa::Any + Send + Sync {
fn expand(&self,
env: &mut MacroExpander,
args: &mut [SpannedExpr<Symbol>])
-> Result<SpannedExpr<Symbol>, Error>;
}
mopafy!(Macro);
impl<F: ::mopa::Any + Clone + Send + Sync> Macro for F
where F: Fn(&mut MacroExpander, &mut [SpannedExpr<Symbol>]) -> Result<SpannedExpr<Symbol>, Error>,
{
fn expand(&self,
env: &mut MacroExpander,
args: &mut [SpannedExpr<Symbol>])
-> Result<SpannedExpr<Symbol>, Error> {
self(env, args)
}
}
pub struct MacroEnv {
macros: RwLock<FnvMap<String, Arc<Macro>>>,
}
impl MacroEnv {
pub fn new() -> MacroEnv {
MacroEnv { macros: RwLock::new(FnvMap::default()) }
}
pub fn insert<M>(&self, name: String, mac: M)
where M: Macro + 'static,
{
self.macros.write().unwrap().insert(name, Arc::new(mac));
}
pub fn get(&self, name: &str) -> Option<Arc<Macro>> {
self.macros.read().unwrap().get(name).cloned()
}
pub fn run(&self, vm: &Thread, expr: &mut SpannedExpr<Symbol>) -> Result<(), Errors<Error>> {
let mut expander = MacroExpander::new(vm);
expander.visit_expr(expr);
expander.finish()
}
}
pub struct MacroExpander<'a> {
pub state: FnvMap<String, Box<Any>>,
pub vm: &'a Thread,
pub errors: Errors<Error>,
macros: &'a MacroEnv,
}
impl<'a> MacroExpander<'a> {
pub fn new(vm: &Thread) -> MacroExpander {
MacroExpander {
vm: vm,
state: FnvMap::default(),
macros: vm.get_macros(),
errors: Errors::new(),
}
}
pub fn finish(self) -> Result<(), Errors<Error>> {
if self.errors.has_errors() {
Err(self.errors)
} else {
Ok(())
}
}
pub fn run(&mut self, expr: &mut SpannedExpr<Symbol>) {
self.visit_expr(expr);
}
}
impl<'a> MutVisitor for MacroExpander<'a> {
type Ident = Symbol;
fn visit_expr(&mut self, expr: &mut SpannedExpr<Symbol>) {
let replacement = match expr.value {
Expr::App(ref mut id, ref mut args) => {
match id.value {
Expr::Ident(ref id) if id.name.as_ref().ends_with('!') => {
let name = id.name.as_ref();
match self.macros.get(&name[..name.len() - 1]) {
Some(m) => {
Some(match m.expand(self, args) {
Ok(e) => e,
Err(err) => {
self.errors.push(err);
pos::spanned(expr.span, Expr::Error)
}
})
}
None => None,
}
}
_ => None,
}
}
_ => None,
};
if let Some(mut e) = replacement {
e.span = expr.span;
*expr = e;
}
ast::walk_mut_expr(self, expr);
}
}