mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-19 06:46:38 +00:00
msdk: Add help functions to get mfxFrameSurface1 from GstBuffer and wrap it as GstMsdkSurface
Note that the memory abstraction for system memory is for windows path. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/2498>
This commit is contained in:
parent
23f5bdcee7
commit
2854af85fd
6 changed files with 281 additions and 0 deletions
151
subprojects/gst-plugins-bad/sys/msdk/gstmsdkallocator.c
Normal file
151
subprojects/gst-plugins-bad/sys/msdk/gstmsdkallocator.c
Normal file
|
@ -0,0 +1,151 @@
|
||||||
|
/*
|
||||||
|
* GStreamer Intel MSDK plugin
|
||||||
|
* Copyright (c) 2022 Intel Corporation. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* 3. Neither the name of the copyright holder nor the names of its contributors
|
||||||
|
* may be used to endorse or promote products derived from this software
|
||||||
|
* without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
||||||
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||||
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
||||||
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
||||||
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
|
||||||
|
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "gstmsdkallocator.h"
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
map_data (GstBuffer * buffer, mfxFrameSurface1 * mfx_surface, GstVideoInfo info)
|
||||||
|
{
|
||||||
|
guint stride;
|
||||||
|
GstVideoFrame frame;
|
||||||
|
|
||||||
|
if (!gst_video_frame_map (&frame, &info, buffer, GST_MAP_READWRITE))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
stride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, 0);
|
||||||
|
|
||||||
|
switch (GST_VIDEO_INFO_FORMAT (&info)) {
|
||||||
|
case GST_VIDEO_FORMAT_NV12:
|
||||||
|
case GST_VIDEO_FORMAT_P010_10LE:
|
||||||
|
case GST_VIDEO_FORMAT_P012_LE:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.UV = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 1);
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_YV12:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 2);
|
||||||
|
mfx_surface->Data.V = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 1);
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_I420:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 1);
|
||||||
|
mfx_surface->Data.V = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 2);
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_YUY2:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = mfx_surface->Data.Y + 1;
|
||||||
|
mfx_surface->Data.V = mfx_surface->Data.Y + 3;
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_UYVY:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = mfx_surface->Data.Y;
|
||||||
|
mfx_surface->Data.V = mfx_surface->Data.U + 2;
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_VUYA:
|
||||||
|
mfx_surface->Data.V = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = mfx_surface->Data.V + 1;
|
||||||
|
mfx_surface->Data.Y = mfx_surface->Data.V + 2;
|
||||||
|
mfx_surface->Data.A = mfx_surface->Data.V + 3;
|
||||||
|
mfx_surface->Data.PitchHigh = (mfxU16) (stride / (1 << 16));
|
||||||
|
mfx_surface->Data.PitchLow = (mfxU16) (stride % (1 << 16));
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_BGRA:
|
||||||
|
case GST_VIDEO_FORMAT_BGRx:
|
||||||
|
mfx_surface->Data.B = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.G = mfx_surface->Data.B + 1;
|
||||||
|
mfx_surface->Data.R = mfx_surface->Data.B + 2;
|
||||||
|
mfx_surface->Data.A = mfx_surface->Data.B + 3;
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_Y210:
|
||||||
|
case GST_VIDEO_FORMAT_Y212_LE:
|
||||||
|
mfx_surface->Data.Y = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.U = mfx_surface->Data.Y + 2;
|
||||||
|
mfx_surface->Data.V = mfx_surface->Data.Y + 6;
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_Y410:
|
||||||
|
mfx_surface->Data.Y410 =
|
||||||
|
(mfxY410 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.Pitch = stride;
|
||||||
|
break;
|
||||||
|
case GST_VIDEO_FORMAT_Y412_LE:
|
||||||
|
mfx_surface->Data.U = (mfxU8 *) GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
|
||||||
|
mfx_surface->Data.Y = mfx_surface->Data.Y + 2;
|
||||||
|
mfx_surface->Data.V = mfx_surface->Data.Y + 4;
|
||||||
|
mfx_surface->Data.A = mfx_surface->Data.Y + 6;
|
||||||
|
mfx_surface->Data.Pitch = (mfxU16) stride;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_assert_not_reached ();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_video_frame_unmap (&frame);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
GstMsdkSurface *
|
||||||
|
gst_msdk_import_sys_mem_to_msdk_surface (GstBuffer * buf, GstVideoInfo info)
|
||||||
|
{
|
||||||
|
GstMsdkSurface *msdk_surface = NULL;
|
||||||
|
GstMapInfo map_info;
|
||||||
|
mfxFrameInfo frame_info = { 0, };
|
||||||
|
mfxFrameSurface1 *mfx_surface = NULL;
|
||||||
|
|
||||||
|
if (!gst_buffer_map (buf, &map_info, GST_MAP_READ)) {
|
||||||
|
GST_ERROR ("Failed to map buffer");
|
||||||
|
return msdk_surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
mfx_surface = g_slice_new0 (mfxFrameSurface1);
|
||||||
|
mfx_surface->Data.MemId = (mfxMemId) map_info.data;
|
||||||
|
|
||||||
|
if (!map_data (buf, mfx_surface, info)) {
|
||||||
|
g_slice_free (mfxFrameSurface1, mfx_surface);
|
||||||
|
return msdk_surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_buffer_unmap (buf, &map_info);
|
||||||
|
|
||||||
|
gst_msdk_set_mfx_frame_info_from_video_info (&frame_info, &info);
|
||||||
|
mfx_surface->Info = frame_info;
|
||||||
|
|
||||||
|
msdk_surface = g_slice_new0 (GstMsdkSurface);
|
||||||
|
msdk_surface->surface = mfx_surface;
|
||||||
|
|
||||||
|
return msdk_surface;
|
||||||
|
}
|
|
@ -39,6 +39,7 @@
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
typedef struct _GstMsdkMemoryID GstMsdkMemoryID;
|
typedef struct _GstMsdkMemoryID GstMsdkMemoryID;
|
||||||
|
typedef struct _GstMsdkSurface GstMsdkSurface;
|
||||||
|
|
||||||
struct _GstMsdkMemoryID {
|
struct _GstMsdkMemoryID {
|
||||||
mfxU32 fourcc;
|
mfxU32 fourcc;
|
||||||
|
@ -56,6 +57,15 @@ struct _GstMsdkMemoryID {
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct _GstMsdkSurface
|
||||||
|
{
|
||||||
|
mfxFrameSurface1 *surface;
|
||||||
|
GstBuffer *buf;
|
||||||
|
};
|
||||||
|
|
||||||
|
GstMsdkSurface *
|
||||||
|
gst_msdk_import_sys_mem_to_msdk_surface (GstBuffer * buf, GstVideoInfo info);
|
||||||
|
|
||||||
mfxStatus gst_msdk_frame_alloc(mfxHDL pthis, mfxFrameAllocRequest *req, mfxFrameAllocResponse *resp);
|
mfxStatus gst_msdk_frame_alloc(mfxHDL pthis, mfxFrameAllocRequest *req, mfxFrameAllocResponse *resp);
|
||||||
mfxStatus gst_msdk_frame_free(mfxHDL pthis, mfxFrameAllocResponse *resp);
|
mfxStatus gst_msdk_frame_free(mfxHDL pthis, mfxFrameAllocResponse *resp);
|
||||||
mfxStatus gst_msdk_frame_lock(mfxHDL pthis, mfxMemId mid, mfxFrameData *ptr);
|
mfxStatus gst_msdk_frame_lock(mfxHDL pthis, mfxMemId mid, mfxFrameData *ptr);
|
||||||
|
|
|
@ -37,6 +37,21 @@
|
||||||
#include "gstmsdkallocator_libva.h"
|
#include "gstmsdkallocator_libva.h"
|
||||||
#include "msdk_libva.h"
|
#include "msdk_libva.h"
|
||||||
|
|
||||||
|
#include <gst/va/gstvaallocator.h>
|
||||||
|
|
||||||
|
#define GST_MSDK_FRAME_SURFACE gst_msdk_frame_surface_quark_get ()
|
||||||
|
static GQuark
|
||||||
|
gst_msdk_frame_surface_quark_get (void)
|
||||||
|
{
|
||||||
|
static gsize g_quark;
|
||||||
|
|
||||||
|
if (g_once_init_enter (&g_quark)) {
|
||||||
|
gsize quark = (gsize) g_quark_from_static_string ("GstMsdkFrameSurface");
|
||||||
|
g_once_init_leave (&g_quark, quark);
|
||||||
|
}
|
||||||
|
return g_quark;
|
||||||
|
}
|
||||||
|
|
||||||
mfxStatus
|
mfxStatus
|
||||||
gst_msdk_frame_alloc (mfxHDL pthis, mfxFrameAllocRequest * req,
|
gst_msdk_frame_alloc (mfxHDL pthis, mfxFrameAllocRequest * req,
|
||||||
mfxFrameAllocResponse * resp)
|
mfxFrameAllocResponse * resp)
|
||||||
|
@ -704,6 +719,105 @@ error_create_surface:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static VASurfaceID
|
||||||
|
_get_va_surface (GstBuffer * buf, GstVideoInfo * info,
|
||||||
|
GstMsdkContext * msdk_context)
|
||||||
|
{
|
||||||
|
VASurfaceID va_surface = VA_INVALID_ID;
|
||||||
|
|
||||||
|
if (!info) {
|
||||||
|
va_surface = gst_va_buffer_get_surface (buf);
|
||||||
|
} else {
|
||||||
|
/* Update offset/stride/size if there is VideoMeta attached to
|
||||||
|
* the dma buffer, which is then used to get vasurface */
|
||||||
|
GstMemory *mem;
|
||||||
|
gint i, fd;
|
||||||
|
GstVideoMeta *vmeta;
|
||||||
|
|
||||||
|
vmeta = gst_buffer_get_video_meta (buf);
|
||||||
|
if (vmeta) {
|
||||||
|
if (GST_VIDEO_INFO_FORMAT (info) != vmeta->format ||
|
||||||
|
GST_VIDEO_INFO_WIDTH (info) != vmeta->width ||
|
||||||
|
GST_VIDEO_INFO_HEIGHT (info) != vmeta->height ||
|
||||||
|
GST_VIDEO_INFO_N_PLANES (info) != vmeta->n_planes) {
|
||||||
|
GST_ERROR ("VideoMeta attached to buffer is not matching"
|
||||||
|
"the negotiated width/height/format");
|
||||||
|
return va_surface;
|
||||||
|
}
|
||||||
|
for (i = 0; i < GST_VIDEO_INFO_N_PLANES (info); ++i) {
|
||||||
|
GST_VIDEO_INFO_PLANE_OFFSET (info, i) = vmeta->offset[i];
|
||||||
|
GST_VIDEO_INFO_PLANE_STRIDE (info, i) = vmeta->stride[i];
|
||||||
|
}
|
||||||
|
GST_VIDEO_INFO_SIZE (info) = gst_buffer_get_size (buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
mem = gst_buffer_peek_memory (buf, 0);
|
||||||
|
fd = gst_dmabuf_memory_get_fd (mem);
|
||||||
|
if (fd < 0)
|
||||||
|
return va_surface;
|
||||||
|
/* export dmabuf to vasurface */
|
||||||
|
if (!gst_msdk_export_dmabuf_to_vasurface (msdk_context, info, fd,
|
||||||
|
&va_surface))
|
||||||
|
return VA_INVALID_ID;
|
||||||
|
}
|
||||||
|
|
||||||
|
return va_surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
GstMsdkSurface *
|
||||||
|
gst_msdk_import_to_msdk_surface (GstBuffer * buf, GstMsdkContext * msdk_context,
|
||||||
|
GstVideoInfo * vinfo)
|
||||||
|
{
|
||||||
|
VASurfaceID va_surface = VA_INVALID_ID;
|
||||||
|
GstMemory *mem = NULL;
|
||||||
|
mfxFrameInfo frame_info = { 0, };
|
||||||
|
GstMsdkSurface *msdk_surface = NULL;
|
||||||
|
mfxFrameSurface1 *mfx_surface = NULL;
|
||||||
|
GstMsdkMemoryID *msdk_mid = NULL;
|
||||||
|
|
||||||
|
mem = gst_buffer_peek_memory (buf, 0);
|
||||||
|
msdk_surface = g_slice_new0 (GstMsdkSurface);
|
||||||
|
|
||||||
|
/* If buffer has qdata pointing to mfxFrameSurface1, directly extract it */
|
||||||
|
if ((mfx_surface = gst_mini_object_get_qdata (GST_MINI_OBJECT_CAST (mem),
|
||||||
|
GST_MSDK_FRAME_SURFACE))) {
|
||||||
|
msdk_surface->surface = mfx_surface;
|
||||||
|
return msdk_surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gst_msdk_is_va_mem (mem)) {
|
||||||
|
va_surface = _get_va_surface (buf, NULL, NULL);
|
||||||
|
} else if (gst_is_dmabuf_memory (mem)) {
|
||||||
|
/* For dma memory, videoinfo is used with dma fd to create va surface. */
|
||||||
|
GstVideoInfo info = *vinfo;
|
||||||
|
va_surface = _get_va_surface (buf, &info, msdk_context);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (va_surface == VA_INVALID_ID) {
|
||||||
|
g_slice_free (GstMsdkSurface, msdk_surface);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
mfx_surface = g_slice_new0 (mfxFrameSurface1);
|
||||||
|
msdk_mid = g_slice_new0 (GstMsdkMemoryID);
|
||||||
|
|
||||||
|
msdk_mid->surface = g_slice_new0 (VASurfaceID);
|
||||||
|
*msdk_mid->surface = va_surface;
|
||||||
|
|
||||||
|
mfx_surface->Data.MemId = (mfxMemId) msdk_mid;
|
||||||
|
|
||||||
|
gst_msdk_set_mfx_frame_info_from_video_info (&frame_info, vinfo);
|
||||||
|
mfx_surface->Info = frame_info;
|
||||||
|
|
||||||
|
/* Set mfxFrameSurface1 as qdata in buffer */
|
||||||
|
gst_mini_object_set_qdata (GST_MINI_OBJECT_CAST (mem),
|
||||||
|
GST_MSDK_FRAME_SURFACE, mfx_surface, NULL);
|
||||||
|
|
||||||
|
msdk_surface->surface = mfx_surface;
|
||||||
|
|
||||||
|
return msdk_surface;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* gst_msdk_replace_mfx_memid:
|
* gst_msdk_replace_mfx_memid:
|
||||||
* This method replace the internal VA Suface in mfxSurface with a new one
|
* This method replace the internal VA Suface in mfxSurface with a new one
|
||||||
|
|
|
@ -45,6 +45,10 @@ gboolean
|
||||||
gst_msdk_export_dmabuf_to_vasurface (GstMsdkContext *context,
|
gst_msdk_export_dmabuf_to_vasurface (GstMsdkContext *context,
|
||||||
GstVideoInfo *vinfo, gint fd, VASurfaceID *surface_id);
|
GstVideoInfo *vinfo, gint fd, VASurfaceID *surface_id);
|
||||||
|
|
||||||
|
GstMsdkSurface *
|
||||||
|
gst_msdk_import_to_msdk_surface (GstBuffer * buf, GstMsdkContext * msdk_context,
|
||||||
|
GstVideoInfo * vinfo);
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_msdk_replace_mfx_memid (GstMsdkContext *context,
|
gst_msdk_replace_mfx_memid (GstMsdkContext *context,
|
||||||
mfxFrameSurface1 *mfx_surface, VASurfaceID surface_id);
|
mfxFrameSurface1 *mfx_surface, VASurfaceID surface_id);
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
msdk_sources = [
|
msdk_sources = [
|
||||||
'gstmsdk.c',
|
'gstmsdk.c',
|
||||||
|
'gstmsdkallocator.c',
|
||||||
'gstmsdkbufferpool.c',
|
'gstmsdkbufferpool.c',
|
||||||
'gstmsdkcontext.c',
|
'gstmsdkcontext.c',
|
||||||
'gstmsdkcontextutil.c',
|
'gstmsdkcontextutil.c',
|
||||||
|
|
|
@ -99,6 +99,7 @@ void GstMFXUnload (mfxLoader loader);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
typedef struct _MsdkSession MsdkSession;
|
typedef struct _MsdkSession MsdkSession;
|
||||||
|
typedef struct _GstMsdkSurface GstMsdkSurface;
|
||||||
|
|
||||||
struct _MsdkSession
|
struct _MsdkSession
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue