forked from mirrors/gstreamer-rs
220 lines
6.2 KiB
Rust
220 lines
6.2 KiB
Rust
// Copyright (C) 2018 Sebastian Dröge <sebastian@centricular.com>
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
use std::fmt;
|
|
|
|
use ffi;
|
|
use glib;
|
|
use glib::translate::{from_glib, ToGlib};
|
|
use gst;
|
|
use gst::prelude::*;
|
|
use gst_ffi;
|
|
|
|
#[repr(C)]
|
|
pub struct VideoMeta(ffi::GstVideoMeta);
|
|
|
|
impl VideoMeta {
|
|
pub fn add(
|
|
buffer: &mut gst::BufferRef,
|
|
flags: ::VideoFrameFlags,
|
|
format: ::VideoFormat,
|
|
width: u32,
|
|
height: u32,
|
|
) -> gst::MetaRefMut<Self, gst::meta::Standalone> {
|
|
let info = ::VideoInfo::new(format, width, height).build().unwrap();
|
|
assert!(buffer.get_size() >= info.size());
|
|
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_video_meta(
|
|
buffer.as_mut_ptr(),
|
|
flags.to_glib(),
|
|
format.to_glib(),
|
|
width,
|
|
height,
|
|
);
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
pub fn add_full<'a>(
|
|
buffer: &'a mut gst::BufferRef,
|
|
flags: ::VideoFrameFlags,
|
|
format: ::VideoFormat,
|
|
width: u32,
|
|
height: u32,
|
|
n_planes: u32,
|
|
offset: &[usize; 4],
|
|
stride: &[i32; 4],
|
|
) -> gst::MetaRefMut<'a, Self, gst::meta::Standalone> {
|
|
let info = ::VideoInfo::new(format, width, height)
|
|
.offset(offset)
|
|
.stride(stride)
|
|
.build()
|
|
.unwrap();
|
|
assert!(buffer.get_size() >= info.size());
|
|
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_video_meta_full(
|
|
buffer.as_mut_ptr(),
|
|
flags.to_glib(),
|
|
format.to_glib(),
|
|
width,
|
|
height,
|
|
n_planes,
|
|
offset.as_ptr() as *mut _,
|
|
stride.as_ptr() as *mut _,
|
|
);
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
pub fn get_flags(&self) -> ::VideoFrameFlags {
|
|
from_glib(self.0.flags)
|
|
}
|
|
|
|
pub fn get_format(&self) -> ::VideoFormat {
|
|
from_glib(self.0.format)
|
|
}
|
|
|
|
pub fn get_id(&self) -> i32 {
|
|
self.0.id
|
|
}
|
|
|
|
pub fn get_width(&self) -> u32 {
|
|
self.0.width
|
|
}
|
|
|
|
pub fn get_height(&self) -> u32 {
|
|
self.0.height
|
|
}
|
|
|
|
pub fn get_n_planes(&self) -> u32 {
|
|
self.0.n_planes
|
|
}
|
|
|
|
pub fn get_offset(&self) -> &[usize] {
|
|
&self.0.offset[0..(self.0.n_planes as usize)]
|
|
}
|
|
|
|
pub fn get_stride(&self) -> &[i32] {
|
|
&self.0.stride[0..(self.0.n_planes as usize)]
|
|
}
|
|
}
|
|
|
|
unsafe impl MetaAPI for VideoMeta {
|
|
type GstType = ffi::GstVideoMeta;
|
|
|
|
fn get_meta_api() -> glib::Type {
|
|
unsafe { from_glib(ffi::gst_video_meta_api_get_type()) }
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for VideoMeta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("VideoMeta")
|
|
.field("id", &self.get_id())
|
|
.field("flags", &self.get_flags())
|
|
.field("format", &self.get_format())
|
|
.field("width", &self.get_width())
|
|
.field("height", &self.get_height())
|
|
.field("n_planes", &self.get_n_planes())
|
|
.field("offset", &self.get_offset())
|
|
.field("stride", &self.get_stride())
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[repr(C)]
|
|
pub struct VideoOverlayCompositionMeta(ffi::GstVideoOverlayCompositionMeta);
|
|
|
|
impl VideoOverlayCompositionMeta {
|
|
pub fn add<'a>(
|
|
buffer: &'a mut gst::BufferRef,
|
|
overlay: &::VideoOverlayComposition,
|
|
) -> gst::MetaRefMut<'a, Self, gst::meta::Standalone> {
|
|
unsafe {
|
|
let meta = ffi::gst_buffer_add_video_overlay_composition_meta(
|
|
buffer.as_mut_ptr(),
|
|
overlay.as_mut_ptr(),
|
|
);
|
|
|
|
Self::from_mut_ptr(buffer, meta)
|
|
}
|
|
}
|
|
|
|
pub fn get_overlay(&self) -> &::VideoOverlayCompositionRef {
|
|
unsafe { ::VideoOverlayCompositionRef::from_ptr(self.0.overlay) }
|
|
}
|
|
|
|
pub fn set_overlay(&mut self, overlay: &::VideoOverlayComposition) {
|
|
#![allow(clippy::cast_ptr_alignment)]
|
|
unsafe {
|
|
gst_ffi::gst_mini_object_unref(self.0.overlay as *mut _);
|
|
self.0.overlay = gst_ffi::gst_mini_object_ref(overlay.as_mut_ptr() as *mut _) as *mut _;
|
|
}
|
|
}
|
|
}
|
|
|
|
unsafe impl MetaAPI for VideoOverlayCompositionMeta {
|
|
type GstType = ffi::GstVideoOverlayCompositionMeta;
|
|
|
|
fn get_meta_api() -> glib::Type {
|
|
unsafe { from_glib(ffi::gst_video_overlay_composition_meta_api_get_type()) }
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for VideoOverlayCompositionMeta {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
f.debug_struct("VideoOverlayCompositionMeta")
|
|
.field("overlay", &self.get_overlay())
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_add_get_meta() {
|
|
gst::init().unwrap();
|
|
|
|
let mut buffer = gst::Buffer::with_size(320 * 240 * 4).unwrap();
|
|
{
|
|
let meta = VideoMeta::add(
|
|
buffer.get_mut().unwrap(),
|
|
::VideoFrameFlags::NONE,
|
|
::VideoFormat::Argb,
|
|
320,
|
|
240,
|
|
);
|
|
assert_eq!(meta.get_id(), 0);
|
|
assert_eq!(meta.get_flags(), ::VideoFrameFlags::NONE);
|
|
assert_eq!(meta.get_format(), ::VideoFormat::Argb);
|
|
assert_eq!(meta.get_width(), 320);
|
|
assert_eq!(meta.get_height(), 240);
|
|
assert_eq!(meta.get_n_planes(), 1);
|
|
assert_eq!(meta.get_offset(), &[0]);
|
|
assert_eq!(meta.get_stride(), &[320 * 4]);
|
|
}
|
|
|
|
{
|
|
let meta = buffer.get_meta::<VideoMeta>().unwrap();
|
|
assert_eq!(meta.get_id(), 0);
|
|
assert_eq!(meta.get_flags(), ::VideoFrameFlags::NONE);
|
|
assert_eq!(meta.get_format(), ::VideoFormat::Argb);
|
|
assert_eq!(meta.get_width(), 320);
|
|
assert_eq!(meta.get_height(), 240);
|
|
assert_eq!(meta.get_n_planes(), 1);
|
|
assert_eq!(meta.get_offset(), &[0]);
|
|
assert_eq!(meta.get_stride(), &[320 * 4]);
|
|
}
|
|
}
|
|
}
|