#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct BootIdentifierUniversallyUniqueIdentifier(BigEndianU128);
impl Deref for BootIdentifierUniversallyUniqueIdentifier
{
type Target = BigEndianU128;
#[inline(always)]
fn deref(&self) -> &Self::Target
{
&self.0
}
}
impl Into<u128> for BootIdentifierUniversallyUniqueIdentifier
{
#[inline(always)]
fn into(self) -> u128
{
self.into_u128()
}
}
impl FromBytes for BootIdentifierUniversallyUniqueIdentifier
{
type Error = ParseNumberError;
fn from_bytes(bytes: &[u8]) -> Result<Self, Self::Error>
{
use self::ParseNumberError::*;
let mut iterator = bytes.split_bytes_n(5, b'-');
#[inline(always)]
fn parse_piece<'a>(iterator: &mut impl Iterator<Item=&'a [u8]>, uuid_index: &mut usize, uuid_big_endian_bytes: &mut BigEndianU128, expected_length: usize) -> Result<(), ParseNumberError>
{
let piece = iterator.next().ok_or(TooShort)?;
if unlikely!(piece.len() != expected_length)
{
return Err(TooSmall)
}
let mut numeric_value_byte = 0;
for nibble_index in 0 .. expected_length
{
let numeric_value_nibble =
{
let byte = piece.get_unchecked_value_safe(nibble_index);
let subtract = match byte
{
b'0'..=b'9' => b'0',
b'a'..=b'f' => (b'a' + 10),
_ => return Err(InvalidByte { byte })
};
byte - subtract
};
let is_upper_nibble = (nibble_index % 2) == 0;
if is_upper_nibble
{
numeric_value_byte = numeric_value_nibble << 4;
}
else
{
numeric_value_byte |= numeric_value_nibble;
uuid_big_endian_bytes.set_unchecked_mut_safe(*uuid_index, numeric_value_byte);
uuid_index.add_assign(1);
numeric_value_byte = 0;
}
}
Ok(())
}
let mut uuid_index = 0;
let mut uuid_big_endian_bytes = unsafe_uninitialized();
parse_piece(&mut iterator, &mut uuid_index, &mut uuid_big_endian_bytes, 8)?;
parse_piece(&mut iterator, &mut uuid_index, &mut uuid_big_endian_bytes, 4)?;
parse_piece(&mut iterator, &mut uuid_index, &mut uuid_big_endian_bytes, 4)?;
parse_piece(&mut iterator, &mut uuid_index, &mut uuid_big_endian_bytes, 4)?;
parse_piece(&mut iterator, &mut uuid_index, &mut uuid_big_endian_bytes, 12)?;
Ok(Self(uuid_big_endian_bytes))
}
}
impl BootIdentifierUniversallyUniqueIdentifier
{
#[inline(always)]
pub fn new(proc_path: &ProcPath) -> io::Result<Self>
{
proc_path.sys_kernel_random_file_path("boot_id").read_value()
}
#[inline(always)]
pub const fn into_u128(&self) -> u128
{
u128::from_be_bytes(self.0)
}
}