♻️ refactor: remove special types BogusArg and BogusEnv
This is a milestone towards actually implementing all of Lua.main
parent
062c6e6a2d
commit
5ab30ac5b4
39
src/state.rs
39
src/state.rs
|
@ -1,7 +1,4 @@
|
||||||
use std::{
|
use std::rc::Rc;
|
||||||
collections::HashMap,
|
|
||||||
rc::Rc,
|
|
||||||
};
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
instruction::Instruction,
|
instruction::Instruction,
|
||||||
|
@ -71,15 +68,16 @@ impl Default for State {
|
||||||
impl State {
|
impl State {
|
||||||
pub fn upvalues_from_args <I: Iterator <Item = String>> (args: I) -> Vec <Value>
|
pub fn upvalues_from_args <I: Iterator <Item = String>> (args: I) -> Vec <Value>
|
||||||
{
|
{
|
||||||
let arg: Vec <_> = args.map (|s| s.to_string ()).collect ();
|
let arg = args.map (|s| Value::from (s)).enumerate ();
|
||||||
|
let arg = Value::from_iter (arg.map (|(i, v)| (Value::from (i), v)));
|
||||||
|
|
||||||
let env = HashMap::from_iter ([
|
let env = [
|
||||||
("arg", Value::BogusArg (arg.into ())),
|
("arg", arg),
|
||||||
("print", Value::BogusPrint),
|
("print", Value::BogusPrint),
|
||||||
].map (|(k, v)| (k.to_string (), v)));
|
].into_iter ().map (|(k, v)| (k.to_string (), v));
|
||||||
|
|
||||||
vec! [
|
vec! [
|
||||||
Value::BogusEnv (env.into ()),
|
Value::from_iter (env.into_iter ()),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -272,30 +270,21 @@ impl State {
|
||||||
let b = usize::try_from (*b).unwrap ();
|
let b = usize::try_from (*b).unwrap ();
|
||||||
let c = usize::try_from (*c).unwrap ();
|
let c = usize::try_from (*c).unwrap ();
|
||||||
|
|
||||||
let env = match upvalues.get (b).unwrap () {
|
let table = upvalues.get (b).unwrap ().as_table ().expect ("GetTabUp only works on tables").borrow ();
|
||||||
Value::BogusEnv (x) => x,
|
|
||||||
_ => panic! ("Only allowed upvalue is BogusEnv"),
|
|
||||||
};
|
|
||||||
|
|
||||||
let key = match k.get (c).unwrap () {
|
let key = match k.get (c).unwrap () {
|
||||||
Value::String (s) => s.as_ref (),
|
Value::String (s) => String::from (s.as_ref()),
|
||||||
_ => panic! ("GetTabUp only supports string keys"),
|
_ => panic! ("GetTabUp only supports string keys"),
|
||||||
};
|
};
|
||||||
|
|
||||||
let value = env.get (key).unwrap ();
|
*self.reg_mut (*a) = table.get (key);
|
||||||
*self.reg_mut (*a) = value.clone ();
|
|
||||||
},
|
},
|
||||||
Instruction::GetI (a, b, c) => {
|
Instruction::GetI (a, b, c) => {
|
||||||
let key = usize::try_from (*c).unwrap ();
|
let key = i64::try_from (*c).unwrap ();
|
||||||
|
|
||||||
let table = self.reg (*b);
|
let value = {
|
||||||
let value = match table {
|
let table = self.reg (*b).as_table ().expect ("GetI only works on tables").borrow ();
|
||||||
Value::BogusArg (arg) => arg.get (key).map (|x| x.as_str().into ()).unwrap_or_default(),
|
table.get (key)
|
||||||
Value::Table (t) => {
|
|
||||||
let key = Value::from (i64::try_from (key).unwrap ());
|
|
||||||
t.borrow ().get (key)
|
|
||||||
},
|
|
||||||
_ => unimplemented! (),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
*self.reg_mut (*a) = value;
|
*self.reg_mut (*a) = value;
|
||||||
|
|
44
src/value.rs
44
src/value.rs
|
@ -29,11 +29,9 @@ pub enum Value {
|
||||||
Table (Rc <RefCell <Table>>),
|
Table (Rc <RefCell <Table>>),
|
||||||
|
|
||||||
// These are all bogus, I haven't figured out how to implement
|
// These are all bogus, I haven't figured out how to implement
|
||||||
// tables and function pointers yet
|
// closures yet
|
||||||
|
|
||||||
BogusArg (Rc <Vec <String>>),
|
|
||||||
BogusClosure (Rc <BogusClosure>),
|
BogusClosure (Rc <BogusClosure>),
|
||||||
BogusEnv (Rc <HashMap <String, Value>>),
|
|
||||||
BogusPrint,
|
BogusPrint,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -54,9 +52,7 @@ impl fmt::Display for Value {
|
||||||
Value::String (s) => write! (f, "{}", s),
|
Value::String (s) => write! (f, "{}", s),
|
||||||
Value::Table (t) => write! (f, "table: {:?}", std::rc::Rc::as_ptr (t)),
|
Value::Table (t) => write! (f, "table: {:?}", std::rc::Rc::as_ptr (t)),
|
||||||
|
|
||||||
Value::BogusArg (_) => write! (f, "BogusArg"),
|
|
||||||
Value::BogusClosure (_) => write! (f, "BogusClosure"),
|
Value::BogusClosure (_) => write! (f, "BogusClosure"),
|
||||||
Value::BogusEnv (_) => write! (f, "BogusEnv"),
|
|
||||||
Value::BogusPrint => write! (f, "BogusPrint"),
|
Value::BogusPrint => write! (f, "BogusPrint"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,12 +94,37 @@ impl From <f64> for Value {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From <usize> for Value {
|
||||||
|
fn from (x: usize) -> Self {
|
||||||
|
Self::Integer (i64::try_from (x).unwrap ())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl From <Table> for Value {
|
impl From <Table> for Value {
|
||||||
fn from (x: Table) -> Self {
|
fn from (x: Table) -> Self {
|
||||||
Self::Table (Rc::new (RefCell::new (x)))
|
Self::Table (Rc::new (RefCell::new (x)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FromIterator <(Value, Value)> for Value {
|
||||||
|
fn from_iter <I: IntoIterator <Item=(Value, Value)>> (i: I) -> Self {
|
||||||
|
let table = Table::from_iter (i);
|
||||||
|
Self::from (table)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromIterator <(String, Value)> for Value {
|
||||||
|
fn from_iter <I: IntoIterator <Item=(String, Value)>> (i: I) -> Self {
|
||||||
|
Self::from_iter (i.into_iter ().map (|(s, v)| (Value::from (s), v)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromIterator <Value> for Value {
|
||||||
|
fn from_iter <I: IntoIterator <Item=Value>> (i: I) -> Self {
|
||||||
|
Self::from_iter ((1..).zip (i.into_iter ()).map (|(i, v)| (Value::from (i), v)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Eq for Value {}
|
impl Eq for Value {}
|
||||||
|
|
||||||
impl std::hash::Hash for Value {
|
impl std::hash::Hash for Value {
|
||||||
|
@ -122,9 +143,7 @@ impl std::hash::Hash for Value {
|
||||||
Self::String (x) => x.hash (state),
|
Self::String (x) => x.hash (state),
|
||||||
Self::Table (x) => Rc::as_ptr (&x).hash (state),
|
Self::Table (x) => Rc::as_ptr (&x).hash (state),
|
||||||
|
|
||||||
Self::BogusArg (_) => panic! ("can't hash Bogus values"),
|
|
||||||
Self::BogusClosure (_) => panic! ("can't hash Bogus values"),
|
Self::BogusClosure (_) => panic! ("can't hash Bogus values"),
|
||||||
Self::BogusEnv (_) => panic! ("can't hash Bogus values"),
|
|
||||||
Self::BogusPrint => panic! ("can't hash Bogus values"),
|
Self::BogusPrint => panic! ("can't hash Bogus values"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -205,6 +224,17 @@ impl Table {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FromIterator <(Value, Value)> for Table {
|
||||||
|
fn from_iter<I: IntoIterator<Item = (Value, Value)>> (i: I) -> Self
|
||||||
|
{
|
||||||
|
let hash = i.into_iter ().collect ();
|
||||||
|
Self {
|
||||||
|
array: Default::default (),
|
||||||
|
hash,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg (test)]
|
#[cfg (test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
Loading…
Reference in New Issue