wasmtime_environ/
stack_map.rs1use cranelift_bitset::ScalarBitSet;
2use object::{Bytes, LittleEndian, U32Bytes};
3
4struct StackMapSection<'a> {
5 pcs: &'a [U32Bytes<LittleEndian>],
6 pointers_to_stack_map: &'a [U32Bytes<LittleEndian>],
7 stack_map_data: &'a [U32Bytes<LittleEndian>],
8}
9
10impl<'a> StackMapSection<'a> {
11 fn parse(section: &'a [u8]) -> Option<StackMapSection<'a>> {
12 let mut section = Bytes(section);
13 let pc_count = section.read::<U32Bytes<LittleEndian>>().ok()?;
16 let pc_count = usize::try_from(pc_count.get(LittleEndian)).ok()?;
17 let (pcs, section) =
18 object::slice_from_bytes::<U32Bytes<LittleEndian>>(section.0, pc_count).ok()?;
19 let (pointers_to_stack_map, section) =
20 object::slice_from_bytes::<U32Bytes<LittleEndian>>(section, pc_count).ok()?;
21 let stack_map_data =
22 object::slice_from_all_bytes::<U32Bytes<LittleEndian>>(section).ok()?;
23 Some(StackMapSection {
24 pcs,
25 pointers_to_stack_map,
26 stack_map_data,
27 })
28 }
29
30 fn lookup(&self, pc: u32) -> Option<StackMap<'a>> {
31 let pc_index = self
32 .pcs
33 .binary_search_by_key(&pc, |v| v.get(LittleEndian))
34 .ok()?;
35 self.get(pc_index)
36 }
37
38 fn into_iter(self) -> impl Iterator<Item = (u32, StackMap<'a>)> + 'a {
39 self.pcs
40 .iter()
41 .enumerate()
42 .map(move |(i, pc)| (pc.get(LittleEndian), self.get(i).unwrap()))
43 }
44
45 fn get(&self, i: usize) -> Option<StackMap<'a>> {
47 let pointer_to_stack_map = self.pointers_to_stack_map[i].get(LittleEndian) as usize;
48 let data = self.stack_map_data.get(pointer_to_stack_map..)?;
49
50 let (frame_size, data) = data.split_first()?;
51 let (count, data) = data.split_first()?;
52 let data = data.get(..count.get(LittleEndian) as usize)?;
53
54 Some(StackMap {
55 frame_size: frame_size.get(LittleEndian),
56 data,
57 })
58 }
59}
60
61pub struct StackMap<'a> {
67 frame_size: u32,
68 data: &'a [U32Bytes<LittleEndian>],
69}
70
71impl<'a> StackMap<'a> {
72 pub fn lookup(pc: u32, section: &'a [u8]) -> Option<StackMap<'a>> {
78 StackMapSection::parse(section)?.lookup(pc)
79 }
80
81 pub fn iter(section: &'a [u8]) -> Option<impl Iterator<Item = (u32, StackMap<'a>)> + 'a> {
90 Some(StackMapSection::parse(section)?.into_iter())
91 }
92
93 pub fn frame_size(&self) -> u32 {
95 self.frame_size
96 }
97
98 pub unsafe fn sp(&self, fp: *mut usize) -> *mut usize {
105 let frame_size = usize::try_from(self.frame_size).unwrap();
106 fp.byte_sub(frame_size)
107 }
108
109 pub unsafe fn live_gc_refs(&self, sp: *mut usize) -> impl Iterator<Item = *mut u32> + '_ {
117 self.offsets().map(move |i| {
118 log::trace!("Live GC ref in frame at frame offset {:#x}", i);
119 let i = usize::try_from(i).unwrap();
120 let ptr_to_gc_ref = sp.byte_add(i);
121
122 assert!({
124 let delta = ptr_to_gc_ref as usize - sp as usize;
125 let frame_size = usize::try_from(self.frame_size).unwrap();
126 delta < frame_size
127 });
128
129 ptr_to_gc_ref.cast::<u32>()
130 })
131 }
132
133 pub fn offsets(&self) -> impl Iterator<Item = u32> + '_ {
135 let bit_positions = self.data.iter().enumerate().flat_map(|(i, word)| {
138 ScalarBitSet(word.get(LittleEndian))
139 .iter()
140 .map(move |bit| (i as u32) * 32 + u32::from(bit))
141 });
142
143 bit_positions.map(|pos| pos * 4)
144 }
145}