From a06d77c57416378adeea9654c9e7e116182c108f Mon Sep 17 00:00:00 2001 From: Ryan Daum Date: Sun, 14 Jan 2024 16:54:04 -0500 Subject: [PATCH] Cleanup/optimize the use of references vs clones when peeking, pushing, updating values on stack, as well as jumping to labels in the activation. 10%ish improvement in tight loop benches when a lot of those kinds of operations are performed. --- crates/kernel/src/builtins/bf_objects.rs | 2 +- crates/kernel/src/vm/activation.rs | 14 +- crates/kernel/src/vm/exec_state.rs | 18 +- crates/kernel/src/vm/vm_execute.rs | 212 ++++++++++++----------- crates/kernel/src/vm/vm_unwind.rs | 12 +- 5 files changed, 138 insertions(+), 120 deletions(-) diff --git a/crates/kernel/src/builtins/bf_objects.rs b/crates/kernel/src/builtins/bf_objects.rs index bfc7b378..a6078ae0 100644 --- a/crates/kernel/src/builtins/bf_objects.rs +++ b/crates/kernel/src/builtins/bf_objects.rs @@ -452,7 +452,7 @@ async fn bf_move<'a>(bf_args: &mut BfCallState<'a>) -> Result { // Accept verb has been called, and returned. Check the result. Should be on stack, // unless short-circuited, in which case we assume *false* let result = if !shortcircuit { - bf_args.exec_state.top().peek_top().unwrap() + bf_args.exec_state.top().peek_top().unwrap().clone() } else { v_int(0) }; diff --git a/crates/kernel/src/vm/activation.rs b/crates/kernel/src/vm/activation.rs index 6c72042e..223b418b 100644 --- a/crates/kernel/src/vm/activation.rs +++ b/crates/kernel/src/vm/activation.rs @@ -321,9 +321,9 @@ impl Activation { #[inline] pub fn next_op(&mut self) -> Op { assert!(self.pc < self.program.main_vector.len(), "pc out of bounds"); - let op = self.program.main_vector[self.pc].clone(); + let op = &self.program.main_vector[self.pc]; self.pc += 1; - op + op.clone() } #[inline] @@ -347,8 +347,8 @@ impl Activation { } #[inline] - pub fn peek_top(&self) -> Option { - self.valstack.last().cloned() + pub fn peek_top(&self) -> Option<&Var> { + self.valstack.last() } #[inline] @@ -363,10 +363,10 @@ impl Activation { } #[inline] - pub fn peek2(&self) -> (Var, Var) { + pub fn peek2(&self) -> (&Var, &Var) { let l = self.valstack.len(); let (a, b) = (&self.valstack[l - 1], &self.valstack[l - 2]); - (a.clone(), b.clone()) + (a, b) } #[inline] @@ -376,7 +376,7 @@ impl Activation { } #[inline] - pub fn jump(&mut self, label_id: Label) { + pub fn jump(&mut self, label_id: &Label) { let label = &self.program.jump_labels[label_id.0 as usize]; self.pc = label.position.0; } diff --git a/crates/kernel/src/vm/exec_state.rs b/crates/kernel/src/vm/exec_state.rs index 43214fc5..47ece204 100644 --- a/crates/kernel/src/vm/exec_state.rs +++ b/crates/kernel/src/vm/exec_state.rs @@ -147,8 +147,8 @@ impl VMExecState { /// Push a value onto the value stack #[inline] - pub(crate) fn push(&mut self, v: &Var) { - self.top_mut().push(v.clone()) + pub(crate) fn push(&mut self, v: Var) { + self.top_mut().push(v) } /// Non-destructively peek a range of values from the value stack. @@ -168,19 +168,19 @@ impl VMExecState { /// Return the top two values on the value stack. #[inline] - pub(crate) fn peek2(&self) -> (Var, Var) { + pub(crate) fn peek2(&self) -> (&Var, &Var) { self.top().peek2() } /// Update at a set offset in the value stack. #[inline] - pub(crate) fn update(&mut self, amt: usize, v: &Var) { - self.top_mut().update(amt, v.clone()) + pub(crate) fn update(&mut self, amt: usize, v: Var) { + self.top_mut().update(amt, v) } /// Return the top of the value stack. #[inline] - pub(crate) fn peek_top(&self) -> Var { + pub(crate) fn peek_top(&self) -> &Var { self.top().peek_top().expect("stack underflow") } @@ -192,19 +192,19 @@ impl VMExecState { /// Jump to the given label. #[inline] - pub(crate) fn jump(&mut self, label: Label) { + pub(crate) fn jump(&mut self, label: &Label) { self.top_mut().jump(label) } /// Return the value of a local variable. #[inline] - pub(crate) fn get_env(&self, id: Name) -> Option<&Var> { + pub(crate) fn get_env(&self, id: &Name) -> Option<&Var> { self.top().environment.get(id.0 as usize) } /// Set the value of a local variable. #[inline] - pub(crate) fn set_env(&mut self, id: Name, v: &Var) { + pub(crate) fn set_env(&mut self, id: &Name, v: Var) { self.top_mut().environment.set(id.0 as usize, v.clone()); } } diff --git a/crates/kernel/src/vm/vm_execute.rs b/crates/kernel/src/vm/vm_execute.rs index fabf99ef..7a04fcf4 100644 --- a/crates/kernel/src/vm/vm_execute.rs +++ b/crates/kernel/src/vm/vm_execute.rs @@ -124,8 +124,8 @@ macro_rules! binary_bool_op { ( $state:ident, $op:tt ) => { let rhs = $state.pop(); let lhs = $state.peek_top(); - let result = if lhs $op rhs { 1 } else { 0 }; - $state.update(0, &v_int(result)) + let result = if lhs $op &rhs { 1 } else { 0 }; + $state.update(0, v_int(result)) }; } @@ -135,7 +135,7 @@ macro_rules! binary_var_op { let lhs = $state.peek_top(); let result = lhs.$op(&rhs); match result { - Ok(result) => $state.update(0, &result), + Ok(result) => $state.update(0, result), Err(err_code) => { $state.pop(); return $vm.push_error($state, err_code); @@ -197,20 +197,20 @@ impl VM { Op::If(label) | Op::Eif(label) | Op::IfQues(label) | Op::While(label) => { let cond = state.pop(); if !cond.is_true() { - state.jump(label); + state.jump(&label); } } Op::Jump { label } => { - state.jump(label); + state.jump(&label); } Op::WhileId { id, end_label: label, } => { - state.set_env(id, &state.peek_top()); + state.set_env(&id, state.peek_top().clone()); let cond = state.pop(); if !cond.is_true() { - state.jump(label); + state.jump(&label); } } Op::ForList { @@ -227,7 +227,7 @@ impl VM { // If the result of raising error was just to push the value -- that is, we // didn't 'throw' and unwind the stack -- we need to get out of the loop. // So we preemptively jump (here and below for List) and then raise the error. - state.jump(label); + state.jump(&label); return self.raise_error(state, E_TYPE); }; let count = *count as usize; @@ -235,7 +235,7 @@ impl VM { state.pop(); state.pop(); - state.jump(label); + state.jump(&label); return self.raise_error(state, E_TYPE); }; @@ -244,80 +244,82 @@ impl VM { state.pop(); state.pop(); - state.jump(label); + state.jump(&label); continue; } // Track iteration count for range; set id to current list element for the count, // then increment the count, rewind the program counter to the top of the loop, and // continue. - state.set_env(id, &l[count]); - state.update(0, &v_int((count + 1) as i64)); + state.set_env(&id, l[count].clone()); + state.update(0, v_int((count + 1) as i64)); } Op::ForRange { end_label: label, id, } => { // Pull the range ends off the stack. - let (to, from) = state.peek2(); + let (from, next_val) = { + let (to, from) = state.peek2(); - // TODO: LambdaMOO has special handling for MAXINT/MAXOBJ - // Given we're 64-bit this is highly unlikely to ever be a concern for us, but - // we also don't want to *crash* on obscene values, so impl that here. + // TODO: LambdaMOO has special handling for MAXINT/MAXOBJ + // Given we're 64-bit this is highly unlikely to ever be a concern for us, but + // we also don't want to *crash* on obscene values, so impl that here. - let next_val = match (to.variant(), from.variant()) { - (Variant::Int(to_i), Variant::Int(from_i)) => { - if from_i > to_i { - state.pop(); - state.pop(); - state.jump(label); + let next_val = match (to.variant(), from.variant()) { + (Variant::Int(to_i), Variant::Int(from_i)) => { + if from_i > to_i { + state.pop(); + state.pop(); + state.jump(&label); - continue; + continue; + } + v_int(from_i + 1) } - v_int(from_i + 1) - } - (Variant::Obj(to_o), Variant::Obj(from_o)) => { - if from_o.0 > to_o.0 { + (Variant::Obj(to_o), Variant::Obj(from_o)) => { + if from_o.0 > to_o.0 { + state.pop(); + state.pop(); + state.jump(&label); + + continue; + } + v_obj(from_o.0 + 1) + } + (_, _) => { state.pop(); state.pop(); - state.jump(label); + // Make sure we've jumped out of the loop before raising the error, + // because in verbs that aren't `d' we could end up continuing on in + // the loop (with a messed up stack) otherwise. + state.jump(&label); - continue; + return self.raise_error(state, E_TYPE); } - v_obj(from_o.0 + 1) - } - (_, _) => { - state.pop(); - state.pop(); - // Make sure we've jumped out of the loop before raising the error, - // because in verbs that aren't `d' we could end up continuing on in - // the loop (with a messed up stack) otherwise. - state.jump(label); - - return self.raise_error(state, E_TYPE); - } + }; + (from.clone(), next_val) }; - - state.update(1, &next_val); - state.set_env(id, &from); + state.update(1, next_val); + state.set_env(&id, from); } Op::Pop => { state.pop(); } Op::ImmNone => { - state.push(&v_none()); + state.push(v_none()); } Op::ImmBigInt(val) => { - state.push(&v_int(val)); + state.push(v_int(val)); } Op::ImmInt(val) => { - state.push(&v_int(val as i64)); + state.push(v_int(val as i64)); } Op::ImmObjid(val) => { - state.push(&v_objid(val)); + state.push(v_objid(val)); } Op::ImmErr(val) => { - state.push(&v_err(val)); + state.push(v_err(val)); } Op::Val(val) => { match state.top().lookahead() { @@ -327,7 +329,7 @@ impl VM { continue; } _ => { - state.push(&val); + state.push(val.clone()); } } } @@ -340,11 +342,11 @@ impl VM { } _ => { let value = &state.top().program.literals[slot.0 as usize].clone(); - state.push(value); + state.push(value.clone()); } } } - Op::ImmEmptyList => state.push(&v_empty_list()), + Op::ImmEmptyList => state.push(v_empty_list()), Op::ListAddTail => { let (tail, list) = (state.pop(), state.peek_top()); let Variant::List(list) = list.variant() else { @@ -353,7 +355,7 @@ impl VM { }; // TODO: quota check SVO_MAX_LIST_CONCAT -> E_QUOTA - state.update(0, &list.push(&tail)); + state.update(0, list.push(&tail)); } Op::ListAppend => { let (tail, list) = (state.pop(), state.peek_top()); @@ -372,7 +374,7 @@ impl VM { // TODO: quota check SVO_MAX_LIST_CONCAT -> E_QUOTA let new_list = list.append(tail); - state.update(0, &new_list); + state.update(0, new_list); } Op::IndexSet => { let (rhs, index, lhs) = (state.pop(), state.pop(), state.peek_top()); @@ -385,7 +387,7 @@ impl VM { }; match lhs.index_set(i, &rhs) { Ok(v) => { - state.update(0, &v); + state.update(0, v); } Err(e) => { state.pop(); @@ -395,14 +397,14 @@ impl VM { } Op::MakeSingletonList => { let v = state.peek_top(); - state.update(0, &v_list(&[v])); + state.update(0, v_list(&[v.clone()])); } Op::PutTemp => { - state.top_mut().temp = state.peek_top(); + state.top_mut().temp = state.peek_top().clone(); } Op::PushTemp => { let tmp = state.top().temp.clone(); - state.push(&tmp); + state.push(tmp); state.top_mut().temp = v_none(); } Op::Eq => { @@ -430,7 +432,7 @@ impl VM { if let Variant::Err(e) = r.variant() { return self.push_error(state, *e); } - state.push(&r); + state.push(r); } Op::Mul => { binary_var_op!(self, state, mul); @@ -460,7 +462,7 @@ impl VM { Op::And(label) => { let v = state.peek_top().is_true(); if !v { - state.jump(label) + state.jump(&label) } else { state.pop(); } @@ -468,14 +470,14 @@ impl VM { Op::Or(label) => { let v = state.peek_top().is_true(); if v { - state.jump(label); + state.jump(&label); } else { state.pop(); } } Op::Not => { let v = !state.peek_top().is_true(); - state.update(0, &v_bool(v)); + state.update(0, v_bool(v)); } Op::UnaryMinus => { let v = state.peek_top(); @@ -484,28 +486,28 @@ impl VM { state.pop(); return self.push_error(state, e); } - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), } } Op::Push(ident) => { - let Some(v) = state.get_env(ident) else { + let Some(v) = state.get_env(&ident) else { return self.push_error(state, E_VARNF); }; - state.push(&v.clone()); + state.push(v.clone()); } Op::Put(ident) => { let v = state.peek_top(); - state.set_env(ident, &v); + state.set_env(&ident, v.clone()); } Op::PushRef => { let (index, list) = state.peek2(); - let index = match one_to_zero_index(&index) { + let index = match one_to_zero_index(index) { Ok(i) => i, Err(e) => return self.push_error(state, e), }; match list.index(index) { Err(e) => return self.push_error(state, e), - Ok(v) => state.push(&v), + Ok(v) => state.push(v), } } Op::Ref => { @@ -523,7 +525,7 @@ impl VM { state.pop(); return self.push_error(state, e); } - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), } } Op::RangeRef => { @@ -534,7 +536,7 @@ impl VM { state.pop(); return self.push_error(state, e); } - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), }, (_, _) => return self.push_error(state, E_TYPE), }; @@ -549,7 +551,7 @@ impl VM { state.pop(); return self.push_error(state, e); } - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), } } _ => { @@ -559,18 +561,18 @@ impl VM { } } Op::GPut { id } => { - state.set_env(id, &state.peek_top()); + state.set_env(&id, state.peek_top().clone()); } Op::GPush { id } => { - let Some(v) = state.get_env(id) else { + let Some(v) = state.get_env(&id) else { return self.push_error(state, E_VARNF); }; - state.push(&v.clone()); + state.push(v.clone()); } Op::Length(offset) => { let v = state.peek_abs(offset.0); match v.len() { - Ok(l) => state.push(&l), + Ok(l) => state.push(l), Err(e) => return self.push_error(state, e), } } @@ -578,10 +580,10 @@ impl VM { let (propname, obj) = (state.pop(), state.peek_top()); match self - .resolve_property(state, world_state, propname, obj) + .resolve_property(state, world_state, propname.clone(), obj.clone()) .await { - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), Err(e) => { state.pop(); return self.push_error(state, e); @@ -591,20 +593,26 @@ impl VM { Op::PushGetProp => { let (propname, obj) = state.peek2(); match self - .resolve_property(state, world_state, propname, obj) + .resolve_property(state, world_state, propname.clone(), obj.clone()) .await { - Ok(v) => state.push(&v), + Ok(v) => state.push(v), Err(e) => return self.push_error(state, e), } } Op::PutProp => { let (rhs, propname, obj) = (state.pop(), state.pop(), state.peek_top()); match self - .set_property(state, world_state, propname, obj, rhs) + .set_property( + state, + world_state, + propname.clone(), + obj.clone(), + rhs.clone(), + ) .await { - Ok(v) => state.update(0, &v), + Ok(v) => state.update(0, v), Err(e) => { state.pop(); return self.push_error(state, e); @@ -699,7 +707,7 @@ impl VM { .push_handler_label(HandlerType::Catch(num_excepts)); } Op::EndCatch(label) | Op::EndExcept(label) => { - let is_catch = op == Op::EndCatch(label); + let is_catch = matches!(op, Op::EndCatch(_)); let v = if is_catch { state.pop() } else { v_none() }; let handler = state @@ -715,9 +723,9 @@ impl VM { state.top_mut().handler_stack.pop(); } if is_catch { - state.push(&v); + state.push(v); } - state.jump(label); + state.jump(&label); } Op::EndFinally => { let Some(finally_handler) = state.top_mut().pop_applicable_handler() else { @@ -726,8 +734,8 @@ impl VM { let HandlerType::Finally(_) = finally_handler.handler_type else { panic!("Handler is not a finally handler") }; - state.push(&v_int(0) /* fallthrough */); - state.push(&v_int(0)); + state.push(v_int(0) /* fallthrough */); + state.push(v_int(0)); } Op::Continue => { let why = state.pop(); @@ -752,11 +760,17 @@ impl VM { } } Op::ExitId(label) => { - state.jump(label); + state.jump(&label); continue; } Op::Exit { stack, label } => { - return self.unwind_stack(state, FinallyReason::Exit { stack, label }); + return self.unwind_stack( + state, + FinallyReason::Exit { + stack, + label, + }, + ); } Op::Scatter { nargs, @@ -767,10 +781,13 @@ impl VM { .. } => { let have_rest = rest <= nargs; - let rhs = state.peek_top(); - let Variant::List(rhs_values) = rhs.variant() else { - state.pop(); - return self.push_error(state, E_TYPE); + let rhs_values = { + let rhs = state.peek_top(); + let Variant::List(rhs_values) = rhs.variant() else { + state.pop(); + return self.push_error(state, E_TYPE); + }; + rhs_values.clone() }; let len = rhs_values.len(); @@ -788,6 +805,7 @@ impl VM { }; let mut jump_where = None; let mut args_iter = rhs_values.iter(); + for label in labels.iter() { match label { ScatterLabel::Rest(id) => { @@ -799,14 +817,14 @@ impl VM { v.push(rest.clone()); } let rest = v_list(&v); - state.set_env(*id, &rest); + state.set_env(id, rest); } ScatterLabel::Required(id) => { let Some(arg) = args_iter.next() else { return self.push_error(state, E_ARGS); }; - state.set_env(*id, arg); + state.set_env(id, arg.clone()); } ScatterLabel::Optional(id, jump_to) => { if nopt_avail > 0 { @@ -814,15 +832,15 @@ impl VM { let Some(arg) = args_iter.next() else { return self.push_error(state, E_ARGS); }; - state.set_env(*id, arg); + state.set_env(id, arg.clone()); } else if jump_where.is_none() && jump_to.is_some() { jump_where = *jump_to; } } } } - match jump_where { - None => state.jump(done), + match &jump_where { + None => state.jump(&done), Some(jump_where) => state.jump(jump_where), } } diff --git a/crates/kernel/src/vm/vm_unwind.rs b/crates/kernel/src/vm/vm_unwind.rs index fd6d3824..75b21f41 100644 --- a/crates/kernel/src/vm/vm_unwind.rs +++ b/crates/kernel/src/vm/vm_unwind.rs @@ -229,7 +229,7 @@ impl VM { /// Push an error to the stack and raise it. pub(crate) fn push_error(&self, state: &mut VMExecState, code: Error) -> ExecutionResult { trace!(?code, "push_error"); - state.push(&v_err(code)); + state.push(v_err(code)); // Check 'd' bit of running verb. If it's set, we raise the error. Otherwise nope. if let Some(activation) = state.stack.last() { if activation @@ -279,7 +279,7 @@ impl VM { msg: String, ) -> ExecutionResult { trace!(?code, msg, "push_error_msg"); - state.push(&v_err(code)); + state.push(v_err(code)); self.raise_error(state, code) } @@ -338,7 +338,7 @@ impl VM { } // Jump to the label pointed to by the finally label and then continue on // executing. - a.jump(label); + a.jump(&label); a.push(v_int(why_num as i64)); trace!(jump = ?label, ?why, "matched finally handler"); return ExecutionResult::More; @@ -353,7 +353,7 @@ impl VM { let Some(handler) = a.pop_applicable_handler() else { continue; }; - let HandlerType::CatchLabel(pushed_label) = handler.handler_type else { + let HandlerType::CatchLabel(pushed_label) = &handler.handler_type else { panic!("Expected CatchLabel"); }; @@ -385,7 +385,7 @@ impl VM { // Exit with a jump.. let's go... if let FinallyReason::Exit { label, .. } = why { trace!("Exit with a jump"); - a.jump(label); + a.jump(&label); return ExecutionResult::More; } @@ -421,7 +421,7 @@ impl VM { // (Unless we're the final activation, in which case that should have been handled // above) if let FinallyReason::Return(value) = &why { - state.push(value); + state.push(value.clone()); return ExecutionResult::More; } }