xref: /wasmtime-44.0.1/winch/codegen/src/stack.rs (revision 7690c500)
114b39bc2SSaúl Cabrera use crate::{isa::reg::Reg, masm::StackSlot};
214b39bc2SSaúl Cabrera use wasmparser::{Ieee32, Ieee64};
314b39bc2SSaúl Cabrera use wasmtime_environ::WasmType;
414b39bc2SSaúl Cabrera 
514b39bc2SSaúl Cabrera /// A typed register value used to track register values in the value
614b39bc2SSaúl Cabrera /// stack.
714b39bc2SSaúl Cabrera #[derive(Debug, Eq, PartialEq, Copy, Clone)]
814b39bc2SSaúl Cabrera pub struct TypedReg {
914b39bc2SSaúl Cabrera     /// The physical register.
1014b39bc2SSaúl Cabrera     pub reg: Reg,
1114b39bc2SSaúl Cabrera     /// The type associated to the physical register.
1214b39bc2SSaúl Cabrera     pub ty: WasmType,
1314b39bc2SSaúl Cabrera }
1414b39bc2SSaúl Cabrera 
1514b39bc2SSaúl Cabrera impl TypedReg {
164b288ba8SSaúl Cabrera     /// Create a new [`TypedReg`].
1714b39bc2SSaúl Cabrera     pub fn new(ty: WasmType, reg: Reg) -> Self {
1814b39bc2SSaúl Cabrera         Self { ty, reg }
1914b39bc2SSaúl Cabrera     }
2014b39bc2SSaúl Cabrera 
214b288ba8SSaúl Cabrera     /// Create an i64 [`TypedReg`].
2214b39bc2SSaúl Cabrera     pub fn i64(reg: Reg) -> Self {
2314b39bc2SSaúl Cabrera         Self {
2414b39bc2SSaúl Cabrera             ty: WasmType::I64,
2514b39bc2SSaúl Cabrera             reg,
2614b39bc2SSaúl Cabrera         }
2714b39bc2SSaúl Cabrera     }
28a109d2abSSaúl Cabrera 
29f0162a40SSaúl Cabrera     /// Create an i32 [`TypedReg`].
30a109d2abSSaúl Cabrera     pub fn i32(reg: Reg) -> Self {
31a109d2abSSaúl Cabrera         Self {
32a109d2abSSaúl Cabrera             ty: WasmType::I32,
33a109d2abSSaúl Cabrera             reg,
34a109d2abSSaúl Cabrera         }
35a109d2abSSaúl Cabrera     }
3617091e6fSJeffrey Charles 
3717091e6fSJeffrey Charles     /// Create an f64 [`TypedReg`].
3817091e6fSJeffrey Charles     pub fn f64(reg: Reg) -> Self {
3917091e6fSJeffrey Charles         Self {
4017091e6fSJeffrey Charles             ty: WasmType::F64,
4117091e6fSJeffrey Charles             reg,
4217091e6fSJeffrey Charles         }
4317091e6fSJeffrey Charles     }
4417091e6fSJeffrey Charles 
4517091e6fSJeffrey Charles     /// Create an f32 [`TypedReg`].
4617091e6fSJeffrey Charles     pub fn f32(reg: Reg) -> Self {
4717091e6fSJeffrey Charles         Self {
4817091e6fSJeffrey Charles             ty: WasmType::F32,
4917091e6fSJeffrey Charles             reg,
5017091e6fSJeffrey Charles         }
5117091e6fSJeffrey Charles     }
5214b39bc2SSaúl Cabrera }
5314b39bc2SSaúl Cabrera 
5414b39bc2SSaúl Cabrera impl From<TypedReg> for Reg {
5514b39bc2SSaúl Cabrera     fn from(tr: TypedReg) -> Self {
5614b39bc2SSaúl Cabrera         tr.reg
5714b39bc2SSaúl Cabrera     }
5814b39bc2SSaúl Cabrera }
5914b39bc2SSaúl Cabrera 
6014b39bc2SSaúl Cabrera /// A local value.
6114b39bc2SSaúl Cabrera #[derive(Debug, Eq, PartialEq, Copy, Clone)]
6214b39bc2SSaúl Cabrera pub struct Local {
6314b39bc2SSaúl Cabrera     /// The index of the local.
6414b39bc2SSaúl Cabrera     pub index: u32,
6514b39bc2SSaúl Cabrera     /// The type of the local.
6614b39bc2SSaúl Cabrera     pub ty: WasmType,
6714b39bc2SSaúl Cabrera }
6814b39bc2SSaúl Cabrera 
6914b39bc2SSaúl Cabrera /// A memory value.
7014b39bc2SSaúl Cabrera #[derive(Debug, Eq, PartialEq, Copy, Clone)]
7114b39bc2SSaúl Cabrera pub struct Memory {
7214b39bc2SSaúl Cabrera     /// The type associated with the memory offset.
7314b39bc2SSaúl Cabrera     pub ty: WasmType,
7414b39bc2SSaúl Cabrera     /// The stack slot corresponding to the memory value.
7514b39bc2SSaúl Cabrera     pub slot: StackSlot,
7614b39bc2SSaúl Cabrera }
77835abbcdSSaúl Cabrera 
78835abbcdSSaúl Cabrera /// Value definition to be used within the shadow stack.
79a50c4972SSaúl Cabrera #[derive(Debug, Eq, PartialEq, Copy, Clone)]
80835abbcdSSaúl Cabrera pub(crate) enum Val {
81835abbcdSSaúl Cabrera     /// I32 Constant.
82835abbcdSSaúl Cabrera     I32(i32),
837c5c7e4bSSaúl Cabrera     /// I64 Constant.
847c5c7e4bSSaúl Cabrera     I64(i64),
8514b39bc2SSaúl Cabrera     /// F32 Constant.
8614b39bc2SSaúl Cabrera     F32(Ieee32),
8714b39bc2SSaúl Cabrera     /// F64 Constant.
8814b39bc2SSaúl Cabrera     F64(Ieee64),
8914b39bc2SSaúl Cabrera     /// A register value.
9014b39bc2SSaúl Cabrera     Reg(TypedReg),
91835abbcdSSaúl Cabrera     /// A local slot.
9214b39bc2SSaúl Cabrera     Local(Local),
93835abbcdSSaúl Cabrera     /// Offset to a memory location.
9414b39bc2SSaúl Cabrera     Memory(Memory),
9514b39bc2SSaúl Cabrera }
9614b39bc2SSaúl Cabrera 
9714b39bc2SSaúl Cabrera impl From<TypedReg> for Val {
9814b39bc2SSaúl Cabrera     fn from(tr: TypedReg) -> Self {
9914b39bc2SSaúl Cabrera         Val::Reg(tr)
10014b39bc2SSaúl Cabrera     }
10114b39bc2SSaúl Cabrera }
10214b39bc2SSaúl Cabrera 
10314b39bc2SSaúl Cabrera impl From<Local> for Val {
10414b39bc2SSaúl Cabrera     fn from(local: Local) -> Self {
10514b39bc2SSaúl Cabrera         Val::Local(local)
10614b39bc2SSaúl Cabrera     }
10714b39bc2SSaúl Cabrera }
10814b39bc2SSaúl Cabrera 
10914b39bc2SSaúl Cabrera impl From<Memory> for Val {
11014b39bc2SSaúl Cabrera     fn from(mem: Memory) -> Self {
11114b39bc2SSaúl Cabrera         Val::Memory(mem)
11214b39bc2SSaúl Cabrera     }
113835abbcdSSaúl Cabrera }
114835abbcdSSaúl Cabrera 
1154b288ba8SSaúl Cabrera impl TryFrom<u32> for Val {
1164b288ba8SSaúl Cabrera     type Error = anyhow::Error;
1174b288ba8SSaúl Cabrera     fn try_from(value: u32) -> Result<Self, Self::Error> {
1184b288ba8SSaúl Cabrera         i32::try_from(value).map(Val::i32).map_err(Into::into)
1194b288ba8SSaúl Cabrera     }
1204b288ba8SSaúl Cabrera }
1214b288ba8SSaúl Cabrera 
122835abbcdSSaúl Cabrera impl Val {
123835abbcdSSaúl Cabrera     /// Create a new I32 constant value.
124835abbcdSSaúl Cabrera     pub fn i32(v: i32) -> Self {
125835abbcdSSaúl Cabrera         Self::I32(v)
126835abbcdSSaúl Cabrera     }
127835abbcdSSaúl Cabrera 
1287c5c7e4bSSaúl Cabrera     /// Create a new I64 constant value.
1297c5c7e4bSSaúl Cabrera     pub fn i64(v: i64) -> Self {
1307c5c7e4bSSaúl Cabrera         Self::I64(v)
1317c5c7e4bSSaúl Cabrera     }
1327c5c7e4bSSaúl Cabrera 
13314b39bc2SSaúl Cabrera     /// Create a new F32 constant value.
13414b39bc2SSaúl Cabrera     pub fn f32(v: Ieee32) -> Self {
13514b39bc2SSaúl Cabrera         Self::F32(v)
13614b39bc2SSaúl Cabrera     }
13714b39bc2SSaúl Cabrera 
13814b39bc2SSaúl Cabrera     pub fn f64(v: Ieee64) -> Self {
13914b39bc2SSaúl Cabrera         Self::F64(v)
14014b39bc2SSaúl Cabrera     }
14114b39bc2SSaúl Cabrera 
142835abbcdSSaúl Cabrera     /// Create a new Reg value.
14314b39bc2SSaúl Cabrera     pub fn reg(reg: Reg, ty: WasmType) -> Self {
14414b39bc2SSaúl Cabrera         Self::Reg(TypedReg { reg, ty })
145835abbcdSSaúl Cabrera     }
146835abbcdSSaúl Cabrera 
147835abbcdSSaúl Cabrera     /// Create a new Local value.
14814b39bc2SSaúl Cabrera     pub fn local(index: u32, ty: WasmType) -> Self {
14914b39bc2SSaúl Cabrera         Self::Local(Local { index, ty })
15014b39bc2SSaúl Cabrera     }
15114b39bc2SSaúl Cabrera 
15214b39bc2SSaúl Cabrera     /// Create a Memory value.
15314b39bc2SSaúl Cabrera     pub fn mem(ty: WasmType, slot: StackSlot) -> Self {
15414b39bc2SSaúl Cabrera         Self::Memory(Memory { ty, slot })
155835abbcdSSaúl Cabrera     }
156835abbcdSSaúl Cabrera 
157835abbcdSSaúl Cabrera     /// Check whether the value is a register.
158835abbcdSSaúl Cabrera     pub fn is_reg(&self) -> bool {
159835abbcdSSaúl Cabrera         match *self {
160835abbcdSSaúl Cabrera             Self::Reg(_) => true,
161835abbcdSSaúl Cabrera             _ => false,
162835abbcdSSaúl Cabrera         }
163835abbcdSSaúl Cabrera     }
164835abbcdSSaúl Cabrera 
165af4d94c8SSaúl Cabrera     /// Check wheter the value is a memory offset.
166af4d94c8SSaúl Cabrera     pub fn is_mem(&self) -> bool {
167af4d94c8SSaúl Cabrera         match *self {
168af4d94c8SSaúl Cabrera             Self::Memory(_) => true,
169af4d94c8SSaúl Cabrera             _ => false,
170af4d94c8SSaúl Cabrera         }
171af4d94c8SSaúl Cabrera     }
172af4d94c8SSaúl Cabrera 
17355f9a4bdSJeffrey Charles     /// Check whether the value is a constant.
17455f9a4bdSJeffrey Charles     pub fn is_const(&self) -> bool {
17555f9a4bdSJeffrey Charles         match *self {
17655f9a4bdSJeffrey Charles             Val::I32(_) | Val::I64(_) | Val::F32(_) | Val::F64(_) => true,
17755f9a4bdSJeffrey Charles             _ => false,
17855f9a4bdSJeffrey Charles         }
17955f9a4bdSJeffrey Charles     }
18055f9a4bdSJeffrey Charles 
181321294a5SJeffrey Charles     /// Check whether the value is local with a particular index.
182321294a5SJeffrey Charles     pub fn is_local_at_index(&self, index: u32) -> bool {
183321294a5SJeffrey Charles         match *self {
184321294a5SJeffrey Charles             Self::Local(Local { index: i, .. }) if i == index => true,
185321294a5SJeffrey Charles             _ => false,
186321294a5SJeffrey Charles         }
187321294a5SJeffrey Charles     }
188321294a5SJeffrey Charles 
189835abbcdSSaúl Cabrera     /// Get the register representation of the value.
190835abbcdSSaúl Cabrera     ///
191835abbcdSSaúl Cabrera     /// # Panics
192835abbcdSSaúl Cabrera     /// This method will panic if the value is not a register.
193f0162a40SSaúl Cabrera     pub fn unwrap_reg(&self) -> TypedReg {
194835abbcdSSaúl Cabrera         match self {
19514b39bc2SSaúl Cabrera             Self::Reg(tr) => *tr,
196835abbcdSSaúl Cabrera             v => panic!("expected value {:?} to be a register", v),
197835abbcdSSaúl Cabrera         }
198835abbcdSSaúl Cabrera     }
199835abbcdSSaúl Cabrera 
200835abbcdSSaúl Cabrera     /// Get the integer representation of the value.
201835abbcdSSaúl Cabrera     ///
202835abbcdSSaúl Cabrera     /// # Panics
203835abbcdSSaúl Cabrera     /// This method will panic if the value is not an i32.
204f0162a40SSaúl Cabrera     pub fn unwrap_i32(&self) -> i32 {
205835abbcdSSaúl Cabrera         match self {
206835abbcdSSaúl Cabrera             Self::I32(v) => *v,
207835abbcdSSaúl Cabrera             v => panic!("expected value {:?} to be i32", v),
208835abbcdSSaúl Cabrera         }
209835abbcdSSaúl Cabrera     }
210835abbcdSSaúl Cabrera 
2117c5c7e4bSSaúl Cabrera     /// Get the integer representation of the value.
2127c5c7e4bSSaúl Cabrera     ///
2137c5c7e4bSSaúl Cabrera     /// # Panics
2147c5c7e4bSSaúl Cabrera     /// This method will panic if the value is not an i64.
215f0162a40SSaúl Cabrera     pub fn unwrap_i64(&self) -> i64 {
2167c5c7e4bSSaúl Cabrera         match self {
2177c5c7e4bSSaúl Cabrera             Self::I64(v) => *v,
2187c5c7e4bSSaúl Cabrera             v => panic!("expected value {:?} to be i64", v),
2197c5c7e4bSSaúl Cabrera         }
2207c5c7e4bSSaúl Cabrera     }
2217c5c7e4bSSaúl Cabrera 
222f0162a40SSaúl Cabrera     /// Returns the underlying memory value if it is one, panics otherwise.
223f0162a40SSaúl Cabrera     pub fn unwrap_mem(&self) -> Memory {
224f0162a40SSaúl Cabrera         match self {
225f0162a40SSaúl Cabrera             Self::Memory(m) => *m,
226f0162a40SSaúl Cabrera             v => panic!("expected value {:?} to be a Memory", v),
227f0162a40SSaúl Cabrera         }
228f0162a40SSaúl Cabrera     }
229f0162a40SSaúl Cabrera 
230835abbcdSSaúl Cabrera     /// Check whether the value is an i32 constant.
231835abbcdSSaúl Cabrera     pub fn is_i32_const(&self) -> bool {
232835abbcdSSaúl Cabrera         match *self {
233835abbcdSSaúl Cabrera             Self::I32(_) => true,
234835abbcdSSaúl Cabrera             _ => false,
235835abbcdSSaúl Cabrera         }
236835abbcdSSaúl Cabrera     }
2377c5c7e4bSSaúl Cabrera 
2387c5c7e4bSSaúl Cabrera     /// Check whether the value is an i64 constant.
2397c5c7e4bSSaúl Cabrera     pub fn is_i64_const(&self) -> bool {
2407c5c7e4bSSaúl Cabrera         match *self {
2417c5c7e4bSSaúl Cabrera             Self::I64(_) => true,
2427c5c7e4bSSaúl Cabrera             _ => false,
2437c5c7e4bSSaúl Cabrera         }
2447c5c7e4bSSaúl Cabrera     }
24514b39bc2SSaúl Cabrera 
24614b39bc2SSaúl Cabrera     /// Get the type of the value.
24714b39bc2SSaúl Cabrera     pub fn ty(&self) -> WasmType {
24814b39bc2SSaúl Cabrera         match self {
24914b39bc2SSaúl Cabrera             Val::I32(_) => WasmType::I32,
25014b39bc2SSaúl Cabrera             Val::I64(_) => WasmType::I64,
25114b39bc2SSaúl Cabrera             Val::F32(_) => WasmType::F32,
25214b39bc2SSaúl Cabrera             Val::F64(_) => WasmType::F64,
25314b39bc2SSaúl Cabrera             Val::Reg(r) => r.ty,
25414b39bc2SSaúl Cabrera             Val::Memory(m) => m.ty,
25514b39bc2SSaúl Cabrera             Val::Local(l) => l.ty,
25614b39bc2SSaúl Cabrera         }
25714b39bc2SSaúl Cabrera     }
258835abbcdSSaúl Cabrera }
259835abbcdSSaúl Cabrera 
260835abbcdSSaúl Cabrera /// The shadow stack used for compilation.
261835abbcdSSaúl Cabrera #[derive(Default, Debug)]
262835abbcdSSaúl Cabrera pub(crate) struct Stack {
263*7690c500STrevor Elliott     inner: Vec<Val>,
264835abbcdSSaúl Cabrera }
265835abbcdSSaúl Cabrera 
266835abbcdSSaúl Cabrera impl Stack {
267835abbcdSSaúl Cabrera     /// Allocate a new stack.
268835abbcdSSaúl Cabrera     pub fn new() -> Self {
269835abbcdSSaúl Cabrera         Self {
270835abbcdSSaúl Cabrera             inner: Default::default(),
271835abbcdSSaúl Cabrera         }
272835abbcdSSaúl Cabrera     }
273835abbcdSSaúl Cabrera 
274321294a5SJeffrey Charles     /// Returns true if the stack contains a local with the provided index
275321294a5SJeffrey Charles     /// except if the only time the local appears is the top element.
276321294a5SJeffrey Charles     pub fn contains_latent_local(&self, index: u32) -> bool {
277321294a5SJeffrey Charles         self.inner
278321294a5SJeffrey Charles             .iter()
279321294a5SJeffrey Charles             // Iterate top-to-bottom so we can skip the top element and stop
280321294a5SJeffrey Charles             // when we see a memory element.
281321294a5SJeffrey Charles             .rev()
282321294a5SJeffrey Charles             // The local is not latent if it's the top element because the top
283321294a5SJeffrey Charles             // element will be popped next which materializes the local.
284321294a5SJeffrey Charles             .skip(1)
285321294a5SJeffrey Charles             // Stop when we see a memory element because that marks where we
286321294a5SJeffrey Charles             // spilled up to so there will not be any locals past this point.
287321294a5SJeffrey Charles             .take_while(|v| !v.is_mem())
288321294a5SJeffrey Charles             .any(|v| v.is_local_at_index(index))
289321294a5SJeffrey Charles     }
290321294a5SJeffrey Charles 
291a109d2abSSaúl Cabrera     /// Extend the stack with the given elements.
292a109d2abSSaúl Cabrera     pub fn extend(&mut self, values: impl IntoIterator<Item = Val>) {
293a109d2abSSaúl Cabrera         self.inner.extend(values);
294a109d2abSSaúl Cabrera     }
295a109d2abSSaúl Cabrera 
296a109d2abSSaúl Cabrera     /// Inserts many values at the given index.
297a109d2abSSaúl Cabrera     pub fn insert_many(&mut self, at: usize, values: impl IntoIterator<Item = Val>) {
298a109d2abSSaúl Cabrera         debug_assert!(at <= self.len());
299a109d2abSSaúl Cabrera         // If last, simply extend.
300a109d2abSSaúl Cabrera         if at == self.inner.len() {
301a109d2abSSaúl Cabrera             self.inner.extend(values);
302a109d2abSSaúl Cabrera         } else {
303a109d2abSSaúl Cabrera             let mut tail = self.inner.split_off(at);
304a109d2abSSaúl Cabrera             self.inner.extend(values);
305a109d2abSSaúl Cabrera             self.inner.append(&mut tail);
306a109d2abSSaúl Cabrera         }
30720c58362SSaúl Cabrera     }
30820c58362SSaúl Cabrera 
309af4d94c8SSaúl Cabrera     /// Get the length of the stack.
310af4d94c8SSaúl Cabrera     pub fn len(&self) -> usize {
311af4d94c8SSaúl Cabrera         self.inner.len()
312af4d94c8SSaúl Cabrera     }
313af4d94c8SSaúl Cabrera 
314835abbcdSSaúl Cabrera     /// Push a value to the stack.
315835abbcdSSaúl Cabrera     pub fn push(&mut self, val: Val) {
316*7690c500STrevor Elliott         self.inner.push(val);
317835abbcdSSaúl Cabrera     }
318835abbcdSSaúl Cabrera 
319835abbcdSSaúl Cabrera     /// Peek into the top in the stack.
3207c5c7e4bSSaúl Cabrera     pub fn peek(&self) -> Option<&Val> {
321*7690c500STrevor Elliott         self.inner.last()
322835abbcdSSaúl Cabrera     }
323835abbcdSSaúl Cabrera 
324af4d94c8SSaúl Cabrera     /// Returns an iterator referencing the last n items of the stack,
325af4d94c8SSaúl Cabrera     /// in bottom-most to top-most order.
326af4d94c8SSaúl Cabrera     pub fn peekn(&self, n: usize) -> impl Iterator<Item = &Val> + '_ {
327af4d94c8SSaúl Cabrera         let len = self.len();
328af4d94c8SSaúl Cabrera         assert!(n <= len);
329af4d94c8SSaúl Cabrera 
330af4d94c8SSaúl Cabrera         let partition = len - n;
331*7690c500STrevor Elliott         self.inner[partition..].into_iter()
332af4d94c8SSaúl Cabrera     }
333af4d94c8SSaúl Cabrera 
334f0162a40SSaúl Cabrera     /// Duplicates the top `n` elements of the stack.
335f0162a40SSaúl Cabrera     // Will be needed for control flow, it's just not integrated yet.
336f0162a40SSaúl Cabrera     #[allow(dead_code)]
337f0162a40SSaúl Cabrera     pub fn dup(&mut self, n: usize) {
338f0162a40SSaúl Cabrera         let len = self.len();
339f0162a40SSaúl Cabrera         assert!(n <= len);
340f0162a40SSaúl Cabrera         let partition = len - n;
341f0162a40SSaúl Cabrera 
342f0162a40SSaúl Cabrera         if n > 0 {
343f0162a40SSaúl Cabrera             for e in partition..len {
344f0162a40SSaúl Cabrera                 if let Some(v) = self.inner.get(e) {
345f0162a40SSaúl Cabrera                     self.push(*v)
346f0162a40SSaúl Cabrera                 }
347f0162a40SSaúl Cabrera             }
348f0162a40SSaúl Cabrera         }
349f0162a40SSaúl Cabrera     }
350f0162a40SSaúl Cabrera 
351835abbcdSSaúl Cabrera     /// Pops the top element of the stack, if any.
352835abbcdSSaúl Cabrera     pub fn pop(&mut self) -> Option<Val> {
353*7690c500STrevor Elliott         self.inner.pop()
354835abbcdSSaúl Cabrera     }
355835abbcdSSaúl Cabrera 
3567c5c7e4bSSaúl Cabrera     /// Pops the element at the top of the stack if it is an i32 const;
357835abbcdSSaúl Cabrera     /// returns `None` otherwise.
358835abbcdSSaúl Cabrera     pub fn pop_i32_const(&mut self) -> Option<i32> {
359835abbcdSSaúl Cabrera         match self.peek() {
360f0162a40SSaúl Cabrera             Some(v) => v.is_i32_const().then(|| self.pop().unwrap().unwrap_i32()),
361835abbcdSSaúl Cabrera             _ => None,
362835abbcdSSaúl Cabrera         }
363835abbcdSSaúl Cabrera     }
364835abbcdSSaúl Cabrera 
3657c5c7e4bSSaúl Cabrera     /// Pops the element at the top of the stack if it is an i64 const;
3667c5c7e4bSSaúl Cabrera     /// returns `None` otherwise.
3677c5c7e4bSSaúl Cabrera     pub fn pop_i64_const(&mut self) -> Option<i64> {
3687c5c7e4bSSaúl Cabrera         match self.peek() {
369f0162a40SSaúl Cabrera             Some(v) => v.is_i64_const().then(|| self.pop().unwrap().unwrap_i64()),
3707c5c7e4bSSaúl Cabrera             _ => None,
3717c5c7e4bSSaúl Cabrera         }
3727c5c7e4bSSaúl Cabrera     }
3737c5c7e4bSSaúl Cabrera 
374835abbcdSSaúl Cabrera     /// Pops the element at the top of the stack if it is a register;
375835abbcdSSaúl Cabrera     /// returns `None` otherwise.
37614b39bc2SSaúl Cabrera     pub fn pop_reg(&mut self) -> Option<TypedReg> {
377835abbcdSSaúl Cabrera         match self.peek() {
378f0162a40SSaúl Cabrera             Some(v) => v.is_reg().then(|| self.pop().unwrap().unwrap_reg()),
379835abbcdSSaúl Cabrera             _ => None,
380835abbcdSSaúl Cabrera         }
381835abbcdSSaúl Cabrera     }
382835abbcdSSaúl Cabrera 
383835abbcdSSaúl Cabrera     /// Pops the given register if it is at the top of the stack;
384835abbcdSSaúl Cabrera     /// returns `None` otherwise.
38514b39bc2SSaúl Cabrera     pub fn pop_named_reg(&mut self, reg: Reg) -> Option<TypedReg> {
386835abbcdSSaúl Cabrera         match self.peek() {
38714b39bc2SSaúl Cabrera             Some(v) => {
388f0162a40SSaúl Cabrera                 (v.is_reg() && v.unwrap_reg().reg == reg).then(|| self.pop().unwrap().unwrap_reg())
38914b39bc2SSaúl Cabrera             }
390835abbcdSSaúl Cabrera             _ => None,
391835abbcdSSaúl Cabrera         }
392835abbcdSSaúl Cabrera     }
393835abbcdSSaúl Cabrera 
394835abbcdSSaúl Cabrera     /// Get a mutable reference to the inner stack representation.
395*7690c500STrevor Elliott     pub fn inner_mut(&mut self) -> &mut Vec<Val> {
396835abbcdSSaúl Cabrera         &mut self.inner
397835abbcdSSaúl Cabrera     }
39850733725SSaúl Cabrera 
39950733725SSaúl Cabrera     /// Calculates the size of, in bytes, of the top n [Memory] entries
40050733725SSaúl Cabrera     /// in the value stack.
40150733725SSaúl Cabrera     pub fn sizeof(&self, top: usize) -> u32 {
40250733725SSaúl Cabrera         self.peekn(top).fold(0, |acc, v| {
40350733725SSaúl Cabrera             if v.is_mem() {
40450733725SSaúl Cabrera                 acc + v.unwrap_mem().slot.size
40550733725SSaúl Cabrera             } else {
40650733725SSaúl Cabrera                 acc
40750733725SSaúl Cabrera             }
40850733725SSaúl Cabrera         })
40950733725SSaúl Cabrera     }
410835abbcdSSaúl Cabrera }
411835abbcdSSaúl Cabrera 
412835abbcdSSaúl Cabrera #[cfg(test)]
413835abbcdSSaúl Cabrera mod tests {
414835abbcdSSaúl Cabrera     use super::{Stack, Val};
415835abbcdSSaúl Cabrera     use crate::isa::reg::Reg;
41614b39bc2SSaúl Cabrera     use wasmtime_environ::WasmType;
417835abbcdSSaúl Cabrera 
418835abbcdSSaúl Cabrera     #[test]
419835abbcdSSaúl Cabrera     fn test_pop_i32_const() {
420835abbcdSSaúl Cabrera         let mut stack = Stack::new();
421835abbcdSSaúl Cabrera         stack.push(Val::i32(33i32));
422835abbcdSSaúl Cabrera         assert_eq!(33, stack.pop_i32_const().unwrap());
423835abbcdSSaúl Cabrera 
42414b39bc2SSaúl Cabrera         stack.push(Val::local(10, WasmType::I32));
425835abbcdSSaúl Cabrera         assert!(stack.pop_i32_const().is_none());
426835abbcdSSaúl Cabrera     }
427835abbcdSSaúl Cabrera 
428835abbcdSSaúl Cabrera     #[test]
429835abbcdSSaúl Cabrera     fn test_pop_reg() {
430835abbcdSSaúl Cabrera         let mut stack = Stack::new();
431835abbcdSSaúl Cabrera         let reg = Reg::int(2usize);
43214b39bc2SSaúl Cabrera         stack.push(Val::reg(reg, WasmType::I32));
433835abbcdSSaúl Cabrera         stack.push(Val::i32(4));
434835abbcdSSaúl Cabrera 
435835abbcdSSaúl Cabrera         assert_eq!(None, stack.pop_reg());
436835abbcdSSaúl Cabrera         let _ = stack.pop().unwrap();
43714b39bc2SSaúl Cabrera         assert_eq!(reg, stack.pop_reg().unwrap().reg);
438835abbcdSSaúl Cabrera     }
439835abbcdSSaúl Cabrera 
440835abbcdSSaúl Cabrera     #[test]
441835abbcdSSaúl Cabrera     fn test_pop_named_reg() {
442835abbcdSSaúl Cabrera         let mut stack = Stack::new();
443835abbcdSSaúl Cabrera         let reg = Reg::int(2usize);
44414b39bc2SSaúl Cabrera         stack.push(Val::reg(reg, WasmType::I32));
44514b39bc2SSaúl Cabrera         stack.push(Val::reg(Reg::int(4), WasmType::I32));
446835abbcdSSaúl Cabrera 
447835abbcdSSaúl Cabrera         assert_eq!(None, stack.pop_named_reg(reg));
448835abbcdSSaúl Cabrera         let _ = stack.pop().unwrap();
44914b39bc2SSaúl Cabrera         assert_eq!(reg, stack.pop_named_reg(reg).unwrap().reg);
450835abbcdSSaúl Cabrera     }
451835abbcdSSaúl Cabrera }
452