Struct StackMapSection

Source
pub struct StackMapSection { /* private fields */ }
Expand description

Builder for the ELF_WASMTIME_STACK_MAP section in compiled executables.

This format is parsed by crate::stack_map.

The current layout of the format is:

┌─────────────────────┬───── 0x00 (relative, not necessarily aligned)
│ count: 4-byte LE    │
├─────────────────────┼───── 0x04
│ pc1: 4-byte LE      │
│ pc2: 4-byte LE      │
│ ...                 │
│ pcN: 4-byte LE      │
├─────────────────────┼───── 0x04 + 4 * count
│ offset1: 4-byte LE  │
│ offset1: 4-byte LE  │
│ ...                 │
│ offsetN: 4-byte LE  │
├─────────────────────┼───── 0x04 + 8 * count
│ data[0]: 4-byte LE  │
│ data[1]: 4-byte LE  │
│ ...                 │
│ data[M]: 4-byte LE  │
└─────────────────────┴───── 0x04 + 8 * count + 4 * M

Here count is the size of the pcN and offsetN arrays. The two arrays are the same size and have corresponding entries in one another. When looking up a stack map for a particular program counter:

  • A binary search is performed on the pcN array.
  • The corresponding offsetM value is looked up once the pcM entry, matching the lookup pc, is found.
  • The offsetM value is used to access data[offsetM] which is an array of 4-byte entries located after the offset* array. This stack map is then encoded as below.

This encoding scheme is chosen so parsing this data structure effectively isn’t required. It’s usable at-rest from a compiled artifact in a section of an executable. Notably having offsets into the data array means that a stack map is just a slice into the data array, and the entire data structure can be “parsed” by reading count and otherwise just making sure various offsets are in-bounds.

A stack map located at data[offsetM] is encoded as:

┌───────────────────────────────────────────────────────┐
│ data[offsetM + 0]: frame_size: 4-byte LE              │
├───────────────────────────────────────────────────────┤
│ data[offsetM + 1]: count: 4-byte LE                   │
├───────────────────────────────────────────────────────┤
│ data[offsetM + 2 + 0]: bitmap: 4-byte LE              │
│ data[offsetM + 2 + 1]: bitmap: 4-byte LE              │
│ ...                                                   │
│ data[offsetM + 2 + count - 1]: bitmap: 4-byte LE      │
└───────────────────────────────────────────────────────┘

Here frame_size and count are always greater than 0. Entries in the bit map represent stack_slot / 4 so must be multiplied by 4 to get the actual stack offset entry. This is because all stack slots are aligned at 4 bytes so by dividing them all by 4 we’re able to compress the bit map that much more.

Implementations§

Source§

impl StackMapSection

Source

pub fn push( &mut self, code_offset: u64, frame_size: u32, frame_offsets: impl ExactSizeIterator<Item = u32>, )

Appends stack map information for code_offset which has the specified frame_size and frame_offsets are the active GC references.

Source

pub fn append_to(self, obj: &mut Object<'_>)

Finishes encoding this section into the Object provided.

Trait Implementations§

Source§

impl Default for StackMapSection

Source§

fn default() -> StackMapSection

Returns the “default value” for a type. Read more

Auto Trait Implementations§

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.