msdkenc: Remove internal vpp function

The internal vpp function and msdkvpp plugin are duplicated,
so remove the internal vpp, please use msdkvpp plugin to do
conversion if necessary.

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4000>
This commit is contained in:
Yinhang Liu 2023-03-30 09:39:31 +08:00 committed by GStreamer Marge Bot
parent 8003acf5bb
commit f292b6061c
7 changed files with 101 additions and 362 deletions

View file

@ -237,6 +237,15 @@ gst_msdkav1enc_set_src_caps (GstMsdkEnc * encoder)
return caps; return caps;
} }
static gboolean
gst_msdkav1enc_is_format_supported (GstMsdkEnc * encoder, GstVideoFormat format)
{
if (format == GST_VIDEO_FORMAT_NV12 || format == GST_VIDEO_FORMAT_P010_10LE)
return TRUE;
return FALSE;
}
static void static void
gst_msdkav1enc_set_property (GObject * object, guint prop_id, gst_msdkav1enc_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec) const GValue * value, GParamSpec * pspec)
@ -363,6 +372,7 @@ gst_msdkav1enc_class_init (gpointer klass, gpointer data)
encoder_class->set_format = gst_msdkav1enc_set_format; encoder_class->set_format = gst_msdkav1enc_set_format;
encoder_class->configure = gst_msdkav1enc_configure; encoder_class->configure = gst_msdkav1enc_configure;
encoder_class->set_src_caps = gst_msdkav1enc_set_src_caps; encoder_class->set_src_caps = gst_msdkav1enc_set_src_caps;
encoder_class->is_format_supported = gst_msdkav1enc_is_format_supported;
encoder_class->qp_max = 255; encoder_class->qp_max = 255;
encoder_class->qp_min = 0; encoder_class->qp_min = 0;

View file

