zerocopy

Trait IntoBytes

Source
pub unsafe trait IntoBytes {
    // Provided methods
    fn as_bytes(&self) -> &[u8]
       where Self: Immutable { ... }
    fn as_mut_bytes(&mut self) -> &mut [u8]
       where Self: FromBytes { ... }
    fn write_to(
        &self,
        dst: &mut [u8],
    ) -> Result<(), SizeError<&Self, &mut [u8]>>
       where Self: Immutable { ... }
    fn write_to_prefix(
        &self,
        dst: &mut [u8],
    ) -> Result<(), SizeError<&Self, &mut [u8]>>
       where Self: Immutable { ... }
    fn write_to_suffix(
        &self,
        dst: &mut [u8],
    ) -> Result<(), SizeError<&Self, &mut [u8]>>
       where Self: Immutable { ... }
}
Expand description

Types that can be converted to an immutable slice of initialized bytes.

Any IntoBytes type can be converted to a slice of initialized bytes of the same size. This is useful for efficiently serializing structured data as raw bytes.

§Implementation

Do not implement this trait yourself! Instead, use #[derive(IntoBytes)]; e.g.:

#[derive(IntoBytes)]
#[repr(C)]
struct MyStruct {
    ...
}

#[derive(IntoBytes)]
#[repr(u8)]
enum MyEnum {
    ...
}

This derive performs a sophisticated, compile-time safety analysis to determine whether a type is IntoBytes. See the derive documentation for guidance on how to interpret error messages produced by the derive’s analysis.

§Safety

This section describes what is required in order for T: IntoBytes, and what unsafe code may assume of such types. If you don’t plan on implementing IntoBytes manually, and you don’t plan on writing unsafe code that operates on IntoBytes types, then you don’t need to read this section.

If T: IntoBytes, then unsafe code may assume that it is sound to treat any t: T as an immutable [u8] of length size_of_val(t). If a type is marked as IntoBytes which violates this contract, it may cause undefined behavior.

#[derive(IntoBytes)] only permits types which satisfy these requirements.

Provided Methods§

Source

fn as_bytes(&self) -> &[u8]
where Self: Immutable,

Gets the bytes of this value.

§Examples
use zerocopy::IntoBytes;

#[derive(IntoBytes, Immutable)]
#[repr(C)]
struct PacketHeader {
    src_port: [u8; 2],
    dst_port: [u8; 2],
    length: [u8; 2],
    checksum: [u8; 2],
}

let header = PacketHeader {
    src_port: [0, 1],
    dst_port: [2, 3],
    length: [4, 5],
    checksum: [6, 7],
};

let bytes = header.as_bytes();

assert_eq!(bytes, [0, 1, 2, 3, 4, 5, 6, 7]);
Source

fn as_mut_bytes(&mut self) -> &mut [u8]
where Self: FromBytes,

Gets the bytes of this value mutably.

§Examples
use zerocopy::IntoBytes;

#[derive(FromBytes, IntoBytes, Immutable)]
#[repr(C)]
struct PacketHeader {
    src_port: [u8; 2],
    dst_port: [u8; 2],
    length: [u8; 2],
    checksum: [u8; 2],
}

let mut header = PacketHeader {
    src_port: [0, 1],
    dst_port: [2, 3],
    length: [4, 5],
    checksum: [6, 7],
};

let bytes = header.as_mut_bytes();

assert_eq!(bytes, [0, 1, 2, 3, 4, 5, 6, 7]);

bytes.reverse();

assert_eq!(header, PacketHeader {
    src_port: [7, 6],
    dst_port: [5, 4],
    length: [3, 2],
    checksum: [1, 0],
});
Source

fn write_to(&self, dst: &mut [u8]) -> Result<(), SizeError<&Self, &mut [u8]>>
where Self: Immutable,

Writes a copy of self to dst.

If dst.len() != size_of_val(self), write_to returns Err.

§Examples
use zerocopy::IntoBytes;

