use crate::serialize::{Deserialize, Serialize};
use crate::wasm_bindgen::*;
use num_derive::FromPrimitive;
use num_traits::{AsPrimitive, PrimInt, Signed};
use std::fmt;
use std::fmt::{Debug, Display};
use std::mem::size_of;
use std::ops::AddAssign;
pub trait CastFromPrimitive<T>: Copy + 'static {
fn cast_from(v: T) -> Self;
}
macro_rules! impl_cast_from_primitive {
( $T:ty => $U:ty ) => {
impl CastFromPrimitive<$U> for $T {
#[inline(always)]
fn cast_from(v: $U) -> Self { v as Self }
}
};
( $T:ty => { $( $U:ty ),* } ) => {
$( impl_cast_from_primitive!($T => $U); )*
};
}
impl_cast_from_primitive!(u8 => { u32, u64, usize });
impl_cast_from_primitive!(u8 => { i8, i64, isize });
impl_cast_from_primitive!(u16 => { u32, u64, usize });
impl_cast_from_primitive!(u16 => { i8, i64, isize });
impl_cast_from_primitive!(i16 => { u32, u64, usize });
impl_cast_from_primitive!(i16 => { i8, i64, isize });
impl_cast_from_primitive!(i32 => { u32, u64, usize });
impl_cast_from_primitive!(i32 => { i8, i64, isize });
pub trait RegisteredPrimitive:
PrimInt
+ AsPrimitive<u8>
+ AsPrimitive<i16>
+ AsPrimitive<u16>
+ AsPrimitive<i32>
+ AsPrimitive<u32>
+ AsPrimitive<usize>
+ CastFromPrimitive<u8>
+ CastFromPrimitive<i16>
+ CastFromPrimitive<u16>
+ CastFromPrimitive<i32>
+ CastFromPrimitive<u32>
+ CastFromPrimitive<usize>
{
}
impl RegisteredPrimitive for u8 {}
impl RegisteredPrimitive for u16 {}
impl RegisteredPrimitive for i16 {}
impl RegisteredPrimitive for i32 {}
macro_rules! impl_cast_from_pixel_to_primitive {
( $T:ty ) => {
impl<T: RegisteredPrimitive> CastFromPrimitive<T> for $T {
#[inline(always)]
fn cast_from(v: T) -> Self {
v.as_()
}
}
};
}
impl_cast_from_pixel_to_primitive!(u8);
impl_cast_from_pixel_to_primitive!(i16);
impl_cast_from_pixel_to_primitive!(u16);
impl_cast_from_pixel_to_primitive!(i32);
impl_cast_from_pixel_to_primitive!(u32);
#[derive(PartialEq, Eq)]
pub enum PixelType {
U8,
U16,
}
pub trait Pixel:
RegisteredPrimitive + Into<u32> + Into<i32> + Debug + Display + Send + Sync + 'static
{
type Coeff: Coefficient;
fn type_enum() -> PixelType;
#[inline]
#[allow(clippy::wrong_self_convention)]
fn to_asm_stride(in_stride: usize) -> isize {
(in_stride * size_of::<Self>()) as isize
}
}
impl Pixel for u8 {
type Coeff = i16;
#[inline]
fn type_enum() -> PixelType {
PixelType::U8
}
}
impl Pixel for u16 {
type Coeff = i32;
#[inline]
fn type_enum() -> PixelType {
PixelType::U16
}
}
pub trait Coefficient:
RegisteredPrimitive + Into<i32> + AddAssign + Signed + Debug + 'static
{
type Pixel: Pixel;
}
impl Coefficient for i16 {
type Pixel = u8;
}
impl Coefficient for i32 {
type Pixel = u16;
}
#[wasm_bindgen]
#[derive(Copy, Clone, Debug, PartialEq, Eq, FromPrimitive, Serialize, Deserialize)]
#[repr(C)]
pub enum ChromaSampling {
Cs420,
Cs422,
Cs444,
Cs400,
}
impl fmt::Display for ChromaSampling {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
write!(
f,
"{}",
match self {
ChromaSampling::Cs420 => "4:2:0",
ChromaSampling::Cs422 => "4:2:2",
ChromaSampling::Cs444 => "4:4:4",
ChromaSampling::Cs400 => "Monochrome",
}
)
}
}
impl Default for ChromaSampling {
fn default() -> Self {
ChromaSampling::Cs420
}
}
impl ChromaSampling {
pub const fn get_decimation(self) -> Option<(usize, usize)> {
use self::ChromaSampling::*;
match self {
Cs420 => Some((1, 1)),
Cs422 => Some((1, 0)),
Cs444 => Some((0, 0)),
Cs400 => None,
}
}
pub const fn get_chroma_dimensions(
self,
luma_width: usize,
luma_height: usize,
) -> (usize, usize) {
if let Some((ss_x, ss_y)) = self.get_decimation() {
((luma_width + ss_x) >> ss_x, (luma_height + ss_y) >> ss_y)
} else {
(0, 0)
}
}
}