@ -496,10 +496,8 @@ gst_msdkenc_init_encoder (GstMsdkEnc * thiz)
GstVideoInfo *info; GstVideoInfo *info;
mfxSession session; mfxSession session;
mfxStatus status; mfxStatus status;
mfxFrameAllocRequest request[2]; mfxFrameAllocRequest request;
guint i; guint i;
gboolean need_vpp = TRUE;
GstVideoFormat encoder_input_fmt;
mfxExtVideoSignalInfo ext_vsi; mfxExtVideoSignalInfo ext_vsi;
if (thiz->initialized) { if (thiz->initialized) {
@ -522,128 +520,18 @@ gst_msdkenc_init_encoder (GstMsdkEnc * thiz)
session = gst_msdk_context_get_session (thiz->context); session = gst_msdk_context_get_session (thiz->context);
thiz->codename = msdk_get_platform_codename (session); thiz->codename = msdk_get_platform_codename (session);
thiz->has_vpp = FALSE;
if (thiz->use_video_memory) if (thiz->use_video_memory)
gst_msdk_set_frame_allocator (thiz->context); gst_msdk_set_frame_allocator (thiz->context);
encoder_input_fmt = GST_VIDEO_INFO_FORMAT (info); #if (MFX_VERSION < 2000)
need_vpp = klass->need_conversion (thiz, info, &encoder_input_fmt); /* check the format for MSDK path */
if (!klass->is_format_supported (thiz, GST_VIDEO_INFO_FORMAT (info))) {
if (need_vpp) { GST_ERROR_OBJECT (thiz,
switch (GST_VIDEO_INFO_FORMAT (info)) { "internal vpp is no longer supported, "
case GST_VIDEO_FORMAT_YV12: "please use msdkvpp plugin to do conversion first");
case GST_VIDEO_FORMAT_I420: goto failed;
thiz->vpp_param.vpp.In.FourCC = MFX_FOURCC_YV12;
thiz->vpp_param.vpp.In.ChromaFormat = MFX_CHROMAFORMAT_YUV420;
break;
case GST_VIDEO_FORMAT_YUY2:
thiz->vpp_param.vpp.In.FourCC = MFX_FOURCC_YUY2;
thiz->vpp_param.vpp.In.ChromaFormat = MFX_CHROMAFORMAT_YUV422;
break;
case GST_VIDEO_FORMAT_UYVY:
thiz->vpp_param.vpp.In.FourCC = MFX_FOURCC_UYVY;
thiz->vpp_param.vpp.In.ChromaFormat = MFX_CHROMAFORMAT_YUV422;
break;
case GST_VIDEO_FORMAT_BGRA:
thiz->vpp_param.vpp.In.FourCC = MFX_FOURCC_RGB4;
thiz->vpp_param.vpp.In.ChromaFormat = MFX_CHROMAFORMAT_YUV444;
break;
default:
g_assert_not_reached ();
break;
}
if (thiz->use_video_memory)
thiz->vpp_param.IOPattern =
MFX_IOPATTERN_IN_VIDEO_MEMORY | MFX_IOPATTERN_OUT_VIDEO_MEMORY;
else
thiz->vpp_param.IOPattern =
MFX_IOPATTERN_IN_SYSTEM_MEMORY | MFX_IOPATTERN_OUT_SYSTEM_MEMORY;
thiz->vpp_param.vpp.In.Width = GST_ROUND_UP_16 (info->width);
thiz->vpp_param.vpp.In.Height = GST_ROUND_UP_32 (info->height);
thiz->vpp_param.vpp.In.CropW = info->width;
thiz->vpp_param.vpp.In.CropH = info->height;
thiz->vpp_param.vpp.In.FrameRateExtN = info->fps_n;
thiz->vpp_param.vpp.In.FrameRateExtD = info->fps_d;
thiz->vpp_param.vpp.In.AspectRatioW = info->par_n;
thiz->vpp_param.vpp.In.AspectRatioH = info->par_d;
thiz->vpp_param.vpp.In.PicStruct = MFX_PICSTRUCT_PROGRESSIVE;
/* work-around to avoid zero fps in msdk structure */
if (0 == thiz->vpp_param.vpp.In.FrameRateExtN)
thiz->vpp_param.vpp.In.FrameRateExtN = 30;
thiz->vpp_param.vpp.Out = thiz->vpp_param.vpp.In;
switch (encoder_input_fmt) {
case GST_VIDEO_FORMAT_P010_10LE:
thiz->vpp_param.vpp.Out.FourCC = MFX_FOURCC_P010;
thiz->vpp_param.vpp.Out.ChromaFormat = MFX_CHROMAFORMAT_YUV420;
break;
case GST_VIDEO_FORMAT_YUY2:
thiz->vpp_param.vpp.Out.FourCC = MFX_FOURCC_YUY2;
thiz->vpp_param.vpp.Out.ChromaFormat = MFX_CHROMAFORMAT_YUV422;
break;
default:
thiz->vpp_param.vpp.Out.FourCC = MFX_FOURCC_NV12;
thiz->vpp_param.vpp.Out.ChromaFormat = MFX_CHROMAFORMAT_YUV420;
break;
}
/* validate parameters and allow MFX to make adjustments */
status = MFXVideoVPP_Query (session, &thiz->vpp_param, &thiz->vpp_param);
if (status < MFX_ERR_NONE) {
GST_ERROR_OBJECT (thiz, "Video VPP Query failed (%s)",
msdk_status_to_string (status));
goto failed;
} else if (status > MFX_ERR_NONE) {
GST_WARNING_OBJECT (thiz, "Video VPP Query returned: %s",
msdk_status_to_string (status));
}
status = MFXVideoVPP_QueryIOSurf (session, &thiz->vpp_param, request);
if (status < MFX_ERR_NONE) {
GST_ERROR_OBJECT (thiz, "VPP Query IO surfaces failed (%s)",
msdk_status_to_string (status));
goto failed;
} else if (status > MFX_ERR_NONE) {
GST_WARNING_OBJECT (thiz, "VPP Query IO surfaces returned: %s",
msdk_status_to_string (status));
}
if (thiz->use_video_memory)
request[0].NumFrameSuggested +=
gst_msdk_context_get_shared_async_depth (thiz->context);
thiz->num_vpp_surfaces = request[0].NumFrameSuggested;
status = MFXVideoVPP_Init (session, &thiz->vpp_param);
if (status < MFX_ERR_NONE) {
GST_ERROR_OBJECT (thiz, "Init failed (%s)",
msdk_status_to_string (status));
} else if (status > MFX_ERR_NONE) {
GST_WARNING_OBJECT (thiz, "Init returned: %s",
msdk_status_to_string (status));
}
status = MFXVideoVPP_GetVideoParam (session, &thiz->vpp_param);
if (status < MFX_ERR_NONE) {
mfxStatus status1;
GST_ERROR_OBJECT (thiz, "Get VPP Parameters failed (%s)",
msdk_status_to_string (status));
status1 = MFXVideoVPP_Close (session);
if (status1 != MFX_ERR_NONE && status1 != MFX_ERR_NOT_INITIALIZED)
GST_WARNING_OBJECT (thiz, "VPP close failed (%s)",
msdk_status_to_string (status1));
} else if (status > MFX_ERR_NONE) {
GST_WARNING_OBJECT (thiz, "Get VPP Parameters returned: %s",
msdk_status_to_string (status));
}
thiz->has_vpp = TRUE;
} }
#endif
thiz->param.AsyncDepth = thiz->async_depth; thiz->param.AsyncDepth = thiz->async_depth;
if (thiz->use_video_memory) if (thiz->use_video_memory)
@ -679,7 +567,7 @@ gst_msdkenc_init_encoder (GstMsdkEnc * thiz)
thiz->param.mfx.FrameInfo.FrameRateExtD, thiz->param.mfx.FrameInfo.FrameRateExtD,
thiz->param.mfx.FrameInfo.FrameRateExtN); thiz->param.mfx.FrameInfo.FrameRateExtN);
switch (encoder_input_fmt) { switch (GST_VIDEO_INFO_FORMAT (info)) {
case GST_VIDEO_FORMAT_P010_10LE: case GST_VIDEO_FORMAT_P010_10LE:
thiz->param.mfx.FrameInfo.FourCC = MFX_FOURCC_P010; thiz->param.mfx.FrameInfo.FourCC = MFX_FOURCC_P010;
thiz->param.mfx.FrameInfo.BitDepthLuma = 10; thiz->param.mfx.FrameInfo.BitDepthLuma = 10;
@ -790,7 +678,7 @@ gst_msdkenc_init_encoder (GstMsdkEnc * thiz)
msdk_status_to_string (status)); msdk_status_to_string (status));
} }
status = MFXVideoENCODE_QueryIOSurf (session, &thiz->param, request); status = MFXVideoENCODE_QueryIOSurf (session, &thiz->param, &request);
if (status < MFX_ERR_NONE) { if (status < MFX_ERR_NONE) {
GST_ERROR_OBJECT (thiz, "Encode Query IO surfaces failed (%s)", GST_ERROR_OBJECT (thiz, "Encode Query IO surfaces failed (%s)",
msdk_status_to_string (status)); msdk_status_to_string (status));
@ -800,27 +688,17 @@ gst_msdkenc_init_encoder (GstMsdkEnc * thiz)
msdk_status_to_string (status)); msdk_status_to_string (status));
} }
request[0].NumFrameSuggested += thiz->num_extra_frames; request.NumFrameSuggested += thiz->num_extra_frames;
if (thiz->has_vpp) if (request.NumFrameSuggested < thiz->param.AsyncDepth) {
request[0].NumFrameSuggested += thiz->num_vpp_surfaces + 1 - 4;
/* Maximum of VPP output and encoder input, if using VPP */
if (thiz->has_vpp)
request[0].NumFrameSuggested =
MAX (request[0].NumFrameSuggested, request[1].NumFrameSuggested);
if (request[0].NumFrameSuggested < thiz->param.AsyncDepth) {
GST_ERROR_OBJECT (thiz, "Required %d surfaces (%d suggested), async %d", GST_ERROR_OBJECT (thiz, "Required %d surfaces (%d suggested), async %d",
request[0].NumFrameMin, request[0].NumFrameSuggested, request.NumFrameMin, request.NumFrameSuggested, thiz->param.AsyncDepth);
thiz->param.AsyncDepth);
goto failed; goto failed;
} }
/* This is VPP output (if any) and encoder input */
thiz->num_surfaces = request[0].NumFrameSuggested;
GST_DEBUG_OBJECT (thiz, "Required %d surfaces (%d suggested), allocated %d", GST_DEBUG_OBJECT (thiz, "Required %d surfaces (%d suggested), allocated %d",
request[0].NumFrameMin, request[0].NumFrameSuggested, thiz->num_surfaces); request.NumFrameMin, request.NumFrameSuggested,
request.NumFrameSuggested);
status = MFXVideoENCODE_Init (session, &thiz->param); status = MFXVideoENCODE_Init (session, &thiz->param);
if (status < MFX_ERR_NONE) { if (status < MFX_ERR_NONE) {
@ -901,16 +779,6 @@ gst_msdkenc_close_encoder (GstMsdkEnc * thiz)
g_free (thiz->tasks); g_free (thiz->tasks);
thiz->tasks = NULL; thiz->tasks = NULL;
/* Close VPP before freeing the surfaces. They are shared between encoder
* and VPP */
if (thiz->has_vpp) {
status = MFXVideoVPP_Close (gst_msdk_context_get_session (thiz->context));
if (status != MFX_ERR_NONE && status != MFX_ERR_NOT_INITIALIZED) {
GST_WARNING_OBJECT (thiz, "VPP close failed (%s)",
msdk_status_to_string (status));
}
}
memset (&thiz->param, 0, sizeof (thiz->param)); memset (&thiz->param, 0, sizeof (thiz->param));
thiz->num_extra_params = 0; thiz->num_extra_params = 0;
thiz->initialized = FALSE; thiz->initialized = FALSE;
@ -951,8 +819,6 @@ gst_msdkenc_free_frame_data (GstMsdkEnc * thiz, FrameData * fdata)
{ {
if (fdata->frame_surface) if (fdata->frame_surface)
gst_msdkenc_free_surface (fdata->frame_surface); gst_msdkenc_free_surface (fdata->frame_surface);
if (thiz->has_vpp)
gst_msdkenc_free_surface (fdata->converted_surface);
gst_video_codec_frame_unref (fdata->frame); gst_video_codec_frame_unref (fdata->frame);
g_slice_free (FrameData, fdata); g_slice_free (FrameData, fdata);
@ -1555,86 +1421,9 @@ gst_msdkenc_set_format (GstVideoEncoder * encoder, GstVideoCodecState * state)
gst_msdkenc_set_latency (thiz); gst_msdkenc_set_latency (thiz);
/* Create another bufferpool if VPP requires */
if (thiz->has_vpp) {
GstVideoInfo *info = &thiz->input_state->info;
GstVideoInfo out_info;
GstVideoFormat out_fmt;
GstCaps *caps;
GstBufferPool *pool = NULL;
gst_video_info_init (&out_info);
out_fmt =
gst_msdk_get_video_format_from_mfx_fourcc (thiz->vpp_param.vpp.
Out.FourCC);
gst_video_info_set_format (&out_info, out_fmt, info->width, info->height);
caps = gst_video_info_to_caps (&out_info);
/* If there's an existing pool try to reuse it when is compatible */
if (thiz->msdk_converted_pool) {
GstStructure *config;
GstCaps *pool_caps;
gboolean is_pool_compatible = FALSE;
config = gst_buffer_pool_get_config (thiz->msdk_converted_pool);
gst_buffer_pool_config_get_params (config, &pool_caps, NULL, NULL, NULL);
if (caps && pool_caps)
is_pool_compatible = gst_caps_is_equal (caps, pool_caps);
gst_structure_free (config);
/* If caps are the same then we are done */
if (is_pool_compatible) {
gst_caps_unref (caps);
goto done;
}
/* Release current pool because we are going to create a new one */
gst_clear_object (&thiz->msdk_converted_pool);
}
/* Otherwise create a new pool */
pool =
gst_msdkenc_create_buffer_pool (thiz, caps, thiz->num_surfaces, FALSE);
thiz->msdk_converted_pool = pool;
gst_caps_unref (caps);
}
done:
return TRUE; return TRUE;
} }
/* This function will be removed later */
static GstMsdkSurface *
gst_msdkenc_get_surface_from_pool_old (GstMsdkEnc * thiz, GstBufferPool * pool,
GstBufferPoolAcquireParams * params)
{
GstBuffer *new_buffer;
GstMsdkSurface *msdk_surface = NULL;
if (!gst_buffer_pool_is_active (pool) &&
!gst_buffer_pool_set_active (pool, TRUE)) {
GST_ERROR_OBJECT (pool, "failed to activate buffer pool");
return NULL;
}
if (gst_buffer_pool_acquire_buffer (pool, &new_buffer, params) != GST_FLOW_OK) {
GST_ERROR_OBJECT (pool, "failed to acquire a buffer from pool");
return NULL;
}
#ifndef _WIN32
msdk_surface = gst_msdk_import_to_msdk_surface (new_buffer, thiz->context,
&thiz->aligned_info, 0);
#else
msdk_surface =
gst_msdk_import_sys_mem_to_msdk_surface (new_buffer, &thiz->aligned_info);
#endif
if (msdk_surface)
msdk_surface->buf = new_buffer;
return msdk_surface;
}
static GstMsdkSurface * static GstMsdkSurface *
gst_msdkenc_get_surface_from_pool (GstMsdkEnc * thiz, gst_msdkenc_get_surface_from_pool (GstMsdkEnc * thiz,
GstVideoCodecFrame * frame, GstBuffer * buf) GstVideoCodecFrame * frame, GstBuffer * buf)
@ -1763,86 +1552,29 @@ gst_msdkenc_handle_frame (GstVideoEncoder * encoder, GstVideoCodecFrame * frame)
if (G_UNLIKELY (thiz->context == NULL)) if (G_UNLIKELY (thiz->context == NULL))
goto not_inited; goto not_inited;
if (thiz->has_vpp) { surface = gst_msdkenc_get_surface_from_frame (thiz, frame);
GstMsdkSurface *vpp_surface; if (!surface)
GstVideoFrame vframe; goto invalid_surface;
mfxSession session;
mfxSyncPoint vpp_sync_point = NULL;
mfxStatus status;
vpp_surface = gst_msdkenc_get_surface_from_frame (thiz, frame); fdata = gst_msdkenc_queue_frame (thiz, frame, info);
if (!vpp_surface) if (!fdata)
goto invalid_surface; goto invalid_frame;
surface =
gst_msdkenc_get_surface_from_pool_old (thiz, thiz->msdk_converted_pool,
NULL);
if (!surface)
goto invalid_surface;
if (!gst_video_frame_map (&vframe, info, frame->input_buffer, GST_MAP_READ)) fdata->frame_surface = surface;
goto invalid_frame;
if (frame->pts != GST_CLOCK_TIME_NONE) { /* It is possible to have input frame without any framerate/pts info,
vpp_surface->surface->Data.TimeStamp = * we need to set the correct pts here. */
gst_util_uint64_scale (frame->pts, 90000, GST_SECOND); if (frame->presentation_frame_number == 0)
surface->surface->Data.TimeStamp = thiz->start_pts = frame->pts;
gst_util_uint64_scale (frame->pts, 90000, GST_SECOND);
} else {
vpp_surface->surface->Data.TimeStamp = MFX_TIMESTAMP_UNKNOWN;
surface->surface->Data.TimeStamp = MFX_TIMESTAMP_UNKNOWN;
}
session = gst_msdk_context_get_session (thiz->context); if (frame->pts != GST_CLOCK_TIME_NONE) {
for (;;) { frame->pts = thiz->start_pts +
status = frame->presentation_frame_number * thiz->frame_duration;
MFXVideoVPP_RunFrameVPPAsync (session, vpp_surface->surface, frame->duration = thiz->frame_duration;
surface->surface, NULL, &vpp_sync_point); surface->surface->Data.TimeStamp =
if (status != MFX_WRN_DEVICE_BUSY) gst_util_uint64_scale (frame->pts, 90000, GST_SECOND);
break;
/* If device is busy, wait 1ms and retry, as per MSDK's recomendation */
g_usleep (1000);
};
gst_video_frame_unmap (&vframe);
if (status != MFX_ERR_NONE && status != MFX_ERR_MORE_DATA) {
GST_ELEMENT_ERROR (thiz, STREAM, ENCODE, ("Converting frame failed."),
("MSDK VPP error (%s)", msdk_status_to_string (status)));
gst_video_encoder_finish_frame (GST_VIDEO_ENCODER (thiz), frame);
return GST_FLOW_ERROR;
}
fdata = g_slice_new0 (FrameData);
fdata->frame = gst_video_codec_frame_ref (frame);
fdata->frame_surface = vpp_surface;
fdata->converted_surface = surface;
thiz->pending_frames = g_list_prepend (thiz->pending_frames, fdata);
} else { } else {
surface = gst_msdkenc_get_surface_from_frame (thiz, frame); surface->surface->Data.TimeStamp = MFX_TIMESTAMP_UNKNOWN;
if (!surface)
goto invalid_surface;
fdata = gst_msdkenc_queue_frame (thiz, frame, info);
if (!fdata)
goto invalid_frame;
fdata->frame_surface = surface;
/* It is possible to have input frame without any framerate/pts info,
* we need to set the correct pts here. */
if (frame->presentation_frame_number == 0)
thiz->start_pts = frame->pts;
if (frame->pts != GST_CLOCK_TIME_NONE) {
frame->pts = thiz->start_pts +
frame->presentation_frame_number * thiz->frame_duration;
frame->duration = thiz->frame_duration;
surface->surface->Data.TimeStamp =
gst_util_uint64_scale (frame->pts, 90000, GST_SECOND);
} else {
surface->surface->Data.TimeStamp = MFX_TIMESTAMP_UNKNOWN;
}
} }
return gst_msdkenc_encode_frame (thiz, surface->surface, frame); return gst_msdkenc_encode_frame (thiz, surface->surface, frame);
@ -2192,27 +1924,9 @@ gst_msdkenc_dispose (GObject * object)
} }
static gboolean static gboolean
gst_msdkenc_need_conversion (GstMsdkEnc * encoder, GstVideoInfo * info, gst_msdkenc_is_format_supported (GstMsdkEnc * encoder, GstVideoFormat format)
GstVideoFormat * out_format)
{ {
switch (GST_VIDEO_INFO_FORMAT (info)) { return format == GST_VIDEO_FORMAT_NV12;
case GST_VIDEO_FORMAT_NV12:
case GST_VIDEO_FORMAT_P010_10LE:
case GST_VIDEO_FORMAT_VUYA:
case GST_VIDEO_FORMAT_BGRx:
#if (MFX_VERSION >= 1027)
case GST_VIDEO_FORMAT_Y410:
case GST_VIDEO_FORMAT_Y210:
#endif
return FALSE;
default:
if (GST_VIDEO_INFO_COMP_DEPTH (info, 0) == 10)
*out_format = GST_VIDEO_FORMAT_P010_10LE;
else
*out_format = GST_VIDEO_FORMAT_NV12;
return TRUE;
}
} }
static gboolean static gboolean
@ -2238,7 +1952,7 @@ gst_msdkenc_class_init (GstMsdkEncClass * klass)
element_class = GST_ELEMENT_CLASS (klass); element_class = GST_ELEMENT_CLASS (klass);
gstencoder_class = GST_VIDEO_ENCODER_CLASS (klass); gstencoder_class = GST_VIDEO_ENCODER_CLASS (klass);
klass->need_conversion = gst_msdkenc_need_conversion; klass->is_format_supported = gst_msdkenc_is_format_supported;
klass->need_reconfig = gst_msdkenc_need_reconfig; klass->need_reconfig = gst_msdkenc_need_reconfig;
klass->set_extra_params = gst_msdkenc_set_extra_params; klass->set_extra_params = gst_msdkenc_set_extra_params;
klass->qp_max = 51; klass->qp_max = 51;

