mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer-rs.git
synced 2025-01-08 16:25:26 +00:00
gl: reimplement gl video frame support
Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer-rs/-/merge_requests/1312>
This commit is contained in:
parent
2a00236a1f
commit
4957921cfa
3 changed files with 193 additions and 44 deletions
|
@ -611,7 +611,7 @@ pub(crate) fn main_loop(app: App) -> Result<(), Error> {
|
||||||
|
|
||||||
let gl = load(&app.windowed_context);
|
let gl = load(&app.windowed_context);
|
||||||
|
|
||||||
let mut curr_frame: Option<gst_video::VideoFrame<gst_video::video_frame::Readable>> = None;
|
let mut curr_frame: Option<gst_gl::GLVideoFrame<gst_gl::gl_video_frame::Readable>> = None;
|
||||||
|
|
||||||
let App {
|
let App {
|
||||||
bus,
|
bus,
|
||||||
|
@ -651,7 +651,7 @@ pub(crate) fn main_loop(app: App) -> Result<(), Error> {
|
||||||
glutin::event::Event::RedrawRequested(_) => needs_redraw = true,
|
glutin::event::Event::RedrawRequested(_) => needs_redraw = true,
|
||||||
// Receive a frame
|
// Receive a frame
|
||||||
glutin::event::Event::UserEvent(Message::Frame(info, buffer)) => {
|
glutin::event::Event::UserEvent(Message::Frame(info, buffer)) => {
|
||||||
if let Ok(frame) = gst_video::VideoFrame::from_buffer_readable_gl(buffer, &info) {
|
if let Ok(frame) = gst_gl::GLVideoFrame::from_buffer_readable(buffer, &info) {
|
||||||
curr_frame = Some(frame);
|
curr_frame = Some(frame);
|
||||||
needs_redraw = true;
|
needs_redraw = true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,52 +1,117 @@
|
||||||
// Take a look at the license at the top of the repository in the LICENSE file.
|
// Take a look at the license at the top of the repository in the LICENSE file.
|
||||||
|
|
||||||
use std::mem;
|
use std::{marker::PhantomData, mem, ptr};
|
||||||
|
|
||||||
use glib::translate::{from_glib, ToGlibPtr};
|
use glib::translate::{from_glib, Borrowed, ToGlibPtr};
|
||||||
use gst_video::video_frame::Readable;
|
use gst_video::VideoFrameExt;
|
||||||
|
|
||||||
pub trait VideoFrameGLExt {
|
pub enum Readable {}
|
||||||
fn from_buffer_readable_gl(
|
|
||||||
buffer: gst::Buffer,
|
|
||||||
info: &gst_video::VideoInfo,
|
|
||||||
) -> Result<gst_video::VideoFrame<Readable>, gst::Buffer>;
|
|
||||||
|
|
||||||
fn from_buffer_ref_readable_gl<'a>(
|
// TODO: implement copy for videoframes. This would need to go through all the individual
|
||||||
buffer: &'a gst::BufferRef,
|
// memoryies and copy them. Some GL textures can be copied, others cannot.
|
||||||
info: &gst_video::VideoInfo,
|
|
||||||
) -> Result<gst_video::VideoFrameRef<&'a gst::BufferRef>, glib::error::BoolError>;
|
|
||||||
|
|
||||||
|
pub struct GLVideoFrame<T> {
|
||||||
|
frame: gst_video::ffi::GstVideoFrame,
|
||||||
|
buffer: gst::Buffer,
|
||||||
|
phantom: PhantomData<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe impl<T> Send for GLVideoFrame<T> {}
|
||||||
|
unsafe impl<T> Sync for GLVideoFrame<T> {}
|
||||||
|
|
||||||
|
// TODO implement Debug for GLVideoFrame
|
||||||
|
|
||||||
|
impl<T> VideoFrameExt for GLVideoFrame<T> {
|
||||||
|
#[inline]
|
||||||
|
fn info(&self) -> &gst_video::VideoInfo {
|
||||||
|
unsafe {
|
||||||
|
&*(&self.frame.info as *const gst_video::ffi::GstVideoInfo
|
||||||
|
as *const gst_video::VideoInfo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn flags(&self) -> gst_video::VideoFrameFlags {
|
||||||
|
unsafe { from_glib(self.frame.flags) }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn id(&self) -> i32 {
|
||||||
|
self.frame.id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> GLVideoFrame<T> {
|
||||||
#[doc(alias = "get_texture_id")]
|
#[doc(alias = "get_texture_id")]
|
||||||
fn texture_id(&self, idx: u32) -> Option<u32>;
|
pub fn texture_id(&self, idx: u32) -> Option<u32> {
|
||||||
}
|
self.as_video_frame_gl_ref().texture_id(idx)
|
||||||
|
|
||||||
impl VideoFrameGLExt for gst_video::VideoFrame<Readable> {
|
|
||||||
fn from_buffer_readable_gl(
|
|
||||||
buffer: gst::Buffer,
|
|
||||||
info: &gst_video::VideoInfo,
|
|
||||||
) -> Result<gst_video::VideoFrame<Readable>, gst::Buffer> {
|
|
||||||
skip_assert_initialized!();
|
|
||||||
gst_video::VideoFrameRef::<&gst::BufferRef>::from_buffer_readable_gl(buffer, info)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn from_buffer_ref_readable_gl<'a>(
|
#[inline]
|
||||||
buffer: &'a gst::BufferRef,
|
pub fn into_buffer(self) -> gst::Buffer {
|
||||||
info: &gst_video::VideoInfo,
|
unsafe {
|
||||||
) -> Result<gst_video::VideoFrameRef<&'a gst::BufferRef>, glib::error::BoolError> {
|
let mut s = mem::ManuallyDrop::new(self);
|
||||||
skip_assert_initialized!();
|
let buffer = ptr::read(&s.buffer);
|
||||||
gst_video::VideoFrameRef::<&gst::BufferRef>::from_buffer_ref_readable_gl(buffer, info)
|
gst_video::ffi::gst_video_frame_unmap(&mut s.frame);
|
||||||
|
buffer
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn texture_id(&self, idx: u32) -> Option<u32> {
|
#[inline]
|
||||||
self.as_video_frame_ref().texture_id(idx)
|
pub fn buffer(&self) -> &gst::BufferRef {
|
||||||
|
unsafe { gst::BufferRef::from_ptr(self.frame.buffer) }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub unsafe fn from_glib_full(frame: gst_video::ffi::GstVideoFrame) -> Self {
|
||||||
|
let buffer = gst::Buffer::from_glib_none(frame.buffer);
|
||||||
|
Self {
|
||||||
|
frame,
|
||||||
|
buffer,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn as_ptr(&self) -> *const gst_video::ffi::GstVideoFrame {
|
||||||
|
&self.frame
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn into_raw(self) -> gst_video::ffi::GstVideoFrame {
|
||||||
|
unsafe {
|
||||||
|
let mut s = mem::ManuallyDrop::new(self);
|
||||||
|
ptr::drop_in_place(&mut s.buffer);
|
||||||
|
s.frame
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn as_video_frame_gl_ref(&self) -> GLVideoFrameRef<&gst::BufferRef> {
|
||||||
|
let frame = unsafe { ptr::read(&self.frame) };
|
||||||
|
GLVideoFrameRef {
|
||||||
|
frame,
|
||||||
|
unmap: false,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> VideoFrameGLExt for gst_video::VideoFrameRef<&'a gst::BufferRef> {
|
impl<T> Drop for GLVideoFrame<T> {
|
||||||
fn from_buffer_readable_gl(
|
#[inline]
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe {
|
||||||
|
gst_video::ffi::gst_video_frame_unmap(&mut self.frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl GLVideoFrame<Readable> {
|
||||||
|
#[inline]
|
||||||
|
pub fn from_buffer_readable(
|
||||||
buffer: gst::Buffer,
|
buffer: gst::Buffer,
|
||||||
info: &gst_video::VideoInfo,
|
info: &gst_video::VideoInfo,
|
||||||
) -> Result<gst_video::VideoFrame<Readable>, gst::Buffer> {
|
) -> Result<Self, gst::Buffer> {
|
||||||
skip_assert_initialized!();
|
skip_assert_initialized!();
|
||||||
|
|
||||||
let n_mem = match buffer_n_gl_memory(buffer.as_ref()) {
|
let n_mem = match buffer_n_gl_memory(buffer.as_ref()) {
|
||||||
|
@ -82,15 +147,80 @@ impl<'a> VideoFrameGLExt for gst_video::VideoFrameRef<&'a gst::BufferRef> {
|
||||||
frame.info.size = 0;
|
frame.info.size = 0;
|
||||||
frame.info.stride.fill(0);
|
frame.info.stride.fill(0);
|
||||||
frame.info.offset.fill(0);
|
frame.info.offset.fill(0);
|
||||||
Ok(gst_video::VideoFrame::from_glib_full(frame))
|
Ok(Self {
|
||||||
|
frame,
|
||||||
|
buffer,
|
||||||
|
phantom: PhantomData,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn from_buffer_ref_readable_gl<'b>(
|
pub struct GLVideoFrameRef<T> {
|
||||||
buffer: &'b gst::BufferRef,
|
frame: gst_video::ffi::GstVideoFrame,
|
||||||
info: &gst_video::VideoInfo,
|
unmap: bool,
|
||||||
) -> Result<gst_video::VideoFrameRef<&'b gst::BufferRef>, glib::error::BoolError> {
|
phantom: PhantomData<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe impl<T> Send for GLVideoFrameRef<T> {}
|
||||||
|
unsafe impl<T> Sync for GLVideoFrameRef<T> {}
|
||||||
|
|
||||||
|
impl<T> VideoFrameExt for GLVideoFrameRef<T> {
|
||||||
|
#[inline]
|
||||||
|
fn info(&self) -> &gst_video::VideoInfo {
|
||||||
|
unsafe {
|
||||||
|
&*(&self.frame.info as *const gst_video::ffi::GstVideoInfo
|
||||||
|
as *const gst_video::VideoInfo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn flags(&self) -> gst_video::VideoFrameFlags {
|
||||||
|
unsafe { from_glib(self.frame.flags) }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn id(&self) -> i32 {
|
||||||
|
self.frame.id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// TODO implement Debug for GLVideoFrameRef
|
||||||
|
//
|
||||||
|
impl<T> GLVideoFrameRef<T> {
|
||||||
|
#[inline]
|
||||||
|
pub fn as_ptr(&self) -> *const gst_video::ffi::GstVideoFrame {
|
||||||
|
&self.frame
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> GLVideoFrameRef<&'a gst::BufferRef> {
|
||||||
|
#[inline]
|
||||||
|
pub unsafe fn from_glib_borrow(frame: *const gst_video::ffi::GstVideoFrame) -> Borrowed<Self> {
|
||||||
|
debug_assert!(!frame.is_null());
|
||||||
|
|
||||||
|
let frame = ptr::read(frame);
|
||||||
|
Borrowed::new(Self {
|
||||||
|
frame,
|
||||||
|
unmap: false,
|
||||||
|
phantom: PhantomData,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub unsafe fn from_glib_full(frame: gst_video::ffi::GstVideoFrame) -> Self {
|
||||||
|
Self {
|
||||||
|
frame,
|
||||||
|
unmap: true,
|
||||||
|
phantom: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn from_buffer_ref_readable<'b>(
|
||||||
|
buffer: &'a gst::BufferRef,
|
||||||
|
info: &'b gst_video::VideoInfo,
|
||||||
|
) -> Result<GLVideoFrameRef<&'a gst::BufferRef>, glib::error::BoolError> {
|
||||||
skip_assert_initialized!();
|
skip_assert_initialized!();
|
||||||
|
|
||||||
let n_mem = match buffer_n_gl_memory(buffer) {
|
let n_mem = match buffer_n_gl_memory(buffer) {
|
||||||
|
@ -130,12 +260,20 @@ impl<'a> VideoFrameGLExt for gst_video::VideoFrameRef<&'a gst::BufferRef> {
|
||||||
frame.info.size = 0;
|
frame.info.size = 0;
|
||||||
frame.info.stride.fill(0);
|
frame.info.stride.fill(0);
|
||||||
frame.info.offset.fill(0);
|
frame.info.offset.fill(0);
|
||||||
Ok(gst_video::VideoFrameRef::from_glib_full(frame))
|
Ok(Self {
|
||||||
|
frame,
|
||||||
|
unmap: true,
|
||||||
|
phantom: PhantomData,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn texture_id(&self, idx: u32) -> Option<u32> {
|
pub fn buffer(&self) -> &gst::BufferRef {
|
||||||
|
unsafe { gst::BufferRef::from_ptr(self.frame.buffer) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn texture_id(&self, idx: u32) -> Option<u32> {
|
||||||
let len = buffer_n_gl_memory(self.buffer())?;
|
let len = buffer_n_gl_memory(self.buffer())?;
|
||||||
|
|
||||||
if idx >= len {
|
if idx >= len {
|
||||||
|
@ -154,6 +292,17 @@ impl<'a> VideoFrameGLExt for gst_video::VideoFrameRef<&'a gst::BufferRef> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T> Drop for GLVideoFrameRef<T> {
|
||||||
|
#[inline]
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe {
|
||||||
|
if self.unmap {
|
||||||
|
gst_video::ffi::gst_video_frame_unmap(&mut self.frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn buffer_n_gl_memory(buffer: &gst::BufferRef) -> Option<u32> {
|
fn buffer_n_gl_memory(buffer: &gst::BufferRef) -> Option<u32> {
|
||||||
skip_assert_initialized!();
|
skip_assert_initialized!();
|
||||||
unsafe {
|
unsafe {
|
||||||
|
|
|
@ -36,8 +36,9 @@ pub use crate::functions::*;
|
||||||
mod gl_context;
|
mod gl_context;
|
||||||
mod gl_display;
|
mod gl_display;
|
||||||
mod gl_sync_meta;
|
mod gl_sync_meta;
|
||||||
mod gl_video_frame;
|
pub mod gl_video_frame;
|
||||||
pub use crate::gl_sync_meta::*;
|
pub use crate::gl_sync_meta::*;
|
||||||
|
pub use crate::gl_video_frame::{GLVideoFrame, GLVideoFrameRef, Readable};
|
||||||
mod gl_base_memory;
|
mod gl_base_memory;
|
||||||
pub use self::gl_base_memory::*;
|
pub use self::gl_base_memory::*;
|
||||||
mod gl_memory;
|
mod gl_memory;
|
||||||
|
@ -55,7 +56,6 @@ pub mod prelude {
|
||||||
pub use crate::{
|
pub use crate::{
|
||||||
auto::traits::*, context::ContextGLExt, gl_context::GLContextExtManual,
|
auto::traits::*, context::ContextGLExt, gl_context::GLContextExtManual,
|
||||||
gl_display::GLDisplayExtManual, gl_framebuffer::GLFramebufferExtManual,
|
gl_display::GLDisplayExtManual, gl_framebuffer::GLFramebufferExtManual,
|
||||||
gl_video_frame::VideoFrameGLExt,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue