mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-05 06:58:49 +00:00
vtdec: Correctly retrieve reorder queue length from SPS
Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4057>
This commit is contained in:
parent
40300172ad
commit
3c739b9b76
3 changed files with 106 additions and 71 deletions
|
@ -115,7 +115,7 @@ if applemedia_found_deps
|
||||||
objcpp_args : gst_plugins_bad_args + applemedia_args + applemedia_objc_args + applemedia_objcpp_args,
|
objcpp_args : gst_plugins_bad_args + applemedia_args + applemedia_objc_args + applemedia_objcpp_args,
|
||||||
link_args : noseh_link_args,
|
link_args : noseh_link_args,
|
||||||
include_directories : [configinc, libsinc],
|
include_directories : [configinc, libsinc],
|
||||||
dependencies : [gstvideo_dep, gstaudio_dep, gstpbutils_dep, gst_dep, gstbase_dep, gstgl_dep, gstglproto_dep] + applemedia_frameworks,
|
dependencies : [gstvideo_dep, gstaudio_dep, gstpbutils_dep, gst_dep, gstbase_dep, gstgl_dep, gstglproto_dep, gstcodecparsers_dep] + applemedia_frameworks,
|
||||||
override_options : ['cpp_std=c++11'],
|
override_options : ['cpp_std=c++11'],
|
||||||
install : true,
|
install : true,
|
||||||
install_dir : plugins_install_dir,
|
install_dir : plugins_install_dir,
|
||||||
|
|
|
@ -1066,54 +1066,6 @@ gst_vtdec_push_frames_if_needed (GstVtdec * vtdec, gboolean drain,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
|
||||||
parse_h264_profile_and_level_from_codec_data (GstVtdec * vtdec,
|
|
||||||
GstBuffer * codec_data, int *profile, int *level)
|
|
||||||
{
|
|
||||||
GstMapInfo map;
|
|
||||||
guint8 *data;
|
|
||||||
gint size;
|
|
||||||
gboolean ret = TRUE;
|
|
||||||
|
|
||||||
gst_buffer_map (codec_data, &map, GST_MAP_READ);
|
|
||||||
data = map.data;
|
|
||||||
size = map.size;
|
|
||||||
|
|
||||||
/* parse the avcC data */
|
|
||||||
if (size < 7)
|
|
||||||
goto avcc_too_small;
|
|
||||||
|
|
||||||
/* parse the version, this must be 1 */
|
|
||||||
if (data[0] != 1)
|
|
||||||
goto wrong_version;
|
|
||||||
|
|
||||||
/* AVCProfileIndication */
|
|
||||||
/* profile_compat */
|
|
||||||
/* AVCLevelIndication */
|
|
||||||
if (profile)
|
|
||||||
*profile = data[1];
|
|
||||||
|
|
||||||
if (level)
|
|
||||||
*level = data[3];
|
|
||||||
|
|
||||||
out:
|
|
||||||
gst_buffer_unmap (codec_data, &map);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
avcc_too_small:
|
|
||||||
GST_ELEMENT_ERROR (vtdec, STREAM, DECODE, (NULL),
|
|
||||||
("invalid codec_data buffer length"));
|
|
||||||
ret = FALSE;
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
wrong_version:
|
|
||||||
GST_ELEMENT_ERROR (vtdec, STREAM, DECODE, (NULL),
|
|
||||||
("wrong avcC version in codec_data"));
|
|
||||||
ret = FALSE;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
get_dpb_max_mb_s_from_level (GstVtdec * vtdec, int level)
|
get_dpb_max_mb_s_from_level (GstVtdec * vtdec, int level)
|
||||||
{
|
{
|
||||||
|
@ -1170,47 +1122,126 @@ gst_vtdec_compute_reorder_queue_length (GstVtdec * vtdec,
|
||||||
vtdec->reorder_queue_length = 0;
|
vtdec->reorder_queue_length = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (vtdec, "Reorder queue length: %d",
|
||||||
|
vtdec->reorder_queue_length);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
parse_h264_decoder_config_record (GstVtdec * vtdec, GstBuffer * codec_data,
|
||||||
|
GstH264DecoderConfigRecord ** config)
|
||||||
|
{
|
||||||
|
GstH264NalParser *parser = gst_h264_nal_parser_new ();
|
||||||
|
GstMapInfo map;
|
||||||
|
gboolean ret = TRUE;
|
||||||
|
|
||||||
|
gst_buffer_map (codec_data, &map, GST_MAP_READ);
|
||||||
|
|
||||||
|
if (gst_h264_parser_parse_decoder_config_record (parser, map.data, map.size,
|
||||||
|
config) != GST_H264_PARSER_OK) {
|
||||||
|
GST_WARNING_OBJECT (vtdec, "Failed to parse codec-data");
|
||||||
|
ret = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_h264_nal_parser_free (parser);
|
||||||
|
gst_buffer_unmap (codec_data, &map);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
get_h264_dpb_size_from_sps (GstVtdec * vtdec, GstH264NalUnit * nalu,
|
||||||
|
gint * dpb_size)
|
||||||
|
{
|
||||||
|
GstH264ParserResult result;
|
||||||
|
GstH264SPS sps;
|
||||||
|
gint width_mb, height_mb;
|
||||||
|
gint max_dpb_frames, max_dpb_size, max_dpb_mbs;
|
||||||
|
|
||||||
|
result = gst_h264_parse_sps (nalu, &sps);
|
||||||
|
if (result != GST_H264_PARSER_OK) {
|
||||||
|
GST_WARNING_OBJECT (vtdec, "Failed to parse SPS, result %d", result);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
max_dpb_mbs = get_dpb_max_mb_s_from_level (vtdec, sps.level_idc);
|
||||||
|
if (max_dpb_mbs == -1) {
|
||||||
|
GST_ELEMENT_ERROR (vtdec, STREAM, DECODE, (NULL),
|
||||||
|
("invalid level found in SPS, could not compute max_dpb_mbs"));
|
||||||
|
gst_h264_sps_clear (&sps);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This formula is specified in sections A.3.1.h and A.3.2.f of the 2009
|
||||||
|
* edition of the standard */
|
||||||
|
width_mb = sps.width / 16;
|
||||||
|
height_mb = sps.height / 16;
|
||||||
|
max_dpb_frames = MIN (max_dpb_mbs / (width_mb * height_mb),
|
||||||
|
GST_VTDEC_DPB_MAX_SIZE);
|
||||||
|
|
||||||
|
if (sps.vui_parameters_present_flag
|
||||||
|
&& sps.vui_parameters.bitstream_restriction_flag)
|
||||||
|
max_dpb_frames = MAX (1, sps.vui_parameters.max_dec_frame_buffering);
|
||||||
|
|
||||||
|
/* Some non-conforming H264 streams may request a number of frames
|
||||||
|
* larger than the calculated limit.
|
||||||
|
* See https://chromium-review.googlesource.com/c/chromium/src/+/760276/
|
||||||
|
*/
|
||||||
|
max_dpb_size = MAX (max_dpb_frames, sps.num_ref_frames);
|
||||||
|
if (max_dpb_size > GST_VTDEC_DPB_MAX_SIZE) {
|
||||||
|
GST_WARNING_OBJECT (vtdec, "Too large calculated DPB size %d",
|
||||||
|
max_dpb_size);
|
||||||
|
max_dpb_size = GST_VTDEC_DPB_MAX_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
*dpb_size = max_dpb_size;
|
||||||
|
|
||||||
|
gst_h264_sps_clear (&sps);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
compute_h264_decode_picture_buffer_length (GstVtdec * vtdec,
|
compute_h264_decode_picture_buffer_length (GstVtdec * vtdec,
|
||||||
GstBuffer * codec_data, int *length)
|
GstBuffer * codec_data, gint * length)
|
||||||
{
|
{
|
||||||
int profile, level;
|
GstH264DecoderConfigRecord *config = NULL;
|
||||||
int dpb_mb_size = 16;
|
GstH264NalUnit *nalu;
|
||||||
int max_dpb_size_frames = 16;
|
guint8 profile, level;
|
||||||
int max_dpb_mb_s = -1;
|
gboolean ret = TRUE;
|
||||||
int width_in_mb_s = GST_ROUND_UP_16 (vtdec->video_info.width) / dpb_mb_size;
|
gint new_length;
|
||||||
int height_in_mb_s = GST_ROUND_UP_16 (vtdec->video_info.height) / dpb_mb_size;
|
guint i;
|
||||||
|
|
||||||
*length = 0;
|
*length = 0;
|
||||||
|
|
||||||
if (!parse_h264_profile_and_level_from_codec_data (vtdec, codec_data,
|
|
||||||
&profile, &level))
|
|
||||||
return FALSE;
|
|
||||||
|
|
||||||
if (vtdec->video_info.width == 0 || vtdec->video_info.height == 0)
|
if (vtdec->video_info.width == 0 || vtdec->video_info.height == 0)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
|
if (!parse_h264_decoder_config_record (vtdec, codec_data, &config))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
profile = config->profile_indication;
|
||||||
|
level = config->level_indication;
|
||||||
GST_INFO_OBJECT (vtdec, "parsed profile %d, level %d", profile, level);
|
GST_INFO_OBJECT (vtdec, "parsed profile %d, level %d", profile, level);
|
||||||
|
|
||||||
if (profile == 66) {
|
if (profile == 66) {
|
||||||
/* baseline or constrained-baseline, we don't need to reorder */
|
/* baseline or constrained-baseline, we don't need to reorder */
|
||||||
return TRUE;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
max_dpb_mb_s = get_dpb_max_mb_s_from_level (vtdec, level);
|
for (i = 0; i < config->sps->len; i++) {
|
||||||
if (max_dpb_mb_s == -1) {
|
nalu = &g_array_index (config->sps, GstH264NalUnit, i);
|
||||||
GST_ELEMENT_ERROR (vtdec, STREAM, DECODE, (NULL),
|
|
||||||
("invalid level in codec_data, could not compute max_dpb_mb_s"));
|
if (nalu->type != GST_H264_NAL_SPS)
|
||||||
return FALSE;
|
continue;
|
||||||
|
|
||||||
|
if (!get_h264_dpb_size_from_sps (vtdec, nalu, &new_length))
|
||||||
|
GST_WARNING_OBJECT (vtdec, "Failed to get DPB size from SPS");
|
||||||
|
else
|
||||||
|
*length = MAX (*length, new_length);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* this formula is specified in sections A.3.1.h and A.3.2.f of the 2009
|
out:
|
||||||
* edition of the standard */
|
gst_h264_decoder_config_record_free (config);
|
||||||
*length = MIN (floor (max_dpb_mb_s / (width_in_mb_s * height_in_mb_s)),
|
return ret;
|
||||||
max_dpb_size_frames);
|
|
||||||
return TRUE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
|
|
|
@ -30,6 +30,8 @@
|
||||||
#if defined(APPLEMEDIA_MOLTENVK)
|
#if defined(APPLEMEDIA_MOLTENVK)
|
||||||
#include <gst/vulkan/vulkan.h>
|
#include <gst/vulkan/vulkan.h>
|
||||||
#endif
|
#endif
|
||||||
|
#define GST_USE_UNSTABLE_API
|
||||||
|
#include <gst/codecparsers/gsth264parser.h>
|
||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
@ -42,6 +44,8 @@ G_BEGIN_DECLS
|
||||||
typedef struct _GstVtdec GstVtdec;
|
typedef struct _GstVtdec GstVtdec;
|
||||||
typedef struct _GstVtdecClass GstVtdecClass;
|
typedef struct _GstVtdecClass GstVtdecClass;
|
||||||
|
|
||||||
|
#define GST_VTDEC_DPB_MAX_SIZE 16
|
||||||
|
|
||||||
struct _GstVtdec
|
struct _GstVtdec
|
||||||
{
|
{
|
||||||
GstVideoDecoder base_vtdec;
|
GstVideoDecoder base_vtdec;
|
||||||
|
|
Loading…
Reference in a new issue