mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer-rs.git
synced 2024-11-26 19:41:09 +00:00
517 lines
14 KiB
Rust
517 lines
14 KiB
Rust
// Take a look at the license at the top of the repository in the LICENSE file.
|
|
|
|
use std::fmt;
|
|
use std::marker::PhantomData;
|
|
use std::ops;
|
|
|
|
use crate::Buffer;
|
|
use crate::BufferRef;
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
use crate::Caps;
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
use crate::CapsRef;
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
use crate::ClockTime;
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
use glib::translate::IntoGlib;
|
|
use glib::translate::{from_glib, from_glib_none, FromGlib, ToGlibPtr};
|
|
|
|
pub unsafe trait MetaAPI: Sync + Send + Sized {
|
|
type GstType;
|
|
|
|
#[doc(alias = "get_meta_api")]
|
|
fn meta_api() -> glib::Type;
|
|
|
|
unsafe fn from_ptr(buffer: &BufferRef, ptr: *const Self::GstType) -> MetaRef<Self> {
|
|
assert!(!ptr.is_null());
|
|
|
|
let meta_api = Self::meta_api();
|
|
if meta_api != glib::Type::INVALID {
|
|
assert_eq!(
|
|
meta_api,
|
|
from_glib((*(*(ptr as *const ffi::GstMeta)).info).api)
|
|
)
|
|
}
|
|
|
|
MetaRef {
|
|
meta: &*(ptr as *const Self),
|
|
buffer,
|
|
}
|
|
}
|
|
|
|
unsafe fn from_mut_ptr<T>(
|
|
buffer: &mut BufferRef,
|
|
ptr: *mut Self::GstType,
|
|
) -> MetaRefMut<Self, T> {
|
|
assert!(!ptr.is_null());
|
|
|
|
let meta_api = Self::meta_api();
|
|
if meta_api != glib::Type::INVALID {
|
|
assert_eq!(
|
|
meta_api,
|
|
from_glib((*(*(ptr as *const ffi::GstMeta)).info).api)
|
|
)
|
|
}
|
|
|
|
MetaRefMut {
|
|
meta: &mut *(ptr as *mut Self),
|
|
buffer,
|
|
mode: PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_16", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_16")))]
|
|
#[derive(Debug, Clone, Copy, PartialOrd, Ord, PartialEq, Eq)]
|
|
pub struct MetaSeqnum(u64);
|
|
|
|
pub struct MetaRef<'a, T: MetaAPI + 'a> {
|
|
meta: &'a T,
|
|
buffer: &'a BufferRef,
|
|
}
|
|
|
|
pub enum Standalone {}
|
|
pub enum Iterated {}
|
|
|
|
pub struct MetaRefMut<'a, T: MetaAPI + 'a, U> {
|
|
meta: &'a mut T,
|
|
buffer: &'a mut BufferRef,
|
|
mode: PhantomData<U>,
|
|
}
|
|
|
|
impl<'a, T: MetaAPI + fmt::Debug + 'a> fmt::Debug for MetaRef<'a, T> {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("MetaRef")
|
|
.field("meta", &self.meta)
|
|
.field("buffer", &self.buffer)
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI + fmt::Debug + 'a, U> fmt::Debug for MetaRefMut<'a, T, U> {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("MetaRef")
|
|
.field("meta", &self.meta)
|
|
.field("buffer", &self.buffer)
|
|
.field("mode", &self.mode)
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI> ops::Deref for MetaRef<'a, T> {
|
|
type Target = T;
|
|
|
|
fn deref(&self) -> &T {
|
|
self.meta
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI> AsRef<MetaRef<'a, T>> for MetaRef<'a, T> {
|
|
fn as_ref(&self) -> &MetaRef<'a, T> {
|
|
self
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI, U> ops::Deref for MetaRefMut<'a, T, U> {
|
|
type Target = T;
|
|
|
|
fn deref(&self) -> &T {
|
|
self.meta
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI, U> ops::DerefMut for MetaRefMut<'a, T, U> {
|
|
fn deref_mut(&mut self) -> &mut T {
|
|
self.meta
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI, U> AsRef<MetaRef<'a, T>> for MetaRefMut<'a, T, U> {
|
|
fn as_ref(&self) -> &MetaRef<'a, T> {
|
|
unsafe { &*(self as *const MetaRefMut<'a, T, U> as *const MetaRef<'a, T>) }
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI> MetaRef<'a, T> {
|
|
#[doc(alias = "get_api")]
|
|
pub fn api(&self) -> glib::Type {
|
|
unsafe {
|
|
let meta = self.meta as *const _ as *const ffi::GstMeta;
|
|
let info = (*meta).info;
|
|
glib::Type::from_glib((*info).api)
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_16", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_16")))]
|
|
#[doc(alias = "get_seqnum")]
|
|
#[doc(alias = "gst_meta_get_seqnum")]
|
|
pub fn seqnum(&self) -> MetaSeqnum {
|
|
unsafe {
|
|
let meta = self.meta as *const _ as *const ffi::GstMeta;
|
|
MetaSeqnum(ffi::gst_meta_get_seqnum(meta))
|
|
}
|
|
}
|
|
|
|
pub fn as_ptr(&self) -> *const T::GstType {
|
|
self.meta as *const _ as *const <T as MetaAPI>::GstType
|
|
}
|
|
}
|
|
|
|
impl<'a> MetaRef<'a, Meta> {
|
|
pub fn downcast_ref<T: MetaAPI>(&self) -> Option<&MetaRef<'a, T>> {
|
|
let target_type = T::meta_api();
|
|
let type_ = self.api();
|
|
|
|
if type_ == glib::Type::INVALID || target_type == type_ {
|
|
Some(unsafe { &*(self as *const MetaRef<'a, Meta> as *const MetaRef<'a, T>) })
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI, U> MetaRefMut<'a, T, U> {
|
|
#[doc(alias = "get_api")]
|
|
pub fn api(&self) -> glib::Type {
|
|
unsafe {
|
|
let meta = self.meta as *const _ as *const ffi::GstMeta;
|
|
let info = (*meta).info;
|
|
glib::Type::from_glib((*info).api)
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_16", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_16")))]
|
|
#[doc(alias = "get_seqnum")]
|
|
#[doc(alias = "gst_meta_get_seqnum")]
|
|
pub fn seqnum(&self) -> u64 {
|
|
unsafe {
|
|
let meta = self.meta as *const _ as *const ffi::GstMeta;
|
|
ffi::gst_meta_get_seqnum(meta)
|
|
}
|
|
}
|
|
|
|
pub fn as_ptr(&self) -> *const T::GstType {
|
|
self.meta as *const _ as *const <T as MetaAPI>::GstType
|
|
}
|
|
|
|
pub fn as_mut_ptr(&mut self) -> *mut T::GstType {
|
|
self.meta as *mut _ as *mut <T as MetaAPI>::GstType
|
|
}
|
|
}
|
|
|
|
impl<'a, T: MetaAPI> MetaRefMut<'a, T, Standalone> {
|
|
#[doc(alias = "gst_buffer_remove_meta")]
|
|
pub fn remove(mut self) {
|
|
unsafe {
|
|
let res = ffi::gst_buffer_remove_meta(
|
|
self.buffer.as_mut_ptr(),
|
|
self.as_mut_ptr() as *mut ffi::GstMeta,
|
|
);
|
|
assert_ne!(res, glib::ffi::GFALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a, U> MetaRefMut<'a, Meta, U> {
|
|
pub fn downcast_ref<T: MetaAPI>(&mut self) -> Option<&MetaRefMut<'a, T, U>> {
|
|
let target_type = T::meta_api();
|
|
let type_ = self.api();
|
|
|
|
if type_ == glib::Type::INVALID || target_type == type_ {
|
|
Some(unsafe { &*(self as *mut MetaRefMut<'a, Meta, U> as *const MetaRefMut<'a, T, U>) })
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
|
|
#[repr(transparent)]
|
|
#[doc(alias = "GstMeta")]
|
|
pub struct Meta(ffi::GstMeta);
|
|
|
|
unsafe impl Send for Meta {}
|
|
unsafe impl Sync for Meta {}
|
|
|
|
impl Meta {
|
|
#[doc(alias = "get_api")]
|
|
fn api(&self) -> glib::Type {
|
|
unsafe { glib::Type::from_glib((*self.0.info).api) }
|
|
}
|
|
}
|
|
|
|
unsafe impl MetaAPI for Meta {
|
|
type GstType = ffi::GstMeta;
|
|
|
|
fn meta_api() -> glib::Type {
|
|
glib::Type::INVALID
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for Meta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("Meta").field("api", &self.api()).finish()
|
|
}
|
|
}
|
|
|
|
#[repr(transparent)]
|
|
#[doc(alias = "GstParentBufferMeta")]
|
|
pub struct ParentBufferMeta(ffi::GstParentBufferMeta);
|
|
|
|
unsafe impl Send for ParentBufferMeta {}
|
|
unsafe impl Sync for ParentBufferMeta {}
|
|
|
|
impl ParentBufferMeta {
|
|
#[doc(alias = "gst_buffer_add_parent_buffer_meta")]
|
|
pub fn add<'a>(buffer: &'a mut BufferRef, parent: &Buffer) -> MetaRefMut<'a, Self, Standalone> {
|
|
skip_assert_initialized!();
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_parent_buffer_meta(
|
|
buffer.as_mut_ptr(),
|
|
parent.to_glib_none().0,
|
|
);
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
#[doc(alias = "get_parent")]
|
|
pub fn parent(&self) -> &BufferRef {
|
|
unsafe { BufferRef::from_ptr(self.0.buffer) }
|
|
}
|
|
|
|
#[doc(alias = "get_parent_owned")]
|
|
pub fn parent_owned(&self) -> Buffer {
|
|
unsafe { from_glib_none(self.0.buffer) }
|
|
}
|
|
}
|
|
|
|
unsafe impl MetaAPI for ParentBufferMeta {
|
|
type GstType = ffi::GstParentBufferMeta;
|
|
|
|
#[doc(alias = "gst_parent_buffer_meta_api_get_type")]
|
|
fn meta_api() -> glib::Type {
|
|
unsafe { from_glib(ffi::gst_parent_buffer_meta_api_get_type()) }
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for ParentBufferMeta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("ParentBufferMeta")
|
|
.field("parent", &self.parent())
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[repr(transparent)]
|
|
#[doc(alias = "GstProtectionMeta")]
|
|
pub struct ProtectionMeta(ffi::GstProtectionMeta);
|
|
|
|
unsafe impl Send for ProtectionMeta {}
|
|
unsafe impl Sync for ProtectionMeta {}
|
|
|
|
impl ProtectionMeta {
|
|
#[doc(alias = "gst_buffer_add_protection_meta")]
|
|
pub fn add(buffer: &mut BufferRef, info: crate::Structure) -> MetaRefMut<Self, Standalone> {
|
|
skip_assert_initialized!();
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_protection_meta(buffer.as_mut_ptr(), info.into_ptr());
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
#[doc(alias = "get_info")]
|
|
pub fn info(&self) -> &crate::StructureRef {
|
|
unsafe { crate::StructureRef::from_glib_borrow(self.0.info) }
|
|
}
|
|
|
|
#[doc(alias = "get_info_mut")]
|
|
pub fn info_mut(&mut self) -> &mut crate::StructureRef {
|
|
unsafe { crate::StructureRef::from_glib_borrow_mut(self.0.info) }
|
|
}
|
|
}
|
|
|
|
unsafe impl MetaAPI for ProtectionMeta {
|
|
type GstType = ffi::GstProtectionMeta;
|
|
|
|
#[doc(alias = "gst_protection_meta_api_get_type")]
|
|
fn meta_api() -> glib::Type {
|
|
unsafe { from_glib(ffi::gst_protection_meta_api_get_type()) }
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for ProtectionMeta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("ProtectionMeta")
|
|
.field("info", &self.info())
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
#[repr(transparent)]
|
|
#[doc(alias = "GstReferenceTimestampMeta")]
|
|
pub struct ReferenceTimestampMeta(ffi::GstReferenceTimestampMeta);
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
unsafe impl Send for ReferenceTimestampMeta {}
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
unsafe impl Sync for ReferenceTimestampMeta {}
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
impl ReferenceTimestampMeta {
|
|
#[doc(alias = "gst_buffer_add_reference_timestamp_meta")]
|
|
pub fn add<'a>(
|
|
buffer: &'a mut BufferRef,
|
|
reference: &Caps,
|
|
timestamp: ClockTime,
|
|
duration: ClockTime,
|
|
) -> MetaRefMut<'a, Self, Standalone> {
|
|
skip_assert_initialized!();
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_reference_timestamp_meta(
|
|
buffer.as_mut_ptr(),
|
|
reference.to_glib_none().0,
|
|
timestamp.into_glib(),
|
|
duration.into_glib(),
|
|
);
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
#[doc(alias = "get_reference")]
|
|
pub fn reference(&self) -> &CapsRef {
|
|
unsafe { CapsRef::from_ptr(self.0.reference) }
|
|
}
|
|
|
|
#[doc(alias = "get_parent_owned")]
|
|
pub fn parent_owned(&self) -> Caps {
|
|
unsafe { from_glib_none(self.0.reference) }
|
|
}
|
|
|
|
#[doc(alias = "get_timestamp")]
|
|
pub fn timestamp(&self) -> ClockTime {
|
|
unsafe { from_glib(self.0.timestamp) }
|
|
}
|
|
|
|
#[doc(alias = "get_duration")]
|
|
pub fn duration(&self) -> ClockTime {
|
|
unsafe { from_glib(self.0.duration) }
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
unsafe impl MetaAPI for ReferenceTimestampMeta {
|
|
type GstType = ffi::GstReferenceTimestampMeta;
|
|
|
|
#[doc(alias = "gst_reference_timestamp_meta_api_get_type")]
|
|
fn meta_api() -> glib::Type {
|
|
unsafe { from_glib(ffi::gst_reference_timestamp_meta_api_get_type()) }
|
|
}
|
|
}
|
|
|
|
#[cfg(any(feature = "v1_14", feature = "dox"))]
|
|
#[cfg_attr(feature = "dox", doc(cfg(feature = "v1_14")))]
|
|
impl fmt::Debug for ReferenceTimestampMeta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("ReferenceTimestampMeta")
|
|
.field("reference", &self.reference())
|
|
.field("timestamp", &self.timestamp())
|
|
.field("duration", &self.duration())
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_add_get_iterate_meta() {
|
|
crate::init().unwrap();
|
|
|
|
let mut buffer = crate::Buffer::new();
|
|
let parent = crate::Buffer::new();
|
|
{
|
|
let meta = ParentBufferMeta::add(buffer.get_mut().unwrap(), &parent);
|
|
unsafe {
|
|
assert_eq!(meta.parent().as_ptr(), parent.as_ptr());
|
|
}
|
|
}
|
|
|
|
{
|
|
let metas = buffer.iter_meta::<Meta>();
|
|
assert_eq!(metas.count(), 1);
|
|
}
|
|
{
|
|
let metas = buffer.get_mut().unwrap().iter_meta_mut::<Meta>();
|
|
assert_eq!(metas.count(), 1);
|
|
}
|
|
{
|
|
let metas = buffer.iter_meta::<ParentBufferMeta>().collect::<Vec<_>>();
|
|
assert_eq!(metas.len(), 1);
|
|
unsafe {
|
|
assert_eq!(metas[0].parent().as_ptr(), parent.as_ptr());
|
|
}
|
|
}
|
|
{
|
|
let metas = buffer
|
|
.get_mut()
|
|
.unwrap()
|
|
.iter_meta_mut::<ParentBufferMeta>()
|
|
.collect::<Vec<_>>();
|
|
assert_eq!(metas.len(), 1);
|
|
unsafe {
|
|
assert_eq!(metas[0].parent().as_ptr(), parent.as_ptr());
|
|
}
|
|
}
|
|
|
|
{
|
|
let meta = buffer
|
|
.get_mut()
|
|
.unwrap()
|
|
.meta_mut::<ParentBufferMeta>()
|
|
.unwrap();
|
|
unsafe {
|
|
assert_eq!(meta.parent().as_ptr(), parent.as_ptr());
|
|
}
|
|
meta.remove();
|
|
}
|
|
|
|
{
|
|
let metas = buffer.iter_meta::<Meta>();
|
|
assert_eq!(metas.count(), 0);
|
|
}
|
|
{
|
|
let metas = buffer.get_mut().unwrap().iter_meta_mut::<Meta>();
|
|
assert_eq!(metas.count(), 0);
|
|
}
|
|
{
|
|
let metas = buffer.iter_meta::<ParentBufferMeta>();
|
|
assert_eq!(metas.count(), 0);
|
|
}
|
|
{
|
|
let metas = buffer
|
|
.get_mut()
|
|
.unwrap()
|
|
.iter_meta_mut::<ParentBufferMeta>();
|
|
assert_eq!(metas.count(), 0);
|
|
}
|
|
|
|
assert!(buffer.meta::<ParentBufferMeta>().is_none());
|
|
}
|
|
}
|