View file

@ -107,7 +107,6 @@ struct _GstMsdkEnc
GstMsdkContext *context; GstMsdkContext *context;
GstMsdkContext *old_context; GstMsdkContext *old_context;
mfxVideoParam param; mfxVideoParam param;
guint num_surfaces;
guint num_tasks; guint num_tasks;
MsdkEncTask *tasks; MsdkEncTask *tasks;
guint next_task; guint next_task;
@ -115,13 +114,6 @@ struct _GstMsdkEnc
* the default value is 0 */ * the default value is 0 */
guint num_extra_frames; guint num_extra_frames;
gboolean has_vpp;
mfxVideoParam vpp_param;
guint num_vpp_surfaces;
/* Input interfaces, output above */
mfxFrameAllocResponse vpp_alloc_resp;
mfxFrameAllocResponse alloc_resp;
mfxExtBuffer *extra_params[MAX_EXTRA_PARAMS]; mfxExtBuffer *extra_params[MAX_EXTRA_PARAMS];
guint num_extra_params; guint num_extra_params;
@ -186,13 +178,7 @@ struct _GstMsdkEncClass
gboolean (*set_format) (GstMsdkEnc * encoder); gboolean (*set_format) (GstMsdkEnc * encoder);
gboolean (*configure) (GstMsdkEnc * encoder); gboolean (*configure) (GstMsdkEnc * encoder);
GstCaps *(*set_src_caps) (GstMsdkEnc * encoder); GstCaps *(*set_src_caps) (GstMsdkEnc * encoder);
/* Return TRUE if vpp is required before encoding gboolean (*is_format_supported) (GstMsdkEnc * encoder, GstVideoFormat format);
* @info (in), input video info
* @out_format (out), a pointer to the output format of vpp, which is set
* when return TRUE
*/
gboolean (*need_conversion) (GstMsdkEnc * encoder, GstVideoInfo * info,
GstVideoFormat * out_format);
/* Return TRUE if sub class requires a recofnig */ /* Return TRUE if sub class requires a recofnig */
gboolean (*need_reconfig) (GstMsdkEnc * encoder, GstVideoCodecFrame * frame); gboolean (*need_reconfig) (GstMsdkEnc * encoder, GstVideoCodecFrame * frame);

