1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
use core::option::Option;
use crate::{Cast, EncastMem, Endian, Flip};
#[cfg(doc)] use crate::BE;
///
/// Defines functions to `encast` and `Flip` on memory.
///
/// Note: In this crate, the term `encast` means decoding a number of
/// bytes to one or more values, the term `decast` means encoding one
/// or more variables to a number of bytes, and the term `endian-flip`
/// means flipping the endianness of value(s).
///
/// # Example
///
/// In the example below, function `encastf` decodes bytes in `bytes1`
/// in Big-Endian ([`BE`]) to variable `udp_hdr2` of type `UdpHdr`.
///
/// ```
/// # fn main() { test(); }
/// # fn test() -> Option<()> {
/// use castflip::experimental::EncastArg;
/// use castflip::{Cast, Flip, BE};
///
/// #[repr(C)]
/// #[derive(Cast, Flip)]
/// struct UdpHdr { // UDP: https://www.rfc-editor.org/rfc/rfc768.txt
/// sport: u16, // UDP Source Port
/// dport: u16, // UDP Destination Port
/// len: u16, // UDP Length
/// sum: u16, // UDP Checksum
/// }
///
/// // Input data: UDP header (8 bytes) + part of DNS header (8 bytes)
/// let bytes1: [u8; 16] = [0xC3, 0xC9, 0x00, 0x35, 0x00, 0x32, 0x82, 0x3F,
/// 0x1A, 0xD1, 0x01, 0x20, 0x00, 0x01, 0x00, 0x00];
///
/// // Decode input `bytes1` to variable `udp_hdr2`.
/// // Because the UDP header is 8 bytes as defined above, only
/// // the first 8 bytes are decoded, remaining 8 bytes are ignored.
/// let udp_hdr2 = UdpHdr::encastf(&bytes1, BE)?; // BE = Big-Endian
///
/// // Check the results (udp_hdr2)
/// assert_eq!(udp_hdr2.sport, 0xC3C9); // = 50121
/// assert_eq!(udp_hdr2.dport, 0x0035); // = 53 (DNS)
/// assert_eq!(udp_hdr2.len, 0x0032); // = 50
/// assert_eq!(udp_hdr2.sum, 0x823F);
/// # Some(())
/// # }
/// ```
///
/// # Description
///
/// All functions in trait `EncastArg` `encast` a number of bytes on
/// the memory to one or more values of the specified type. The
/// endianness of resulting value(s) is flipped when required and
/// necessary. Currently, only an implementation for `[u8]` is
/// provided.
///
/// The bytes in argument `bytes` should be larger than or equal to
/// the specified number of value(s) of the specified type `T`. If
/// there are enough bytes, the required number of bytes at the head
/// of `bytes` are decoded to the specified type of value(s), which
/// is/are returned in `Some`(). The remaining bytes are ignored. If
/// there are not enough bytes, `None` is returned.
///
/// When argument `endian` is specified, the endianness of value(s) is
/// flipped if necessary.
///
/// The API of this trait is an older form of [`EncastMem`].
///
pub trait EncastArg: Cast {
/// Decodes the bytes at the head of `bytes` to a value of type `T`.
/// The endianness of the resulting value is not flipped.
fn encast(bytes: &[u8]) -> Option<Self>;
/// Decodes the bytes at the head of `bytes` to a value of type `T`.
/// The endianness of the resulting value is flipped if necessary.
/// The endianness of the bytes is specified in `endian`.
fn encastf(bytes: &[u8], endian: Endian) -> Option<Self>
where
Self: Flip;
/// Decodes the bytes at the head of `bytes` to a vector of
/// value(s) of type `T`. The endianness of the resulting
/// value(s) is not flipped. The number of elements in the
/// resulting vecotr is specified in `nelem`.
#[cfg(feature = "std")]
fn encastv(bytes: &[u8], nelem: usize) -> Option<Vec<Self>>;
/// Decodes the bytes at the head of `bytes` to a vector of
/// value(s) of type `T`. The endianness of the resulting
/// value(s) is flipped if necessary. The endianness of the bytes
/// is specified in `endian`. The number of elements in the
/// resulting vecotr is specified in `nelem`.
#[cfg(feature = "std")]
fn encastvf(bytes: &[u8], nelem: usize, endian: Endian)
-> Option<Vec<Self>>
where
Self: Flip;
}
impl<T> EncastArg for T
where
T: Cast
{
fn encast(bytes: &[u8]) -> Option<Self>
{
bytes.encast()
}
fn encastf(bytes: &[u8], endian: Endian) -> Option<Self>
where
Self: Flip
{
bytes.encastf(endian)
}
#[cfg(feature = "std")]
fn encastv(bytes: &[u8], nelem: usize) -> Option<Vec<Self>>
{
bytes.encastv(nelem)
}
#[cfg(feature = "std")]
fn encastvf(bytes: &[u8], nelem: usize, endian: Endian)
-> Option<Vec<Self>>
where
Self: Flip
{
bytes.encastvf(nelem, endian)
}
}