From eb742c8ed0dad1a62bef8f65b4fac78746513c2a Mon Sep 17 00:00:00 2001 From: Mengkejiergeli Ba Date: Wed, 17 Nov 2021 17:05:09 +0800 Subject: [PATCH] msdkdec: Add SFC codes SFC refers to Scaler Format Converter, more details here: https://01.org/sites/default/files/documentation/intel-gfx-prm-osrc-kbl-vol15-sfc.pdf By attaching mfxExtDecVideoProcessing buffer to mfxVideoParam, decoder can do SFC for csc and scaling. MSDK will check if surface bitdepth equals to input when no SFC is triggered, which means in the case of SFC, there is no need to update surface bitdepth via gst_msdk_update_mfx_frame_info_from_mfx_video_param. Part-of: --- .../gst-plugins-bad/sys/msdk/gstmsdkdec.c | 54 +++++++++++++++++-- .../gst-plugins-bad/sys/msdk/gstmsdkdec.h | 5 ++ 2 files changed, 56 insertions(+), 3 deletions(-) diff --git a/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.c b/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.c index 0f121d7c7c..098ddcfd39 100644 --- a/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.c +++ b/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.c @@ -92,6 +92,15 @@ gst_msdkdec_add_bs_extra_param (GstMsdkDec * thiz, mfxExtBuffer * param) } } +void +gst_msdkdec_add_video_extra_param (GstMsdkDec * thiz, mfxExtBuffer * param) +{ + if (thiz->num_video_extra_params < MAX_VIDEO_EXTRA_PARAMS) { + thiz->video_extra_params[thiz->num_video_extra_params] = param; + thiz->num_video_extra_params++; + } +} + static GstVideoCodecFrame * gst_msdkdec_get_oldest_frame (GstVideoDecoder * decoder) { @@ -253,8 +262,9 @@ get_surface (GstMsdkDec * thiz, GstBuffer * buffer) goto failed_unref_buffer2; } - gst_msdk_update_mfx_frame_info_from_mfx_video_param (&i->surface->Info, - &thiz->param); + if (!thiz->sfc) + gst_msdk_update_mfx_frame_info_from_mfx_video_param (&i->surface->Info, + &thiz->param); thiz->locked_msdk_surfaces = g_list_append (thiz->locked_msdk_surfaces, i); return i; @@ -298,6 +308,7 @@ gst_msdkdec_close_decoder (GstMsdkDec * thiz, gboolean reset_param) memset (&thiz->param, 0, sizeof (thiz->param)); thiz->num_bs_extra_params = 0; + thiz->num_video_extra_params = 0; thiz->initialized = FALSE; gst_adapter_clear (thiz->adapter); } @@ -327,10 +338,16 @@ static gboolean gst_msdkdec_init_decoder (GstMsdkDec * thiz) { GstMsdkDecClass *klass = GST_MSDKDEC_GET_CLASS (thiz); - GstVideoInfo *info; + GstVideoInfo *info, *output_info; mfxSession session; mfxStatus status; mfxFrameAllocRequest request; +#if (MFX_VERSION >= 1022) + mfxExtDecVideoProcessing ext_dec_video_proc; +#endif + + GstVideoCodecState *output_state = + gst_video_decoder_get_output_state (GST_VIDEO_DECODER (thiz)); if (thiz->initialized) return TRUE; @@ -345,6 +362,7 @@ gst_msdkdec_init_decoder (GstMsdkDec * thiz) return FALSE; } info = &thiz->input_state->info; + output_info = &output_state->info; GST_OBJECT_LOCK (thiz); @@ -389,6 +407,35 @@ gst_msdkdec_init_decoder (GstMsdkDec * thiz) thiz->param.mfx.FrameInfo.ChromaFormat ? thiz->param.mfx. FrameInfo.ChromaFormat : MFX_CHROMAFORMAT_YUV420; +#if (MFX_VERSION >= 1022) + if (output_info && thiz->sfc) { + memset (&ext_dec_video_proc, 0, sizeof (ext_dec_video_proc)); + ext_dec_video_proc.Header.BufferId = MFX_EXTBUFF_DEC_VIDEO_PROCESSING; + ext_dec_video_proc.Header.BufferSz = sizeof (ext_dec_video_proc); + ext_dec_video_proc.In.CropW = thiz->param.mfx.FrameInfo.CropW; + ext_dec_video_proc.In.CropH = thiz->param.mfx.FrameInfo.CropH; + ext_dec_video_proc.In.CropX = 0; + ext_dec_video_proc.In.CropY = 0; + ext_dec_video_proc.Out.FourCC = + gst_msdk_get_mfx_fourcc_from_format (output_info->finfo->format); + ext_dec_video_proc.Out.ChromaFormat = + gst_msdk_get_mfx_chroma_from_format (output_info->finfo->format); + ext_dec_video_proc.Out.Width = GST_ROUND_UP_16 (output_info->width); + ext_dec_video_proc.Out.Height = GST_ROUND_UP_32 (output_info->height); + ext_dec_video_proc.Out.CropW = output_info->width; + ext_dec_video_proc.Out.CropH = output_info->height; + ext_dec_video_proc.Out.CropX = 0; + ext_dec_video_proc.Out.CropY = 0; + gst_msdkdec_add_video_extra_param (thiz, + (mfxExtBuffer *) & ext_dec_video_proc); + } +#endif + + if (thiz->num_video_extra_params) { + thiz->param.NumExtParam = thiz->num_video_extra_params; + thiz->param.ExtParam = thiz->video_extra_params; + } + session = gst_msdk_context_get_session (thiz->context); /* validate parameters and allow MFX to make adjustments */ status = MFXVideoDECODE_Query (session, &thiz->param, &thiz->param); @@ -1955,6 +2002,7 @@ gst_msdkdec_init (GstMsdkDec * thiz) thiz->do_realloc = TRUE; thiz->force_reset_on_res_change = TRUE; thiz->report_error = FALSE; + thiz->sfc = FALSE; thiz->adapter = gst_adapter_new (); thiz->input_state = NULL; thiz->pool = NULL; diff --git a/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.h b/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.h index 638edfb008..e5b25d4ed7 100644 --- a/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.h +++ b/subprojects/gst-plugins-bad/sys/msdk/gstmsdkdec.h @@ -55,6 +55,7 @@ G_BEGIN_DECLS (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_MSDKDEC)) #define MAX_BS_EXTRA_PARAMS 8 +#define MAX_VIDEO_EXTRA_PARAMS 8 typedef struct _GstMsdkDec GstMsdkDec; typedef struct _GstMsdkDecClass GstMsdkDecClass; @@ -74,6 +75,7 @@ struct _GstMsdkDec gboolean use_video_memory; gboolean use_dmabuf; gboolean initialized; + gboolean sfc; /* for packetization */ GstAdapter *adapter; @@ -105,6 +107,9 @@ struct _GstMsdkDec mfxExtBuffer *bs_extra_params[MAX_BS_EXTRA_PARAMS]; guint num_bs_extra_params; + mfxExtBuffer *video_extra_params[MAX_VIDEO_EXTRA_PARAMS]; + guint num_video_extra_params; + #if (MFX_VERSION >= 1025) mfxExtDecodeErrorReport error_report; #endif