View file

@ -789,6 +789,23 @@ gst_msdkh264enc_set_extra_params (GstMsdkEnc * encoder,
gst_msdkenc_add_extra_param (encoder, (mfxExtBuffer *) & h264enc->roi[0]); gst_msdkenc_add_extra_param (encoder, (mfxExtBuffer *) & h264enc->roi[0]);
} }
static gboolean
gst_msdkh264enc_is_format_supported (GstMsdkEnc * encoder,
GstVideoFormat format)
{
switch (format) {
case GST_VIDEO_FORMAT_NV12:
case GST_VIDEO_FORMAT_YUY2:
case GST_VIDEO_FORMAT_VUYA:
case GST_VIDEO_FORMAT_UYVY:
case GST_VIDEO_FORMAT_BGRA:
case GST_VIDEO_FORMAT_BGRx:
return TRUE;
default:
return FALSE;
}
}
static void static void
_msdkh264enc_install_properties (GObjectClass * gobject_class, _msdkh264enc_install_properties (GObjectClass * gobject_class,
GstMsdkEncClass * encoder_class) GstMsdkEncClass * encoder_class)
@ -986,6 +1003,7 @@ gst_msdkh264enc_class_init (gpointer klass, gpointer data)
encoder_class->set_src_caps = gst_msdkh264enc_set_src_caps; encoder_class->set_src_caps = gst_msdkh264enc_set_src_caps;
encoder_class->need_reconfig = gst_msdkh264enc_need_reconfig; encoder_class->need_reconfig = gst_msdkh264enc_need_reconfig;
encoder_class->set_extra_params = gst_msdkh264enc_set_extra_params; encoder_class->set_extra_params = gst_msdkh264enc_set_extra_params;
encoder_class->is_format_supported = gst_msdkh264enc_is_format_supported;
_msdkh264enc_install_properties (gobject_class, encoder_class); _msdkh264enc_install_properties (gobject_class, encoder_class);