#[derive(IntoBytes, Immutable)]
#[repr(C)]
struct PacketHeader {
    src_port: [u8; 2],
    dst_port: [u8; 2],
    length: [u8; 2],
    checksum: [u8; 2],
}

let header = PacketHeader {
    src_port: [0, 1],
    dst_port: [2, 3],
    length: [4, 5],
    checksum: [6, 7],
};

let mut bytes = [0, 0, 0, 0, 0, 0, 0, 0];

header.write_to(&mut bytes[..]);

assert_eq!(bytes, [0, 1, 2, 3, 4, 5, 6, 7]);

If too many or too few target bytes are provided, write_to returns Err and leaves the target bytes unmodified:

let mut excessive_bytes = &mut [0u8; 128][..];

let write_result = header.write_to(excessive_bytes);

assert!(write_result.is_err());
assert_eq!(excessive_bytes, [0u8; 128]);
Source

fn write_to_prefix( &self, dst: &mut [u8], ) -> Result<(), SizeError<&Self, &mut [u8]>>
where Self: Immutable,

Writes a copy of self to the prefix of dst.

write_to_prefix writes self to the first size_of_val(self) bytes of dst. If dst.len() < size_of_val(self), it returns Err.

§Examples
use zerocopy::IntoBytes;

#[derive(IntoBytes, Immutable)]
#[repr(C)]
struct PacketHeader {
    src_port: [u8; 2],
    dst_port: [u8; 2],
    length: [u8; 2],
    checksum: [u8; 2],
}

let header = PacketHeader {
    src_port: [0, 1],
    dst_port: [2, 3],
    length: [4, 5],
    checksum: [6, 7],
};

let mut bytes = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0];

header.write_to_prefix(&mut bytes[..]);

assert_eq!(bytes, [0, 1, 2, 3, 4, 5, 6, 7, 0, 0]);

If insufficient target bytes are provided, write_to_prefix returns Err and leaves the target bytes unmodified:

let mut insufficent_bytes = &mut [0, 0][..];

let write_result = header.write_to_suffix(insufficent_bytes);

assert!(write_result.is_err());
assert_eq!(insufficent_bytes, [0, 0]);
Source

fn write_to_suffix( &self, dst: &mut [u8], ) -> Result<(), SizeError<&Self, &mut [u8]>>
where Self: Immutable,

Writes a copy of self to the suffix of dst.

write_to_suffix writes self to the last size_of_val(self) bytes of dst. If dst.len() < size_of_val(self), it returns Err.

§Examples
use zerocopy::IntoBytes;

#[derive(IntoBytes, Immutable)]
#[repr(C)]
struct PacketHeader {
    src_port: [u8; 2],
    dst_port: [u8; 2],
    length: [u8; 2],
    checksum: [u8; 2],
}

let header = PacketHeader {
    src_port: [0, 1],
    dst_port: [2, 3],
    length: [4, 5],
    checksum: [6, 7],
};

let mut bytes = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0];

header.write_to_suffix(&mut bytes[..]);

assert_eq!(bytes, [0, 0, 0, 1, 2, 3, 4, 5, 6, 7]);

let mut insufficent_bytes = &mut [0, 0][..];

let write_result = header.write_to_suffix(insufficent_bytes);

assert!(write_result.is_err());
assert_eq!(insufficent_bytes, [0, 0]);

If insufficient target bytes are provided, write_to_suffix returns Err and leaves the target bytes unmodified:

let mut insufficent_bytes = &mut [0, 0][..];

let write_result = header.write_to_suffix(insufficent_bytes);

assert!(write_result.is_err());
assert_eq!(insufficent_bytes, [0, 0]);

Dyn Compatibility§

This trait is not dyn compatible.

In older versions of Rust, dyn compatibility was called "object safety", so this trait is not object safe.

Implementations on Foreign Types§

Source§

impl IntoBytes for Option<NonZeroI8>

Source§

impl IntoBytes for Option<NonZeroI16>

Source§

impl IntoBytes for Option<NonZeroI32>

Source§

impl IntoBytes for Option<NonZeroI64>

