replace mutex with rwlock

Signed-off-by: Ava Hahn <ava@aidanis.online>
This commit is contained in:
Ava Hahn 2023-02-20 19:42:48 -08:00
parent c7d0bba928
commit e055f26e90
Signed by untrusted user who does not match committer: affine
GPG key ID: 3A4645B8CF806069
5 changed files with 72 additions and 54 deletions

View file

@ -36,11 +36,10 @@ pub fn eval (
let mut cdr = Box::from(Ctr::None); let mut cdr = Box::from(Ctr::None);
// lets me redirect the input // lets me redirect the input
let table_handle = SYM_TABLE.read().unwrap();
let mut arg_car = &ast.car; let mut arg_car = &ast.car;
let mut arg_cdr = &ast.cdr; let mut arg_cdr = &ast.cdr;
let table_handle = SYM_TABLE.lock().unwrap();
// iterate over ast and build out ret // iterate over ast and build out ret
let mut none = false; let mut none = false;
while !none { while !none {
@ -68,7 +67,6 @@ pub fn eval (
} }
} }
// im tired please simplify this
Ctr::Symbol(_) => { Ctr::Symbol(_) => {
if simplify_function_branches && first { if simplify_function_branches && first {
if let Some(func) = prefetched_function { if let Some(func) = prefetched_function {

View file

@ -45,7 +45,7 @@ pub static LIB_STORE_NO_ENV: Symbol = Symbol {
// TODO : declare function if arg list is long enough // TODO : declare function if arg list is long enough
fn _store_callback (ast: &Seg, env_cfg: bool) -> Ctr { fn _store_callback (ast: &Seg, env_cfg: bool) -> Ctr {
let mut table_handle = SYM_TABLE.lock().unwrap(); let mut table_handle = SYM_TABLE.write().unwrap();
let is_var = ast.len() == 2; let is_var = ast.len() == 2;
if let Ctr::Symbol(ref identifier) = *ast.car { if let Ctr::Symbol(ref identifier) = *ast.car {
match &*ast.cdr { match &*ast.cdr {

View file

@ -18,14 +18,14 @@
use crate::eval::eval; use crate::eval::eval;
use crate::segment::{Seg, Ctr, Type}; use crate::segment::{Seg, Ctr, Type};
use std::collections::HashMap; use std::collections::HashMap;
use std::sync::Mutex; use std::sync::RwLock;
use lazy_static::lazy_static; use lazy_static::lazy_static;
pub type SymTable = HashMap<String, Symbol>; pub type SymTable = HashMap<String, Symbol>;
lazy_static! { lazy_static! {
pub static ref SYM_TABLE: Mutex<SymTable> = { pub static ref SYM_TABLE: RwLock<SymTable> = {
Mutex::new(SymTable::new()) RwLock::new(SymTable::new())
}; };
} }
@ -172,7 +172,7 @@ impl Symbol {
// Prep var table for function execution // Prep var table for function execution
for n in 0..f.arg_syms.len() { for n in 0..f.arg_syms.len() {
if let Some(old) = SYM_TABLE.lock().unwrap() if let Some(old) = SYM_TABLE.write().unwrap()
.insert(f.arg_syms[n].clone(), Symbol{ .insert(f.arg_syms[n].clone(), Symbol{
name: f.arg_syms[n].clone(), name: f.arg_syms[n].clone(),
value: ValueType::VarForm(Box::new(args[n].clone())), value: ValueType::VarForm(Box::new(args[n].clone())),
@ -206,9 +206,9 @@ impl Symbol {
// clear local vars and restore previous values // clear local vars and restore previous values
for n in 0..f.arg_syms.len() { for n in 0..f.arg_syms.len() {
SYM_TABLE.lock().unwrap().remove(&f.arg_syms[n]); SYM_TABLE.write().unwrap().remove(&f.arg_syms[n]);
if let Some(val) = holding_table.remove(&f.arg_syms[n]) { if let Some(val) = holding_table.remove(&f.arg_syms[n]) {
SYM_TABLE.lock().unwrap().insert(f.arg_syms[n].clone(), val); SYM_TABLE.write().unwrap().insert(f.arg_syms[n].clone(), val);
} }
} }

View file

@ -51,9 +51,11 @@ mod eval_tests {
#[test] #[test]
fn eval_function_call() { fn eval_function_call() {
let mut table_handle = SYM_TABLE.lock().unwrap();
let test_doc = "('one' (echo 'unwrap_me'))".to_string(); let test_doc = "('one' (echo 'unwrap_me'))".to_string();
let output = "('one' 'unwrap_me')"; let output = "('one' 'unwrap_me')";
{ // we want the write lock to expire before eval
let mut table_handle = SYM_TABLE.write().unwrap();
let test_external_func: Symbol = Symbol { let test_external_func: Symbol = Symbol {
name: String::from("echo"), name: String::from("echo"),
args: Args::Lazy(1), args: Args::Lazy(1),
@ -70,6 +72,7 @@ mod eval_tests {
}; };
table_handle.insert(String::from("echo"), test_external_func); table_handle.insert(String::from("echo"), test_external_func);
}
match lex(&test_doc) { match lex(&test_doc) {
Err(e) => { Err(e) => {
println!("Lexing error: {}\n", e); println!("Lexing error: {}\n", e);
@ -91,9 +94,11 @@ mod eval_tests {
#[test] #[test]
fn eval_embedded_func_calls() { fn eval_embedded_func_calls() {
let mut table_handle = SYM_TABLE.lock().unwrap();
let test_doc = "('one' (echo (echo 'unwrap_me')))".to_string(); let test_doc = "('one' (echo (echo 'unwrap_me')))".to_string();
let output = "('one' 'unwrap_me')"; let output = "('one' 'unwrap_me')";
{
let mut table_handle = SYM_TABLE.write().unwrap();
let test_external_func: Symbol = Symbol{ let test_external_func: Symbol = Symbol{
name: String::from("echo"), name: String::from("echo"),
args: Args::Lazy(1), args: Args::Lazy(1),
@ -110,6 +115,7 @@ mod eval_tests {
}; };
table_handle.insert(String::from("echo"), test_external_func); table_handle.insert(String::from("echo"), test_external_func);
}
match lex(&test_doc) { match lex(&test_doc) {
Err(e) => { Err(e) => {
println!("Lexing error: {}\n", e); println!("Lexing error: {}\n", e);

View file

@ -5,7 +5,6 @@ mod func_tests {
#[test] #[test]
fn decl_and_call_internal_func() { fn decl_and_call_internal_func() {
let mut table_handle = SYM_TABLE.lock().unwrap();
let test_internal_func: Symbol = Symbol { let test_internal_func: Symbol = Symbol {
name: String::from("test_func_in"), name: String::from("test_func_in"),
has_undefined_symbols: false, has_undefined_symbols: false,
@ -17,7 +16,6 @@ mod func_tests {
if let Ctr::Bool(_) = *inner.car { if let Ctr::Bool(_) = *inner.car {
is_bool = true; is_bool = true;
} }
Ctr::Bool(is_bool) Ctr::Bool(is_bool)
}, },
)), )),
@ -27,7 +25,12 @@ mod func_tests {
Box::new(Ctr::None) Box::new(Ctr::None)
); );
{
let mut table_handle = SYM_TABLE.write().unwrap();
table_handle.insert(String::from("test_func_in"), test_internal_func); table_handle.insert(String::from("test_func_in"), test_internal_func);
}
let table_handle = SYM_TABLE.read().unwrap();
let func: &Symbol; let func: &Symbol;
if let Some(f) = table_handle.get(&"test_func_in".to_string()) { if let Some(f) = table_handle.get(&"test_func_in".to_string()) {
func = f; func = f;
@ -53,7 +56,6 @@ mod func_tests {
#[test] #[test]
fn decl_and_call_external_func_singlet() { fn decl_and_call_external_func_singlet() {
let mut table_handle = SYM_TABLE.lock().unwrap();
match lex(&"((input))".to_string()) { match lex(&"((input))".to_string()) {
Err(e) => panic!("{}", e), Err(e) => panic!("{}", e),
Ok(finner) => { Ok(finner) => {
@ -72,7 +74,12 @@ mod func_tests {
Box::new(Ctr::None) Box::new(Ctr::None)
); );
{
let mut table_handle = SYM_TABLE.write().unwrap();
table_handle.insert(String::from("test_func_in"), test_external_func); table_handle.insert(String::from("test_func_in"), test_external_func);
}
let table_handle = SYM_TABLE.read().unwrap();
let func: &Symbol; let func: &Symbol;
if let Some(f) = table_handle.get(&"test_func_in".to_string()) { if let Some(f) = table_handle.get(&"test_func_in".to_string()) {
func = f; func = f;
@ -101,7 +108,6 @@ mod func_tests {
#[test] #[test]
fn decl_and_call_external_func_multi_body() { fn decl_and_call_external_func_multi_body() {
let mut table_handle = SYM_TABLE.lock().unwrap();
match lex(&"((input) (input))".to_string()) { match lex(&"((input) (input))".to_string()) {
Err(e) => panic!("{}", e), Err(e) => panic!("{}", e),
Ok(finner) => { Ok(finner) => {
@ -120,7 +126,12 @@ mod func_tests {
Box::new(Ctr::None) Box::new(Ctr::None)
); );
{
let mut table_handle = SYM_TABLE.write().unwrap();
table_handle.insert(String::from("echo_2"), test_external_func); table_handle.insert(String::from("echo_2"), test_external_func);
}
let table_handle = SYM_TABLE.read().unwrap();
let func: &Symbol; let func: &Symbol;
if let Some(f) = table_handle.get(&"echo_2".to_string()) { if let Some(f) = table_handle.get(&"echo_2".to_string()) {
func = f; func = f;
@ -143,7 +154,6 @@ mod func_tests {
#[test] #[test]
fn decl_and_call_func_with_nested_call() { fn decl_and_call_func_with_nested_call() {
let mut table_handle = SYM_TABLE.lock().unwrap();
let inner_func: Symbol = Symbol { let inner_func: Symbol = Symbol {
name: String::from("test_inner"), name: String::from("test_inner"),
has_undefined_symbols: false, has_undefined_symbols: false,
@ -177,15 +187,19 @@ mod func_tests {
}), }),
}; };
let args = Seg::from( let args = Seg::from(
Box::new(Ctr::Bool(true)), Box::new(Ctr::Bool(true)),
Box::new(Ctr::None) Box::new(Ctr::None)
); );
{
let mut table_handle = SYM_TABLE.write().unwrap();
table_handle.insert(String::from("test_inner"), inner_func); table_handle.insert(String::from("test_inner"), inner_func);
table_handle.insert(String::from("test_outer"), outer_func); table_handle.insert(String::from("test_outer"), outer_func);
}
let table_handle = SYM_TABLE.read().unwrap();
let func: &Symbol; let func: &Symbol;
if let Some(f) = table_handle.get(&"test_outer".to_string()) { if let Some(f) = table_handle.get(&"test_outer".to_string()) {
func = f; func = f;