View file

@ -898,16 +898,19 @@ gst_msdkh265enc_set_extra_params (GstMsdkEnc * encoder,
} }
static gboolean static gboolean
gst_msdkh265enc_need_conversion (GstMsdkEnc * encoder, GstVideoInfo * info, gst_msdkh265enc_is_format_supported (GstMsdkEnc * encoder,
GstVideoFormat * out_format) GstVideoFormat format)
{ {
GstMsdkH265Enc *h265enc = GST_MSDKH265ENC (encoder); GstMsdkH265Enc *h265enc = GST_MSDKH265ENC (encoder);
switch (GST_VIDEO_INFO_FORMAT (info)) { switch (format) {
case GST_VIDEO_FORMAT_NV12: case GST_VIDEO_FORMAT_NV12:
case GST_VIDEO_FORMAT_VUYA:
case GST_VIDEO_FORMAT_BGRA:
case GST_VIDEO_FORMAT_BGRx:
case GST_VIDEO_FORMAT_Y212_LE:
case GST_VIDEO_FORMAT_BGR10A2_LE: case GST_VIDEO_FORMAT_BGR10A2_LE:
case GST_VIDEO_FORMAT_P010_10LE: case GST_VIDEO_FORMAT_P010_10LE:
case GST_VIDEO_FORMAT_VUYA:
#if (MFX_VERSION >= 1027) #if (MFX_VERSION >= 1027)
case GST_VIDEO_FORMAT_Y410: case GST_VIDEO_FORMAT_Y410:
case GST_VIDEO_FORMAT_Y210: case GST_VIDEO_FORMAT_Y210:
@ -915,20 +918,15 @@ gst_msdkh265enc_need_conversion (GstMsdkEnc * encoder, GstVideoInfo * info,
#if (MFX_VERSION >= 1031) #if (MFX_VERSION >= 1031)
case GST_VIDEO_FORMAT_P012_LE: case GST_VIDEO_FORMAT_P012_LE:
#endif #endif
return FALSE; return TRUE;
case GST_VIDEO_FORMAT_YUY2: case GST_VIDEO_FORMAT_YUY2:
#if (MFX_VERSION >= 1027) #if (MFX_VERSION >= 1027)
if (encoder->codename >= MFX_PLATFORM_ICELAKE && if (encoder->codename >= MFX_PLATFORM_ICELAKE &&
h265enc->tune_mode == MFX_CODINGOPTION_OFF) h265enc->tune_mode == MFX_CODINGOPTION_OFF)
return FALSE; return TRUE;
#endif #endif
default: default:
if (GST_VIDEO_INFO_COMP_DEPTH (info, 0) == 10) return FALSE;
*out_format = GST_VIDEO_FORMAT_P010_10LE;
else
*out_format = GST_VIDEO_FORMAT_NV12;
return TRUE;
} }
} }
@ -1123,7 +1121,7 @@ gst_msdkh265enc_class_init (gpointer klass, gpointer data)
encoder_class->set_src_caps = gst_msdkh265enc_set_src_caps; encoder_class->set_src_caps = gst_msdkh265enc_set_src_caps;
encoder_class->need_reconfig = gst_msdkh265enc_need_reconfig; encoder_class->need_reconfig = gst_msdkh265enc_need_reconfig;
encoder_class->set_extra_params = gst_msdkh265enc_set_extra_params; encoder_class->set_extra_params = gst_msdkh265enc_set_extra_params;
encoder_class->need_conversion = gst_msdkh265enc_need_conversion; encoder_class->is_format_supported = gst_msdkh265enc_is_format_supported;
_msdkh265enc_install_properties (gobject_class, encoder_class); _msdkh265enc_install_properties (gobject_class, encoder_class);

View file

@ -159,22 +159,18 @@ gst_msdkmjpegenc_set_property (GObject * object, guint prop_id,
} }
static gboolean static gboolean
gst_msdkmjpegenc_need_conversion (GstMsdkEnc * encoder, GstVideoInfo * info, gst_msdkmjpegenc_is_format_supported (GstMsdkEnc * encoder,
GstVideoFormat * out_format) GstVideoFormat format)
{ {
switch (GST_VIDEO_INFO_FORMAT (info)) { switch (format) {
case GST_VIDEO_FORMAT_NV12: case GST_VIDEO_FORMAT_NV12:
case GST_VIDEO_FORMAT_YUY2: case GST_VIDEO_FORMAT_YUY2:
case GST_VIDEO_FORMAT_BGRA:
return FALSE;
case GST_VIDEO_FORMAT_UYVY: case GST_VIDEO_FORMAT_UYVY:
*out_format = GST_VIDEO_FORMAT_YUY2; case GST_VIDEO_FORMAT_BGRA:
case GST_VIDEO_FORMAT_BGRx:
return TRUE; return TRUE;
default: default:
*out_format = GST_VIDEO_FORMAT_NV12; return FALSE;
return TRUE;
} }
} }
@ -195,7 +191,7 @@ gst_msdkmjpegenc_class_init (gpointer klass, gpointer data)
encoder_class->set_format = gst_msdkmjpegenc_set_format; encoder_class->set_format = gst_msdkmjpegenc_set_format;
encoder_class->configure = gst_msdkmjpegenc_configure; encoder_class->configure = gst_msdkmjpegenc_configure;
encoder_class->set_src_caps = gst_msdkmjpegenc_set_src_caps; encoder_class->set_src_caps = gst_msdkmjpegenc_set_src_caps;
encoder_class->need_conversion = gst_msdkmjpegenc_need_conversion; encoder_class->is_format_supported = gst_msdkmjpegenc_is_format_supported;
gobject_class->get_property = gst_msdkmjpegenc_get_property; gobject_class->get_property = gst_msdkmjpegenc_get_property;
gobject_class->set_property = gst_msdkmjpegenc_set_property; gobject_class->set_property = gst_msdkmjpegenc_set_property;

