videoframe 0.2.0

A common vocabulary of pixel-format and color-metadata types for video processing pipelines.
Documentation
//! Tier 12 — Packed 12-bit CIE XYZ source frames (`AV_PIX_FMT_XYZ12LE` /
//! `AV_PIX_FMT_XYZ12BE`).
//!
//! Despite the inventory doc note "3 planes (X / Y / Z)", FFmpeg's actual
//! `AV_PIX_FMT_XYZ12LE` descriptor is **packed**: one stream of `u16`
//! triples in `X, Y, Z` order, **high-bit-packed** per the FFmpeg spec
//! ("the same as RGB48LE/BE, but the lower 4 bits of each component are
//! zero"). The active 12-bit code lives in bits `[15:4]` of each `u16`;
//! bits `[3:0]` are reserved zero. Equivalently, the wire `u16` value
//! is `code << 4` for a 12-bit code in `[0, 4095]`. This matches the
//! DCDM JPEG2000 cinema container format that decoders like OpenJPEG
//! expand into.
//!
//! # Stride semantics
//!
//! **Stride is in samples (`u16` elements)**, not bytes. Each row needs
//! at least `3 * width` u16 samples. Callers with a raw FFmpeg byte
//! buffer should cast via [`bytemuck::cast_slice`] (which checks
//! alignment) and divide `linesize[0]` by 2.
//!
//! # Sample-value validation
//!
//! `try_new` validates geometry only. Every kernel applies an
//! endian-aware load (`from_le` / `from_be`) followed by `>> 4` to
//! recover the active 12-bit code from the high-bit-packed `u16`,
//! then defensively masks with `& 0x0FFF`. Producers that violate the
//! spec by setting bits `[3:0]` see those bits silently discarded —
//! matches `Yuv420pFrame16` / `GbrpHighBitFrame` precedent (scanning
//! every sample at video rates is prohibitive).
//!
//! # Endianness
//!
//! The const-generic `BE: bool` parameter selects whether the wire-format
//! u16 samples are little-endian (`BE = false`) or big-endian (`BE =
//! true`). Type aliases [`Xyz12LeFrame`] and [`Xyz12BeFrame`] cover the
//! two FFmpeg variants. The byte-swap is a compile-time const branch in
//! every row kernel; the `BE = false` path is a no-op.

use super::{
  GeometryOverflow, InsufficientPlane, InsufficientStride, WidthOverflow, ZeroDimension,
};
use derive_more::{IsVariant, TryUnwrap, Unwrap};
use thiserror::Error;

/// Errors returned by [`Xyz12Frame::try_new`].
///
/// Variant shape mirrors [`super::Rgbf32FrameError`] but with all sizes
/// expressed in **samples** (`u16` elements) instead of bytes.
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, IsVariant, TryUnwrap, Unwrap, Error)]
#[non_exhaustive]
#[unwrap(ref, ref_mut)]
#[try_unwrap(ref, ref_mut)]
pub enum Xyz12FrameError {
  /// `width` or `height` was zero.
  #[error(transparent)]
  ZeroDimension(ZeroDimension),

  /// `stride < 3 * width` (in u16 elements).
  #[error(transparent)]
  InsufficientStride(InsufficientStride),

  /// Plane is shorter than `stride * height` u16 elements.
  #[error(transparent)]
  InsufficientPlane(InsufficientPlane),

  /// `stride * height` overflows `usize`.
  #[error(transparent)]
  GeometryOverflow(GeometryOverflow),

  /// `3 * width` overflows `u32`.
  #[error(transparent)]
  WidthOverflow(WidthOverflow),
}

/// A validated packed **XYZ12** frame (`AV_PIX_FMT_XYZ12LE` /
/// `AV_PIX_FMT_XYZ12BE`).
///
/// Each pixel occupies 3 × `u16` (six bytes), in **`X, Y, Z`** order.
/// Samples are 12-bit codes stored **high-bit-packed**: active 12 bits
/// in `[15:4]` of each `u16`, low 4 bits reserved zero (per the
/// FFmpeg `AV_PIX_FMT_XYZ12LE` / `AV_PIX_FMT_XYZ12BE` spec — "the same
/// as RGB48LE/BE, but the lower 4 bits of each component are zero").
/// Producers that violate the convention by setting bits `[3:0]` are
/// tolerated: every row kernel applies `>> 4` after the endian-aware
/// load, silently discarding the dirty low bits.
///
/// `stride` is in **u16 elements** (≥ `3 * width`), matching the
/// per-format convention that stride aligns with the underlying slice
/// element type. No width parity constraint.
///
/// The `BE: bool` const parameter selects little-endian (`false`) or
/// big-endian (`true`) wire-format encoding of each `u16`. Use the
/// type aliases [`Xyz12LeFrame`] / [`Xyz12BeFrame`] at call sites.
#[derive(Debug, Clone, Copy)]
pub struct Xyz12Frame<'a, const BE: bool = false> {
  xyz: &'a [u16],
  width: u32,
  height: u32,
  stride: u32,
}

impl<'a, const BE: bool> Xyz12Frame<'a, BE> {
  /// Constructs a new [`Xyz12Frame`], validating dimensions and plane
  /// length.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn try_new(
    xyz: &'a [u16],
    width: u32,
    height: u32,
    stride: u32,
  ) -> Result<Self, Xyz12FrameError> {
    if width == 0 || height == 0 {
      return Err(Xyz12FrameError::ZeroDimension(ZeroDimension::new(
        width, height,
      )));
    }
    let min_stride = match width.checked_mul(3) {
      Some(v) => v,
      None => return Err(Xyz12FrameError::WidthOverflow(WidthOverflow::new(width))),
    };
    if stride < min_stride {
      return Err(Xyz12FrameError::InsufficientStride(
        InsufficientStride::new(stride, min_stride),
      ));
    }
    let plane_min = match (stride as usize).checked_mul(height as usize) {
      Some(v) => v,
      None => {
        return Err(Xyz12FrameError::GeometryOverflow(GeometryOverflow::new(
          stride, height,
        )));
      }
    };
    if xyz.len() < plane_min {
      return Err(Xyz12FrameError::InsufficientPlane(InsufficientPlane::new(
        plane_min,
        xyz.len(),
      )));
    }
    Ok(Self {
      xyz,
      width,
      height,
      stride,
    })
  }

  /// Constructs a new [`Xyz12Frame`], panicking on invalid inputs.
  /// Prefer [`Self::try_new`] when inputs may be invalid at runtime.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn new(xyz: &'a [u16], width: u32, height: u32, stride: u32) -> Self {
    match Self::try_new(xyz, width, height, stride) {
      Ok(frame) => frame,
      Err(_) => panic!("invalid Xyz12Frame dimensions or plane length"),
    }
  }

  /// Packed `X, Y, Z` plane — `width * 3` u16 elements per row.
  /// Samples are 12-bit codes in bits `[15:4]` of each `u16`
  /// (high-bit-packed per FFmpeg `AV_PIX_FMT_XYZ12LE/BE`); bits `[3:0]`
  /// are reserved zero. Every row kernel right-shifts by 4 after the
  /// endian-aware load to recover the active code.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn xyz(&self) -> &'a [u16] {
    self.xyz
  }
  /// Frame width in pixels.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn width(&self) -> u32 {
    self.width
  }
  /// Frame height in pixels.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn height(&self) -> u32 {
    self.height
  }
  /// Stride in **u16 elements** (≥ `3 * width`).
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn stride(&self) -> u32 {
    self.stride
  }
  /// Returns whether wire-format `u16` samples are big-endian (`true`)
  /// or little-endian (`false`). Mirrors the const-generic parameter.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn big_endian(&self) -> bool {
    BE
  }
}

/// Type alias for a validated packed XYZ12 frame, **little-endian**
/// wire format (`AV_PIX_FMT_XYZ12LE`).
pub type Xyz12LeFrame<'a> = Xyz12Frame<'a, false>;

/// Type alias for a validated packed XYZ12 frame, **big-endian**
/// wire format (`AV_PIX_FMT_XYZ12BE`).
pub type Xyz12BeFrame<'a> = Xyz12Frame<'a, true>;