#![allow(
unsafe_code,
reason = "The `bytemuck` marker traits are `unsafe` and require `unsafe impl`."
)]
use crate::GenericFamily;
use bytemuck::{Contiguous, NoUninit, Zeroable, checked::CheckedBitPattern};
unsafe impl NoUninit for GenericFamily {}
unsafe impl Zeroable for GenericFamily {}
unsafe impl CheckedBitPattern for GenericFamily {
type Bits = u8;
fn is_valid_bit_pattern(bits: &u8) -> bool {
*bits <= Self::MAX_VALUE
}
}
unsafe impl Contiguous for GenericFamily {
type Int = u8;
const MIN_VALUE: u8 = Self::Serif as u8;
#[allow(
clippy::use_self,
reason = "Using `Self::MAX_VALUE` here would refer to `Contiguous::MAX_VALUE` (self-reference)."
)]
const MAX_VALUE: u8 = GenericFamily::MAX_VALUE;
}
#[cfg(test)]
mod tests {
use super::GenericFamily;
use bytemuck::{Contiguous, Zeroable, checked::try_from_bytes};
use core::ptr;
#[test]
fn checked_bit_pattern() {
let valid = bytemuck::bytes_of(&2_u8);
let invalid = bytemuck::bytes_of(&200_u8);
assert_eq!(
Ok(&GenericFamily::Monospace),
try_from_bytes::<GenericFamily>(valid)
);
assert!(try_from_bytes::<GenericFamily>(invalid).is_err());
}
#[test]
fn contiguous() {
let hd1 = GenericFamily::SansSerif;
let hd2 = GenericFamily::from_integer(hd1.into_integer());
assert_eq!(Some(hd1), hd2);
assert_eq!(None, GenericFamily::from_integer(255));
}
#[test]
fn zeroable() {
let hd = GenericFamily::zeroed();
assert_eq!(hd, GenericFamily::Serif);
}
const _: () = {
let mut value = 0;
while value <= GenericFamily::MAX_VALUE {
let it: GenericFamily = unsafe { ptr::read((&raw const value).cast()) };
if it as u8 != value {
unreachable!();
}
value += 1;
}
};
}
#[cfg(doctest)]
mod doctests {
const _GENERIC_FAMILY: () = {};
}