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
use std::str::FromStr;
use crate::{reflect::Eval, Value};
use super::*;
#[derive(Debug, Copy, Clone, PartialEq)]
#[repr(u8)]
pub(crate) enum Fun {
IsNone,
IsSome,
UnWrap,
UnWrapOr = 1 << F,
And = (1 << F) + 1,
Or = (1 << F) + 2,
Xor = (1 << F) + 3,
}
use Fun::*;
const F: u8 = 6;
const L: u8 = 1;
impl FromStr for Fun {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
use Fun::*;
match s {
"is_some" => Ok(IsSome),
"is_none" => Ok(IsNone),
"unwrap" => Ok(UnWrap),
"unwrap_or" => Ok(UnWrapOr),
"and" => Ok(And),
"or" => Ok(Or),
"xor" => Ok(Xor),
_ => Err(()),
}
}
}
macro_rules! unpack {
($e:expr) => {
match $e {
Value::Option(x) => *x,
v => Some(v),
}
};
}
impl Eval for Fun {
fn eval(self, stack: &mut Vec<Value>) -> Result<(), ()> {
macro_rules! bool {
($fun:ident) => {{
let e = fun!($fun, unpack, stack);
stack.push(Value::Bool(e));
return Ok(());
}};
}
let e = match self {
And => fun_arg!(and, unpack, stack),
IsNone => bool!(is_none),
IsSome => bool!(is_some),
Or => fun_arg!(or, unpack, stack),
UnWrap => {
let op1 = unpack!(stack.pop().ok_or(())?);
return op1.ok_or(()).map(|e| stack.push(e));
}
UnWrapOr => {
let op2 = stack.pop().ok_or(())?;
let op1 = unpack!(stack.pop().ok_or(())?);
stack.push(op1.unwrap_or(op2));
return Ok(());
}
Xor => fun_arg!(xor, unpack, stack),
};
stack.push(Value::Option(Box::new(e)));
Ok(())
}
}
impl HasArg for Fun {
#[inline]
fn has_arg(self) -> bool {
(self as u8).leading_zeros() as u8 == L
}
}