Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Override component model Lower::store_list and Lift::load_list for f32/f64 #9892

Draft
wants to merge 1 commit into
base: main
Choose a base branch
from
Draft
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
78 changes: 75 additions & 3 deletions crates/wasmtime/src/runtime/component/func/typed.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ use core::marker;
use core::mem::{self, MaybeUninit};
use core::ptr::NonNull;
use core::str;
use std::iter;
use wasmtime_environ::component::{
CanonicalAbiInfo, ComponentTypes, InterfaceType, StringEncoding, VariantInfo, MAX_FLAT_PARAMS,
MAX_FLAT_RESULTS,
Expand Down Expand Up @@ -874,7 +875,7 @@ integers! {
}

macro_rules! floats {
($($float:ident/$get_float:ident = $ty:ident with abi:$abi:ident)*) => ($(const _: () = {
($($float:ident/$get_float:ident = $ty:ident with abi:$abi:ident $integer:ident)*) => ($(const _: () = {
unsafe impl ComponentType for $float {
type Lower = ValRaw;

Expand Down Expand Up @@ -914,6 +915,44 @@ macro_rules! floats {
*ptr = self.to_bits().to_le_bytes();
Ok(())
}

fn store_list<T>(
cx: &mut LowerContext<'_, T>,
ty: InterfaceType,
offset: usize,
items: &[Self],
) -> Result<()> {
debug_assert!(matches!(ty, InterfaceType::$ty));

// Double-check that the CM alignment is at least the host's
// alignment for this type which should be true for all
// platforms.
assert!((Self::ALIGN32 as usize) >= mem::align_of::<Self>());

// Slice `cx`'s memory to the window that we'll be modifying.
// This should all have already been verified in terms of
// alignment and sizing meaning that these assertions here are
// not truly necessary but are instead double-checks.
//
// Note that we're casting a `[u8]` slice to `[$integer]` (with
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This paragraph (and the code that follows) refers to $integer, but we really care about $float, right? We should rewrite this to talk about floating point numbers instead of integers, e.g. "all u8 patterns are valid $float patterns since $float is a IEEE 754 floating point type."

// $integer having the same size as Self) with `align_to_mut`
// which is not safe in general but is safe in our specific
// case as all `u8` patterns are valid `$integer` patterns
// since `$integer` is an integral type.
let dst = &mut cx.as_slice_mut()[offset..][..items.len() * Self::SIZE32];
let (before, middle, end) = unsafe { dst.align_to_mut::<$integer>() };
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
let (before, middle, end) = unsafe { dst.align_to_mut::<$integer>() };
let (before, middle, end) = unsafe { dst.align_to_mut::<$float>() };

Presumably the alignment of e.g. f32 and u32 should be the same, but we might as well be precise here (and in similar code below).

assert!(before.is_empty() && end.is_empty());
assert_eq!(middle.len(), items.len());

// And with all that out of the way perform the copying loop.
// This is not a `copy_from_slice` because endianness needs to
// be handled here, but LLVM should pretty easily transform this
// into a memcpy on little-endian platforms.
for (dst, src) in iter::zip(middle, items) {
*dst = src.to_bits().to_le();
}
Ok(())
}
}

unsafe impl Lift for $float {
Expand All @@ -929,13 +968,46 @@ macro_rules! floats {
debug_assert!((bytes.as_ptr() as usize) % Self::SIZE32 == 0);
Ok($float::from_le_bytes(bytes.try_into().unwrap()))
}

fn load_list(cx: &mut LiftContext<'_>, list: &WasmList<Self>) -> Result<Vec<Self>> where Self: Sized {
// See comments in `WasmList::get` for the panicking indexing
let byte_size = list.len * mem::size_of::<$integer>();
let bytes = &cx.memory()[list.ptr..][..byte_size];

// The canonical ABI requires that everything is aligned to its
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Again, these comments should be updated to discuss floating point types rather than integer types.

// own size, so this should be an aligned array. Furthermore the
// alignment of primitive integers for hosts should be smaller
// than or equal to the size of the primitive itself, meaning
// that a wasm canonical-abi-aligned list is also aligned for
// the host. That should mean that the head/tail slices here are
// empty.
//
// Also note that the `unsafe` here is needed since the type
// we're aligning to isn't guaranteed to be valid, but in our
// case it's just integers and bytes so this should be safe.

let slice = unsafe {
let (head, body, tail) = bytes.align_to::<$integer>();
assert!(head.is_empty() && tail.is_empty());
body
};

// Copy the resulting slice to a new Vec, handling endianness
// in the process
Ok(
slice
.iter()
.map(|i| $float::from_bits($integer::from_le(*i)))
.collect()
)
}
}
};)*)
}

floats! {
f32/get_f32 = Float32 with abi:SCALAR4
f64/get_f64 = Float64 with abi:SCALAR8
f32/get_f32 = Float32 with abi:SCALAR4 u32
f64/get_f64 = Float64 with abi:SCALAR8 u64
}

unsafe impl ComponentType for bool {
Expand Down
Loading