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
use std::collections::HashMap;
use gc::Gc;
use context::Context;
use symbol::Symbol;
use util::{as_sym_list, is_shl};
use value::Value;
#[derive(Clone, Debug, Eq, Finalize, Hash, PartialEq, Trace)]
pub enum ModuleHeaderError {
BadImportForm,
BadModuleForm,
NoModuleForm,
}
pub fn parse_module_header<C: 'static + Context>(
mut values: Vec<Gc<Value<C>>>,
) -> Result<(Symbol, Vec<Symbol>, HashMap<Symbol, Vec<Symbol>>, Vec<Gc<Value<C>>>), ModuleHeaderError> {
fn parse_header_form<C: 'static + Context>(
import: bool,
value: Gc<Value<C>>,
) -> Result<(Symbol, Vec<Symbol>), ModuleHeaderError> {
if let Some(mut sym_list) = as_sym_list(value) {
if sym_list.len() >= 2 {
let head = sym_list.remove(0);
if import && head != Symbol::from("import") {
Err(ModuleHeaderError::BadImportForm)
} else if !import && head != Symbol::from("module") {
Err(ModuleHeaderError::BadModuleForm)
} else {
let name = sym_list.remove(0);
Ok((name, sym_list))
}
} else {
Err(if import {
ModuleHeaderError::BadImportForm
} else {
ModuleHeaderError::BadModuleForm
})
}
} else {
Err(if import {
ModuleHeaderError::BadImportForm
} else {
ModuleHeaderError::BadModuleForm
})
}
}
if values.len() == 0 {
return Err(ModuleHeaderError::NoModuleForm);
}
let import_sym = Symbol::from("import");
let (name, exports) = parse_header_form(false, values.remove(0))?;
let end = values
.iter()
.cloned()
.position(|x| !is_shl(import_sym, x))
.unwrap_or_else(|| values.len());
let imports = values
.drain(..end)
.map(|v| parse_header_form(true, v))
.collect::<Result<_, _>>()?;
Ok((name, exports, imports, values))
}