diff --git a/Cargo.toml b/Cargo.toml index f725b8da48..4e9173ca49 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -74,6 +74,8 @@ alloc = [] derive = ["zerocopy-derive"] simd = [] simd-nightly = ["simd"] +# This feature enables support for the unstable f16 and f128 types +float-nightly = [] std = ["alloc"] # This feature depends on all other features that work on the stable compiler. # We make no stability guarantees about this feature; it may be modified or diff --git a/src/impls.rs b/src/impls.rs index 8d05537d2b..8e4f5be9e0 100644 --- a/src/impls.rs +++ b/src/impls.rs @@ -72,6 +72,8 @@ safety_comment! { unsafe_impl!(isize: Immutable, TryFromBytes, FromZeros, FromBytes, IntoBytes); unsafe_impl!(f32: Immutable, TryFromBytes, FromZeros, FromBytes, IntoBytes); unsafe_impl!(f64: Immutable, TryFromBytes, FromZeros, FromBytes, IntoBytes); + unsafe_impl!(#[cfg(feature="float-nightly")] f16: Immutable, TryFromBytes, FromZeros, FromBytes, IntoBytes); + unsafe_impl!(#[cfg(feature="float-nightly")] f128: Immutable, TryFromBytes, FromZeros, FromBytes, IntoBytes); } safety_comment! { diff --git a/src/lib.rs b/src/lib.rs index 1f9e519eef..e443c9de05 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -303,6 +303,7 @@ all(feature = "simd-nightly", any(target_arch = "powerpc", target_arch = "powerpc64")), feature(stdarch_powerpc) )] +#![cfg_attr(feature = "float-nightly", feature(f16, f128))] #![cfg_attr(doc_cfg, feature(doc_cfg))] #![cfg_attr( __ZEROCOPY_INTERNAL_USE_ONLY_NIGHTLY_FEATURES_IN_TESTS, diff --git a/src/util/macros.rs b/src/util/macros.rs index 4745f9556c..a96c5760e9 100644 --- a/src/util/macros.rs +++ b/src/util/macros.rs @@ -66,10 +66,37 @@ macro_rules! unsafe_impl { unsafe_impl!(@method $trait $(; |$candidate: MaybeAligned<$repr>| $is_bit_valid)?); } }; + // Implement all `$traits` for `$ty` with no bounds. - ($ty:ty: $($traits:ident),*) => { - $( unsafe_impl!($ty: $traits); )* + // + // The 2 arms under this one are there so we can apply + // N attributes for each one of M trait implementations. + // The simple solution of: + // + // ($(#[$attrs:meta])* $ty:ty: $($traits:ident),*) => { + // $( unsafe_impl!( $(#[$attrs])* $ty: $traits ) );* + // } + // + // Won't work. The macro processor sees that the outer repetition + // contains both $attrs and $traits and expects them to match the same + // amount of fragments. + // + // To solve this we must: + // 1. Pack the attributes into a single token tree fragment we can match over + // 2. Expand the traits. + // 3. Unpack and expand the attributes. + ($(#[$attrs:meta])* $ty:ty: $($traits:ident),*) => { + unsafe_impl!(@impl_traits_with_packed_attrs { $( #[$attrs])* } $ty: $($traits),*) }; + + (@impl_traits_with_packed_attrs $attrs:tt $ty:ty: $($traits:ident),*) => { + $( unsafe_impl!(@unpack_attrs $attrs $ty: $traits); )* + }; + + (@unpack_attrs { $(#[$attrs:meta])* } $ty:ty: $traits:ident) => { + unsafe_impl!($(#[$attrs])* $ty: $traits); + }; + // This arm is identical to the following one, except it contains a // preceding `const`. If we attempt to handle these with a single arm, there // is an inherent ambiguity between `const` (the keyword) and `const` (the