mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 17:18:15 +00:00
907aca399f
drm_fourcc.h should be picked up via the pkgconfig include, not the system includedir directly. Also consolidate the libdrm usage in va and msdk. All this allows it to be picked up consistently (via the subproject, for example). Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/6932>
2126 lines
57 KiB
C
2126 lines
57 KiB
C
/* GStreamer Intel MSDK plugin
|
|
* Copyright (c) 2023, 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 "gstmsdkcaps.h"
|
|
|
|
#ifndef _WIN32
|
|
#include <drm_fourcc.h>
|
|
#include <gst/video/video-info-dma.h>
|
|
#include "gstmsdkallocator_libva.h"
|
|
#include <gst/va/gstvavideoformat.h>
|
|
#endif
|
|
|
|
#define DEFAULT_DELIMITER ", "
|
|
#define PROFILE_DELIMITER DEFAULT_DELIMITER
|
|
|
|
#define DEFAULT_VIDEO_FORMAT GST_VIDEO_FORMAT_NV12
|
|
|
|
#define ENC_IOPATTERN MFX_IOPATTERN_IN_VIDEO_MEMORY
|
|
#define DEC_IOPATTERN MFX_IOPATTERN_OUT_VIDEO_MEMORY
|
|
#define VPP_IOPATTERN \
|
|
MFX_IOPATTERN_IN_VIDEO_MEMORY | MFX_IOPATTERN_OUT_VIDEO_MEMORY
|
|
|
|
#ifdef _WIN32
|
|
/* fix "unreferenced local variable" for windows */
|
|
#define VAR_UNUSED(v) (void)(v)
|
|
#endif
|
|
|
|
#if (MFX_VERSION >= 2000)
|
|
static guint default_width = GST_ROUND_UP_16 (320);
|
|
static guint default_height = GST_ROUND_UP_16 (240);
|
|
|
|
static guint max_res_widths[] = { 640, 1280, 1920, 2048, 4096, 8192, 16384 };
|
|
|
|
static guint max_res_heights[] = {
|
|
480, 720, 1080, 1920, 2048, 4096, 8192, 12288, 16384
|
|
};
|
|
#endif
|
|
|
|
typedef struct
|
|
{
|
|
mfxU32 id;
|
|
const gchar *names;
|
|
} Profile;
|
|
|
|
static const Profile profs_avc[] = {
|
|
{MFX_PROFILE_AVC_MAIN, "main"},
|
|
{MFX_PROFILE_AVC_BASELINE, "baseline"},
|
|
{MFX_PROFILE_AVC_EXTENDED, "extended"},
|
|
{MFX_PROFILE_AVC_HIGH, "high"},
|
|
{MFX_PROFILE_AVC_CONSTRAINED_BASELINE, "constrained-baseline"},
|
|
{MFX_PROFILE_AVC_CONSTRAINED_HIGH, "constrained-high"},
|
|
{MFX_PROFILE_AVC_PROGRESSIVE_HIGH, "progressive-high"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_hevc[] = {
|
|
{MFX_PROFILE_HEVC_MAIN, "main"},
|
|
{MFX_PROFILE_HEVC_MAIN10, "main-10, main-10-still-picture"},
|
|
{MFX_PROFILE_HEVC_MAINSP, "main-still-picture"},
|
|
{MFX_PROFILE_HEVC_REXT,
|
|
"main-444, main-444-10, main-422-10, main-12, main-422-12"},
|
|
#if (MFX_VERSION >= 1032)
|
|
{MFX_PROFILE_HEVC_SCC, "screen-extended-main, screen-extended-main-10, "
|
|
"screen-extended-main-444, screen-extended-main-444-10"},
|
|
#endif
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_mpeg2[] = {
|
|
{MFX_PROFILE_MPEG2_MAIN, "main"},
|
|
{MFX_PROFILE_MPEG2_SIMPLE, "simple"},
|
|
{MFX_PROFILE_MPEG2_HIGH, "high"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_vc1[] = {
|
|
{MFX_PROFILE_VC1_MAIN, "main"},
|
|
{MFX_PROFILE_VC1_SIMPLE, "simple"},
|
|
{MFX_PROFILE_VC1_ADVANCED, "advanced"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_vp8[] = {
|
|
{MFX_PROFILE_VP8_0, "0"},
|
|
{MFX_PROFILE_VP8_1, "1"},
|
|
{MFX_PROFILE_VP8_2, "2"},
|
|
{MFX_PROFILE_VP8_3, "3"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_vp9[] = {
|
|
{MFX_PROFILE_VP9_0, "0"},
|
|
{MFX_PROFILE_VP9_1, "1"},
|
|
{MFX_PROFILE_VP9_2, "2"},
|
|
{MFX_PROFILE_VP9_3, "3"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_av1[] = {
|
|
#if (MFX_VERSION >= 1034)
|
|
{MFX_PROFILE_AV1_MAIN, "main"},
|
|
{MFX_PROFILE_AV1_HIGH, "high"},
|
|
{MFX_PROFILE_AV1_PRO, "pro"},
|
|
#endif
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const Profile profs_jpeg[] = {
|
|
{MFX_PROFILE_JPEG_BASELINE, "baseline"},
|
|
{MFX_PROFILE_UNKNOWN, NULL}
|
|
};
|
|
|
|
static const struct CodecProfiles
|
|
{
|
|
guint codec;
|
|
const gchar *media_type;
|
|
const Profile *profiles;
|
|
} codec_profs[] = {
|
|
{MFX_CODEC_AVC, "video/x-h264", profs_avc},
|
|
{MFX_CODEC_HEVC, "video/x-h265", profs_hevc},
|
|
{MFX_CODEC_MPEG2, "video/mpeg", profs_mpeg2},
|
|
{MFX_CODEC_VC1, "video/x-wmv", profs_vc1},
|
|
{MFX_CODEC_VP8, "video/x-vp8", profs_vp8},
|
|
{MFX_CODEC_VP9, "video/x-vp9", profs_vp9},
|
|
{MFX_CODEC_AV1, "video/x-av1", profs_av1},
|
|
{MFX_CODEC_JPEG, "image/jpeg", profs_jpeg}
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
guint min_width;
|
|
guint max_width;
|
|
guint min_height;
|
|
guint max_height;
|
|
} ResolutionRange;
|
|
|
|
typedef gboolean (*IsParamSupportedFunc) (mfxSession * session,
|
|
mfxVideoParam * in, mfxVideoParam * out);
|
|
|
|
gboolean
|
|
gst_msdkcaps_has_feature (const GstCaps * caps, const gchar * feature)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 0; i < gst_caps_get_size (caps); i++) {
|
|
GstCapsFeatures *const features = gst_caps_get_features (caps, i);
|
|
/* Skip ANY features, we need an exact match for correct evaluation */
|
|
if (gst_caps_features_is_any (features))
|
|
continue;
|
|
if (gst_caps_features_contains (features, feature))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
_list_append_string (GValue * list, const gchar * str)
|
|
{
|
|
GValue gval = G_VALUE_INIT;
|
|
|
|
g_return_if_fail (list != NULL);
|
|
g_return_if_fail (str != NULL);
|
|
|
|
g_value_init (&gval, G_TYPE_STRING);
|
|
g_value_set_string (&gval, str);
|
|
|
|
gst_value_list_append_value (list, &gval);
|
|
g_value_unset (&gval);
|
|
}
|
|
|
|
static gboolean
|
|
_strings_to_list (const gchar * strings, GValue * list)
|
|
{
|
|
gchar **strs = NULL;
|
|
|
|
if (!strings || !list)
|
|
return FALSE;
|
|
|
|
strs = g_strsplit (strings, DEFAULT_DELIMITER, 0);
|
|
for (guint i = 0; strs[i]; i++)
|
|
_list_append_string (list, strs[i]);
|
|
g_strfreev (strs);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static const gchar *
|
|
_get_media_type (guint codec)
|
|
{
|
|
for (int c = 0; c < G_N_ELEMENTS (codec_profs); c++) {
|
|
if (codec_profs[c].codec == codec)
|
|
return codec_profs[c].media_type;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
/* For RGB cases, the byte orders are different between vaImageFormat (LSB) and
|
|
* GStreamer video format (MSB). This function corrects the mapping between
|
|
* different order manners.
|
|
*/
|
|
static void
|
|
_fix_map (GstMsdkContext * context)
|
|
{
|
|
GstVaDisplay *display = NULL;
|
|
VAImageFormat *va_formats;
|
|
VADisplay dpy;
|
|
VAStatus status;
|
|
int max, num = 0;
|
|
|
|
display = (GstVaDisplay *) gst_msdk_context_get_va_display (context);
|
|
dpy = gst_va_display_get_va_dpy (display);
|
|
gst_object_unref (display);
|
|
|
|
max = vaMaxNumImageFormats (dpy);
|
|
if (max == 0)
|
|
return;
|
|
|
|
va_formats = g_new (VAImageFormat, max);
|
|
status = vaQueryImageFormats (dpy, va_formats, &num);
|
|
gst_va_video_format_fix_map (va_formats, num);
|
|
|
|
if (status != VA_STATUS_SUCCESS)
|
|
GST_WARNING ("vaQueryImageFormats: %s", vaErrorStr (status));
|
|
|
|
g_free (va_formats);
|
|
return;
|
|
}
|
|
|
|
static gboolean
|
|
_dma_fmt_to_dma_drm_fmts (GstMsdkContext * context,
|
|
GstMsdkContextJobType job_type,
|
|
const GValue * dma_fmts, GValue * dma_drm_fmts)
|
|
{
|
|
const gchar *fmt_str;
|
|
gchar *drm_fmt_str;
|
|
guint32 drm_fourcc;
|
|
guint64 modifier;
|
|
GstVideoFormat fmt;
|
|
GValue gval = G_VALUE_INIT;
|
|
GValue mods = G_VALUE_INIT;
|
|
|
|
g_return_val_if_fail (dma_fmts != NULL, FALSE);
|
|
g_return_val_if_fail (context != NULL, FALSE);
|
|
|
|
fmt_str = g_value_get_string (dma_fmts);
|
|
fmt = gst_video_format_from_string (fmt_str);
|
|
|
|
g_return_val_if_fail (fmt != GST_VIDEO_FORMAT_UNKNOWN, FALSE);
|
|
|
|
_fix_map (context);
|
|
|
|
drm_fourcc = gst_va_drm_fourcc_from_video_format (fmt);
|
|
if (drm_fourcc == DRM_FORMAT_INVALID)
|
|
return FALSE;
|
|
|
|
g_value_init (&mods, GST_TYPE_LIST);
|
|
g_value_init (&gval, G_TYPE_STRING);
|
|
|
|
gst_msdk_get_supported_modifiers (context, job_type, fmt, &mods);
|
|
|
|
for (gint m = 0; m < gst_value_list_get_size (&mods); m++) {
|
|
const GValue *gmod = gst_value_list_get_value (&mods, m);
|
|
modifier = g_value_get_uint64 (gmod);
|
|
|
|
drm_fmt_str = gst_video_dma_drm_fourcc_to_string (drm_fourcc, modifier);
|
|
if (!drm_fmt_str)
|
|
continue;
|
|
|
|
g_value_set_string (&gval, drm_fmt_str);
|
|
gst_value_list_append_value (dma_drm_fmts, &gval);
|
|
|
|
GST_DEBUG ("Got modifier: %s", drm_fmt_str);
|
|
g_free (drm_fmt_str);
|
|
}
|
|
g_value_unset (&mods);
|
|
g_value_unset (&gval);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_dma_fmts_to_dma_drm_fmts (GstMsdkContext * context,
|
|
GstMsdkContextJobType job_type,
|
|
const GValue * dma_fmts, GValue * dma_drm_fmts)
|
|
{
|
|
gint size = gst_value_list_get_size (dma_fmts);
|
|
|
|
for (gint f = 0; f < size; f++) {
|
|
const GValue *dma_fmt = gst_value_list_get_value (dma_fmts, f);
|
|
if (!dma_fmt)
|
|
continue;
|
|
|
|
_dma_fmt_to_dma_drm_fmts (context, job_type, dma_fmt, dma_drm_fmts);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstCaps *
|
|
_create_dma_drm_caps (GstMsdkContext * context,
|
|
GstMsdkContextJobType job_type, const GValue * dma_formats)
|
|
{
|
|
GstCaps *dma_drm_caps = NULL;
|
|
GValue dma_drm_fmts = G_VALUE_INIT;
|
|
|
|
g_return_val_if_fail (context != NULL, FALSE);
|
|
g_return_val_if_fail (dma_formats != NULL, FALSE);
|
|
|
|
g_value_init (&dma_drm_fmts, GST_TYPE_LIST);
|
|
|
|
if (GST_VALUE_HOLDS_LIST (dma_formats))
|
|
_dma_fmts_to_dma_drm_fmts (context, job_type, dma_formats, &dma_drm_fmts);
|
|
else if (G_VALUE_HOLDS_STRING (dma_formats))
|
|
_dma_fmt_to_dma_drm_fmts (context, job_type, dma_formats, &dma_drm_fmts);
|
|
|
|
if (gst_value_list_get_size (&dma_drm_fmts) > 0) {
|
|
dma_drm_caps = gst_caps_from_string ("video/x-raw(memory:DMABuf)");
|
|
gst_caps_set_simple (dma_drm_caps, "format", G_TYPE_STRING, "DMA_DRM",
|
|
NULL);
|
|
gst_caps_set_value (dma_drm_caps, "drm-format", &dma_drm_fmts);
|
|
}
|
|
|
|
g_value_unset (&dma_drm_fmts);
|
|
|
|
return dma_drm_caps;
|
|
}
|
|
#endif
|
|
|
|
#if (MFX_VERSION >= 2000)
|
|
|
|
static gboolean
|
|
_fill_mfxframeinfo (GstVideoFormat format, mfxFrameInfo * frameinfo)
|
|
{
|
|
if (format == GST_VIDEO_FORMAT_UNKNOWN)
|
|
return FALSE;
|
|
|
|
frameinfo->ChromaFormat = gst_msdk_get_mfx_chroma_from_format (format);
|
|
frameinfo->FourCC = gst_msdk_get_mfx_fourcc_from_format (format);
|
|
|
|
switch (format) {
|
|
case GST_VIDEO_FORMAT_NV12:
|
|
case GST_VIDEO_FORMAT_YV12:
|
|
case GST_VIDEO_FORMAT_I420:
|
|
case GST_VIDEO_FORMAT_YUY2:
|
|
case GST_VIDEO_FORMAT_UYVY:
|
|
case GST_VIDEO_FORMAT_BGRA:
|
|
case GST_VIDEO_FORMAT_ABGR:
|
|
case GST_VIDEO_FORMAT_BGRx:
|
|
case GST_VIDEO_FORMAT_VUYA:
|
|
frameinfo->BitDepthLuma = 8;
|
|
frameinfo->BitDepthChroma = 8;
|
|
frameinfo->Shift = 0;
|
|
break;
|
|
case GST_VIDEO_FORMAT_BGR10A2_LE:
|
|
frameinfo->BitDepthLuma = 10;
|
|
frameinfo->BitDepthChroma = 10;
|
|
break;
|
|
case GST_VIDEO_FORMAT_P010_10LE:
|
|
frameinfo->BitDepthLuma = 10;
|
|
frameinfo->BitDepthChroma = 10;
|
|
frameinfo->Shift = 1;
|
|
break;
|
|
#if (MFX_VERSION >= 1027)
|
|
case GST_VIDEO_FORMAT_Y210:
|
|
frameinfo->BitDepthLuma = 10;
|
|
frameinfo->BitDepthChroma = 10;
|
|
frameinfo->Shift = 1;
|
|
break;
|
|
case GST_VIDEO_FORMAT_Y410:
|
|
frameinfo->BitDepthLuma = 10;
|
|
frameinfo->BitDepthChroma = 10;
|
|
frameinfo->Shift = 0;
|
|
break;
|
|
#endif
|
|
#if (MFX_VERSION >= 1031)
|
|
case GST_VIDEO_FORMAT_P012_LE:
|
|
case GST_VIDEO_FORMAT_Y212_LE:
|
|
case GST_VIDEO_FORMAT_Y412_LE:
|
|
frameinfo->BitDepthLuma = 12;
|
|
frameinfo->BitDepthChroma = 12;
|
|
frameinfo->Shift = 1;
|
|
break;
|
|
#endif
|
|
#if (MFX_VERSION >=2004)
|
|
case GST_VIDEO_FORMAT_RGBP:
|
|
case GST_VIDEO_FORMAT_BGRP:
|
|
frameinfo->BitDepthLuma = 8;
|
|
frameinfo->BitDepthChroma = 8;
|
|
frameinfo->Shift = 0;
|
|
break;
|
|
#endif
|
|
case GST_VIDEO_FORMAT_RGB16:
|
|
/* Ignore parameters settings, do nothing */
|
|
break;
|
|
default:
|
|
GST_WARNING ("Unsupported format %s",
|
|
gst_video_format_to_string (format));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static const gchar *
|
|
_profile_to_string (guint codec, mfxU32 profile)
|
|
{
|
|
if (profile == MFX_PROFILE_UNKNOWN)
|
|
return NULL;
|
|
|
|
for (int c = 0; c < G_N_ELEMENTS (codec_profs); c++) {
|
|
if (codec_profs[c].codec == codec) {
|
|
const Profile *p = codec_profs[c].profiles;
|
|
for (; p->id != MFX_PROFILE_UNKNOWN; p++) {
|
|
if (p->id == profile)
|
|
return p->names;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static mfxU16
|
|
_get_main_codec_profile (guint codec)
|
|
{
|
|
for (int c = 0; c < G_N_ELEMENTS (codec_profs); c++) {
|
|
if (codec_profs[c].codec == codec) {
|
|
return codec_profs[c].profiles->id;
|
|
}
|
|
}
|
|
|
|
return MFX_PROFILE_UNKNOWN;
|
|
}
|
|
|
|
static void
|
|
_codec_init_param (mfxVideoParam * param,
|
|
guint codec_id, mfxU16 pattern, GstVideoFormat format)
|
|
{
|
|
g_return_if_fail (param != NULL);
|
|
|
|
memset (param, 0, sizeof (mfxVideoParam));
|
|
param->IOPattern = pattern;
|
|
param->mfx.CodecId = codec_id;
|
|
param->mfx.CodecProfile = _get_main_codec_profile (codec_id);
|
|
param->mfx.FrameInfo.Width = default_width;
|
|
param->mfx.FrameInfo.Height = default_height;
|
|
param->mfx.FrameInfo.CropW = param->mfx.FrameInfo.Width;
|
|
param->mfx.FrameInfo.CropH = param->mfx.FrameInfo.Height;
|
|
param->mfx.FrameInfo.PicStruct = MFX_PICSTRUCT_PROGRESSIVE;
|
|
param->mfx.FrameInfo.FrameRateExtN = 30;
|
|
param->mfx.FrameInfo.FrameRateExtD = 1;
|
|
param->mfx.FrameInfo.AspectRatioW = 1;
|
|
param->mfx.FrameInfo.AspectRatioH = 1;
|
|
|
|
_fill_mfxframeinfo (format, ¶m->mfx.FrameInfo);
|
|
}
|
|
|
|
static gboolean
|
|
_get_min_width (mfxSession * session, mfxVideoParam * in,
|
|
mfxVideoParam * out, IsParamSupportedFunc func,
|
|
const mfxRange32U * width, guint * min_w)
|
|
{
|
|
g_return_val_if_fail (func != NULL, FALSE);
|
|
|
|
in->mfx.FrameInfo.Height = default_height;
|
|
in->mfx.FrameInfo.CropH = default_height;
|
|
out->mfx.FrameInfo.Height = in->mfx.FrameInfo.Height;
|
|
out->mfx.FrameInfo.CropH = in->mfx.FrameInfo.CropH;
|
|
|
|
for (guint w = width->Min; w < *min_w;) {
|
|
in->mfx.FrameInfo.Width = w;
|
|
in->mfx.FrameInfo.CropW = w;
|
|
out->mfx.FrameInfo.Width = in->mfx.FrameInfo.Width;
|
|
out->mfx.FrameInfo.CropW = in->mfx.FrameInfo.CropW;
|
|
|
|
if (func (session, in, out)) {
|
|
*min_w = in->mfx.FrameInfo.Width;
|
|
return TRUE;
|
|
} else {
|
|
if (out->mfx.FrameInfo.Width != 0) {
|
|
if (func (session, out, out)) {
|
|
*min_w = out->mfx.FrameInfo.Width;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
w += width->Step;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_get_min_height (mfxSession * session, mfxVideoParam * in,
|
|
mfxVideoParam * out, IsParamSupportedFunc func,
|
|
const mfxRange32U * height, guint * min_h)
|
|
{
|
|
g_return_val_if_fail (func != NULL, FALSE);
|
|
|
|
in->mfx.FrameInfo.Width = default_width;
|
|
in->mfx.FrameInfo.CropW = default_width;
|
|
out->mfx.FrameInfo.Width = in->mfx.FrameInfo.Width;
|
|
out->mfx.FrameInfo.CropW = in->mfx.FrameInfo.CropW;
|
|
|
|
for (guint h = height->Min; h < *min_h;) {
|
|
in->mfx.FrameInfo.Height = h;
|
|
in->mfx.FrameInfo.CropH = h;
|
|
out->mfx.FrameInfo.Height = h;
|
|
out->mfx.FrameInfo.CropH = h;
|
|
|
|
if (func (session, in, out)) {
|
|
*min_h = in->mfx.FrameInfo.Height;
|
|
return TRUE;
|
|
} else {
|
|
if (out->mfx.FrameInfo.Height != 0) {
|
|
if (func (session, out, out)) {
|
|
*min_h = out->mfx.FrameInfo.Height;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
h += height->Step;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static guint
|
|
_get_smaller_res_width (guint cur_res_width)
|
|
{
|
|
guint i = G_N_ELEMENTS (max_res_widths) - 1;
|
|
|
|
for (; i >= 0; i--) {
|
|
if (max_res_widths[i] < cur_res_width)
|
|
return max_res_widths[i];
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gboolean
|
|
_get_max_width (mfxSession * session, mfxVideoParam * in,
|
|
mfxVideoParam * out, IsParamSupportedFunc func,
|
|
const mfxRange32U * width, guint * max_w)
|
|
{
|
|
guint w;
|
|
|
|
g_return_val_if_fail (func != NULL, FALSE);
|
|
|
|
in->mfx.FrameInfo.Height = default_height;
|
|
in->mfx.FrameInfo.CropH = default_height;
|
|
out->mfx.FrameInfo.Height = in->mfx.FrameInfo.Height;
|
|
out->mfx.FrameInfo.CropH = in->mfx.FrameInfo.CropH;
|
|
|
|
w = width->Max;
|
|
do {
|
|
in->mfx.FrameInfo.Width = w;
|
|
in->mfx.FrameInfo.CropW = w;
|
|
out->mfx.FrameInfo.Width = in->mfx.FrameInfo.Width;
|
|
out->mfx.FrameInfo.CropW = in->mfx.FrameInfo.CropW;
|
|
|
|
if (func (session, in, out)) {
|
|
*max_w = in->mfx.FrameInfo.Width;
|
|
return TRUE;
|
|
} else {
|
|
if (out->mfx.FrameInfo.Width != 0) {
|
|
if (func (session, out, out)) {
|
|
*max_w = out->mfx.FrameInfo.Width;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
w = _get_smaller_res_width (w);
|
|
} while (w);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static guint
|
|
_get_smaller_res_height (guint cur_res_height)
|
|
{
|
|
guint i = G_N_ELEMENTS (max_res_heights) - 1;
|
|
|
|
for (; i >= 0; i--) {
|
|
if (max_res_heights[i] < cur_res_height)
|
|
return max_res_heights[i];
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gboolean
|
|
_get_max_height (mfxSession * session, mfxVideoParam * in,
|
|
mfxVideoParam * out, IsParamSupportedFunc func,
|
|
const mfxRange32U * height, guint * max_h)
|
|
{
|
|
guint h;
|
|
|
|
g_return_val_if_fail (func != NULL, FALSE);
|
|
|
|
in->mfx.FrameInfo.Width = default_width;
|
|
in->mfx.FrameInfo.CropW = default_width;
|
|
out->mfx.FrameInfo.Width = in->mfx.FrameInfo.Width;
|
|
out->mfx.FrameInfo.CropW = in->mfx.FrameInfo.CropW;
|
|
|
|
h = height->Max;
|
|
do {
|
|
in->mfx.FrameInfo.Height = h;
|
|
in->mfx.FrameInfo.CropH = h;
|
|
out->mfx.FrameInfo.Height = in->mfx.FrameInfo.Height;
|
|
out->mfx.FrameInfo.CropH = in->mfx.FrameInfo.CropH;
|
|
|
|
if (func (session, in, out)) {
|
|
*max_h = in->mfx.FrameInfo.Height;
|
|
return TRUE;
|
|
} else {
|
|
if (out->mfx.FrameInfo.Height != 0) {
|
|
if (func (session, out, out)) {
|
|
*max_h = out->mfx.FrameInfo.Height;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
h = _get_smaller_res_height (h);
|
|
} while (h);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_format_in_list (GstVideoFormat format, GValue * list)
|
|
{
|
|
const GValue *gfmt;
|
|
const gchar *fmt_str = NULL;
|
|
guint size = gst_value_list_get_size (list);
|
|
|
|
if (format == GST_VIDEO_FORMAT_UNKNOWN)
|
|
return FALSE;
|
|
|
|
for (guint i = 0; i < size; i++) {
|
|
gfmt = gst_value_list_get_value (list, i);
|
|
fmt_str = g_value_get_string (gfmt);
|
|
|
|
if (format == gst_video_format_from_string (fmt_str))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_fourcc_in_array (mfxU32 fourcc, mfxU32 * array, mfxU16 num)
|
|
{
|
|
for (guint i = 0; i < num; i++) {
|
|
if (array[i] == fourcc)
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gboolean
|
|
_enc_is_param_supported (mfxSession * session,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
mfxStatus status = MFXVideoENCODE_Query (*session, in, out);
|
|
if (status == MFX_ERR_NONE)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gboolean
|
|
_enc_ensure_codec (mfxEncoderDescription * enc_desc, guint c)
|
|
{
|
|
for (guint p = 0; p < enc_desc->Codecs[c].NumProfiles; p++) {
|
|
if (enc_desc->Codecs[c].Profiles[p].MemDesc->NumColorFormats)
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gint
|
|
_enc_get_codec_index (mfxEncoderDescription * enc_desc, guint codec_id)
|
|
{
|
|
gint c;
|
|
|
|
for (c = 0; c < enc_desc->NumCodecs; c++) {
|
|
if (enc_desc->Codecs[c].CodecID == codec_id) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (c >= enc_desc->NumCodecs)
|
|
goto failed;
|
|
|
|
if (!_enc_ensure_codec (enc_desc, c))
|
|
goto failed;
|
|
|
|
return c;
|
|
|
|
failed:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
return -1;
|
|
}
|
|
|
|
static gboolean
|
|
_enc_get_resolution_range (mfxSession * session,
|
|
mfxEncoderDescription * enc_desc, guint codec_id,
|
|
ResolutionRange * res_range)
|
|
{
|
|
guint c;
|
|
mfxVideoParam in, out;
|
|
mfxRange32U *width, *height;
|
|
ResolutionRange res = { default_width, default_width,
|
|
default_height, default_height
|
|
};
|
|
|
|
g_return_val_if_fail (res_range != NULL, FALSE);
|
|
|
|
c = _enc_get_codec_index (enc_desc, codec_id);
|
|
width = &enc_desc->Codecs[c].Profiles->MemDesc->Width;
|
|
height = &enc_desc->Codecs[c].Profiles->MemDesc->Height;
|
|
|
|
_codec_init_param (&in, codec_id, ENC_IOPATTERN, DEFAULT_VIDEO_FORMAT);
|
|
if (codec_id == MFX_CODEC_AV1)
|
|
in.mfx.CodecLevel = MFX_LEVEL_AV1_41;
|
|
out = in;
|
|
|
|
IsParamSupportedFunc func = _enc_is_param_supported;
|
|
if (!_get_min_width (session, &in, &out, func, width, &res.min_width) ||
|
|
!_get_max_width (session, &in, &out, func, width, &res.max_width) ||
|
|
!_get_min_height (session, &in, &out, func, height, &res.min_height) ||
|
|
!_get_max_height (session, &in, &out, func, height, &res.max_height))
|
|
return FALSE;
|
|
|
|
GST_DEBUG ("Got %" GST_FOURCC_FORMAT
|
|
" ENC supported resolution range width: [%d, %d], height: [%d, %d]",
|
|
GST_FOURCC_ARGS (enc_desc->Codecs[c].CodecID),
|
|
res.min_width, res.max_width, res.min_height, res.max_height);
|
|
|
|
res_range->min_width = res.min_width;
|
|
res_range->max_width = res.max_width;
|
|
res_range->min_height = res.min_height;
|
|
res_range->max_height = res.max_height;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_enc_is_format_supported (mfxSession * session,
|
|
guint codec_id, GstVideoFormat format,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
if (!_fill_mfxframeinfo (format, &in->mfx.FrameInfo))
|
|
return FALSE;
|
|
|
|
in->mfx.LowPower = MFX_CODINGOPTION_UNKNOWN;
|
|
if (!_enc_is_param_supported (session, in, out)) {
|
|
in->mfx.LowPower = (out->mfx.LowPower == MFX_CODINGOPTION_ON) ?
|
|
MFX_CODINGOPTION_OFF : MFX_CODINGOPTION_ON;
|
|
|
|
if (!_enc_is_param_supported (session, in, out))
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_enc_get_supported_formats_and_profiles (mfxSession * session,
|
|
mfxEncoderDescription * enc_desc, guint codec_id,
|
|
GValue * supported_fmts, GValue * supported_profs)
|
|
{
|
|
guint size;
|
|
const GValue *gfmt;
|
|
GValue fmts = G_VALUE_INIT;
|
|
GstVideoFormat fmt = DEFAULT_VIDEO_FORMAT;
|
|
mfxVideoParam in, out;
|
|
gboolean prof_supported;
|
|
const gchar *prof_str = NULL;
|
|
gchar **profs = NULL;
|
|
gint c = _enc_get_codec_index (enc_desc, codec_id);
|
|
|
|
_codec_init_param (&in, codec_id, ENC_IOPATTERN, DEFAULT_VIDEO_FORMAT);
|
|
if (codec_id == MFX_CODEC_AV1)
|
|
in.mfx.CodecLevel = MFX_LEVEL_AV1_41;
|
|
out = in;
|
|
|
|
g_value_init (&fmts, GST_TYPE_LIST);
|
|
gst_msdk_get_video_format_list (&fmts);
|
|
size = gst_value_list_get_size (&fmts);
|
|
|
|
for (guint p = 0; p < enc_desc->Codecs[c].NumProfiles; p++) {
|
|
in.mfx.CodecProfile = enc_desc->Codecs[c].Profiles[p].Profile;
|
|
prof_supported = FALSE;
|
|
|
|
for (guint i = 0; i < size; i++) {
|
|
gfmt = gst_value_list_get_value (&fmts, i);
|
|
fmt = g_value_get_uint (gfmt);
|
|
|
|
if (_format_in_list (fmt, supported_fmts))
|
|
continue;
|
|
|
|
if (!_fourcc_in_array (gst_msdk_get_mfx_fourcc_from_format (fmt),
|
|
enc_desc->Codecs[c].Profiles[p].MemDesc->ColorFormats,
|
|
enc_desc->Codecs[c].Profiles[p].MemDesc->NumColorFormats))
|
|
continue;
|
|
|
|
if (!_enc_is_format_supported (session, codec_id, fmt, &in, &out))
|
|
continue;
|
|
|
|
_list_append_string (supported_fmts, gst_video_format_to_string (fmt));
|
|
prof_supported = TRUE;
|
|
}
|
|
|
|
if (!prof_supported) {
|
|
for (guint f = 0;
|
|
f < enc_desc->Codecs[c].Profiles[p].MemDesc->NumColorFormats; f++) {
|
|
fmt =
|
|
gst_msdk_get_video_format_from_mfx_fourcc (enc_desc->
|
|
Codecs[c].Profiles[p].MemDesc->ColorFormats[f]);
|
|
if (_enc_is_format_supported (session, codec_id, fmt, &in, &out)) {
|
|
prof_supported = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!prof_supported)
|
|
continue;
|
|
|
|
prof_str = _profile_to_string (codec_id, in.mfx.CodecProfile);
|
|
if (!prof_str)
|
|
continue;
|
|
|
|
profs = g_strsplit (prof_str, PROFILE_DELIMITER, 0);
|
|
for (guint i = 0; profs[i]; i++)
|
|
_list_append_string (supported_profs, profs[i]);
|
|
g_strfreev (profs);
|
|
}
|
|
|
|
g_value_unset (&fmts);
|
|
|
|
if (gst_value_list_get_size (supported_fmts) == 0 ||
|
|
gst_value_list_get_size (supported_profs) == 0)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstCaps *
|
|
_enc_create_sink_caps (GstMsdkContext * context, guint codec_id,
|
|
const ResolutionRange * res, GValue * supported_formats)
|
|
{
|
|
GstCaps *caps, *dma_caps, *raw_caps;
|
|
|
|
#ifndef _WIN32
|
|
caps = gst_caps_from_string
|
|
("video/x-raw(memory:VAMemory), format=(string){ NV12 }");
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_ENCODER,
|
|
supported_formats);
|
|
gst_caps_append (caps, dma_caps);
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
caps = gst_caps_from_string
|
|
("video/x-raw(memory:D3D11Memory), format=(string){ NV12 }");
|
|
#endif
|
|
|
|
raw_caps = gst_caps_from_string ("video/x-raw");
|
|
gst_caps_set_value (raw_caps, "format", supported_formats);
|
|
gst_caps_append (caps, raw_caps);
|
|
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, res->min_width, res->max_width,
|
|
"height", GST_TYPE_INT_RANGE, res->min_height, res->max_height,
|
|
"interlace-mode", G_TYPE_STRING, "progressive", NULL);
|
|
|
|
GST_DEBUG ("Create %" GST_FOURCC_FORMAT " ENC sink_caps %" GST_PTR_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id), caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
static GstCaps *
|
|
_enc_create_src_caps (guint codec_id,
|
|
const ResolutionRange * res, GValue * supported_profiles)
|
|
{
|
|
GstCaps *caps;
|
|
const gchar *media_type = NULL;
|
|
|
|
media_type = _get_media_type (codec_id);
|
|
if (!media_type)
|
|
return NULL;
|
|
|
|
caps = gst_caps_new_empty_simple (media_type);
|
|
gst_caps_set_value (caps, "profile", supported_profiles);
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, res->min_width, res->max_width,
|
|
"height", GST_TYPE_INT_RANGE, res->min_height, res->max_height, NULL);
|
|
|
|
GST_DEBUG ("Create %" GST_FOURCC_FORMAT " ENC src_caps %" GST_PTR_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id), caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_enc_create_caps (GstMsdkContext * context,
|
|
gpointer enc_description, guint codec_id,
|
|
GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
mfxEncoderDescription *enc_desc;
|
|
GstCaps *in_caps = NULL, *out_caps = NULL;
|
|
ResolutionRange res_range = { 1, G_MAXUINT16, 1, G_MAXUINT16 };
|
|
GValue supported_fmts = G_VALUE_INIT;
|
|
GValue supported_profs = G_VALUE_INIT;
|
|
mfxSession session;
|
|
|
|
g_return_val_if_fail (context, FALSE);
|
|
g_return_val_if_fail (enc_description, FALSE);
|
|
|
|
session = gst_msdk_context_get_session (context);
|
|
enc_desc = (mfxEncoderDescription *) enc_description;
|
|
|
|
if (_enc_get_codec_index (enc_desc, codec_id) < 0)
|
|
goto failed;
|
|
|
|
g_value_init (&supported_fmts, GST_TYPE_LIST);
|
|
g_value_init (&supported_profs, GST_TYPE_LIST);
|
|
if (!_enc_get_supported_formats_and_profiles (&session,
|
|
enc_desc, codec_id, &supported_fmts, &supported_profs))
|
|
goto failed;
|
|
|
|
if (!_enc_get_resolution_range (&session, enc_desc, codec_id, &res_range))
|
|
goto failed;
|
|
|
|
in_caps = _enc_create_sink_caps (context,
|
|
codec_id, &res_range, &supported_fmts);
|
|
g_value_unset (&supported_fmts);
|
|
if (!in_caps)
|
|
goto failed;
|
|
|
|
out_caps = _enc_create_src_caps (codec_id, &res_range, &supported_profs);
|
|
g_value_unset (&supported_profs);
|
|
if (!out_caps)
|
|
goto failed;
|
|
|
|
*sink_caps = in_caps;
|
|
*src_caps = out_caps;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_WARNING ("Failed to create caps for %" GST_FOURCC_FORMAT " ENC",
|
|
GST_FOURCC_ARGS (codec_id));
|
|
|
|
g_value_unset (&supported_fmts);
|
|
g_value_unset (&supported_profs);
|
|
if (in_caps)
|
|
gst_caps_unref (in_caps);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gboolean
|
|
_dec_is_param_supported (mfxSession * session,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
mfxStatus status = MFXVideoDECODE_Query (*session, in, out);
|
|
if (status == MFX_ERR_NONE)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gboolean
|
|
_dec_ensure_codec (mfxDecoderDescription * dec_desc, guint c)
|
|
{
|
|
for (guint p = 0; p < dec_desc->Codecs[c].NumProfiles; p++) {
|
|
if (dec_desc->Codecs[c].Profiles[p].MemDesc->NumColorFormats)
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gint
|
|
_dec_get_codec_index (mfxDecoderDescription * dec_desc, guint codec_id)
|
|
{
|
|
gint c;
|
|
|
|
for (c = 0; c < dec_desc->NumCodecs; c++) {
|
|
if (dec_desc->Codecs[c].CodecID == codec_id) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (c >= dec_desc->NumCodecs)
|
|
goto failed;
|
|
|
|
if (!_dec_ensure_codec (dec_desc, c))
|
|
goto failed;
|
|
|
|
return c;
|
|
|
|
failed:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
return -1;
|
|
}
|
|
|
|
static void
|
|
_jpegdec_set_color_format (mfxVideoParam * param, GstVideoFormat format)
|
|
{
|
|
param->mfx.JPEGChromaFormat = param->mfx.FrameInfo.ChromaFormat;
|
|
|
|
switch (format) {
|
|
case GST_VIDEO_FORMAT_NV12:
|
|
case GST_VIDEO_FORMAT_YUY2:
|
|
param->mfx.JPEGColorFormat = MFX_JPEG_COLORFORMAT_YCbCr;
|
|
break;
|
|
case GST_VIDEO_FORMAT_BGRA:
|
|
case GST_VIDEO_FORMAT_BGRx:
|
|
param->mfx.JPEGColorFormat = MFX_JPEG_COLORFORMAT_RGB;
|
|
break;
|
|
default:
|
|
GST_WARNING ("Jpegdec unsupported format %s",
|
|
gst_video_format_to_string (format));
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
_dec_get_resolution_range (mfxSession * session,
|
|
mfxDecoderDescription * dec_desc, guint codec_id,
|
|
ResolutionRange * res_range)
|
|
{
|
|
guint c;
|
|
mfxVideoParam in, out;
|
|
mfxRange32U *width, *height;
|
|
ResolutionRange res = { default_width, default_width,
|
|
default_height, default_height
|
|
};
|
|
|
|
g_return_val_if_fail (res_range != NULL, FALSE);
|
|
|
|
c = _dec_get_codec_index (dec_desc, codec_id);
|
|
width = &dec_desc->Codecs[c].Profiles->MemDesc->Width;
|
|
height = &dec_desc->Codecs[c].Profiles->MemDesc->Height;
|
|
|
|
_codec_init_param (&in, codec_id, DEC_IOPATTERN, DEFAULT_VIDEO_FORMAT);
|
|
if (codec_id == MFX_CODEC_AV1)
|
|
in.mfx.CodecLevel = MFX_LEVEL_AV1_41;
|
|
if (codec_id == MFX_CODEC_JPEG)
|
|
_jpegdec_set_color_format (&in, GST_VIDEO_FORMAT_NV12);
|
|
out = in;
|
|
|
|
IsParamSupportedFunc func = _dec_is_param_supported;
|
|
if (!_get_min_width (session, &in, &out, func, width, &res.min_width) ||
|
|
!_get_max_width (session, &in, &out, func, width, &res.max_width) ||
|
|
!_get_min_height (session, &in, &out, func, height, &res.min_height) ||
|
|
!_get_max_height (session, &in, &out, func, height, &res.max_height))
|
|
return FALSE;
|
|
|
|
GST_DEBUG ("Got %" GST_FOURCC_FORMAT
|
|
" DEC supported resolution range width: [%d, %d], height: [%d, %d]",
|
|
GST_FOURCC_ARGS (dec_desc->Codecs[c].CodecID),
|
|
res.min_width, res.max_width, res.min_height, res.max_height);
|
|
|
|
res_range->min_width = res.min_width;
|
|
res_range->max_width = res.max_width;
|
|
res_range->min_height = res.min_height;
|
|
res_range->max_height = res.max_height;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_dec_is_format_supported (mfxSession * session,
|
|
guint codec_id, GstVideoFormat format,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
if (!_fill_mfxframeinfo (format, &in->mfx.FrameInfo))
|
|
return FALSE;
|
|
|
|
if (codec_id == MFX_CODEC_JPEG)
|
|
_jpegdec_set_color_format (in, format);
|
|
|
|
in->mfx.LowPower = MFX_CODINGOPTION_UNKNOWN;
|
|
if (!_dec_is_param_supported (session, in, out)) {
|
|
in->mfx.LowPower = (out->mfx.LowPower == MFX_CODINGOPTION_ON) ?
|
|
MFX_CODINGOPTION_OFF : MFX_CODINGOPTION_ON;
|
|
|
|
if (!_dec_is_param_supported (session, in, out))
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_dec_get_supported_formats (mfxSession * session,
|
|
mfxDecoderDescription * dec_desc, guint codec_id, GValue * supported_fmts)
|
|
{
|
|
guint size;
|
|
const GValue *gfmt;
|
|
GValue fmts = G_VALUE_INIT;
|
|
GstVideoFormat fmt = DEFAULT_VIDEO_FORMAT;
|
|
mfxVideoParam in, out;
|
|
gint c;
|
|
|
|
_codec_init_param (&in, codec_id, DEC_IOPATTERN, DEFAULT_VIDEO_FORMAT);
|
|
if (codec_id == MFX_CODEC_AV1)
|
|
in.mfx.CodecLevel = MFX_LEVEL_AV1_41;
|
|
out = in;
|
|
|
|
g_value_init (&fmts, GST_TYPE_LIST);
|
|
gst_msdk_get_video_format_list (&fmts);
|
|
size = gst_value_list_get_size (&fmts);
|
|
|
|
c = _dec_get_codec_index (dec_desc, codec_id);
|
|
|
|
for (guint p = 0; p < dec_desc->Codecs[c].NumProfiles; p++) {
|
|
in.mfx.CodecProfile = dec_desc->Codecs[c].Profiles[p].Profile;
|
|
|
|
for (guint i = 0; i < size; i++) {
|
|
gfmt = gst_value_list_get_value (&fmts, i);
|
|
fmt = g_value_get_uint (gfmt);
|
|
|
|
if (_format_in_list (fmt, supported_fmts))
|
|
continue;
|
|
|
|
if (!_fourcc_in_array (gst_msdk_get_mfx_fourcc_from_format (fmt),
|
|
dec_desc->Codecs[c].Profiles[p].MemDesc->ColorFormats,
|
|
dec_desc->Codecs[c].Profiles[p].MemDesc->NumColorFormats))
|
|
continue;
|
|
|
|
if (!_dec_is_format_supported (session, codec_id, fmt, &in, &out))
|
|
continue;
|
|
|
|
_list_append_string (supported_fmts, gst_video_format_to_string (fmt));
|
|
}
|
|
}
|
|
|
|
g_value_unset (&fmts);
|
|
|
|
return (gst_value_list_get_size (supported_fmts) == 0) ? FALSE : TRUE;
|
|
}
|
|
|
|
static GstCaps *
|
|
_dec_create_sink_caps (guint codec_id)
|
|
{
|
|
GstCaps *caps;
|
|
const gchar *media_type = NULL;
|
|
|
|
media_type = _get_media_type (codec_id);
|
|
if (!media_type)
|
|
return NULL;
|
|
|
|
caps = gst_caps_new_empty_simple (media_type);
|
|
|
|
GST_DEBUG ("Create %" GST_FOURCC_FORMAT " DEC sink_caps %" GST_PTR_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id), caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
static GstCaps *
|
|
_dec_create_src_caps (GstMsdkContext * context,
|
|
mfxSession * session, guint codec_id,
|
|
mfxDecoderDescription * dec_desc, GValue * supported_formats)
|
|
{
|
|
GstCaps *caps, *dma_caps, *raw_caps;
|
|
ResolutionRange res = { 1, G_MAXUINT16, 1, G_MAXUINT16 };
|
|
|
|
if (!_dec_get_resolution_range (session, dec_desc, codec_id, &res))
|
|
return NULL;
|
|
|
|
#ifndef _WIN32
|
|
caps = gst_caps_from_string
|
|
("video/x-raw(memory:VAMemory), format=(string){ NV12 }");
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_DECODER,
|
|
supported_formats);
|
|
gst_caps_append (caps, dma_caps);
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
caps = gst_caps_from_string
|
|
("video/x-raw(memory:D3D11Memory), format=(string){ NV12 }");
|
|
#endif
|
|
|
|
raw_caps = gst_caps_from_string ("video/x-raw");
|
|
gst_caps_set_value (raw_caps, "format", supported_formats);
|
|
gst_caps_append (caps, raw_caps);
|
|
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, res.min_width, res.max_width,
|
|
"height", GST_TYPE_INT_RANGE, res.min_height, res.max_height,
|
|
"interlace-mode", G_TYPE_STRING, "progressive", NULL);
|
|
|
|
GST_DEBUG ("Create %" GST_FOURCC_FORMAT " DEC src_caps %" GST_PTR_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id), caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_dec_create_caps (GstMsdkContext * context,
|
|
gpointer dec_description, guint codec_id,
|
|
GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
mfxDecoderDescription *dec_desc;
|
|
GstCaps *in_caps = NULL, *out_caps = NULL;
|
|
GValue supported_fmts = G_VALUE_INIT;
|
|
mfxSession session;
|
|
|
|
g_return_val_if_fail (context, FALSE);
|
|
g_return_val_if_fail (dec_description, FALSE);
|
|
|
|
session = gst_msdk_context_get_session (context);
|
|
dec_desc = (mfxDecoderDescription *) dec_description;
|
|
|
|
if (_dec_get_codec_index (dec_desc, codec_id) < 0)
|
|
goto failed;
|
|
|
|
g_value_init (&supported_fmts, GST_TYPE_LIST);
|
|
if (!_dec_get_supported_formats (&session,
|
|
dec_desc, codec_id, &supported_fmts))
|
|
goto failed;
|
|
|
|
in_caps = _dec_create_sink_caps (codec_id);
|
|
if (!in_caps)
|
|
goto failed;
|
|
|
|
out_caps = _dec_create_src_caps (context,
|
|
&session, codec_id, dec_desc, &supported_fmts);
|
|
g_value_unset (&supported_fmts);
|
|
if (!out_caps)
|
|
goto failed;
|
|
|
|
*sink_caps = in_caps;
|
|
*src_caps = out_caps;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_WARNING ("Failed to create caps for %" GST_FOURCC_FORMAT " DEC",
|
|
GST_FOURCC_ARGS (codec_id));
|
|
|
|
g_value_unset (&supported_fmts);
|
|
if (in_caps)
|
|
gst_caps_unref (in_caps);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
_vpp_init_param (mfxVideoParam * param,
|
|
GstVideoFormat infmt, GstVideoFormat outfmt)
|
|
{
|
|
g_return_if_fail (param != NULL);
|
|
|
|
memset (param, 0, sizeof (mfxVideoParam));
|
|
param->IOPattern = VPP_IOPATTERN;
|
|
param->vpp.In.Width = default_width;
|
|
param->vpp.In.Height = default_height;
|
|
param->vpp.In.CropW = param->mfx.FrameInfo.Width;
|
|
param->vpp.In.CropH = param->mfx.FrameInfo.Height;
|
|
param->vpp.In.PicStruct = MFX_PICSTRUCT_PROGRESSIVE;
|
|
param->vpp.In.FrameRateExtN = 30;
|
|
param->vpp.In.FrameRateExtD = 1;
|
|
param->vpp.In.AspectRatioW = 1;
|
|
param->vpp.In.AspectRatioH = 1;
|
|
|
|
param->vpp.Out = param->vpp.In;
|
|
|
|
_fill_mfxframeinfo (infmt, ¶m->vpp.In);
|
|
_fill_mfxframeinfo (outfmt, ¶m->vpp.Out);
|
|
}
|
|
|
|
static inline gboolean
|
|
_vpp_is_param_supported (mfxSession * session,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
mfxStatus status = MFXVideoVPP_Query (*session, in, out);
|
|
if (status == MFX_ERR_NONE)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_vpp_are_formats_supported (mfxSession * session,
|
|
GstVideoFormat infmt, GstVideoFormat outfmt,
|
|
mfxVideoParam * in, mfxVideoParam * out)
|
|
{
|
|
if (!_fill_mfxframeinfo (infmt, &in->vpp.In))
|
|
return FALSE;
|
|
|
|
if (!_fill_mfxframeinfo (outfmt, &in->vpp.Out))
|
|
return FALSE;
|
|
|
|
if (!_vpp_is_param_supported (session, in, out))
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_vpp_get_supported_formats (mfxSession * session,
|
|
GValue * supported_in_fmts, GValue * supported_out_fmts)
|
|
{
|
|
guint size;
|
|
const GValue *gfmt;
|
|
GValue fmts = G_VALUE_INIT;
|
|
GstVideoFormat infmt = DEFAULT_VIDEO_FORMAT;
|
|
GstVideoFormat outfmt = DEFAULT_VIDEO_FORMAT;
|
|
gboolean infmt_in_list, outfmt_in_list;
|
|
mfxVideoParam in, out;
|
|
|
|
_vpp_init_param (&in, infmt, outfmt);
|
|
out = in;
|
|
|
|
g_value_init (&fmts, GST_TYPE_LIST);
|
|
gst_msdk_get_video_format_list (&fmts);
|
|
|
|
size = gst_value_list_get_size (&fmts);
|
|
for (guint i = 0; i < size; i++) {
|
|
gfmt = gst_value_list_get_value (&fmts, i);
|
|
infmt = g_value_get_uint (gfmt);
|
|
|
|
for (guint j = 0; j < size; j++) {
|
|
gfmt = gst_value_list_get_value (&fmts, j);
|
|
outfmt = g_value_get_uint (gfmt);
|
|
|
|
infmt_in_list = _format_in_list (infmt, supported_in_fmts);
|
|
outfmt_in_list = _format_in_list (outfmt, supported_out_fmts);
|
|
if (infmt_in_list && outfmt_in_list)
|
|
continue;
|
|
|
|
if (!_vpp_are_formats_supported (session, infmt, outfmt, &in, &out))
|
|
continue;
|
|
|
|
if (!infmt_in_list)
|
|
_list_append_string (supported_in_fmts,
|
|
gst_video_format_to_string (infmt));
|
|
|
|
if (!outfmt_in_list)
|
|
_list_append_string (supported_out_fmts,
|
|
gst_video_format_to_string (outfmt));
|
|
}
|
|
}
|
|
|
|
g_value_unset (&fmts);
|
|
|
|
if (gst_value_list_get_size (supported_in_fmts) == 0 ||
|
|
gst_value_list_get_size (supported_out_fmts) == 0)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_vpp_get_desc_image_range (mfxVPPDescription * vpp_desc,
|
|
GstVideoFormat format, mfxRange32U * width, mfxRange32U * height)
|
|
{
|
|
mfxU32 infmt = gst_msdk_get_mfx_fourcc_from_format (format);
|
|
|
|
for (guint f = 0; f < vpp_desc->NumFilters; f++) {
|
|
for (guint i = 0; i < vpp_desc->NumFilters; i++) {
|
|
if (vpp_desc->Filters[f].MemDesc->Formats[i].InFormat != infmt)
|
|
continue;
|
|
|
|
*width = vpp_desc->Filters[f].MemDesc->Width;
|
|
*height = vpp_desc->Filters[f].MemDesc->Height;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_vpp_get_resolution_range (mfxSession * session,
|
|
mfxVPPDescription * vpp_desc, ResolutionRange * res_range)
|
|
{
|
|
mfxVideoParam in, out;
|
|
mfxRange32U width, height;
|
|
ResolutionRange res = { default_width, default_width,
|
|
default_height, default_height
|
|
};
|
|
|
|
g_return_val_if_fail (res_range != NULL, FALSE);
|
|
|
|
if (!_vpp_get_desc_image_range (vpp_desc,
|
|
DEFAULT_VIDEO_FORMAT, &width, &height))
|
|
return FALSE;
|
|
|
|
_vpp_init_param (&in, DEFAULT_VIDEO_FORMAT, DEFAULT_VIDEO_FORMAT);
|
|
out = in;
|
|
|
|
IsParamSupportedFunc func = _vpp_is_param_supported;
|
|
if (!_get_min_width (session, &in, &out, func, &width, &res.min_width) ||
|
|
!_get_max_width (session, &in, &out, func, &width, &res.max_width) ||
|
|
!_get_min_height (session, &in, &out, func, &height, &res.min_height) ||
|
|
!_get_max_height (session, &in, &out, func, &height, &res.max_height))
|
|
return FALSE;
|
|
|
|
GST_DEBUG ("Got VPP supported resolution range "
|
|
"width: [%d, %d], height: [%d, %d]",
|
|
res.min_width, res.max_width, res.min_height, res.max_height);
|
|
|
|
res_range->min_width = res.min_width;
|
|
res_range->max_width = res.max_width;
|
|
res_range->min_height = res.min_height;
|
|
res_range->max_height = res.max_height;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstCaps *
|
|
_vpp_create_caps (GstMsdkContext * context,
|
|
GValue * supported_fmts, ResolutionRange * res)
|
|
{
|
|
GstCaps *caps, *dma_caps, *raw_caps;
|
|
|
|
#ifndef _WIN32
|
|
caps = gst_caps_from_string ("video/x-raw(memory:VAMemory), "
|
|
"format=(string){ NV12, VUYA, P010_10LE }");
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_VPP, supported_fmts);
|
|
gst_caps_append (caps, dma_caps);
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
caps = gst_caps_from_string ("video/x-raw(memory:D3D11Memory), "
|
|
"format=(string){ NV12, VUYA, P010_10LE }");
|
|
#endif
|
|
raw_caps = gst_caps_from_string ("video/x-raw");
|
|
gst_caps_set_value (raw_caps, "format", supported_fmts);
|
|
gst_caps_append (caps, raw_caps);
|
|
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, res->min_width, res->max_width,
|
|
"height", GST_TYPE_INT_RANGE, res->min_height, res->max_height, NULL);
|
|
|
|
gst_msdkcaps_set_strings (caps, "memory:SystemMemory",
|
|
"interlace-mode", "progressive, interleaved, mixed");
|
|
|
|
GST_DEBUG ("Create VPP caps %" GST_PTR_FORMAT, caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_vpp_create_caps (GstMsdkContext * context,
|
|
gpointer vpp_description, GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
mfxVPPDescription *vpp_desc;
|
|
GstCaps *in_caps = NULL, *out_caps = NULL;
|
|
GValue supported_in_fmts = G_VALUE_INIT;
|
|
GValue supported_out_fmts = G_VALUE_INIT;
|
|
ResolutionRange res_range = { 1, G_MAXUINT16, 1, G_MAXUINT16 };
|
|
mfxSession session;
|
|
|
|
g_return_val_if_fail (context, FALSE);
|
|
g_return_val_if_fail (vpp_description, FALSE);
|
|
|
|
vpp_desc = (mfxVPPDescription *) vpp_description;
|
|
if (vpp_desc->NumFilters == 0)
|
|
return FALSE;
|
|
|
|
session = gst_msdk_context_get_session (context);
|
|
|
|
g_value_init (&supported_in_fmts, GST_TYPE_LIST);
|
|
g_value_init (&supported_out_fmts, GST_TYPE_LIST);
|
|
|
|
if (!_vpp_get_supported_formats (&session,
|
|
&supported_in_fmts, &supported_out_fmts))
|
|
goto failed;
|
|
|
|
if (!_vpp_get_resolution_range (&session, vpp_desc, &res_range))
|
|
goto failed;
|
|
|
|
in_caps = _vpp_create_caps (context, &supported_in_fmts, &res_range);
|
|
g_value_unset (&supported_in_fmts);
|
|
if (!in_caps)
|
|
goto failed;
|
|
|
|
out_caps = _vpp_create_caps (context, &supported_out_fmts, &res_range);
|
|
g_value_unset (&supported_out_fmts);
|
|
if (!out_caps)
|
|
goto failed;
|
|
|
|
*sink_caps = in_caps;
|
|
*src_caps = out_caps;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_WARNING ("Failed to create caps for VPP");
|
|
|
|
g_value_unset (&supported_in_fmts);
|
|
g_value_unset (&supported_out_fmts);
|
|
if (in_caps)
|
|
gst_caps_unref (in_caps);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
#else
|
|
|
|
gboolean
|
|
gst_msdkcaps_enc_create_caps (GstMsdkContext * context,
|
|
gpointer enc_description, guint codec_id,
|
|
GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_dec_create_caps (GstMsdkContext * context,
|
|
gpointer dec_description, guint codec_id,
|
|
GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_vpp_create_caps (GstMsdkContext * context,
|
|
gpointer vpp_description, GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
#endif
|
|
|
|
static gboolean
|
|
_get_profiles (guint codec_id, GValue * supported_profs)
|
|
{
|
|
const Profile *profiles = NULL;
|
|
gchar **profs = NULL;
|
|
|
|
g_return_val_if_fail (supported_profs != NULL, FALSE);
|
|
|
|
for (guint c = 0; c < G_N_ELEMENTS (codec_profs); c++) {
|
|
if (codec_profs[c].codec == codec_id) {
|
|
profiles = codec_profs[c].profiles;
|
|
break;
|
|
}
|
|
}
|
|
if (!profiles)
|
|
return FALSE;
|
|
|
|
for (; profiles->id != MFX_PROFILE_UNKNOWN; profiles++) {
|
|
profs = g_strsplit (profiles->names, PROFILE_DELIMITER, 0);
|
|
for (guint p = 0; profs[p]; p++)
|
|
_list_append_string (supported_profs, profs[p]);
|
|
g_strfreev (profs);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static const char *
|
|
_enc_get_static_raw_formats (guint codec_id)
|
|
{
|
|
switch (codec_id) {
|
|
case MFX_CODEC_AVC:
|
|
return "NV12, YUY2, UYVY, BGRA";
|
|
case MFX_CODEC_HEVC:
|
|
return "NV12, YUY2, BGRA, BGR10A2_LE, P010_10LE, VUYA, Y410, Y210, "
|
|
"P012_LE, Y212_LE";
|
|
case MFX_CODEC_MPEG2:
|
|
return "NV12";
|
|
case MFX_CODEC_VP9:
|
|
return "NV12, P010_10LE, VUYA, Y410";
|
|
case MFX_CODEC_AV1:
|
|
return "NV12, P010_10LE";
|
|
case MFX_CODEC_JPEG:
|
|
return "NV12, YUY2, UYVY, BGRA";
|
|
default:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
static const char *
|
|
_enc_get_static_dma_formats (guint codec_id)
|
|
{
|
|
switch (codec_id) {
|
|
case MFX_CODEC_AVC:
|
|
return "NV12, BGRx";
|
|
case MFX_CODEC_HEVC:
|
|
return "NV12, P010_10LE";
|
|
case MFX_CODEC_MPEG2:
|
|
return "NV12";
|
|
case MFX_CODEC_VP9:
|
|
return "NV12, P010_10LE";
|
|
case MFX_CODEC_AV1:
|
|
return "NV12, P010_10LE";
|
|
case MFX_CODEC_JPEG:
|
|
return "NV12, BGRx";
|
|
default:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
gboolean
|
|
gst_msdkcaps_enc_create_static_caps (GstMsdkContext * context,
|
|
guint codec_id, GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
GstCaps *in_caps = NULL, *out_caps = NULL;
|
|
GstCaps *dma_caps = NULL;
|
|
gchar *raw_caps_str;
|
|
const gchar *media_type = NULL;
|
|
const char *raw_fmts = NULL;
|
|
const char *dma_fmts_str = NULL;
|
|
GValue dma_fmts = G_VALUE_INIT;
|
|
GValue supported_profs = G_VALUE_INIT;
|
|
|
|
raw_fmts = _enc_get_static_raw_formats (codec_id);
|
|
if (!raw_fmts)
|
|
goto failed;
|
|
raw_caps_str = g_strdup_printf ("video/x-raw, format=(string){ %s }",
|
|
raw_fmts);
|
|
in_caps = gst_caps_from_string (raw_caps_str);
|
|
g_free (raw_caps_str);
|
|
|
|
#ifndef _WIN32
|
|
dma_fmts_str = _enc_get_static_dma_formats (codec_id);
|
|
if (!dma_fmts_str)
|
|
goto failed;
|
|
|
|
g_value_init (&dma_fmts, GST_TYPE_LIST);
|
|
_strings_to_list (dma_fmts_str, &dma_fmts);
|
|
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_ENCODER, &dma_fmts);
|
|
g_value_unset (&dma_fmts);
|
|
gst_caps_append (in_caps, dma_caps);
|
|
|
|
gst_caps_append (in_caps,
|
|
gst_caps_from_string
|
|
("video/x-raw(memory:VAMemory), format=(string){ NV12 }"));
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
VAR_UNUSED (dma_fmts_str);
|
|
VAR_UNUSED (dma_fmts);
|
|
gst_caps_append (in_caps,
|
|
gst_caps_from_string
|
|
("video/x-raw(memory:D3D11Memory), format=(string){ NV12 }"));
|
|
#endif
|
|
|
|
gst_caps_set_simple (in_caps,
|
|
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1,
|
|
"interlace-mode", G_TYPE_STRING, "progressive", NULL);
|
|
|
|
media_type = _get_media_type (codec_id);
|
|
if (!media_type)
|
|
goto failed;
|
|
|
|
out_caps = gst_caps_new_empty_simple (media_type);
|
|
|
|
g_value_init (&supported_profs, GST_TYPE_LIST);
|
|
if (!_get_profiles (codec_id, &supported_profs))
|
|
goto failed;
|
|
|
|
gst_caps_set_value (out_caps, "profile", &supported_profs);
|
|
g_value_unset (&supported_profs);
|
|
|
|
gst_caps_set_simple (out_caps,
|
|
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL);
|
|
|
|
*sink_caps = in_caps;
|
|
*src_caps = out_caps;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_WARNING ("Failed to create caps for %" GST_FOURCC_FORMAT " ENC",
|
|
GST_FOURCC_ARGS (codec_id));
|
|
|
|
g_value_unset (&supported_profs);
|
|
if (in_caps)
|
|
gst_caps_unref (in_caps);
|
|
if (out_caps)
|
|
gst_caps_unref (out_caps);
|
|
if (dma_caps)
|
|
gst_caps_unref (dma_caps);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static const char *
|
|
_dec_get_static_raw_formats (guint codec_id)
|
|
{
|
|
switch (codec_id) {
|
|
case MFX_CODEC_AVC:
|
|
return "NV12, BGRA, BGRx";
|
|
case MFX_CODEC_HEVC:
|
|
return "NV12, P010_10LE, YUY2, Y210, VUYA, Y410, P012_LE, "
|
|
"Y212_LE, Y412_LE, BGRA, BGRx";
|
|
case MFX_CODEC_MPEG2:
|
|
return "NV12";
|
|
case MFX_CODEC_VP9:
|
|
return "NV12, P010_10LE, VUYA, Y410, P012_LE, Y412_LE";
|
|
case MFX_CODEC_AV1:
|
|
return "NV12, P010_10LE, VUYA, Y410";
|
|
case MFX_CODEC_JPEG:
|
|
return "NV12, YUY2";
|
|
case MFX_CODEC_VP8:
|
|
return "NV12";
|
|
case MFX_CODEC_VC1:
|
|
return "NV12";
|
|
default:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
static const char *
|
|
_dec_get_static_dma_formats (guint codec_id)
|
|
{
|
|
switch (codec_id) {
|
|
case MFX_CODEC_AVC:
|
|
return "NV12, BGRA, BGRx";
|
|
case MFX_CODEC_HEVC:
|
|
return "NV12, P010_10LE, YUY2, Y210, VUYA, Y410, P012_LE, "
|
|
"Y212_LE, Y412_LE, BGRA, BGRx";
|
|
case MFX_CODEC_MPEG2:
|
|
return "NV12";
|
|
case MFX_CODEC_VP9:
|
|
return "NV12, P010_10LE, VUYA, Y410, P012_LE, Y412_LE";
|
|
case MFX_CODEC_AV1:
|
|
return "NV12, P010_10LE, VUYA, Y410";
|
|
case MFX_CODEC_JPEG:
|
|
return "NV12, YUY2";
|
|
case MFX_CODEC_VP8:
|
|
return "NV12";
|
|
case MFX_CODEC_VC1:
|
|
return "NV12";
|
|
default:
|
|
GST_WARNING ("Unsupported codec %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_id));
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
gboolean
|
|
gst_msdkcaps_dec_create_static_caps (GstMsdkContext * context,
|
|
guint codec_id, GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
GstCaps *in_caps = NULL, *out_caps = NULL;
|
|
GstCaps *dma_caps = NULL;
|
|
gchar *raw_caps_str;
|
|
const gchar *media_type = NULL;
|
|
const char *raw_fmts = NULL;
|
|
const char *dma_fmts_str = NULL;
|
|
GValue dma_fmts = G_VALUE_INIT;
|
|
|
|
media_type = _get_media_type (codec_id);
|
|
if (!media_type)
|
|
goto failed;
|
|
|
|
in_caps = gst_caps_new_empty_simple (media_type);
|
|
|
|
raw_fmts = _dec_get_static_raw_formats (codec_id);
|
|
if (!raw_fmts)
|
|
goto failed;
|
|
raw_caps_str = g_strdup_printf ("video/x-raw, format=(string){ %s }",
|
|
raw_fmts);
|
|
out_caps = gst_caps_from_string (raw_caps_str);
|
|
g_free (raw_caps_str);
|
|
|
|
#ifndef _WIN32
|
|
dma_fmts_str = _dec_get_static_dma_formats (codec_id);
|
|
if (!dma_fmts_str)
|
|
goto failed;
|
|
|
|
g_value_init (&dma_fmts, GST_TYPE_LIST);
|
|
_strings_to_list (dma_fmts_str, &dma_fmts);
|
|
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_DECODER, &dma_fmts);
|
|
g_value_unset (&dma_fmts);
|
|
gst_caps_append (out_caps, dma_caps);
|
|
|
|
gst_caps_append (out_caps,
|
|
gst_caps_from_string
|
|
("video/x-raw(memory:VAMemory), format=(string){ NV12 }"));
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
VAR_UNUSED (dma_fmts_str);
|
|
VAR_UNUSED (dma_fmts);
|
|
gst_caps_append (out_caps,
|
|
gst_caps_from_string
|
|
("video/x-raw(memory:D3D11Memory), format=(string){ NV12 }"));
|
|
#endif
|
|
|
|
gst_caps_set_simple (out_caps,
|
|
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1,
|
|
"interlace-mode", G_TYPE_STRING, "progressive", NULL);
|
|
|
|
*sink_caps = in_caps;
|
|
*src_caps = out_caps;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_WARNING ("Failed to create caps for %" GST_FOURCC_FORMAT " DEC",
|
|
GST_FOURCC_ARGS (codec_id));
|
|
|
|
if (in_caps)
|
|
gst_caps_unref (in_caps);
|
|
if (out_caps)
|
|
gst_caps_unref (out_caps);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static const char *
|
|
_vpp_get_static_raw_formats (GstPadDirection direction)
|
|
{
|
|
switch (direction) {
|
|
case GST_PAD_SINK:
|
|
return "NV12, YV12, I420, YUY2, UYVY, VUYA, BGRA, BGRx, P010_10LE, "
|
|
"RGB16, Y410, Y210, P012_LE, Y212_LE, Y412_LE";
|
|
case GST_PAD_SRC:
|
|
return "NV12, BGRA, YUY2, UYVY, VUYA, BGRx, P010_10LE, BGR10A2_LE, "
|
|
"YV12, Y410, Y210, RGBP, BGRP, P012_LE, Y212_LE, Y412_LE";
|
|
default:
|
|
GST_WARNING ("Unsupported VPP direction");
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
static const char *
|
|
_vpp_get_static_dma_formats (GstPadDirection direction)
|
|
{
|
|
switch (direction) {
|
|
case GST_PAD_SINK:
|
|
return "NV12, BGRA, YUY2, UYVY, VUYA, P010_10LE, RGB16, Y410, Y210, "
|
|
"P012_LE, Y212_LE, Y412_LE";
|
|
case GST_PAD_SRC:
|
|
return "NV12, BGRA, YUY2, UYVY, VUYA, BGRx, P010_10LE, BGR10A2_LE, "
|
|
"YV12, Y410, Y210, RGBP, BGRP, P012_LE, Y212_LE, Y412_LE";
|
|
default:
|
|
GST_WARNING ("Unsupported VPP direction");
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
static GstCaps *
|
|
_vpp_create_static_caps (GstMsdkContext * context, GstPadDirection direction)
|
|
{
|
|
GstCaps *caps = NULL, *dma_caps = NULL;
|
|
gchar *caps_str;
|
|
GValue dma_fmts = G_VALUE_INIT;
|
|
|
|
caps_str = g_strdup_printf ("video/x-raw, format=(string){ %s }",
|
|
_vpp_get_static_raw_formats (direction));
|
|
caps = gst_caps_from_string (caps_str);
|
|
g_free (caps_str);
|
|
|
|
#ifndef _WIN32
|
|
g_value_init (&dma_fmts, GST_TYPE_LIST);
|
|
_strings_to_list (_vpp_get_static_dma_formats (direction), &dma_fmts);
|
|
|
|
dma_caps = _create_dma_drm_caps (context, GST_MSDK_JOB_VPP, &dma_fmts);
|
|
g_value_unset (&dma_fmts);
|
|
gst_caps_append (caps, dma_caps);
|
|
|
|
gst_caps_append (caps, gst_caps_from_string ("video/x-raw(memory:VAMemory), "
|
|
"format=(string){ NV12, VUYA, P010_10LE }"));
|
|
#else
|
|
VAR_UNUSED (dma_caps);
|
|
VAR_UNUSED (dma_fmts);
|
|
|
|
gst_caps_append (caps,
|
|
gst_caps_from_string ("video/x-raw(memory:D3D11Memory), "
|
|
"format=(string){ NV12, VUYA, P010_10LE }"));
|
|
#endif
|
|
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL);
|
|
|
|
gst_msdkcaps_set_strings (caps, "memory:SystemMemory",
|
|
"interlace-mode", "progressive, interleaved, mixed");
|
|
|
|
return caps;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_vpp_create_static_caps (GstMsdkContext * context,
|
|
GstCaps ** sink_caps, GstCaps ** src_caps)
|
|
{
|
|
*sink_caps = _vpp_create_static_caps (context, GST_PAD_SINK);
|
|
*src_caps = _vpp_create_static_caps (context, GST_PAD_SRC);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
_pad_template_init (GstElementClass * klass,
|
|
const gchar * name_template, GstPadDirection direction,
|
|
GstCaps * caps, const gchar * doc_caps_str)
|
|
{
|
|
GstPadTemplate *pad_templ;
|
|
|
|
if (!caps)
|
|
return;
|
|
|
|
pad_templ = gst_pad_template_new (name_template,
|
|
direction, GST_PAD_ALWAYS, caps);
|
|
if (doc_caps_str) {
|
|
GstCaps *doc_caps = gst_caps_from_string (doc_caps_str);
|
|
gst_pad_template_set_documentation_caps (pad_templ, doc_caps);
|
|
gst_caps_unref (doc_caps);
|
|
}
|
|
|
|
gst_element_class_add_pad_template (klass, pad_templ);
|
|
}
|
|
|
|
void
|
|
gst_msdkcaps_pad_template_init (GstElementClass * klass,
|
|
GstCaps * sink_caps, GstCaps * src_caps,
|
|
const gchar * doc_sink_caps_str, const gchar * doc_src_caps_str)
|
|
{
|
|
_pad_template_init (klass,
|
|
"sink", GST_PAD_SINK, sink_caps, doc_sink_caps_str);
|
|
|
|
_pad_template_init (klass, "src", GST_PAD_SRC, src_caps, doc_src_caps_str);
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_set_strings (GstCaps * caps,
|
|
const gchar * features, const char *field, const gchar * strings)
|
|
{
|
|
GValue list = G_VALUE_INIT;
|
|
guint size = gst_caps_get_size (caps);
|
|
|
|
g_return_val_if_fail (GST_IS_CAPS (caps), FALSE);
|
|
g_return_val_if_fail (field != NULL, FALSE);
|
|
g_return_val_if_fail (strings != NULL, FALSE);
|
|
|
|
g_value_init (&list, GST_TYPE_LIST);
|
|
_strings_to_list (strings, &list);
|
|
|
|
if (features) {
|
|
GstStructure *s = NULL;
|
|
GstCapsFeatures *f = gst_caps_features_from_string (features);
|
|
|
|
for (guint i = 0; i < size; i++) {
|
|
if (gst_caps_features_is_equal (f, gst_caps_get_features (caps, i))) {
|
|
s = gst_caps_get_structure (caps, i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
gst_caps_features_free (f);
|
|
|
|
if (!s)
|
|
return FALSE;
|
|
|
|
/* When we use this function to get a fixated caps, we should set
|
|
* a single value instead of a list in the corresponding field.
|
|
*/
|
|
if (gst_value_list_get_size (&list) == 1)
|
|
gst_structure_set_value (s, field, gst_value_list_get_value (&list, 0));
|
|
else
|
|
gst_structure_set_value (s, field, &list);
|
|
} else {
|
|
gst_caps_set_value (caps, field, &list);
|
|
}
|
|
|
|
g_value_unset (&list);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_msdkcaps_video_info_from_caps (const GstCaps * caps,
|
|
GstVideoInfo * info, guint64 * modifier)
|
|
{
|
|
g_return_val_if_fail (caps != NULL, FALSE);
|
|
g_return_val_if_fail (info != NULL, FALSE);
|
|
|
|
#ifndef _WIN32
|
|
if (gst_video_is_dma_drm_caps (caps)) {
|
|
GstVideoInfoDmaDrm *drm_info = gst_video_info_dma_drm_new_from_caps (caps);
|
|
if (!drm_info)
|
|
goto failed;
|
|
|
|
if (!gst_video_info_dma_drm_to_video_info (drm_info, info)) {
|
|
gst_video_info_dma_drm_free (drm_info);
|
|
goto failed;
|
|
}
|
|
if (modifier)
|
|
*modifier = drm_info->drm_modifier;
|
|
|
|
gst_video_info_dma_drm_free (drm_info);
|
|
} else
|
|
#endif
|
|
|
|
if (!gst_video_info_from_caps (info, caps))
|
|
goto failed;
|
|
|
|
return TRUE;
|
|
|
|
failed:
|
|
GST_ERROR_OBJECT (caps, "Failed to get video info fom caps");
|
|
return FALSE;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
GstCaps *
|
|
gst_msdkcaps_video_info_to_drm_caps (GstVideoInfo * info, guint64 modifier)
|
|
{
|
|
GstVideoInfoDmaDrm drm_info;
|
|
|
|
gst_video_info_dma_drm_init (&drm_info);
|
|
drm_info.vinfo = *info;
|
|
drm_info.drm_fourcc =
|
|
gst_va_drm_fourcc_from_video_format (GST_VIDEO_INFO_FORMAT (info));
|
|
drm_info.drm_modifier = modifier;
|
|
|
|
return gst_video_info_dma_drm_to_caps (&drm_info);
|
|
}
|
|
|
|
guint64
|
|
get_msdkcaps_get_modifier (const GstCaps * caps)
|
|
{
|
|
guint64 modifier = DRM_FORMAT_MOD_INVALID;
|
|
guint size = gst_caps_get_size (caps);
|
|
|
|
for (guint i = 0; i < size; i++) {
|
|
GstCapsFeatures *f = gst_caps_get_features (caps, i);
|
|
|
|
if (gst_caps_features_contains (f, GST_CAPS_FEATURE_MEMORY_DMABUF)) {
|
|
GstStructure *s = gst_caps_get_structure (caps, i);
|
|
const GValue *drm_fmts = gst_structure_get_value (s, "drm-format");
|
|
const gchar *drm_str = NULL;
|
|
|
|
if (!drm_fmts)
|
|
continue;
|
|
|
|
if (G_VALUE_HOLDS_STRING (drm_fmts))
|
|
drm_str = g_value_get_string (drm_fmts);
|
|
else if (GST_VALUE_HOLDS_LIST (drm_fmts)) {
|
|
const GValue *val = gst_value_list_get_value (drm_fmts, 0);
|
|
drm_str = g_value_get_string (val);
|
|
}
|
|
|
|
gst_video_dma_drm_fourcc_from_string (drm_str, &modifier);
|
|
}
|
|
}
|
|
|
|
GST_DEBUG ("got modifier: 0x%016lx", modifier);
|
|
|
|
return modifier;
|
|
}
|
|
#endif
|