use super::*;
#[cfg(feature = "serde")]
use alloc::string::{String, ToString};
#[cfg(feature = "mem_dbg")]
use mem_dbg::{MemDbg, MemSize};
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
#[cfg_attr(feature = "mem_dbg", derive(MemDbg, MemSize))]
#[cfg_attr(feature = "mem_dbg", mem_size(flat))]
#[cfg_attr(feature = "fuzz", derive(arbitrary::Arbitrary))]
#[non_exhaustive]
pub enum Codes {
Unary,
Gamma,
Delta,
Omega,
VByteLe,
VByteBe,
Zeta(usize),
Pi(usize),
Golomb(u64),
ExpGolomb(usize),
Rice(usize),
}
impl Codes {
#[must_use]
pub const fn canonicalize(self) -> Self {
match self {
Self::Zeta(1) | Self::ExpGolomb(0) | Self::Pi(0) => Self::Gamma,
Self::Rice(0) | Self::Golomb(1) => Self::Unary,
Self::Golomb(b) if b.is_power_of_two() => Self::Rice(b.trailing_zeros() as usize),
other => other,
}
}
#[inline(always)]
pub fn read<E: Endianness, CR: CodesRead<E> + ?Sized>(
&self,
reader: &mut CR,
) -> Result<u64, CR::Error> {
DynamicCodeRead::read(self, reader)
}
#[inline(always)]
pub fn write<E: Endianness, CW: CodesWrite<E> + ?Sized>(
&self,
writer: &mut CW,
n: u64,
) -> Result<usize, CW::Error> {
DynamicCodeWrite::write(self, writer, n)
}
pub const fn to_code_const(&self) -> Result<usize, DispatchError> {
Ok(match self.canonicalize() {
Self::Unary => code_consts::UNARY,
Self::Gamma => code_consts::GAMMA,
Self::Delta => code_consts::DELTA,
Self::Omega => code_consts::OMEGA,
Self::VByteLe => code_consts::VBYTE_LE,
Self::VByteBe => code_consts::VBYTE_BE,
Self::Zeta(2) => code_consts::ZETA2,
Self::Zeta(3) => code_consts::ZETA3,
Self::Zeta(4) => code_consts::ZETA4,
Self::Zeta(5) => code_consts::ZETA5,
Self::Zeta(6) => code_consts::ZETA6,
Self::Zeta(7) => code_consts::ZETA7,
Self::Zeta(8) => code_consts::ZETA8,
Self::Zeta(9) => code_consts::ZETA9,
Self::Zeta(10) => code_consts::ZETA10,
Self::Rice(1) => code_consts::RICE1,
Self::Rice(2) => code_consts::RICE2,
Self::Rice(3) => code_consts::RICE3,
Self::Rice(4) => code_consts::RICE4,
Self::Rice(5) => code_consts::RICE5,
Self::Rice(6) => code_consts::RICE6,
Self::Rice(7) => code_consts::RICE7,
Self::Rice(8) => code_consts::RICE8,
Self::Rice(9) => code_consts::RICE9,
Self::Rice(10) => code_consts::RICE10,
Self::Pi(1) => code_consts::PI1,
Self::Pi(2) => code_consts::PI2,
Self::Pi(3) => code_consts::PI3,
Self::Pi(4) => code_consts::PI4,
Self::Pi(5) => code_consts::PI5,
Self::Pi(6) => code_consts::PI6,
Self::Pi(7) => code_consts::PI7,
Self::Pi(8) => code_consts::PI8,
Self::Pi(9) => code_consts::PI9,
Self::Pi(10) => code_consts::PI10,
Self::Golomb(3) => code_consts::GOLOMB3,
Self::Golomb(5) => code_consts::GOLOMB5,
Self::Golomb(6) => code_consts::GOLOMB6,
Self::Golomb(7) => code_consts::GOLOMB7,
Self::Golomb(9) => code_consts::GOLOMB9,
Self::Golomb(10) => code_consts::GOLOMB10,
Self::ExpGolomb(1) => code_consts::EXP_GOLOMB1,
Self::ExpGolomb(2) => code_consts::EXP_GOLOMB2,
Self::ExpGolomb(3) => code_consts::EXP_GOLOMB3,
Self::ExpGolomb(4) => code_consts::EXP_GOLOMB4,
Self::ExpGolomb(5) => code_consts::EXP_GOLOMB5,
Self::ExpGolomb(6) => code_consts::EXP_GOLOMB6,
Self::ExpGolomb(7) => code_consts::EXP_GOLOMB7,
Self::ExpGolomb(8) => code_consts::EXP_GOLOMB8,
Self::ExpGolomb(9) => code_consts::EXP_GOLOMB9,
Self::ExpGolomb(10) => code_consts::EXP_GOLOMB10,
_ => {
return Err(DispatchError::UnsupportedCode(*self));
}
})
}
pub const fn from_code_const(const_code: usize) -> Result<Self, DispatchError> {
Ok(match const_code {
code_consts::UNARY => Self::Unary,
code_consts::GAMMA => Self::Gamma,
code_consts::DELTA => Self::Delta,
code_consts::OMEGA => Self::Omega,
code_consts::VBYTE_LE => Self::VByteLe,
code_consts::VBYTE_BE => Self::VByteBe,
code_consts::ZETA2 => Self::Zeta(2),
code_consts::ZETA3 => Self::Zeta(3),
code_consts::ZETA4 => Self::Zeta(4),
code_consts::ZETA5 => Self::Zeta(5),
code_consts::ZETA6 => Self::Zeta(6),
code_consts::ZETA7 => Self::Zeta(7),
code_consts::ZETA8 => Self::Zeta(8),
code_consts::ZETA9 => Self::Zeta(9),
code_consts::ZETA10 => Self::Zeta(10),
code_consts::RICE1 => Self::Rice(1),
code_consts::RICE2 => Self::Rice(2),
code_consts::RICE3 => Self::Rice(3),
code_consts::RICE4 => Self::Rice(4),
code_consts::RICE5 => Self::Rice(5),
code_consts::RICE6 => Self::Rice(6),
code_consts::RICE7 => Self::Rice(7),
code_consts::RICE8 => Self::Rice(8),
code_consts::RICE9 => Self::Rice(9),
code_consts::RICE10 => Self::Rice(10),
code_consts::PI1 => Self::Pi(1),
code_consts::PI2 => Self::Pi(2),
code_consts::PI3 => Self::Pi(3),
code_consts::PI4 => Self::Pi(4),
code_consts::PI5 => Self::Pi(5),
code_consts::PI6 => Self::Pi(6),
code_consts::PI7 => Self::Pi(7),
code_consts::PI8 => Self::Pi(8),
code_consts::PI9 => Self::Pi(9),
code_consts::PI10 => Self::Pi(10),
code_consts::GOLOMB3 => Self::Golomb(3),
code_consts::GOLOMB5 => Self::Golomb(5),
code_consts::GOLOMB6 => Self::Golomb(6),
code_consts::GOLOMB7 => Self::Golomb(7),
code_consts::GOLOMB9 => Self::Golomb(9),
code_consts::GOLOMB10 => Self::Golomb(10),
code_consts::EXP_GOLOMB1 => Self::ExpGolomb(1),
code_consts::EXP_GOLOMB2 => Self::ExpGolomb(2),
code_consts::EXP_GOLOMB3 => Self::ExpGolomb(3),
code_consts::EXP_GOLOMB4 => Self::ExpGolomb(4),
code_consts::EXP_GOLOMB5 => Self::ExpGolomb(5),
code_consts::EXP_GOLOMB6 => Self::ExpGolomb(6),
code_consts::EXP_GOLOMB7 => Self::ExpGolomb(7),
code_consts::EXP_GOLOMB8 => Self::ExpGolomb(8),
code_consts::EXP_GOLOMB9 => Self::ExpGolomb(9),
code_consts::EXP_GOLOMB10 => Self::ExpGolomb(10),
_ => return Err(DispatchError::UnsupportedCodeConst(const_code)),
})
}
}
impl DynamicCodeRead for Codes {
#[inline]
fn read<E: Endianness, CR: CodesRead<E> + ?Sized>(
&self,
reader: &mut CR,
) -> Result<u64, CR::Error> {
Ok(match self.canonicalize() {
Codes::Unary => reader.read_unary()?,
Codes::Gamma => reader.read_gamma()?,
Codes::Delta => reader.read_delta()?,
Codes::Omega => reader.read_omega()?,
Codes::VByteBe => reader.read_vbyte_be()?,
Codes::VByteLe => reader.read_vbyte_le()?,
Codes::Zeta(3) => reader.read_zeta3()?,
Codes::Zeta(k) => reader.read_zeta(k)?,
Codes::Pi(2) => reader.read_pi2()?,
Codes::Pi(k) => reader.read_pi(k)?,
Codes::Golomb(b) => reader.read_golomb(b)?,
Codes::ExpGolomb(k) => reader.read_exp_golomb(k)?,
Codes::Rice(log2_b) => reader.read_rice(log2_b)?,
})
}
}
impl DynamicCodeWrite for Codes {
#[inline]
fn write<E: Endianness, CW: CodesWrite<E> + ?Sized>(
&self,
writer: &mut CW,
n: u64,
) -> Result<usize, CW::Error> {
Ok(match self.canonicalize() {
Codes::Unary => writer.write_unary(n)?,
Codes::Gamma => writer.write_gamma(n)?,
Codes::Delta => writer.write_delta(n)?,
Codes::Omega => writer.write_omega(n)?,
Codes::VByteBe => writer.write_vbyte_be(n)?,
Codes::VByteLe => writer.write_vbyte_le(n)?,
Codes::Zeta(3) => writer.write_zeta3(n)?,
Codes::Zeta(k) => writer.write_zeta(n, k)?,
Codes::Pi(2) => writer.write_pi2(n)?,
Codes::Pi(k) => writer.write_pi(n, k)?,
Codes::Golomb(b) => writer.write_golomb(n, b)?,
Codes::ExpGolomb(k) => writer.write_exp_golomb(n, k)?,
Codes::Rice(log2_b) => writer.write_rice(n, log2_b)?,
})
}
}
impl<E: Endianness, CR: CodesRead<E> + ?Sized> StaticCodeRead<E, CR> for Codes {
#[inline(always)]
fn read(&self, reader: &mut CR) -> Result<u64, CR::Error> {
<Self as DynamicCodeRead>::read(self, reader)
}
}
impl<E: Endianness, CW: CodesWrite<E> + ?Sized> StaticCodeWrite<E, CW> for Codes {
#[inline(always)]
fn write(&self, writer: &mut CW, n: u64) -> Result<usize, CW::Error> {
<Self as DynamicCodeWrite>::write(self, writer, n)
}
}
impl CodeLen for Codes {
#[inline]
fn len(&self, n: u64) -> usize {
match self.canonicalize() {
Codes::Unary => n as usize + 1,
Codes::Gamma => len_gamma(n),
Codes::Delta => len_delta(n),
Codes::Omega => len_omega(n),
Codes::VByteLe | Codes::VByteBe => bit_len_vbyte(n),
Codes::Zeta(k) => len_zeta(n, k),
Codes::Pi(k) => len_pi(n, k),
Codes::Golomb(b) => len_golomb(n, b),
Codes::ExpGolomb(k) => len_exp_golomb(n, k),
Codes::Rice(log2_b) => len_rice(n, log2_b),
}
}
}
#[derive(Debug, Clone)]
pub enum CodeError {
ParseError(core::num::ParseIntError),
UnknownCode([u8; 32]),
}
impl core::error::Error for CodeError {}
impl core::fmt::Display for CodeError {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
match self {
CodeError::ParseError(e) => write!(f, "parse error: {}", e),
CodeError::UnknownCode(s) => {
write!(f, "unknown code: ")?;
for c in s {
if *c == 0 {
break;
}
write!(f, "{}", *c as char)?;
}
Ok(())
}
}
}
}
impl From<core::num::ParseIntError> for CodeError {
fn from(e: core::num::ParseIntError) -> Self {
CodeError::ParseError(e)
}
}
impl core::fmt::Display for Codes {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
match self {
Codes::Unary => write!(f, "Unary"),
Codes::Gamma => write!(f, "Gamma"),
Codes::Delta => write!(f, "Delta"),
Codes::Omega => write!(f, "Omega"),
Codes::VByteBe => write!(f, "VByteBe"),
Codes::VByteLe => write!(f, "VByteLe"),
Codes::Zeta(k) => write!(f, "Zeta({})", k),
Codes::Pi(k) => write!(f, "Pi({})", k),
Codes::Golomb(b) => write!(f, "Golomb({})", b),
Codes::ExpGolomb(k) => write!(f, "ExpGolomb({})", k),
Codes::Rice(log2_b) => write!(f, "Rice({})", log2_b),
}
}
}
fn array_format_error(s: &str) -> [u8; 32] {
let mut error_buffer = [0u8; 32];
let len = s.len().min(32);
error_buffer[..len].copy_from_slice(&s.as_bytes()[..len]);
error_buffer
}
impl core::str::FromStr for Codes {
type Err = CodeError;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"Unary" => Ok(Codes::Unary),
"Gamma" => Ok(Codes::Gamma),
"Delta" => Ok(Codes::Delta),
"Omega" => Ok(Codes::Omega),
"VByteBe" => Ok(Codes::VByteBe),
"VByteLe" => Ok(Codes::VByteLe),
_ => {
let mut parts = s.split('(');
let name = parts
.next()
.ok_or_else(|| CodeError::UnknownCode(array_format_error(s)))?;
let k = parts
.next()
.ok_or_else(|| CodeError::UnknownCode(array_format_error(s)))?
.split(')')
.next()
.ok_or_else(|| CodeError::UnknownCode(array_format_error(s)))?;
match name {
"Zeta" => Ok(Codes::Zeta(k.parse()?)),
"Pi" => Ok(Codes::Pi(k.parse()?)),
"Golomb" => Ok(Codes::Golomb(k.parse()?)),
"ExpGolomb" => Ok(Codes::ExpGolomb(k.parse()?)),
"Rice" => Ok(Codes::Rice(k.parse()?)),
_ => Err(CodeError::UnknownCode(array_format_error(name))),
}
}
}
}
}
#[cfg(feature = "serde")]
impl serde::Serialize for Codes {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str(&self.to_string())
}
}
#[cfg(feature = "serde")]
impl<'de> serde::Deserialize<'de> for Codes {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
s.parse().map_err(serde::de::Error::custom)
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
pub struct MinimalBinary(
pub u64,
);
impl DynamicCodeRead for MinimalBinary {
fn read<E: Endianness, R: CodesRead<E> + ?Sized>(
&self,
reader: &mut R,
) -> Result<u64, R::Error> {
reader.read_minimal_binary(self.0)
}
}
impl DynamicCodeWrite for MinimalBinary {
fn write<E: Endianness, W: CodesWrite<E> + ?Sized>(
&self,
writer: &mut W,
n: u64,
) -> Result<usize, W::Error> {
writer.write_minimal_binary(n, self.0)
}
}
impl<E: Endianness, CR: CodesRead<E> + ?Sized> StaticCodeRead<E, CR> for MinimalBinary {
fn read(&self, reader: &mut CR) -> Result<u64, CR::Error> {
<Self as DynamicCodeRead>::read(self, reader)
}
}
impl<E: Endianness, CW: CodesWrite<E> + ?Sized> StaticCodeWrite<E, CW> for MinimalBinary {
fn write(&self, writer: &mut CW, n: u64) -> Result<usize, CW::Error> {
<Self as DynamicCodeWrite>::write(self, writer, n)
}
}
impl CodeLen for MinimalBinary {
fn len(&self, n: u64) -> usize {
len_minimal_binary(n, self.0)
}
}