Source§

impl IntoBytes for Option<NonZeroI128>

Source§

impl IntoBytes for Option<NonZeroIsize>

Source§

impl IntoBytes for Option<NonZeroU8>

Source§

impl IntoBytes for Option<NonZeroU16>

Source§

impl IntoBytes for Option<NonZeroU32>

Source§

impl IntoBytes for Option<NonZeroU64>

Source§

impl IntoBytes for Option<NonZeroU128>

Source§

impl IntoBytes for Option<NonZeroUsize>

Source§

impl IntoBytes for bool

Source§

impl IntoBytes for char

Source§

impl IntoBytes for f32

Source§

impl IntoBytes for f64

Source§

impl IntoBytes for i8

Source§

impl IntoBytes for i16

Source§

impl IntoBytes for i32

Source§

impl IntoBytes for i64

Source§

impl IntoBytes for i128

Source§

impl IntoBytes for isize

Source§

impl IntoBytes for str

Source§

impl IntoBytes for u8

Source§

impl IntoBytes for u16

Source§

impl IntoBytes for u32

Source§

impl IntoBytes for u64

Source§

impl IntoBytes for u128

Source§

impl IntoBytes for ()

Source§

impl IntoBytes for usize

Source§

impl IntoBytes for __m128

Source§

impl IntoBytes for __m128d

Source§

impl IntoBytes for __m128i

Source§

impl IntoBytes for __m256

Source§

impl IntoBytes for __m256d

Source§

impl IntoBytes for __m256i

Source§

impl IntoBytes for AtomicBool

Source§

impl IntoBytes for AtomicI8

Source§

impl IntoBytes for AtomicI16

Source§

impl IntoBytes for AtomicI32

Source§

impl IntoBytes for AtomicI64

Source§

impl IntoBytes for AtomicIsize

Source§

impl IntoBytes for AtomicU8

Source§

impl IntoBytes for AtomicU16

Source§

impl IntoBytes for AtomicU32

Source§

impl IntoBytes for AtomicU64

Source§

impl IntoBytes for AtomicUsize

Source§

impl IntoBytes for NonZeroI8

Source§

impl IntoBytes for NonZeroI16

Source§

impl IntoBytes for NonZeroI32

Source§

impl IntoBytes for NonZeroI64

Source§

impl IntoBytes for NonZeroI128

Source§

impl IntoBytes for NonZeroIsize

Source§

impl IntoBytes for NonZeroU8

Source§

impl IntoBytes for NonZeroU16

Source§

impl IntoBytes for NonZeroU32

Source§

impl IntoBytes for NonZeroU64

Source§

impl IntoBytes for NonZeroU128

Source§

impl IntoBytes for NonZeroUsize

Source§

impl<T: IntoBytes> IntoBytes for [T]

Source§

impl<T: IntoBytes> IntoBytes for Wrapping<T>

Source§

impl<T: IntoBytes, const N: usize> IntoBytes for [T; N]

Source§

impl<T: ?Sized + IntoBytes> IntoBytes for UnsafeCell<T>

Source§

impl<T: ?Sized + IntoBytes> IntoBytes for ManuallyDrop<T>

Source§

impl<T: ?Sized> IntoBytes for PhantomData<T>

Implementors§

Source§

impl<O> IntoBytes for F32<O>

Source§

impl<O> IntoBytes for F64<O>

Source§

impl<O> IntoBytes for I16<O>

Source§

impl<O> IntoBytes for I32<O>

Source§

impl<O> IntoBytes for I64<O>

Source§

impl<O> IntoBytes for I128<O>

Source§

impl<O> IntoBytes for Isize<O>

Source§

impl<O> IntoBytes for U16<O>

Source§

impl<O> IntoBytes for U32<O>

Source§

impl<O> IntoBytes for U64<O>

Source§

impl<O> IntoBytes for U128<O>

Source§

impl<O> IntoBytes for Usize<O>

Source§

impl<T> IntoBytes for Unalign<T>
where T: IntoBytes,