// Copyright (C) 2017 Sebastian Dröge // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use gst_video_sys; use std::ffi::CStr; use std::fmt; use std::str; use glib; use glib::translate::{from_glib, ToGlib}; pub struct VideoFormatInfo(&'static gst_video_sys::GstVideoFormatInfo); impl VideoFormatInfo { pub fn from_format(format: ::VideoFormat) -> VideoFormatInfo { assert_initialized_main_thread!(); unsafe { let info = gst_video_sys::gst_video_format_get_info(format.to_glib()); assert!(!info.is_null()); VideoFormatInfo(&*info) } } pub fn format(&self) -> ::VideoFormat { from_glib(self.0.format) } pub fn name<'a>(&self) -> &'a str { unsafe { CStr::from_ptr(self.0.name).to_str().unwrap() } } pub fn description<'a>(&self) -> &'a str { unsafe { CStr::from_ptr(self.0.description).to_str().unwrap() } } pub fn flags(&self) -> ::VideoFormatFlags { from_glib(self.0.flags) } pub fn bits(&self) -> u32 { self.0.bits } pub fn n_components(&self) -> u32 { self.0.n_components } pub fn shift(&self) -> &[u32] { &self.0.shift[0..(self.0.n_components as usize)] } pub fn depth(&self) -> &[u32] { &self.0.depth[0..(self.0.n_components as usize)] } pub fn pixel_stride(&self) -> &[i32] { &self.0.pixel_stride[0..(self.0.n_components as usize)] } pub fn n_planes(&self) -> u32 { self.0.n_planes } pub fn plane(&self) -> &[u32] { &self.0.plane[0..(self.0.n_components as usize)] } pub fn poffset(&self) -> &[u32] { &self.0.poffset[0..(self.0.n_components as usize)] } pub fn w_sub(&self) -> &[u32] { &self.0.w_sub[0..(self.0.n_components as usize)] } pub fn h_sub(&self) -> &[u32] { &self.0.h_sub[0..(self.0.n_components as usize)] } pub fn tile_mode(&self) -> ::VideoTileMode { from_glib(self.0.tile_mode) } pub fn tile_ws(&self) -> u32 { self.0.tile_ws } pub fn tile_hs(&self) -> u32 { self.0.tile_hs } pub fn unpack_format(&self) -> ::VideoFormat { from_glib(self.0.unpack_format) } pub fn pack_lines(&self) -> i32 { self.0.pack_lines } pub fn has_alpha(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_ALPHA != 0 } pub fn has_palette(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_PALETTE != 0 } pub fn is_complex(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_COMPLEX != 0 } pub fn is_gray(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_GRAY != 0 } pub fn is_le(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_LE != 0 } pub fn is_rgb(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_RGB != 0 } pub fn is_tiled(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_TILED != 0 } pub fn is_yuv(&self) -> bool { self.0.flags & gst_video_sys::GST_VIDEO_FORMAT_FLAG_YUV != 0 } pub fn scale_width(&self, component: u8, width: u32) -> u32 { (-((-(i64::from(width))) >> self.w_sub()[component as usize])) as u32 } pub fn scale_height(&self, component: u8, height: u32) -> u32 { (-((-(i64::from(height))) >> self.h_sub()[component as usize])) as u32 } #[allow(clippy::too_many_arguments)] pub fn unpack( &self, flags: ::VideoPackFlags, dest: &mut [u8], src: &[&[u8]], stride: &[i32], x: i32, y: i32, width: i32, ) { let unpack_format = Self::from_format(self.unpack_format()); if unpack_format.pixel_stride()[0] == 0 || self.0.unpack_func.is_none() { panic!("No unpack format for {:?}", self); } if src.len() != self.n_planes() as usize { panic!( "Wrong number of planes provided for format: {} != {}", src.len(), self.n_planes() ); } if stride.len() != self.n_planes() as usize { panic!( "Wrong number of strides provided for format: {} != {}", stride.len(), self.n_planes() ); } if dest.len() < unpack_format.pixel_stride()[0] as usize * width as usize { panic!("Too small destination slice"); } for plane in 0..(self.n_planes()) { if stride[plane as usize] < self.scale_width(plane as u8, width as u32) as i32 * self.pixel_stride()[plane as usize] { panic!("Too small source stride for plane {}", plane); } let plane_size = y * stride[plane as usize] + self.scale_width(plane as u8, (x + width) as u32) as i32 * self.pixel_stride()[plane as usize]; if src[plane as usize].len() < plane_size as usize { panic!("Too small source plane size for plane {}", plane); } } unsafe { use std::ptr; let mut src_ptr = [ptr::null() as *const u8; gst_video_sys::GST_VIDEO_MAX_PLANES as usize]; for plane in 0..(self.n_planes()) { src_ptr[plane as usize] = src[plane as usize].as_ptr(); } (self.0.unpack_func.as_ref().unwrap())( self.0, flags.to_glib(), dest.as_mut_ptr() as *mut _, src_ptr.as_ptr() as *const _, stride.as_ptr() as *const i32, x, y, width, ); } } #[allow(clippy::too_many_arguments)] pub fn pack( &self, flags: ::VideoPackFlags, src: &[u8], src_stride: i32, dest: &mut [&mut [u8]], dest_stride: &[i32], chroma_site: ::VideoChromaSite, y: i32, width: i32, ) { let unpack_format = Self::from_format(self.unpack_format()); if unpack_format.pixel_stride()[0] == 0 || self.0.unpack_func.is_none() { panic!("No unpack format for {:?}", self); } if dest.len() != self.n_planes() as usize { panic!( "Wrong number of planes provided for format: {} != {}", dest.len(), self.n_planes() ); } if dest_stride.len() != self.n_planes() as usize { panic!( "Wrong number of strides provided for format: {} != {}", dest_stride.len(), self.n_planes() ); } if src.len() < unpack_format.pixel_stride()[0] as usize * width as usize { panic!("Too small source slice"); } for plane in 0..(self.n_planes()) { if dest_stride[plane as usize] < self.scale_width(plane as u8, width as u32) as i32 * self.pixel_stride()[plane as usize] { panic!("Too small destination stride for plane {}", plane); } let plane_size = y * dest_stride[plane as usize] + self.scale_width(plane as u8, width as u32) as i32 * self.pixel_stride()[plane as usize]; if dest[plane as usize].len() < plane_size as usize { panic!("Too small destination plane size for plane {}", plane); } } unsafe { use std::ptr; let mut dest_ptr = [ptr::null_mut() as *mut u8; gst_video_sys::GST_VIDEO_MAX_PLANES as usize]; for plane in 0..(self.n_planes()) { dest_ptr[plane as usize] = dest[plane as usize].as_mut_ptr(); } (self.0.pack_func.as_ref().unwrap())( self.0, flags.to_glib(), src.as_ptr() as *mut _, src_stride, dest_ptr.as_mut_ptr() as *mut _, dest_stride.as_ptr() as *const i32, chroma_site.to_glib(), y, width, ); } } } unsafe impl Sync for VideoFormatInfo {} unsafe impl Send for VideoFormatInfo {} impl PartialEq for VideoFormatInfo { fn eq(&self, other: &Self) -> bool { self.format() == other.format() } } impl Eq for VideoFormatInfo {} impl fmt::Debug for VideoFormatInfo { fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> { f.debug_struct("VideoFormatInfo") .field("format", &self.format()) .field("name", &self.name()) .field("description", &self.description()) .field("flags", &self.flags()) .field("bits", &self.bits()) .field("n-components", &self.n_components()) .field("shift", &self.shift()) .field("depth", &self.depth()) .field("pixel-stride", &self.pixel_stride()) .field("n-planes", &self.n_planes()) .field("plane", &self.plane()) .field("poffset", &self.poffset()) .field("w-sub", &self.w_sub()) .field("h-sub", &self.h_sub()) .field("unpack-format", &self.unpack_format()) .field("pack-lines", &self.pack_lines()) .field("tile-mode", &self.tile_mode()) .field("tile-ws", &self.tile_ws()) .field("tile-hs", &self.tile_hs()) .finish() } } impl fmt::Display for VideoFormatInfo { fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> { f.write_str(self.name()) } } impl str::FromStr for ::VideoFormatInfo { type Err = glib::BoolError; fn from_str(s: &str) -> Result { skip_assert_initialized!(); let format = s.parse()?; Ok(VideoFormatInfo::from_format(format)) } } impl From<::VideoFormat> for VideoFormatInfo { fn from(f: ::VideoFormat) -> Self { skip_assert_initialized!(); Self::from_format(f) } } #[doc(hidden)] impl glib::translate::GlibPtrDefault for VideoFormatInfo { type GlibType = *mut gst_video_sys::GstVideoFormatInfo; } #[doc(hidden)] impl<'a> glib::translate::ToGlibPtr<'a, *const gst_video_sys::GstVideoFormatInfo> for VideoFormatInfo { type Storage = &'a VideoFormatInfo; fn to_glib_none( &'a self, ) -> glib::translate::Stash<'a, *const gst_video_sys::GstVideoFormatInfo, Self> { glib::translate::Stash(self.0, self) } fn to_glib_full(&self) -> *const gst_video_sys::GstVideoFormatInfo { unimplemented!() } } #[doc(hidden)] impl glib::translate::FromGlibPtrNone<*mut gst_video_sys::GstVideoFormatInfo> for VideoFormatInfo { #[inline] unsafe fn from_glib_none(ptr: *mut gst_video_sys::GstVideoFormatInfo) -> Self { VideoFormatInfo(&*ptr) } } #[cfg(test)] mod tests { use super::*; use gst; #[test] fn test_get() { gst::init().unwrap(); let info = VideoFormatInfo::from_format(::VideoFormat::I420); assert_eq!(info.name(), "I420"); let other_info = "I420".parse().unwrap(); assert_eq!(info, other_info); assert_eq!(info.scale_width(0, 128), 128); assert_eq!(info.scale_width(1, 128), 64); assert_eq!(info.scale_width(2, 128), 64); } #[test] fn test_unpack() { gst::init().unwrap(); // One line black 320 pixel I420 let input = &[&[0; 320][..], &[128; 160][..], &[128; 160][..]]; // One line of AYUV let intermediate = &mut [0; 320 * 4][..]; // One line of 320 pixel I420 let output = &mut [&mut [0; 320][..], &mut [0; 160][..], &mut [0; 160][..]]; let info = VideoFormatInfo::from_format(::VideoFormat::I420); assert_eq!(info.unpack_format(), ::VideoFormat::Ayuv); info.unpack( ::VideoPackFlags::empty(), intermediate, input, &[320, 160, 160][..], 0, 0, 320, ); for pixel in intermediate.chunks_exact(4) { assert_eq!(&[255, 0, 128, 128][..], pixel); } info.pack( ::VideoPackFlags::empty(), &intermediate[..(4 * 320)], 4 * 320, output, &[320, 160, 160][..], ::VideoChromaSite::NONE, 0, 320, ); assert_eq!(input, output); } }