mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-22 23:28:16 +00:00
1108 lines
40 KiB
C
1108 lines
40 KiB
C
|
/*
|
||
|
* gsth2642json.c - H.264 parsed bistream to json
|
||
|
*
|
||
|
* Copyright (C) 2023 Collabora
|
||
|
* Author: Benjamin Gaignard <benjamin.gaignard@collabora.com>
|
||
|
*
|
||
|
* This library is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU Library General Public
|
||
|
* License as published by the Free Software Foundation; either
|
||
|
* version 2 of the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This library is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
|
* Library General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Library General Public
|
||
|
* License along with this library; if not, write to the
|
||
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||
|
* Boston, MA 02110-1301, USA.
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* SECTION:element-h2642json
|
||
|
* @title: h2642json
|
||
|
*
|
||
|
* Convert H.264 bitstream parameters to JSON formated text.
|
||
|
*
|
||
|
* ## Example launch line
|
||
|
* ```
|
||
|
* gst-launch-1.0 filesrc location=/path/to/h.264/file ! parsebin ! h2642json ! filesink location=/path/to/json/file
|
||
|
* ```
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
|
||
|
#ifdef HAVE_CONFIG_H
|
||
|
#include <config.h>
|
||
|
#endif
|
||
|
|
||
|
#include <gst/base/base.h>
|
||
|
#include <json-glib/json-glib.h>
|
||
|
|
||
|
#include "gsth2642json.h"
|
||
|
|
||
|
GST_DEBUG_CATEGORY (gst_h264_2_json_debug);
|
||
|
#define GST_CAT_DEFAULT gst_h264_2_json_debug
|
||
|
|
||
|
struct _GstH2642json
|
||
|
{
|
||
|
GstElement parent;
|
||
|
|
||
|
GstPad *sinkpad, *srcpad;
|
||
|
GstH264NalParser *parser;
|
||
|
|
||
|
guint nal_length_size;
|
||
|
|
||
|
gboolean use_avc;
|
||
|
|
||
|
JsonObject *json;
|
||
|
};
|
||
|
|
||
|
static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
|
||
|
GST_PAD_SINK,
|
||
|
GST_PAD_ALWAYS,
|
||
|
GST_STATIC_CAPS ("video/x-h264")
|
||
|
);
|
||
|
|
||
|
static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
|
||
|
GST_PAD_SRC,
|
||
|
GST_PAD_ALWAYS,
|
||
|
GST_STATIC_CAPS ("text/x-json,format=h264"));
|
||
|
|
||
|
G_DEFINE_TYPE_WITH_CODE (GstH2642json, gst_h264_2_json,
|
||
|
GST_TYPE_ELEMENT,
|
||
|
GST_DEBUG_CATEGORY_INIT (gst_h264_2_json_debug, "h2642json", 0,
|
||
|
"H.264 to json"));
|
||
|
|
||
|
static void
|
||
|
gst_h264_2_json_finalize (GObject * object)
|
||
|
{
|
||
|
GstH2642json *self = GST_H264_2_JSON (object);
|
||
|
|
||
|
json_object_unref (self->json);
|
||
|
gst_h264_nal_parser_free (self->parser);
|
||
|
}
|
||
|
|
||
|
static gchar *
|
||
|
get_string_from_json_object (JsonObject * object)
|
||
|
{
|
||
|
JsonNode *root;
|
||
|
JsonGenerator *generator;
|
||
|
gchar *text;
|
||
|
|
||
|
/* Make it the root node */
|
||
|
root = json_node_init_object (json_node_alloc (), object);
|
||
|
generator = json_generator_new ();
|
||
|
json_generator_set_indent (generator, 2);
|
||
|
json_generator_set_indent_char (generator, ' ');
|
||
|
json_generator_set_pretty (generator, TRUE);
|
||
|
json_generator_set_root (generator, root);
|
||
|
text = json_generator_to_data (generator, NULL);
|
||
|
|
||
|
/* Release everything */
|
||
|
g_object_unref (generator);
|
||
|
json_node_free (root);
|
||
|
return text;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_parse_sps (GstH2642json * self, GstH264NalUnit * nalu)
|
||
|
{
|
||
|
JsonObject *json = self->json;
|
||
|
JsonObject *sps;
|
||
|
JsonArray *scaling_lists_4x4, *scaling_lists_8x8, *offset_for_ref_frame;
|
||
|
GstH264SPS h264_sps;
|
||
|
GstH264ParserResult pres;
|
||
|
GstFlowReturn ret = GST_FLOW_OK;
|
||
|
gint i, j;
|
||
|
|
||
|
pres = gst_h264_parse_sps (nalu, &h264_sps);
|
||
|
if (pres != GST_H264_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to parse SPS, result %d", pres);
|
||
|
return GST_FLOW_ERROR;
|
||
|
}
|
||
|
|
||
|
GST_LOG_OBJECT (self, "SPS parsed");
|
||
|
|
||
|
if (gst_h264_parser_update_sps (self->parser,
|
||
|
&h264_sps) != GST_H264_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to update SPS");
|
||
|
ret = GST_FLOW_ERROR;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
sps = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (sps, "id", h264_sps.id);
|
||
|
json_object_set_int_member (sps, "profile idc", h264_sps.profile_idc);
|
||
|
json_object_set_boolean_member (sps, "constraint set0 flag",
|
||
|
h264_sps.constraint_set0_flag);
|
||
|
json_object_set_boolean_member (sps, "constraint set1 flag",
|
||
|
h264_sps.constraint_set1_flag);
|
||
|
json_object_set_boolean_member (sps, "constraint set2 flag",
|
||
|
h264_sps.constraint_set2_flag);
|
||
|
json_object_set_boolean_member (sps, "constraint set3 flag",
|
||
|
h264_sps.constraint_set3_flag);
|
||
|
json_object_set_boolean_member (sps, "constraint set4 flag",
|
||
|
h264_sps.constraint_set4_flag);
|
||
|
json_object_set_boolean_member (sps, "constraint set5 flag",
|
||
|
h264_sps.constraint_set5_flag);
|
||
|
json_object_set_int_member (sps, "level idc", h264_sps.level_idc);
|
||
|
json_object_set_int_member (sps, "chroma format idc",
|
||
|
h264_sps.chroma_format_idc);
|
||
|
json_object_set_boolean_member (sps, "separate colour plane flag",
|
||
|
h264_sps.separate_colour_plane_flag);
|
||
|
json_object_set_int_member (sps, "bit depth luma minus8",
|
||
|
h264_sps.bit_depth_luma_minus8);
|
||
|
json_object_set_int_member (sps, "bit depth chroma minus8",
|
||
|
h264_sps.bit_depth_chroma_minus8);
|
||
|
json_object_set_boolean_member (sps, "qpprime y zero transform bypass flag",
|
||
|
h264_sps.qpprime_y_zero_transform_bypass_flag);
|
||
|
|
||
|
json_object_set_boolean_member (sps, "scaling matrix present flag",
|
||
|
h264_sps.scaling_matrix_present_flag);
|
||
|
|
||
|
scaling_lists_4x4 = json_array_new ();
|
||
|
for (i = 0; i < 6; i++)
|
||
|
for (j = 0; j < 16; j++)
|
||
|
json_array_add_int_element (scaling_lists_4x4,
|
||
|
h264_sps.scaling_lists_4x4[i][j]);
|
||
|
json_object_set_array_member (sps, "scaling lists 4x4", scaling_lists_4x4);
|
||
|
|
||
|
scaling_lists_8x8 = json_array_new ();
|
||
|
for (i = 0; i < 6; i++)
|
||
|
for (j = 0; j < 64; j++)
|
||
|
json_array_add_int_element (scaling_lists_8x8,
|
||
|
h264_sps.scaling_lists_8x8[i][j]);
|
||
|
json_object_set_array_member (sps, "scaling lists 8x8", scaling_lists_8x8);
|
||
|
|
||
|
json_object_set_int_member (sps, "log2 max frame num minus4",
|
||
|
h264_sps.log2_max_frame_num_minus4);
|
||
|
json_object_set_int_member (sps, "pic order cnt type",
|
||
|
h264_sps.pic_order_cnt_type);
|
||
|
json_object_set_int_member (sps, "log2 max pic order cnt lsb minus4",
|
||
|
h264_sps.log2_max_pic_order_cnt_lsb_minus4);
|
||
|
json_object_set_boolean_member (sps, "delta pic order always zero flag",
|
||
|
h264_sps.delta_pic_order_always_zero_flag);
|
||
|
json_object_set_int_member (sps, "offset for non ref pic",
|
||
|
h264_sps.offset_for_non_ref_pic);
|
||
|
json_object_set_int_member (sps, "offset for top to bottom field",
|
||
|
h264_sps.offset_for_top_to_bottom_field);
|
||
|
json_object_set_int_member (sps, "num ref frames in pic order cnt cycle",
|
||
|
h264_sps.num_ref_frames_in_pic_order_cnt_cycle);
|
||
|
|
||
|
offset_for_ref_frame = json_array_new ();
|
||
|
for (i = 0; i < 255; i++)
|
||
|
json_array_add_int_element (offset_for_ref_frame,
|
||
|
h264_sps.offset_for_ref_frame[i]);
|
||
|
json_object_set_array_member (sps, "offset for ref frame",
|
||
|
offset_for_ref_frame);
|
||
|
|
||
|
json_object_set_int_member (sps, "max num ref frames",
|
||
|
h264_sps.num_ref_frames);
|
||
|
json_object_set_boolean_member (sps, "gaps in frame num value allowed flag",
|
||
|
h264_sps.gaps_in_frame_num_value_allowed_flag);
|
||
|
|
||
|
json_object_set_int_member (sps, "pic width in mbs minus1",
|
||
|
h264_sps.pic_width_in_mbs_minus1);
|
||
|
json_object_set_int_member (sps, "pic height in map units minus1",
|
||
|
h264_sps.pic_height_in_map_units_minus1);
|
||
|
json_object_set_boolean_member (sps, "frame mbs only flag",
|
||
|
h264_sps.frame_mbs_only_flag);
|
||
|
|
||
|
json_object_set_boolean_member (sps, "mb adaptive frame field flag",
|
||
|
h264_sps.mb_adaptive_frame_field_flag);
|
||
|
json_object_set_boolean_member (sps, "direct 8x8 inference flag",
|
||
|
h264_sps.direct_8x8_inference_flag);
|
||
|
json_object_set_boolean_member (sps, "frame cropping flag",
|
||
|
h264_sps.frame_cropping_flag);
|
||
|
|
||
|
json_object_set_int_member (sps, "frame crop left offset",
|
||
|
h264_sps.frame_crop_left_offset);
|
||
|
json_object_set_int_member (sps, "frame crop right offset",
|
||
|
h264_sps.frame_crop_right_offset);
|
||
|
json_object_set_int_member (sps, "frame crop top offset",
|
||
|
h264_sps.frame_crop_top_offset);
|
||
|
json_object_set_int_member (sps, "frame crop bottom offset",
|
||
|
h264_sps.frame_crop_bottom_offset);
|
||
|
|
||
|
json_object_set_boolean_member (sps, "vui parameters present flag",
|
||
|
h264_sps.vui_parameters_present_flag);
|
||
|
|
||
|
if (h264_sps.vui_parameters_present_flag) {
|
||
|
JsonObject *vui = json_object_new ();
|
||
|
GstH264VUIParams *vui_parameters = &h264_sps.vui_parameters;
|
||
|
|
||
|
json_object_set_boolean_member (vui, "aspect ratio info present flag",
|
||
|
vui_parameters->aspect_ratio_info_present_flag);
|
||
|
json_object_set_int_member (vui, "aspect ratio idc",
|
||
|
vui_parameters->aspect_ratio_idc);
|
||
|
if (vui_parameters->aspect_ratio_idc == 255) {
|
||
|
json_object_set_int_member (vui, "sar width", vui_parameters->sar_width);
|
||
|
json_object_set_int_member (vui, "sar height",
|
||
|
vui_parameters->sar_height);
|
||
|
}
|
||
|
|
||
|
json_object_set_boolean_member (vui, "overscan info present flag",
|
||
|
vui_parameters->overscan_info_present_flag);
|
||
|
if (vui_parameters->overscan_info_present_flag)
|
||
|
json_object_set_boolean_member (vui, "overscan appropriate flag",
|
||
|
vui_parameters->overscan_appropriate_flag);
|
||
|
|
||
|
json_object_set_boolean_member (vui, "video signal type present flag",
|
||
|
vui_parameters->video_signal_type_present_flag);
|
||
|
json_object_set_int_member (vui, "video_format",
|
||
|
vui_parameters->video_format);
|
||
|
json_object_set_boolean_member (vui, "video_full_range_flag",
|
||
|
vui_parameters->video_full_range_flag);
|
||
|
json_object_set_boolean_member (vui, "colour description present flag",
|
||
|
vui_parameters->colour_description_present_flag);
|
||
|
json_object_set_int_member (vui, "colour primaries",
|
||
|
vui_parameters->colour_primaries);
|
||
|
json_object_set_int_member (vui, "transfer characteristics",
|
||
|
vui_parameters->transfer_characteristics);
|
||
|
json_object_set_int_member (vui, "matrix coefficients",
|
||
|
vui_parameters->matrix_coefficients);
|
||
|
json_object_set_boolean_member (vui, "chroma loc info present flag",
|
||
|
vui_parameters->chroma_loc_info_present_flag);
|
||
|
json_object_set_int_member (vui, "chroma sample loc type top field",
|
||
|
vui_parameters->chroma_sample_loc_type_top_field);
|
||
|
json_object_set_int_member (vui, "chroma sample loc type bottom field",
|
||
|
vui_parameters->chroma_sample_loc_type_bottom_field);
|
||
|
|
||
|
json_object_set_boolean_member (vui, "timing_info_present_flag",
|
||
|
vui_parameters->timing_info_present_flag);
|
||
|
if (vui_parameters->timing_info_present_flag) {
|
||
|
json_object_set_int_member (vui, "num units in tick",
|
||
|
vui_parameters->num_units_in_tick);
|
||
|
json_object_set_int_member (vui, "time scale",
|
||
|
vui_parameters->time_scale);
|
||
|
json_object_set_boolean_member (vui, "fixed frame rate flag",
|
||
|
vui_parameters->fixed_frame_rate_flag);
|
||
|
}
|
||
|
|
||
|
json_object_set_boolean_member (vui, "nal hrd parameters present flag",
|
||
|
vui_parameters->nal_hrd_parameters_present_flag);
|
||
|
if (vui_parameters->nal_hrd_parameters_present_flag) {
|
||
|
JsonObject *nal_hrd_parameters = json_object_new ();
|
||
|
JsonArray *bit_rate_value_minus1, *cpb_size_value_minus1, *cbr_flag;
|
||
|
|
||
|
json_object_set_int_member (nal_hrd_parameters, "cpb cnt minus1",
|
||
|
vui_parameters->nal_hrd_parameters.cpb_cnt_minus1);
|
||
|
json_object_set_int_member (nal_hrd_parameters, "bit rate scale",
|
||
|
vui_parameters->nal_hrd_parameters.bit_rate_scale);
|
||
|
json_object_set_int_member (nal_hrd_parameters, "cpb size scale",
|
||
|
vui_parameters->nal_hrd_parameters.cpb_size_scale);
|
||
|
|
||
|
bit_rate_value_minus1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_int_element (bit_rate_value_minus1,
|
||
|
vui_parameters->nal_hrd_parameters.bit_rate_value_minus1[i]);
|
||
|
json_object_set_array_member (nal_hrd_parameters, "bit rate value minus1",
|
||
|
bit_rate_value_minus1);
|
||
|
|
||
|
cpb_size_value_minus1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_int_element (cpb_size_value_minus1,
|
||
|
vui_parameters->nal_hrd_parameters.cpb_size_value_minus1[i]);
|
||
|
json_object_set_array_member (nal_hrd_parameters, "cpb size value minus1",
|
||
|
cpb_size_value_minus1);
|
||
|
|
||
|
cbr_flag = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_boolean_element (cbr_flag,
|
||
|
vui_parameters->nal_hrd_parameters.cbr_flag[i]);
|
||
|
json_object_set_array_member (nal_hrd_parameters, "cbr flag", cbr_flag);
|
||
|
|
||
|
json_object_set_int_member (nal_hrd_parameters,
|
||
|
"initial cpb removal delay length minus1",
|
||
|
vui_parameters->
|
||
|
nal_hrd_parameters.initial_cpb_removal_delay_length_minus1);
|
||
|
json_object_set_int_member (nal_hrd_parameters,
|
||
|
"cpb removal delay length minus1",
|
||
|
vui_parameters->nal_hrd_parameters.cpb_removal_delay_length_minus1);
|
||
|
json_object_set_int_member (nal_hrd_parameters,
|
||
|
"dpb output delay length minus1",
|
||
|
vui_parameters->nal_hrd_parameters.dpb_output_delay_length_minus1);
|
||
|
json_object_set_int_member (nal_hrd_parameters, "time offset length",
|
||
|
vui_parameters->nal_hrd_parameters.time_offset_length);
|
||
|
|
||
|
json_object_set_object_member (vui, "nal hrd parameters",
|
||
|
nal_hrd_parameters);
|
||
|
}
|
||
|
|
||
|
json_object_set_boolean_member (vui, "vcl_hrd_parameters_present_flag",
|
||
|
vui_parameters->vcl_hrd_parameters_present_flag);
|
||
|
if (vui_parameters->vcl_hrd_parameters_present_flag) {
|
||
|
JsonObject *vcl_hrd_parameters = json_object_new ();
|
||
|
JsonArray *bit_rate_value_minus1, *cpb_size_value_minus1, *cbr_flag;
|
||
|
|
||
|
json_object_set_int_member (vcl_hrd_parameters, "cpb cnt minus1",
|
||
|
vui_parameters->vcl_hrd_parameters.cpb_cnt_minus1);
|
||
|
json_object_set_int_member (vcl_hrd_parameters, "bit rate scale",
|
||
|
vui_parameters->vcl_hrd_parameters.bit_rate_scale);
|
||
|
json_object_set_int_member (vcl_hrd_parameters, "cpb size scale",
|
||
|
vui_parameters->vcl_hrd_parameters.cpb_size_scale);
|
||
|
|
||
|
bit_rate_value_minus1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_int_element (bit_rate_value_minus1,
|
||
|
vui_parameters->vcl_hrd_parameters.bit_rate_value_minus1[i]);
|
||
|
json_object_set_array_member (vcl_hrd_parameters, "bit rate value minus1",
|
||
|
bit_rate_value_minus1);
|
||
|
|
||
|
cpb_size_value_minus1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_int_element (cpb_size_value_minus1,
|
||
|
vui_parameters->vcl_hrd_parameters.cpb_size_value_minus1[i]);
|
||
|
json_object_set_array_member (vcl_hrd_parameters, "cpb size value minus1",
|
||
|
cpb_size_value_minus1);
|
||
|
|
||
|
cbr_flag = json_array_new ();
|
||
|
for (i = 0; i < 32; i++)
|
||
|
json_array_add_boolean_element (cbr_flag,
|
||
|
vui_parameters->vcl_hrd_parameters.cbr_flag[i]);
|
||
|
json_object_set_array_member (vcl_hrd_parameters, "cbr flag", cbr_flag);
|
||
|
|
||
|
json_object_set_int_member (vcl_hrd_parameters,
|
||
|
"initial cpb removal delay length minus1",
|
||
|
vui_parameters->
|
||
|
vcl_hrd_parameters.initial_cpb_removal_delay_length_minus1);
|
||
|
json_object_set_int_member (vcl_hrd_parameters,
|
||
|
"cpb removal delay length minus1",
|
||
|
vui_parameters->vcl_hrd_parameters.cpb_removal_delay_length_minus1);
|
||
|
json_object_set_int_member (vcl_hrd_parameters,
|
||
|
"dpb output delay length minus1",
|
||
|
vui_parameters->vcl_hrd_parameters.dpb_output_delay_length_minus1);
|
||
|
json_object_set_int_member (vcl_hrd_parameters, "time offset length",
|
||
|
vui_parameters->vcl_hrd_parameters.time_offset_length);
|
||
|
|
||
|
json_object_set_object_member (vui, "vcl hrd parameters",
|
||
|
vcl_hrd_parameters);
|
||
|
}
|
||
|
|
||
|
json_object_set_boolean_member (vui, "low delay hrd flag",
|
||
|
vui_parameters->low_delay_hrd_flag);
|
||
|
json_object_set_boolean_member (vui, "pic struct present flag",
|
||
|
vui_parameters->pic_struct_present_flag);
|
||
|
|
||
|
json_object_set_boolean_member (vui, "bitstream restriction flag",
|
||
|
vui_parameters->bitstream_restriction_flag);
|
||
|
if (vui_parameters->bitstream_restriction_flag) {
|
||
|
json_object_set_boolean_member (vui,
|
||
|
"motion vectors over pic boundaries flag",
|
||
|
vui_parameters->motion_vectors_over_pic_boundaries_flag);
|
||
|
json_object_set_int_member (vui, "max bytes per pic denom",
|
||
|
vui_parameters->max_bytes_per_pic_denom);
|
||
|
json_object_set_int_member (vui, "max bits per mb denom",
|
||
|
vui_parameters->max_bits_per_mb_denom);
|
||
|
json_object_set_int_member (vui, "log2 max mv length horizontal",
|
||
|
vui_parameters->log2_max_mv_length_horizontal);
|
||
|
json_object_set_int_member (vui, "log2 max mv length vertical",
|
||
|
vui_parameters->log2_max_mv_length_vertical);
|
||
|
json_object_set_int_member (vui, "num reorder frames",
|
||
|
vui_parameters->num_reorder_frames);
|
||
|
json_object_set_int_member (vui, "max dec frame buffering",
|
||
|
vui_parameters->max_dec_frame_buffering);
|
||
|
}
|
||
|
json_object_set_object_member (sps, "VUI params", vui);
|
||
|
}
|
||
|
|
||
|
json_object_set_int_member (sps, "extension type", h264_sps.extension_type);
|
||
|
json_object_set_object_member (json, "sps", sps);
|
||
|
|
||
|
error:
|
||
|
gst_h264_sps_clear (&h264_sps);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_parse_pps (GstH2642json * self, GstH264NalUnit * nalu)
|
||
|
{
|
||
|
GstH264PPS h264_pps;
|
||
|
JsonObject *pps;
|
||
|
GstH264ParserResult pres;
|
||
|
JsonObject *json = self->json;
|
||
|
GstFlowReturn ret = GST_FLOW_OK;
|
||
|
JsonArray *scaling_lists_4x4, *scaling_lists_8x8;
|
||
|
gint i, j;
|
||
|
|
||
|
pres = gst_h264_parse_pps (self->parser, nalu, &h264_pps);
|
||
|
if (pres != GST_H264_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to parse PPS, result %d", pres);
|
||
|
return GST_FLOW_ERROR;
|
||
|
}
|
||
|
|
||
|
GST_LOG_OBJECT (self, "PPS parsed");
|
||
|
|
||
|
if (h264_pps.num_slice_groups_minus1 > 0) {
|
||
|
GST_FIXME_OBJECT (self, "FMO is not supported");
|
||
|
ret = GST_FLOW_ERROR;
|
||
|
goto error;
|
||
|
} else if (gst_h264_parser_update_pps (self->parser,
|
||
|
&h264_pps) != GST_H264_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to update PPS");
|
||
|
ret = GST_FLOW_ERROR;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
pps = json_object_new ();
|
||
|
|
||
|
json_object_set_boolean_member (pps, "entropy coding mode flag",
|
||
|
h264_pps.entropy_coding_mode_flag);
|
||
|
json_object_set_boolean_member (pps, "pic order present flag",
|
||
|
h264_pps.pic_order_present_flag);
|
||
|
|
||
|
json_object_set_int_member (pps, "num slice groups minus1",
|
||
|
h264_pps.num_slice_groups_minus1);
|
||
|
if (h264_pps.num_slice_groups_minus1 > 0) {
|
||
|
json_object_set_int_member (pps, "slice group map type",
|
||
|
h264_pps.slice_group_map_type);
|
||
|
switch (h264_pps.slice_group_map_type) {
|
||
|
case 0:
|
||
|
{
|
||
|
JsonArray *run_length_minus1 = json_array_new ();
|
||
|
|
||
|
for (i = 0; i < 8; i++)
|
||
|
json_array_add_int_element (run_length_minus1,
|
||
|
h264_pps.run_length_minus1[i]);
|
||
|
json_object_set_array_member (pps, "run lengthminus1",
|
||
|
run_length_minus1);
|
||
|
break;
|
||
|
}
|
||
|
case 2:
|
||
|
{
|
||
|
JsonArray *top_left = json_array_new ();
|
||
|
JsonArray *bottom_right = json_array_new ();
|
||
|
for (i = 0; i < 8; i++) {
|
||
|
json_array_add_int_element (top_left, h264_pps.top_left[i]);
|
||
|
json_array_add_int_element (bottom_right, h264_pps.bottom_right[i]);
|
||
|
}
|
||
|
json_object_set_array_member (pps, "top left", top_left);
|
||
|
json_object_set_array_member (pps, "bottom right", bottom_right);
|
||
|
break;
|
||
|
}
|
||
|
case 3:
|
||
|
case 4:
|
||
|
case 5:
|
||
|
{
|
||
|
json_object_set_boolean_member (pps,
|
||
|
"slice group change direction flag",
|
||
|
h264_pps.slice_group_change_direction_flag);
|
||
|
json_object_set_int_member (pps, "slice group change rate minus1",
|
||
|
h264_pps.slice_group_change_rate_minus1);
|
||
|
break;
|
||
|
}
|
||
|
case 6:
|
||
|
{
|
||
|
json_object_set_int_member (pps, "pic size in map units minus1",
|
||
|
h264_pps.pic_size_in_map_units_minus1);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
json_object_set_int_member (pps, "num ref idx l0 default active minus1",
|
||
|
h264_pps.num_ref_idx_l0_active_minus1);
|
||
|
json_object_set_int_member (pps, "num ref idx l1 default active minus1",
|
||
|
h264_pps.num_ref_idx_l1_active_minus1);
|
||
|
json_object_set_boolean_member (pps, "weighted pred flag",
|
||
|
h264_pps.weighted_pred_flag);
|
||
|
json_object_set_int_member (pps, "weighted bipred idc",
|
||
|
h264_pps.weighted_bipred_idc);
|
||
|
json_object_set_int_member (pps, "pic init qp minus26",
|
||
|
h264_pps.pic_init_qp_minus26);
|
||
|
json_object_set_int_member (pps, "pic init qs minus26",
|
||
|
h264_pps.pic_init_qs_minus26);
|
||
|
json_object_set_int_member (pps, "chroma qp index offset",
|
||
|
h264_pps.chroma_qp_index_offset);
|
||
|
json_object_set_boolean_member (pps, "deblocking filter control present flag",
|
||
|
h264_pps.deblocking_filter_control_present_flag);
|
||
|
json_object_set_boolean_member (pps, "constrained intra pred flag",
|
||
|
h264_pps.constrained_intra_pred_flag);
|
||
|
json_object_set_boolean_member (pps, "redundant pic cnt present flag",
|
||
|
h264_pps.redundant_pic_cnt_present_flag);
|
||
|
|
||
|
json_object_set_boolean_member (pps, "transform 8x8 mode flag",
|
||
|
h264_pps.transform_8x8_mode_flag);
|
||
|
|
||
|
json_object_set_int_member (pps, "second chroma qp index offset",
|
||
|
h264_pps.second_chroma_qp_index_offset);
|
||
|
json_object_set_boolean_member (pps, "pic scaling matrix present flag",
|
||
|
h264_pps.pic_scaling_matrix_present_flag);
|
||
|
|
||
|
scaling_lists_4x4 = json_array_new ();
|
||
|
for (i = 0; i < 6; i++)
|
||
|
for (j = 0; j < 16; j++)
|
||
|
json_array_add_int_element (scaling_lists_4x4,
|
||
|
h264_pps.scaling_lists_4x4[i][j]);
|
||
|
json_object_set_array_member (pps, "scaling lists 4x4", scaling_lists_4x4);
|
||
|
|
||
|
scaling_lists_8x8 = json_array_new ();
|
||
|
for (i = 0; i < 6; i++)
|
||
|
for (j = 0; j < 64; j++)
|
||
|
json_array_add_int_element (scaling_lists_8x8,
|
||
|
h264_pps.scaling_lists_8x8[i][j]);
|
||
|
json_object_set_array_member (pps, "scaling lists 8x8", scaling_lists_8x8);
|
||
|
|
||
|
json_object_set_object_member (json, "pps", pps);
|
||
|
|
||
|
error:
|
||
|
gst_h264_pps_clear (&h264_pps);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_parse_slice (GstH2642json * self, GstH264NalUnit * nalu)
|
||
|
{
|
||
|
GstH264SliceHdr slice;
|
||
|
GstH264PPS *pps;
|
||
|
GstH264SPS *sps;
|
||
|
GstH264ParserResult pres = GST_H264_PARSER_OK;
|
||
|
JsonObject *json = self->json;
|
||
|
JsonArray *delta_pic_order_cnt, *ref_pic_list_modification_l0,
|
||
|
*ref_pic_list_modification_l1, *luma_weight_l0, *luma_offset_l0;
|
||
|
JsonObject *hdr, *pred_weight_table;
|
||
|
gint i, j;
|
||
|
|
||
|
pres =
|
||
|
gst_h264_parser_parse_slice_hdr (self->parser, nalu, &slice, TRUE, TRUE);
|
||
|
|
||
|
if (pres != GST_H264_PARSER_OK) {
|
||
|
GST_ERROR_OBJECT (self, "Failed to parse slice header, ret %d", pres);
|
||
|
return GST_FLOW_ERROR;
|
||
|
}
|
||
|
|
||
|
pps = slice.pps;
|
||
|
sps = pps->sequence;
|
||
|
|
||
|
hdr = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (hdr, "first mb in slice",
|
||
|
slice.first_mb_in_slice);
|
||
|
json_object_set_int_member (hdr, "type", slice.type);
|
||
|
|
||
|
if (sps->separate_colour_plane_flag)
|
||
|
json_object_set_int_member (hdr, "colour plane id", slice.colour_plane_id);
|
||
|
|
||
|
json_object_set_int_member (hdr, "frame num", slice.frame_num);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "field pic flag", slice.field_pic_flag);
|
||
|
json_object_set_boolean_member (hdr, "bottom field flag",
|
||
|
slice.bottom_field_flag);
|
||
|
|
||
|
if (nalu->type == GST_H264_NAL_SLICE_IDR)
|
||
|
json_object_set_int_member (hdr, "idr pic id", slice.idr_pic_id);
|
||
|
|
||
|
if (sps->pic_order_cnt_type == 0)
|
||
|
json_object_set_int_member (hdr, "pic order cnt lsb",
|
||
|
slice.pic_order_cnt_lsb);
|
||
|
|
||
|
if (pps->pic_order_present_flag && !slice.field_pic_flag)
|
||
|
json_object_set_int_member (hdr, "delta pic order cnt bottom",
|
||
|
slice.delta_pic_order_cnt_bottom);
|
||
|
|
||
|
delta_pic_order_cnt = json_array_new ();
|
||
|
for (i = 0; i < 2; i++)
|
||
|
json_array_add_int_element (delta_pic_order_cnt,
|
||
|
slice.delta_pic_order_cnt[i]);
|
||
|
json_object_set_array_member (hdr, "delta pic order cnt",
|
||
|
delta_pic_order_cnt);
|
||
|
|
||
|
json_object_set_int_member (hdr, "redundant pic cnt",
|
||
|
slice.redundant_pic_cnt);
|
||
|
|
||
|
if (GST_H264_IS_B_SLICE (&slice))
|
||
|
json_object_set_boolean_member (hdr, "direct spatial mv pred flag",
|
||
|
slice.direct_spatial_mv_pred_flag);
|
||
|
|
||
|
json_object_set_int_member (hdr, "num ref idx l0 active minus1",
|
||
|
slice.num_ref_idx_l0_active_minus1);
|
||
|
json_object_set_int_member (hdr, "num ref idx l1 active minus1",
|
||
|
slice.num_ref_idx_l1_active_minus1);
|
||
|
|
||
|
json_object_set_int_member (hdr, "ref pic list modification flag l0",
|
||
|
slice.ref_pic_list_modification_flag_l0);
|
||
|
json_object_set_int_member (hdr, "n ref pic list modification l0",
|
||
|
slice.n_ref_pic_list_modification_l0);
|
||
|
ref_pic_list_modification_l0 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
JsonObject *modification = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (modification, "modification of pic nums idc",
|
||
|
slice.ref_pic_list_modification_l0[i].modification_of_pic_nums_idc);
|
||
|
switch (slice.ref_pic_list_modification_l0[i].modification_of_pic_nums_idc) {
|
||
|
case 0:
|
||
|
case 1:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "abs diff pic num minus1",
|
||
|
slice.ref_pic_list_modification_l0[i].
|
||
|
value.abs_diff_pic_num_minus1);
|
||
|
break;
|
||
|
}
|
||
|
case 2:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "long term pic num",
|
||
|
slice.ref_pic_list_modification_l0[i].value.long_term_pic_num);
|
||
|
break;
|
||
|
}
|
||
|
case 4:
|
||
|
case 5:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "abs diff view idx minus1",
|
||
|
slice.ref_pic_list_modification_l0[i].
|
||
|
value.abs_diff_view_idx_minus1);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
json_array_add_object_element (ref_pic_list_modification_l0, modification);
|
||
|
}
|
||
|
json_object_set_array_member (hdr, "ref pic list modification l0",
|
||
|
ref_pic_list_modification_l0);
|
||
|
|
||
|
json_object_set_int_member (hdr, "ref pic list modification flag l0",
|
||
|
slice.ref_pic_list_modification_flag_l1);
|
||
|
json_object_set_int_member (hdr, "n ref pic list modification l0",
|
||
|
slice.n_ref_pic_list_modification_l1);
|
||
|
ref_pic_list_modification_l1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
JsonObject *modification = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (modification, "modification of pic nums idc",
|
||
|
slice.ref_pic_list_modification_l1[i].modification_of_pic_nums_idc);
|
||
|
switch (slice.ref_pic_list_modification_l1[i].modification_of_pic_nums_idc) {
|
||
|
case 0:
|
||
|
case 1:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "abs diff pic num minus1",
|
||
|
slice.ref_pic_list_modification_l1[i].
|
||
|
value.abs_diff_pic_num_minus1);
|
||
|
break;
|
||
|
}
|
||
|
case 2:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "long term pic num",
|
||
|
slice.ref_pic_list_modification_l1[i].value.long_term_pic_num);
|
||
|
break;
|
||
|
}
|
||
|
case 4:
|
||
|
case 5:
|
||
|
{
|
||
|
json_object_set_int_member (modification, "abs diff view idx minus1",
|
||
|
slice.ref_pic_list_modification_l1[i].
|
||
|
value.abs_diff_view_idx_minus1);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
json_array_add_object_element (ref_pic_list_modification_l1, modification);
|
||
|
}
|
||
|
json_object_set_array_member (hdr, "ref pic list modification l1",
|
||
|
ref_pic_list_modification_l1);
|
||
|
|
||
|
pred_weight_table = json_object_new ();
|
||
|
json_object_set_int_member (pred_weight_table, "luma log2 weight denom",
|
||
|
slice.pred_weight_table.luma_log2_weight_denom);
|
||
|
json_object_set_int_member (pred_weight_table, "chroma log2 weight denom",
|
||
|
slice.pred_weight_table.chroma_log2_weight_denom);
|
||
|
|
||
|
luma_weight_l0 = json_array_new ();
|
||
|
luma_offset_l0 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
json_array_add_int_element (luma_weight_l0,
|
||
|
slice.pred_weight_table.luma_weight_l0[i]);
|
||
|
json_array_add_int_element (luma_offset_l0,
|
||
|
slice.pred_weight_table.luma_offset_l0[i]);
|
||
|
}
|
||
|
json_object_set_array_member (pred_weight_table, "luma weight l0",
|
||
|
luma_weight_l0);
|
||
|
json_object_set_array_member (pred_weight_table, "luma offset l0",
|
||
|
luma_offset_l0);
|
||
|
|
||
|
if (sps->chroma_array_type != 0) {
|
||
|
JsonArray *chroma_weight_l0 = json_array_new ();
|
||
|
JsonArray *chroma_offset_l0 = json_array_new ();
|
||
|
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
for (j = 0; j < 2; j++) {
|
||
|
json_array_add_int_element (chroma_weight_l0,
|
||
|
slice.pred_weight_table.chroma_weight_l0[i][j]);
|
||
|
json_array_add_int_element (chroma_offset_l0,
|
||
|
slice.pred_weight_table.chroma_offset_l0[i][j]);
|
||
|
}
|
||
|
}
|
||
|
json_object_set_array_member (pred_weight_table, "chroma weight l0",
|
||
|
chroma_weight_l0);
|
||
|
json_object_set_array_member (pred_weight_table, "chroma offset l0",
|
||
|
chroma_offset_l0);
|
||
|
}
|
||
|
|
||
|
if (GST_H264_IS_B_SLICE (&slice)) {
|
||
|
JsonArray *luma_weight_l1 = json_array_new ();
|
||
|
JsonArray *luma_offset_l1 = json_array_new ();
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
json_array_add_int_element (luma_weight_l1,
|
||
|
slice.pred_weight_table.luma_weight_l1[i]);
|
||
|
json_array_add_int_element (luma_offset_l1,
|
||
|
slice.pred_weight_table.luma_offset_l1[i]);
|
||
|
}
|
||
|
json_object_set_array_member (pred_weight_table, "luma weight l1",
|
||
|
luma_weight_l1);
|
||
|
json_object_set_array_member (pred_weight_table, "luma offset l1",
|
||
|
luma_offset_l1);
|
||
|
|
||
|
if (sps->chroma_array_type != 0) {
|
||
|
JsonArray *chroma_weight_l1 = json_array_new ();
|
||
|
JsonArray *chroma_offset_l1 = json_array_new ();
|
||
|
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
for (j = 0; j < 2; j++) {
|
||
|
json_array_add_int_element (chroma_weight_l1,
|
||
|
slice.pred_weight_table.chroma_weight_l1[i][j]);
|
||
|
json_array_add_int_element (chroma_offset_l1,
|
||
|
slice.pred_weight_table.chroma_offset_l1[i][j]);
|
||
|
}
|
||
|
}
|
||
|
json_object_set_array_member (pred_weight_table, "chroma weight l1",
|
||
|
chroma_weight_l1);
|
||
|
json_object_set_array_member (pred_weight_table, "chroma offset l1",
|
||
|
chroma_offset_l1);
|
||
|
}
|
||
|
}
|
||
|
json_object_set_object_member (hdr, "pred weight table", pred_weight_table);
|
||
|
|
||
|
if (nalu->ref_idc != 0) {
|
||
|
JsonObject *dec_ref_pic_marking = json_object_new ();
|
||
|
JsonArray *ref_pic_marking = json_array_new ();
|
||
|
|
||
|
if (nalu->idr_pic_flag) {
|
||
|
json_object_set_boolean_member (dec_ref_pic_marking,
|
||
|
"no output of prior pics flag",
|
||
|
slice.dec_ref_pic_marking.no_output_of_prior_pics_flag);
|
||
|
json_object_set_boolean_member (dec_ref_pic_marking,
|
||
|
"long term reference flag",
|
||
|
slice.dec_ref_pic_marking.long_term_reference_flag);
|
||
|
}
|
||
|
json_object_set_boolean_member (dec_ref_pic_marking,
|
||
|
"adaptive ref pic marking mode flag",
|
||
|
slice.dec_ref_pic_marking.adaptive_ref_pic_marking_mode_flag);
|
||
|
|
||
|
for (i = 0; i < 10; i++) {
|
||
|
GstH264RefPicMarking *m = &slice.dec_ref_pic_marking.ref_pic_marking[i];
|
||
|
JsonObject *marking = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (marking,
|
||
|
"memory management control operation",
|
||
|
m->memory_management_control_operation);
|
||
|
json_object_set_int_member (marking, "difference of pic nums minus1",
|
||
|
m->difference_of_pic_nums_minus1);
|
||
|
json_object_set_int_member (marking, "long term pic num",
|
||
|
m->long_term_pic_num);
|
||
|
json_object_set_int_member (marking, "long term frame idx",
|
||
|
m->long_term_frame_idx);
|
||
|
json_object_set_int_member (marking, "max long term frame idx plus1",
|
||
|
m->max_long_term_frame_idx_plus1);
|
||
|
json_array_add_object_element (ref_pic_marking, marking);
|
||
|
}
|
||
|
json_object_set_array_member (dec_ref_pic_marking, "ref pic marking",
|
||
|
ref_pic_marking);
|
||
|
|
||
|
json_object_set_int_member (dec_ref_pic_marking, "n ref pic marking",
|
||
|
slice.dec_ref_pic_marking.n_ref_pic_marking);
|
||
|
json_object_set_int_member (dec_ref_pic_marking, "bit size",
|
||
|
slice.dec_ref_pic_marking.bit_size);
|
||
|
|
||
|
json_object_set_object_member (hdr, "dec ref pic marking",
|
||
|
dec_ref_pic_marking);
|
||
|
}
|
||
|
|
||
|
json_object_set_int_member (hdr, "cabac init idc", slice.cabac_init_idc);
|
||
|
json_object_set_int_member (hdr, "slice qp delta", slice.slice_qp_delta);
|
||
|
json_object_set_int_member (hdr, "slice qs delta", slice.slice_qs_delta);
|
||
|
json_object_set_int_member (hdr, "disable deblocking filter idc",
|
||
|
slice.disable_deblocking_filter_idc);
|
||
|
json_object_set_int_member (hdr, "slice alpha c0 offset div2",
|
||
|
slice.slice_alpha_c0_offset_div2);
|
||
|
json_object_set_int_member (hdr, "slice beta offset div2",
|
||
|
slice.slice_beta_offset_div2);
|
||
|
json_object_set_int_member (hdr, "slice group change cycle",
|
||
|
slice.slice_group_change_cycle);
|
||
|
|
||
|
json_object_set_object_member (json, "slice header", hdr);
|
||
|
|
||
|
return GST_FLOW_OK;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_decode_nal (GstH2642json * self, GstH264NalUnit * nalu)
|
||
|
{
|
||
|
GstFlowReturn ret = GST_FLOW_OK;
|
||
|
|
||
|
GST_LOG_OBJECT (self, "Parsed nal type: %d, offset %d, size %d",
|
||
|
nalu->type, nalu->offset, nalu->size);
|
||
|
|
||
|
switch (nalu->type) {
|
||
|
case GST_H264_NAL_SPS:
|
||
|
ret = gst_h264_2_json_parse_sps (self, nalu);
|
||
|
break;
|
||
|
case GST_H264_NAL_PPS:
|
||
|
ret = gst_h264_2_json_parse_pps (self, nalu);
|
||
|
break;
|
||
|
case GST_H264_NAL_SLICE:
|
||
|
case GST_H264_NAL_SLICE_DPA:
|
||
|
case GST_H264_NAL_SLICE_DPB:
|
||
|
case GST_H264_NAL_SLICE_DPC:
|
||
|
case GST_H264_NAL_SLICE_IDR:
|
||
|
case GST_H264_NAL_SLICE_EXT:
|
||
|
ret = gst_h264_2_json_parse_slice (self, nalu);
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_chain (GstPad * sinkpad, GstObject * object, GstBuffer * in_buf)
|
||
|
{
|
||
|
GstH2642json *self = GST_H264_2_JSON (object);
|
||
|
JsonObject *json = self->json;
|
||
|
GstBuffer *out_buf;
|
||
|
gchar *json_string;
|
||
|
guint length;
|
||
|
GstH264NalUnit nalu;
|
||
|
GstH264ParserResult pres;
|
||
|
GstMapInfo in_map, out_map;
|
||
|
GstFlowReturn ret = GST_FLOW_OK;
|
||
|
|
||
|
if (!gst_buffer_map (in_buf, &in_map, GST_MAP_READ)) {
|
||
|
GST_ERROR_OBJECT (self, "Cannot map buffer");
|
||
|
return GST_FLOW_ERROR;
|
||
|
}
|
||
|
|
||
|
if (self->use_avc) {
|
||
|
pres = gst_h264_parser_identify_nalu_avc (self->parser,
|
||
|
in_map.data, 0, in_map.size, self->nal_length_size, &nalu);
|
||
|
|
||
|
while (pres == GST_H264_PARSER_OK && ret == GST_FLOW_OK) {
|
||
|
ret = gst_h264_2_json_decode_nal (self, &nalu);
|
||
|
|
||
|
pres = gst_h264_parser_identify_nalu_avc (self->parser,
|
||
|
in_map.data, nalu.offset + nalu.size, in_map.size,
|
||
|
self->nal_length_size, &nalu);
|
||
|
}
|
||
|
} else {
|
||
|
pres = gst_h264_parser_identify_nalu (self->parser,
|
||
|
in_map.data, 0, in_map.size, &nalu);
|
||
|
|
||
|
if (pres == GST_H264_PARSER_NO_NAL_END)
|
||
|
pres = GST_H264_PARSER_OK;
|
||
|
|
||
|
while (pres == GST_H264_PARSER_OK && ret == GST_FLOW_OK) {
|
||
|
ret = gst_h264_2_json_decode_nal (self, &nalu);
|
||
|
|
||
|
pres = gst_h264_parser_identify_nalu (self->parser,
|
||
|
in_map.data, nalu.offset + nalu.size, in_map.size, &nalu);
|
||
|
|
||
|
if (pres == GST_H264_PARSER_NO_NAL_END)
|
||
|
pres = GST_H264_PARSER_OK;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
json_string = get_string_from_json_object (json);
|
||
|
length = strlen (json_string);
|
||
|
out_buf = gst_buffer_new_allocate (NULL, length, NULL);
|
||
|
gst_buffer_map (out_buf, &out_map, GST_MAP_WRITE);
|
||
|
if (length)
|
||
|
memcpy (&out_map.data[0], json_string, length);
|
||
|
gst_buffer_unmap (out_buf, &out_map);
|
||
|
|
||
|
g_free (json_string);
|
||
|
|
||
|
gst_buffer_copy_into (out_buf, in_buf,
|
||
|
GST_BUFFER_COPY_FLAGS | GST_BUFFER_COPY_TIMESTAMPS |
|
||
|
GST_BUFFER_COPY_METADATA, 0, -1);
|
||
|
ret = gst_pad_push (self->srcpad, out_buf);
|
||
|
|
||
|
gst_buffer_unmap (in_buf, &in_map);
|
||
|
gst_buffer_unref (in_buf);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_h264_2_json_parse_codec_data (GstH2642json * self, const guint8 * data,
|
||
|
gsize size)
|
||
|
{
|
||
|
GstH264DecoderConfigRecord *config = NULL;
|
||
|
GstFlowReturn ret = GST_FLOW_OK;
|
||
|
GstH264NalUnit *nalu;
|
||
|
guint i;
|
||
|
|
||
|
if (gst_h264_parser_parse_decoder_config_record (self->parser, data, size,
|
||
|
&config) != GST_H264_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to parse codec-data");
|
||
|
return GST_FLOW_ERROR;
|
||
|
}
|
||
|
|
||
|
self->nal_length_size = config->length_size_minus_one + 1;
|
||
|
for (i = 0; i < config->sps->len; i++) {
|
||
|
nalu = &g_array_index (config->sps, GstH264NalUnit, i);
|
||
|
|
||
|
/* TODO: handle subset sps for SVC/MVC. That would need to be stored in
|
||
|
* separate array instead of putting SPS/subset-SPS into a single array */
|
||
|
if (nalu->type != GST_H264_NAL_SPS)
|
||
|
continue;
|
||
|
|
||
|
ret = gst_h264_2_json_parse_sps (self, nalu);
|
||
|
if (ret != GST_FLOW_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to parse SPS");
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < config->pps->len; i++) {
|
||
|
nalu = &g_array_index (config->pps, GstH264NalUnit, i);
|
||
|
if (nalu->type != GST_H264_NAL_PPS)
|
||
|
continue;
|
||
|
|
||
|
ret = gst_h264_2_json_parse_pps (self, nalu);
|
||
|
if (ret != GST_FLOW_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Failed to parse PPS");
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
out:
|
||
|
gst_h264_decoder_config_record_free (config);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_h264_2_json_get_codec_data (GstH2642json * self, GstCaps * caps)
|
||
|
{
|
||
|
if (caps && gst_caps_get_size (caps) > 0) {
|
||
|
GstStructure *s = gst_caps_get_structure (caps, 0);
|
||
|
|
||
|
if (gst_structure_has_field (s, "codec_data")) {
|
||
|
GST_WARNING_OBJECT (self, "get codec-data");
|
||
|
|
||
|
const GValue *h = gst_structure_get_value (s, "codec_data");
|
||
|
GstBuffer *codec_data = gst_value_get_buffer (h);
|
||
|
GstMapInfo map;
|
||
|
|
||
|
gst_buffer_map (codec_data, &map, GST_MAP_READ);
|
||
|
if (gst_h264_2_json_parse_codec_data (self, map.data,
|
||
|
map.size) != GST_FLOW_OK) {
|
||
|
/* keep going without error.
|
||
|
* Probably inband SPS/PPS might be valid data */
|
||
|
GST_WARNING_OBJECT (self, "Failed to handle codec data");
|
||
|
}
|
||
|
gst_buffer_unmap (codec_data, &map);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_h264_2_json_use_avc (GstH2642json * self, GstCaps * caps)
|
||
|
{
|
||
|
if (caps && gst_caps_get_size (caps) > 0) {
|
||
|
GstStructure *s = gst_caps_get_structure (caps, 0);
|
||
|
const gchar *str_stream = NULL;
|
||
|
|
||
|
str_stream = gst_structure_get_string (s, "stream-format");
|
||
|
|
||
|
self->use_avc = FALSE;
|
||
|
if (str_stream && (g_strcmp0 (str_stream, "avc") == 0
|
||
|
|| g_strcmp0 (str_stream, "avc3") == 0)) {
|
||
|
self->use_avc = TRUE;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
gst_h264_2_json_set_caps (GstH2642json * self, GstCaps * caps)
|
||
|
{
|
||
|
GstCaps *src_caps =
|
||
|
gst_caps_new_simple ("text/x-json", "format", G_TYPE_STRING, "h264",
|
||
|
NULL);
|
||
|
GstEvent *event;
|
||
|
|
||
|
event = gst_event_new_caps (src_caps);
|
||
|
gst_caps_unref (src_caps);
|
||
|
|
||
|
gst_h264_2_json_use_avc (self, caps);
|
||
|
|
||
|
gst_h264_2_json_get_codec_data (self, caps);
|
||
|
|
||
|
return gst_pad_push_event (self->srcpad, event);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
gst_h264_2_json_sink_event (GstPad * pad, GstObject * parent, GstEvent * event)
|
||
|
{
|
||
|
GstH2642json *self = GST_H264_2_JSON (parent);
|
||
|
gboolean res = TRUE;
|
||
|
|
||
|
switch (GST_EVENT_TYPE (event)) {
|
||
|
case GST_EVENT_CAPS:
|
||
|
{
|
||
|
GstCaps *caps;
|
||
|
|
||
|
gst_event_parse_caps (event, &caps);
|
||
|
res = gst_h264_2_json_set_caps (self, caps);
|
||
|
gst_event_unref (event);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
res = gst_pad_event_default (pad, parent, event);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_h264_2_json_class_init (GstH2642jsonClass * klass)
|
||
|
{
|
||
|
GObjectClass *gobject_class;
|
||
|
GstElementClass *gstelement_class;
|
||
|
|
||
|
gobject_class = (GObjectClass *) klass;
|
||
|
gstelement_class = (GstElementClass *) klass;
|
||
|
|
||
|
gobject_class->finalize = gst_h264_2_json_finalize;
|
||
|
|
||
|
gst_element_class_add_static_pad_template (gstelement_class, &src_factory);
|
||
|
gst_element_class_add_static_pad_template (gstelement_class, &sink_factory);
|
||
|
|
||
|
gst_element_class_set_static_metadata (gstelement_class, "H2642json",
|
||
|
"Transform",
|
||
|
"H264 to json element",
|
||
|
"Benjamin Gaignard <benjamin.gaignard@collabora.com>");
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_h264_2_json_init (GstH2642json * self)
|
||
|
{
|
||
|
self->sinkpad = gst_pad_new_from_static_template (&sink_factory, "sink");
|
||
|
gst_pad_set_chain_function (self->sinkpad, gst_h264_2_json_chain);
|
||
|
gst_element_add_pad (GST_ELEMENT (self), self->sinkpad);
|
||
|
gst_pad_set_event_function (self->sinkpad,
|
||
|
GST_DEBUG_FUNCPTR (gst_h264_2_json_sink_event));
|
||
|
|
||
|
self->srcpad = gst_pad_new_from_static_template (&src_factory, "src");
|
||
|
gst_element_add_pad (GST_ELEMENT (self), self->srcpad);
|
||
|
|
||
|
self->json = json_object_new ();
|
||
|
|
||
|
self->parser = gst_h264_nal_parser_new ();
|
||
|
self->use_avc = FALSE;
|
||
|
self->nal_length_size = 4;
|
||
|
}
|