View file

@ -232,6 +232,22 @@ gst_msdkvp9enc_set_src_caps (GstMsdkEnc * encoder)
return caps; return caps;
} }
static gboolean
gst_msdkvp9enc_is_format_supported (GstMsdkEnc * encoder, GstVideoFormat format)
{
switch (format) {
case GST_VIDEO_FORMAT_NV12:
case GST_VIDEO_FORMAT_VUYA:
case GST_VIDEO_FORMAT_P010_10LE:
#if (MFX_VERSION >= 1027)
case GST_VIDEO_FORMAT_Y410:
#endif
return TRUE;
default:
return FALSE;
}
}
static void static void
gst_msdkvp9enc_set_property (GObject * object, guint prop_id, gst_msdkvp9enc_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec) const GValue * value, GParamSpec * pspec)
@ -272,6 +288,7 @@ gst_msdkvp9enc_class_init (gpointer klass, gpointer data)
encoder_class->set_format = gst_msdkvp9enc_set_format; encoder_class->set_format = gst_msdkvp9enc_set_format;
encoder_class->configure = gst_msdkvp9enc_configure; encoder_class->configure = gst_msdkvp9enc_configure;
encoder_class->set_src_caps = gst_msdkvp9enc_set_src_caps; encoder_class->set_src_caps = gst_msdkvp9enc_set_src_caps;
encoder_class->is_format_supported = gst_msdkvp9enc_is_format_supported;
encoder_class->qp_max = 255; encoder_class->qp_max = 255;
encoder_class->qp_min = 0; encoder_class->qp_min = 0;