mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-03 22:18:50 +00:00
1037 lines
42 KiB
C
1037 lines
42 KiB
C
|
/*
|
||
|
* gstav12json.c - AV1 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-av12json
|
||
|
* @title: av12json
|
||
|
*
|
||
|
* Convert AV1 bitstream parameters to JSON formated text.
|
||
|
*
|
||
|
* ## Example launch line
|
||
|
* ```
|
||
|
* gst-launch-1.0 filesrc location=/path/to/av1/file ! parsebin ! av12json ! 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 "gstav12json.h"
|
||
|
|
||
|
GST_DEBUG_CATEGORY (gst_av1_2_json_debug);
|
||
|
#define GST_CAT_DEFAULT gst_av1_2_json_debug
|
||
|
|
||
|
struct _GstAV12json
|
||
|
{
|
||
|
GstElement parent;
|
||
|
|
||
|
GstPad *sinkpad, *srcpad;
|
||
|
GstAV1Parser *parser;
|
||
|
gboolean use_annex_b;
|
||
|
|
||
|
JsonObject *json;
|
||
|
};
|
||
|
|
||
|
static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
|
||
|
GST_PAD_SINK,
|
||
|
GST_PAD_ALWAYS,
|
||
|
GST_STATIC_CAPS ("video/x-av1")
|
||
|
);
|
||
|
|
||
|
static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
|
||
|
GST_PAD_SRC,
|
||
|
GST_PAD_ALWAYS,
|
||
|
GST_STATIC_CAPS ("text/x-json,format=av1"));
|
||
|
|
||
|
G_DEFINE_TYPE_WITH_CODE (GstAV12json, gst_av1_2_json,
|
||
|
GST_TYPE_ELEMENT,
|
||
|
GST_DEBUG_CATEGORY_INIT (gst_av1_2_json_debug, "av12json", 0,
|
||
|
"AV1 to json"));
|
||
|
|
||
|
static void
|
||
|
gst_av1_2_json_finalize (GObject * object)
|
||
|
{
|
||
|
GstAV12json *self = GST_AV1_2_JSON (object);
|
||
|
|
||
|
gst_av1_parser_free (self->parser);
|
||
|
json_object_unref (self->json);
|
||
|
}
|
||
|
|
||
|
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 void
|
||
|
gst_av1_2_json_sequence_header (GstAV12json * self,
|
||
|
GstAV1SequenceHeaderOBU * seq_header)
|
||
|
{
|
||
|
JsonObject *json = self->json;
|
||
|
JsonObject *hdr = json_object_new ();
|
||
|
JsonArray *operating_points;
|
||
|
JsonObject *color_config;
|
||
|
guint i;
|
||
|
|
||
|
json_object_set_int_member (hdr, "seq profile", seq_header->seq_profile);
|
||
|
json_object_set_boolean_member (hdr, "still picture",
|
||
|
seq_header->still_picture);
|
||
|
json_object_set_int_member (hdr, "reduced still picture header",
|
||
|
seq_header->reduced_still_picture_header);
|
||
|
json_object_set_int_member (hdr, "frame width bits minus 1",
|
||
|
seq_header->frame_width_bits_minus_1);
|
||
|
json_object_set_int_member (hdr, "frame height bits minus 1",
|
||
|
seq_header->frame_height_bits_minus_1);
|
||
|
json_object_set_int_member (hdr, "max frame width minus 1",
|
||
|
seq_header->max_frame_width_minus_1);
|
||
|
json_object_set_int_member (hdr, "max frame height minus 1",
|
||
|
seq_header->max_frame_height_minus_1);
|
||
|
json_object_set_boolean_member (hdr, "frame id numbers present flag",
|
||
|
seq_header->frame_id_numbers_present_flag);
|
||
|
json_object_set_int_member (hdr, "delta frame id length minus 2",
|
||
|
seq_header->delta_frame_id_length_minus_2);
|
||
|
json_object_set_int_member (hdr, "additional frame id length minus 1",
|
||
|
seq_header->additional_frame_id_length_minus_1);
|
||
|
json_object_set_boolean_member (hdr, "use 128x128 superblock",
|
||
|
seq_header->use_128x128_superblock);
|
||
|
json_object_set_boolean_member (hdr, "enable filter intra",
|
||
|
seq_header->enable_filter_intra);
|
||
|
json_object_set_boolean_member (hdr, "enable intra edge filter",
|
||
|
seq_header->enable_intra_edge_filter);
|
||
|
json_object_set_boolean_member (hdr, "enable interintra compound",
|
||
|
seq_header->enable_interintra_compound);
|
||
|
json_object_set_boolean_member (hdr, "enable masked compound",
|
||
|
seq_header->enable_masked_compound);
|
||
|
json_object_set_boolean_member (hdr, "enable warped motion",
|
||
|
seq_header->enable_warped_motion);
|
||
|
json_object_set_boolean_member (hdr, "enable order hint",
|
||
|
seq_header->enable_order_hint);
|
||
|
json_object_set_boolean_member (hdr, "enable dual filter",
|
||
|
seq_header->enable_dual_filter);
|
||
|
json_object_set_boolean_member (hdr, "enable jnt comp",
|
||
|
seq_header->enable_jnt_comp);
|
||
|
json_object_set_boolean_member (hdr, "enable ref frame mvs",
|
||
|
seq_header->enable_ref_frame_mvs);
|
||
|
json_object_set_boolean_member (hdr, "seq choose screen content tools",
|
||
|
seq_header->seq_choose_screen_content_tools);
|
||
|
json_object_set_int_member (hdr, "seq force screen content tools",
|
||
|
seq_header->seq_force_screen_content_tools);
|
||
|
json_object_set_boolean_member (hdr, "seq choose integer mv",
|
||
|
seq_header->seq_choose_integer_mv);
|
||
|
json_object_set_int_member (hdr, "seq force integer mv",
|
||
|
seq_header->seq_force_integer_mv);
|
||
|
json_object_set_int_member (hdr, "order hint bits minus 1",
|
||
|
seq_header->order_hint_bits_minus_1);
|
||
|
json_object_set_boolean_member (hdr, "enable superres",
|
||
|
seq_header->enable_superres);
|
||
|
json_object_set_boolean_member (hdr, "enable cdef", seq_header->enable_cdef);
|
||
|
json_object_set_boolean_member (hdr, "enable restoration",
|
||
|
seq_header->enable_restoration);
|
||
|
json_object_set_int_member (hdr, "film grain params present",
|
||
|
seq_header->film_grain_params_present);
|
||
|
json_object_set_int_member (hdr, "operating points cnt minus 1",
|
||
|
seq_header->operating_points_cnt_minus_1);
|
||
|
|
||
|
operating_points = json_array_new ();
|
||
|
for (i = 0; i < seq_header->operating_points_cnt_minus_1 + 1; i++) {
|
||
|
JsonObject *operating_point = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (operating_point, "seq level idx",
|
||
|
seq_header->operating_points[i].seq_level_idx);
|
||
|
json_object_set_int_member (operating_point, "seq tier",
|
||
|
seq_header->operating_points[i].seq_tier);
|
||
|
json_object_set_int_member (operating_point, "idc",
|
||
|
seq_header->operating_points[i].idc);
|
||
|
json_object_set_boolean_member (operating_point,
|
||
|
"decoder model present for this op",
|
||
|
seq_header->operating_points[i].decoder_model_present_for_this_op);
|
||
|
json_object_set_int_member (operating_point, "decoder buffer delay",
|
||
|
seq_header->operating_points[i].decoder_buffer_delay);
|
||
|
json_object_set_int_member (operating_point, "encoder buffer delay",
|
||
|
seq_header->operating_points[i].encoder_buffer_delay);
|
||
|
json_object_set_boolean_member (operating_point, "low delay mode flag",
|
||
|
seq_header->operating_points[i].low_delay_mode_flag);
|
||
|
json_object_set_boolean_member (operating_point,
|
||
|
"initial display delay present for this op",
|
||
|
seq_header->
|
||
|
operating_points[i].initial_display_delay_present_for_this_op);
|
||
|
json_object_set_int_member (operating_point,
|
||
|
"initial display delay minus 1",
|
||
|
seq_header->operating_points[i].initial_display_delay_minus_1);
|
||
|
json_array_add_object_element (operating_points, operating_point);
|
||
|
}
|
||
|
json_object_set_array_member (hdr, "operating points", operating_points);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "decoder model info present flag",
|
||
|
seq_header->decoder_model_info_present_flag);
|
||
|
if (seq_header->decoder_model_info_present_flag) {
|
||
|
JsonObject *decoder_model_info = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (decoder_model_info,
|
||
|
"buffer delay length minus 1",
|
||
|
seq_header->decoder_model_info.buffer_delay_length_minus_1);
|
||
|
json_object_set_int_member (decoder_model_info,
|
||
|
"num units in decoding tick",
|
||
|
seq_header->decoder_model_info.num_units_in_decoding_tick);
|
||
|
json_object_set_int_member (decoder_model_info,
|
||
|
"buffer removal time length minus 1",
|
||
|
seq_header->decoder_model_info.buffer_removal_time_length_minus_1);
|
||
|
json_object_set_int_member (decoder_model_info,
|
||
|
"frame presentation time length minus 1",
|
||
|
seq_header->decoder_model_info.frame_presentation_time_length_minus_1);
|
||
|
json_object_set_object_member (hdr, "decoder model info",
|
||
|
decoder_model_info);
|
||
|
}
|
||
|
|
||
|
json_object_set_int_member (hdr, "initial display delay present flag",
|
||
|
seq_header->initial_display_delay_present_flag);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "timing info present flag",
|
||
|
seq_header->timing_info_present_flag);
|
||
|
if (seq_header->timing_info_present_flag) {
|
||
|
JsonObject *timing_info = json_object_new ();
|
||
|
|
||
|
json_object_set_int_member (timing_info, "num units in display tick",
|
||
|
seq_header->timing_info.num_units_in_display_tick);
|
||
|
json_object_set_int_member (timing_info, "time scale",
|
||
|
seq_header->timing_info.time_scale);
|
||
|
json_object_set_boolean_member (timing_info, "equal picture interval",
|
||
|
seq_header->timing_info.equal_picture_interval);
|
||
|
json_object_set_int_member (timing_info, "num ticks per picture minus 1",
|
||
|
seq_header->timing_info.num_ticks_per_picture_minus_1);
|
||
|
json_object_set_object_member (hdr, "timing info", timing_info);
|
||
|
}
|
||
|
|
||
|
color_config = json_object_new ();
|
||
|
json_object_set_boolean_member (color_config, "high bitdepth",
|
||
|
seq_header->color_config.high_bitdepth);
|
||
|
json_object_set_boolean_member (color_config, "twelve bit",
|
||
|
seq_header->color_config.twelve_bit);
|
||
|
json_object_set_boolean_member (color_config, "mono chrome",
|
||
|
seq_header->color_config.mono_chrome);
|
||
|
json_object_set_boolean_member (color_config,
|
||
|
"color description present flag",
|
||
|
seq_header->color_config.color_description_present_flag);
|
||
|
json_object_set_int_member (color_config, "color primaries",
|
||
|
seq_header->color_config.color_primaries);
|
||
|
json_object_set_int_member (color_config, "transfer characteristics",
|
||
|
seq_header->color_config.transfer_characteristics);
|
||
|
json_object_set_int_member (color_config, "matrix coefficients",
|
||
|
seq_header->color_config.matrix_coefficients);
|
||
|
json_object_set_boolean_member (color_config, "color range",
|
||
|
seq_header->color_config.color_range);
|
||
|
json_object_set_int_member (color_config, "subsampling x",
|
||
|
seq_header->color_config.subsampling_x);
|
||
|
json_object_set_int_member (color_config, "subsampling y",
|
||
|
seq_header->color_config.subsampling_y);
|
||
|
json_object_set_int_member (color_config, "chroma sample position",
|
||
|
seq_header->color_config.chroma_sample_position);
|
||
|
json_object_set_boolean_member (color_config, "separate uv delta q",
|
||
|
seq_header->color_config.separate_uv_delta_q);
|
||
|
json_object_set_object_member (hdr, "color config", color_config);
|
||
|
|
||
|
json_object_set_int_member (hdr, "order hint bits",
|
||
|
seq_header->order_hint_bits);
|
||
|
json_object_set_int_member (hdr, "bit depth", seq_header->bit_depth);
|
||
|
json_object_set_int_member (hdr, "num planes", seq_header->num_planes);
|
||
|
|
||
|
json_object_set_object_member (json, "sequence header", hdr);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_av1_2_json_frame_header (GstAV12json * self,
|
||
|
GstAV1FrameHeaderOBU * frame_header)
|
||
|
{
|
||
|
JsonObject *json = self->json;
|
||
|
JsonArray *buffer_removal_time, *ref_order_hint, *ref_frame_idx,
|
||
|
*loop_filter_level, *loop_filter_ref_deltas, *loop_filter_mode_deltas,
|
||
|
*feature_enabled, *feature_data, *width_in_sbs_minus_1,
|
||
|
*height_in_sbs_minus_1, *mi_col_starts, *mi_row_starts,
|
||
|
*cdef_y_pri_strength, *cdef_y_sec_strength, *cdef_uv_pri_strength,
|
||
|
*cdef_uv_sec_strength, *frame_restoration_type, *loop_restoration_size,
|
||
|
*is_global, *is_rot_zoom, *is_translation, *gm_params, *gm_type, *invalid,
|
||
|
*point_y_value, *point_y_scaling, *point_cb_value, *point_cb_scaling,
|
||
|
*point_cr_value, *point_cr_scaling, *ar_coeffs_y_plus_128,
|
||
|
*ar_coeffs_cb_plus_128, *ar_coeffs_cr_plus_128, *order_hints,
|
||
|
*ref_frame_sign_bias, *lossless_array, *seg_qm_level, *skip_mode_frame;
|
||
|
JsonObject *hdr = json_object_new ();
|
||
|
JsonObject *loop_filter_params, *quantization_params, *segmentation_params,
|
||
|
*tile_info, *cdef_params, *loop_restoration_params, *global_motion_params,
|
||
|
*film_grain_params;
|
||
|
guint i, j;
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "show existing frame",
|
||
|
frame_header->show_existing_frame);
|
||
|
json_object_set_int_member (hdr, "frame to show map idx",
|
||
|
frame_header->frame_to_show_map_idx);
|
||
|
json_object_set_int_member (hdr, "frame presentation time",
|
||
|
frame_header->frame_presentation_time);
|
||
|
json_object_set_int_member (hdr, "tu presentation delay",
|
||
|
frame_header->tu_presentation_delay);
|
||
|
json_object_set_int_member (hdr, "display frame id",
|
||
|
frame_header->display_frame_id);
|
||
|
switch (frame_header->frame_type) {
|
||
|
case GST_AV1_KEY_FRAME:
|
||
|
json_object_set_string_member (hdr, "frame type", "key frame");
|
||
|
break;
|
||
|
case GST_AV1_INTER_FRAME:
|
||
|
json_object_set_string_member (hdr, "frame type", "inter frame");
|
||
|
break;
|
||
|
case GST_AV1_INTRA_ONLY_FRAME:
|
||
|
json_object_set_string_member (hdr, "frame type", "intra only frame");
|
||
|
break;
|
||
|
case GST_AV1_SWITCH_FRAME:
|
||
|
json_object_set_string_member (hdr, "frame type", "switch frame");
|
||
|
break;
|
||
|
}
|
||
|
json_object_set_boolean_member (hdr, "show frame", frame_header->show_frame);
|
||
|
json_object_set_boolean_member (hdr, "showable frame",
|
||
|
frame_header->showable_frame);
|
||
|
json_object_set_boolean_member (hdr, "error resilient mode",
|
||
|
frame_header->error_resilient_mode);
|
||
|
json_object_set_boolean_member (hdr, "disable cdf update",
|
||
|
frame_header->disable_cdf_update);
|
||
|
json_object_set_int_member (hdr, "allow screen content tools",
|
||
|
frame_header->allow_screen_content_tools);
|
||
|
json_object_set_boolean_member (hdr, "force integer_mv",
|
||
|
frame_header->force_integer_mv);
|
||
|
json_object_set_int_member (hdr, "current frame id",
|
||
|
frame_header->current_frame_id);
|
||
|
json_object_set_boolean_member (hdr, "frame size override flag",
|
||
|
frame_header->frame_size_override_flag);
|
||
|
json_object_set_int_member (hdr, "order hint", frame_header->order_hint);
|
||
|
json_object_set_int_member (hdr, "primary ref_frame",
|
||
|
frame_header->primary_ref_frame);
|
||
|
json_object_set_boolean_member (hdr, "buffer removal time present flag",
|
||
|
frame_header->buffer_removal_time_present_flag);
|
||
|
|
||
|
buffer_removal_time = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_OPERATING_POINTS; i++)
|
||
|
json_array_add_int_element (buffer_removal_time,
|
||
|
frame_header->buffer_removal_time[i]);
|
||
|
json_object_set_array_member (hdr, "buffer removal time",
|
||
|
buffer_removal_time);
|
||
|
|
||
|
json_object_set_int_member (hdr, "refresh frame flags",
|
||
|
frame_header->refresh_frame_flags);
|
||
|
|
||
|
ref_order_hint = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_NUM_REF_FRAMES; i++)
|
||
|
json_array_add_int_element (ref_order_hint,
|
||
|
frame_header->ref_order_hint[i]);
|
||
|
json_object_set_array_member (hdr, "ref order hint", ref_order_hint);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "allow intrabc",
|
||
|
frame_header->allow_intrabc);
|
||
|
json_object_set_boolean_member (hdr, "frame refs short signaling",
|
||
|
frame_header->frame_refs_short_signaling);
|
||
|
json_object_set_int_member (hdr, "last frame idx",
|
||
|
frame_header->last_frame_idx);
|
||
|
json_object_set_int_member (hdr, "gold frame idx",
|
||
|
frame_header->gold_frame_idx);
|
||
|
|
||
|
ref_frame_idx = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_REFS_PER_FRAME; i++)
|
||
|
json_array_add_int_element (ref_frame_idx, frame_header->ref_frame_idx[i]);
|
||
|
json_object_set_array_member (hdr, "ref frame idx", ref_frame_idx);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "allow high precision mv",
|
||
|
frame_header->allow_high_precision_mv);
|
||
|
json_object_set_boolean_member (hdr, "is motion mode switchable",
|
||
|
frame_header->is_motion_mode_switchable);
|
||
|
json_object_set_boolean_member (hdr, "use ref frame mvs",
|
||
|
frame_header->use_ref_frame_mvs);
|
||
|
json_object_set_boolean_member (hdr, "disable frame end update cdf",
|
||
|
frame_header->disable_frame_end_update_cdf);
|
||
|
json_object_set_boolean_member (hdr, "allow warped motion",
|
||
|
frame_header->allow_warped_motion);
|
||
|
json_object_set_boolean_member (hdr, "reduced tx set",
|
||
|
frame_header->reduced_tx_set);
|
||
|
json_object_set_boolean_member (hdr, "render and frame size different",
|
||
|
frame_header->render_and_frame_size_different);
|
||
|
json_object_set_boolean_member (hdr, "use superres",
|
||
|
frame_header->use_superres);
|
||
|
json_object_set_boolean_member (hdr, "is filter switchable",
|
||
|
frame_header->is_filter_switchable);
|
||
|
json_object_set_int_member (hdr, "interpolation filter",
|
||
|
frame_header->interpolation_filter);
|
||
|
|
||
|
loop_filter_params = json_object_new ();
|
||
|
loop_filter_level = json_array_new ();
|
||
|
for (i = 0; i < 4; i++)
|
||
|
json_array_add_int_element (loop_filter_level,
|
||
|
frame_header->loop_filter_params.loop_filter_level[i]);
|
||
|
json_object_set_array_member (loop_filter_params, "loop filter level",
|
||
|
loop_filter_level);
|
||
|
json_object_set_int_member (loop_filter_params, "loop filter sharpness",
|
||
|
frame_header->loop_filter_params.loop_filter_sharpness);
|
||
|
json_object_set_boolean_member (loop_filter_params,
|
||
|
"loop filter delta enabled",
|
||
|
frame_header->loop_filter_params.loop_filter_delta_enabled);
|
||
|
json_object_set_boolean_member (loop_filter_params,
|
||
|
"loop filter delta update",
|
||
|
frame_header->loop_filter_params.loop_filter_delta_update);
|
||
|
|
||
|
loop_filter_ref_deltas = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_TOTAL_REFS_PER_FRAME; i++)
|
||
|
json_array_add_int_element (loop_filter_ref_deltas,
|
||
|
frame_header->loop_filter_params.loop_filter_ref_deltas[i]);
|
||
|
json_object_set_array_member (loop_filter_params, "loop filter ref deltas",
|
||
|
loop_filter_ref_deltas);
|
||
|
|
||
|
loop_filter_mode_deltas = json_array_new ();
|
||
|
for (i = 0; i < 2; i++)
|
||
|
json_array_add_int_element (loop_filter_mode_deltas,
|
||
|
frame_header->loop_filter_params.loop_filter_mode_deltas[i]);
|
||
|
json_object_set_array_member (loop_filter_params, "loop filter mode deltas",
|
||
|
loop_filter_mode_deltas);
|
||
|
json_object_set_boolean_member (loop_filter_params, "delta lf present",
|
||
|
frame_header->loop_filter_params.delta_lf_present);
|
||
|
json_object_set_int_member (loop_filter_params, "delta lf res",
|
||
|
frame_header->loop_filter_params.delta_lf_res);
|
||
|
json_object_set_int_member (loop_filter_params, "delta lf multi",
|
||
|
frame_header->loop_filter_params.delta_lf_multi);
|
||
|
json_object_set_object_member (hdr, "loop filter params", loop_filter_params);
|
||
|
|
||
|
quantization_params = json_object_new ();
|
||
|
json_object_set_int_member (quantization_params, "base q idx",
|
||
|
frame_header->quantization_params.base_q_idx);
|
||
|
json_object_set_boolean_member (quantization_params, "diff uv delta",
|
||
|
frame_header->quantization_params.diff_uv_delta);
|
||
|
json_object_set_boolean_member (quantization_params, "using qmatrix",
|
||
|
frame_header->quantization_params.using_qmatrix);
|
||
|
json_object_set_int_member (quantization_params, "qm y",
|
||
|
frame_header->quantization_params.qm_y);
|
||
|
json_object_set_int_member (quantization_params, "qm u",
|
||
|
frame_header->quantization_params.qm_u);
|
||
|
json_object_set_int_member (quantization_params, "qm v",
|
||
|
frame_header->quantization_params.qm_v);
|
||
|
json_object_set_boolean_member (quantization_params, "delta q present",
|
||
|
frame_header->quantization_params.delta_q_present);
|
||
|
json_object_set_int_member (quantization_params, "delta q res",
|
||
|
frame_header->quantization_params.delta_q_res);
|
||
|
json_object_set_int_member (quantization_params, "delta q y dc",
|
||
|
frame_header->quantization_params.delta_q_y_dc);
|
||
|
json_object_set_int_member (quantization_params, "delta q u dc",
|
||
|
frame_header->quantization_params.delta_q_u_dc);
|
||
|
json_object_set_int_member (quantization_params, "delta q u ac",
|
||
|
frame_header->quantization_params.delta_q_u_ac);
|
||
|
json_object_set_int_member (quantization_params, "delta q v dc",
|
||
|
frame_header->quantization_params.delta_q_v_dc);
|
||
|
json_object_set_int_member (quantization_params, "delta q v ac",
|
||
|
frame_header->quantization_params.delta_q_v_ac);
|
||
|
json_object_set_object_member (hdr, "quantization params",
|
||
|
quantization_params);
|
||
|
|
||
|
segmentation_params = json_object_new ();
|
||
|
json_object_set_boolean_member (segmentation_params, "segmentation enabled",
|
||
|
frame_header->segmentation_params.segmentation_enabled);
|
||
|
json_object_set_int_member (segmentation_params, "segmentation update map",
|
||
|
frame_header->segmentation_params.segmentation_update_map);
|
||
|
json_object_set_int_member (segmentation_params,
|
||
|
"segmentation temporal update",
|
||
|
frame_header->segmentation_params.segmentation_temporal_update);
|
||
|
json_object_set_int_member (segmentation_params, "segmentation update data",
|
||
|
frame_header->segmentation_params.segmentation_update_data);
|
||
|
feature_enabled = json_array_new ();
|
||
|
feature_data = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_SEGMENTS; i++) {
|
||
|
for (j = 0; j < GST_AV1_SEG_LVL_MAX; j++) {
|
||
|
json_array_add_int_element (feature_enabled,
|
||
|
frame_header->segmentation_params.feature_enabled[i][j]);
|
||
|
json_array_add_int_element (feature_data,
|
||
|
frame_header->segmentation_params.feature_data[i][j]);
|
||
|
}
|
||
|
}
|
||
|
json_object_set_array_member (segmentation_params, "feature enabled",
|
||
|
feature_enabled);
|
||
|
json_object_set_array_member (segmentation_params, "feature data",
|
||
|
feature_data);
|
||
|
json_object_set_int_member (segmentation_params, "seg id pre skip",
|
||
|
frame_header->segmentation_params.seg_id_pre_skip);
|
||
|
json_object_set_int_member (segmentation_params, "last active seg id",
|
||
|
frame_header->segmentation_params.last_active_seg_id);
|
||
|
json_object_set_object_member (hdr, "segmentation params",
|
||
|
segmentation_params);
|
||
|
|
||
|
tile_info = json_object_new ();
|
||
|
json_object_set_int_member (tile_info, "uniform tile spacing flag",
|
||
|
frame_header->tile_info.uniform_tile_spacing_flag);
|
||
|
json_object_set_int_member (tile_info, "increment tile rows log2",
|
||
|
frame_header->tile_info.increment_tile_rows_log2);
|
||
|
width_in_sbs_minus_1 = json_array_new ();
|
||
|
height_in_sbs_minus_1 = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_TILE_COLS; i++) {
|
||
|
json_array_add_int_element (width_in_sbs_minus_1,
|
||
|
frame_header->tile_info.width_in_sbs_minus_1[i]);
|
||
|
json_array_add_int_element (height_in_sbs_minus_1,
|
||
|
frame_header->tile_info.height_in_sbs_minus_1[i]);
|
||
|
}
|
||
|
json_object_set_array_member (tile_info, "width in sbs minus 1",
|
||
|
width_in_sbs_minus_1);
|
||
|
json_object_set_array_member (tile_info, "height in sbs minus 1",
|
||
|
height_in_sbs_minus_1);
|
||
|
json_object_set_int_member (tile_info, "tile size bytes minus 1",
|
||
|
frame_header->tile_info.tile_size_bytes_minus_1);
|
||
|
json_object_set_int_member (tile_info, "context update tile id",
|
||
|
frame_header->tile_info.context_update_tile_id);
|
||
|
mi_col_starts = json_array_new ();
|
||
|
mi_row_starts = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_TILE_COLS + 1; i++) {
|
||
|
json_array_add_int_element (mi_col_starts,
|
||
|
frame_header->tile_info.mi_col_starts[i]);
|
||
|
json_array_add_int_element (mi_row_starts,
|
||
|
frame_header->tile_info.mi_row_starts[i]);
|
||
|
}
|
||
|
json_object_set_array_member (tile_info, "mi col starts", mi_col_starts);
|
||
|
json_object_set_array_member (tile_info, "mi row starts", mi_row_starts);
|
||
|
json_object_set_int_member (tile_info, "tile cols log2",
|
||
|
frame_header->tile_info.tile_cols_log2);
|
||
|
json_object_set_int_member (tile_info, "tile cols",
|
||
|
frame_header->tile_info.tile_cols);
|
||
|
json_object_set_int_member (tile_info, "tile rows log2",
|
||
|
frame_header->tile_info.tile_rows_log2);
|
||
|
json_object_set_int_member (tile_info, "tile rows",
|
||
|
frame_header->tile_info.tile_rows);
|
||
|
json_object_set_int_member (tile_info, "tile size bytes",
|
||
|
frame_header->tile_info.tile_size_bytes);
|
||
|
json_object_set_object_member (hdr, "tile_info", tile_info);
|
||
|
|
||
|
cdef_params = json_object_new ();
|
||
|
json_object_set_int_member (cdef_params, "cdef damping",
|
||
|
frame_header->cdef_params.cdef_damping);
|
||
|
json_object_set_int_member (cdef_params, "cdef bits",
|
||
|
frame_header->cdef_params.cdef_bits);
|
||
|
cdef_y_pri_strength = json_array_new ();
|
||
|
cdef_y_sec_strength = json_array_new ();
|
||
|
cdef_uv_pri_strength = json_array_new ();
|
||
|
cdef_uv_sec_strength = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_CDEF_MAX; i++) {
|
||
|
json_array_add_int_element (cdef_y_pri_strength,
|
||
|
frame_header->cdef_params.cdef_y_pri_strength[i]);
|
||
|
json_array_add_int_element (cdef_y_sec_strength,
|
||
|
frame_header->cdef_params.cdef_y_sec_strength[i]);
|
||
|
json_array_add_int_element (cdef_uv_pri_strength,
|
||
|
frame_header->cdef_params.cdef_uv_pri_strength[i]);
|
||
|
json_array_add_int_element (cdef_uv_sec_strength,
|
||
|
frame_header->cdef_params.cdef_uv_sec_strength[i]);
|
||
|
}
|
||
|
json_object_set_array_member (cdef_params, "cdef y pri strength",
|
||
|
cdef_y_pri_strength);
|
||
|
json_object_set_array_member (cdef_params, "cdef y sec strength",
|
||
|
cdef_y_sec_strength);
|
||
|
json_object_set_array_member (cdef_params, "cdef uv pri_strength",
|
||
|
cdef_uv_pri_strength);
|
||
|
json_object_set_array_member (cdef_params, "cdef uv sec_strength",
|
||
|
cdef_uv_sec_strength);
|
||
|
json_object_set_object_member (hdr, "cdef params", cdef_params);
|
||
|
|
||
|
loop_restoration_params = json_object_new ();
|
||
|
json_object_set_int_member (loop_restoration_params, "lr unit shift",
|
||
|
frame_header->loop_restoration_params.lr_unit_shift);
|
||
|
json_object_set_int_member (loop_restoration_params, "lr uv shift",
|
||
|
frame_header->loop_restoration_params.lr_uv_shift);
|
||
|
frame_restoration_type = json_array_new ();
|
||
|
loop_restoration_size = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_NUM_PLANES; i++) {
|
||
|
json_array_add_int_element (frame_restoration_type,
|
||
|
frame_header->loop_restoration_params.frame_restoration_type[i]);
|
||
|
json_array_add_int_element (loop_restoration_size,
|
||
|
frame_header->loop_restoration_params.loop_restoration_size[i]);
|
||
|
}
|
||
|
json_object_set_array_member (loop_restoration_params,
|
||
|
"frame restoration type", frame_restoration_type);
|
||
|
json_object_set_array_member (loop_restoration_params,
|
||
|
"loop restoration size", loop_restoration_size);
|
||
|
json_object_set_int_member (loop_restoration_params, "uses lr",
|
||
|
frame_header->loop_restoration_params.uses_lr);
|
||
|
json_object_set_object_member (hdr, "loop restoration params",
|
||
|
loop_restoration_params);
|
||
|
|
||
|
json_object_set_boolean_member (hdr, "tx mode select",
|
||
|
frame_header->tx_mode_select);
|
||
|
json_object_set_boolean_member (hdr, "skip mode present",
|
||
|
frame_header->skip_mode_present);
|
||
|
json_object_set_boolean_member (hdr, "reference select",
|
||
|
frame_header->reference_select);
|
||
|
|
||
|
global_motion_params = json_object_new ();
|
||
|
is_global = json_array_new ();
|
||
|
is_rot_zoom = json_array_new ();
|
||
|
is_translation = json_array_new ();
|
||
|
gm_params = json_array_new ();
|
||
|
gm_type = json_array_new ();
|
||
|
invalid = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_NUM_REF_FRAMES; i++) {
|
||
|
json_array_add_boolean_element (is_global,
|
||
|
frame_header->global_motion_params.is_global[i]);
|
||
|
json_array_add_boolean_element (is_rot_zoom,
|
||
|
frame_header->global_motion_params.is_rot_zoom[i]);
|
||
|
json_array_add_boolean_element (is_translation,
|
||
|
frame_header->global_motion_params.is_translation[i]);
|
||
|
for (j = 0; j < 6; j++)
|
||
|
json_array_add_int_element (gm_params,
|
||
|
frame_header->global_motion_params.gm_params[i][j]);
|
||
|
json_array_add_int_element (gm_type,
|
||
|
frame_header->global_motion_params.gm_type[i]);
|
||
|
json_array_add_boolean_element (invalid,
|
||
|
frame_header->global_motion_params.invalid[i]);
|
||
|
}
|
||
|
json_object_set_array_member (global_motion_params, "is global", is_global);
|
||
|
json_object_set_array_member (global_motion_params, "is rot zoom",
|
||
|
is_rot_zoom);
|
||
|
json_object_set_array_member (global_motion_params, "is translation",
|
||
|
is_translation);
|
||
|
json_object_set_array_member (global_motion_params, "gm params", gm_params);
|
||
|
json_object_set_array_member (global_motion_params, "gm type", gm_type);
|
||
|
json_object_set_array_member (global_motion_params, "invalid", invalid);
|
||
|
json_object_set_object_member (hdr, "global motion params",
|
||
|
global_motion_params);
|
||
|
|
||
|
film_grain_params = json_object_new ();
|
||
|
json_object_set_boolean_member (film_grain_params, "apply grain",
|
||
|
frame_header->film_grain_params.apply_grain);
|
||
|
json_object_set_int_member (film_grain_params, "grain seed",
|
||
|
frame_header->film_grain_params.grain_seed);
|
||
|
json_object_set_boolean_member (film_grain_params, "update grain",
|
||
|
frame_header->film_grain_params.update_grain);
|
||
|
json_object_set_int_member (film_grain_params, "film grain params ref idx",
|
||
|
frame_header->film_grain_params.film_grain_params_ref_idx);
|
||
|
json_object_set_int_member (film_grain_params, "num y points",
|
||
|
frame_header->film_grain_params.num_y_points);
|
||
|
point_y_value = json_array_new ();
|
||
|
point_y_scaling = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_NUM_Y_POINTS; i++) {
|
||
|
json_array_add_int_element (point_y_value,
|
||
|
frame_header->film_grain_params.point_y_value[i]);
|
||
|
json_array_add_int_element (point_y_scaling,
|
||
|
frame_header->film_grain_params.point_y_scaling[i]);
|
||
|
}
|
||
|
json_object_set_array_member (film_grain_params, "point y value",
|
||
|
point_y_value);
|
||
|
json_object_set_array_member (film_grain_params, "point y scaling",
|
||
|
point_y_scaling);
|
||
|
json_object_set_int_member (film_grain_params, "chroma scaling from luma",
|
||
|
frame_header->film_grain_params.chroma_scaling_from_luma);
|
||
|
json_object_set_int_member (film_grain_params, "num cb points",
|
||
|
frame_header->film_grain_params.num_cb_points);
|
||
|
point_cb_value = json_array_new ();
|
||
|
point_cb_scaling = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_NUM_CB_POINTS; i++) {
|
||
|
json_array_add_int_element (point_cb_value,
|
||
|
frame_header->film_grain_params.point_cb_value[i]);
|
||
|
json_array_add_int_element (point_cb_scaling,
|
||
|
frame_header->film_grain_params.point_cb_scaling[i]);
|
||
|
}
|
||
|
json_object_set_array_member (film_grain_params, "point cb value",
|
||
|
point_cb_value);
|
||
|
json_object_set_array_member (film_grain_params, "point cb scaling",
|
||
|
point_cb_scaling);
|
||
|
json_object_set_int_member (film_grain_params, "num cr points",
|
||
|
frame_header->film_grain_params.num_cr_points);
|
||
|
point_cr_value = json_array_new ();
|
||
|
point_cr_scaling = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_NUM_CR_POINTS; i++) {
|
||
|
json_array_add_int_element (point_cr_value,
|
||
|
frame_header->film_grain_params.point_cr_value[i]);
|
||
|
json_array_add_int_element (point_cr_scaling,
|
||
|
frame_header->film_grain_params.point_cr_scaling[i]);
|
||
|
}
|
||
|
json_object_set_array_member (film_grain_params, "point cr value",
|
||
|
point_cr_value);
|
||
|
json_object_set_array_member (film_grain_params, "point cr scaling",
|
||
|
point_cr_scaling);
|
||
|
json_object_set_int_member (film_grain_params, "grain scaling minus 8",
|
||
|
frame_header->film_grain_params.grain_scaling_minus_8);
|
||
|
json_object_set_int_member (film_grain_params, "ar coeff lag",
|
||
|
frame_header->film_grain_params.ar_coeff_lag);
|
||
|
ar_coeffs_y_plus_128 = json_array_new ();
|
||
|
ar_coeffs_cb_plus_128 = json_array_new ();
|
||
|
ar_coeffs_cr_plus_128 = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_NUM_POS_LUMA; i++) {
|
||
|
json_array_add_int_element (ar_coeffs_y_plus_128,
|
||
|
frame_header->film_grain_params.ar_coeffs_y_plus_128[i]);
|
||
|
json_array_add_int_element (ar_coeffs_cb_plus_128,
|
||
|
frame_header->film_grain_params.ar_coeffs_cb_plus_128[i]);
|
||
|
json_array_add_int_element (ar_coeffs_cr_plus_128,
|
||
|
frame_header->film_grain_params.ar_coeffs_cr_plus_128[i]);
|
||
|
}
|
||
|
json_object_set_array_member (film_grain_params, "ar coeffs y plus 128",
|
||
|
ar_coeffs_y_plus_128);
|
||
|
json_object_set_array_member (film_grain_params, "ar coeffs cb plus 128",
|
||
|
ar_coeffs_cb_plus_128);
|
||
|
json_object_set_array_member (film_grain_params, "ar coeffs cr plus 128",
|
||
|
ar_coeffs_cr_plus_128);
|
||
|
json_object_set_int_member (film_grain_params, "ar coeff shift minus 6",
|
||
|
frame_header->film_grain_params.ar_coeff_shift_minus_6);
|
||
|
json_object_set_int_member (film_grain_params, "grain scale shift",
|
||
|
frame_header->film_grain_params.grain_scale_shift);
|
||
|
json_object_set_int_member (film_grain_params, "cb mult",
|
||
|
frame_header->film_grain_params.cb_mult);
|
||
|
json_object_set_int_member (film_grain_params, "cb luma mult",
|
||
|
frame_header->film_grain_params.cb_luma_mult);
|
||
|
json_object_set_int_member (film_grain_params, "cb offset",
|
||
|
frame_header->film_grain_params.cb_offset);
|
||
|
json_object_set_int_member (film_grain_params, "cr mult",
|
||
|
frame_header->film_grain_params.cr_mult);
|
||
|
json_object_set_int_member (film_grain_params, "cr luma mult",
|
||
|
frame_header->film_grain_params.cr_luma_mult);
|
||
|
json_object_set_int_member (film_grain_params, "cr offset",
|
||
|
frame_header->film_grain_params.cr_offset);
|
||
|
json_object_set_boolean_member (film_grain_params, "overlap flag",
|
||
|
frame_header->film_grain_params.overlap_flag);
|
||
|
json_object_set_boolean_member (film_grain_params, "clip to restricted range",
|
||
|
frame_header->film_grain_params.clip_to_restricted_range);
|
||
|
json_object_set_object_member (hdr, "film grain params", film_grain_params);
|
||
|
|
||
|
json_object_set_int_member (hdr, "superres denom",
|
||
|
frame_header->superres_denom);
|
||
|
json_object_set_int_member (hdr, "frame is intra",
|
||
|
frame_header->frame_is_intra);
|
||
|
|
||
|
order_hints = json_array_new ();
|
||
|
ref_frame_sign_bias = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_NUM_REF_FRAMES; i++) {
|
||
|
json_array_add_int_element (order_hints, frame_header->order_hints[i]);
|
||
|
json_array_add_int_element (ref_frame_sign_bias,
|
||
|
frame_header->ref_frame_sign_bias[i]);
|
||
|
}
|
||
|
json_object_set_array_member (hdr, "order hints", order_hints);
|
||
|
json_object_set_array_member (hdr, "ref frame sign bias",
|
||
|
ref_frame_sign_bias);
|
||
|
|
||
|
json_object_set_int_member (hdr, "coded lossless",
|
||
|
frame_header->coded_lossless);
|
||
|
json_object_set_int_member (hdr, "all lossless", frame_header->all_lossless);
|
||
|
|
||
|
lossless_array = json_array_new ();
|
||
|
for (i = 0; i < GST_AV1_MAX_SEGMENTS; i++)
|
||
|
json_array_add_int_element (lossless_array,
|
||
|
frame_header->lossless_array[i]);
|
||
|
json_object_set_array_member (hdr, "lossless array", lossless_array);
|
||
|
|
||
|
seg_qm_level = json_array_new ();
|
||
|
for (i = 0; i < 3; i++)
|
||
|
for (j = 0; j < GST_AV1_MAX_SEGMENTS; j++)
|
||
|
json_array_add_int_element (seg_qm_level,
|
||
|
frame_header->seg_qm_Level[i][j]);
|
||
|
json_object_set_array_member (hdr, "seg qm level", seg_qm_level);
|
||
|
|
||
|
json_object_set_int_member (hdr, "upscaled width",
|
||
|
frame_header->upscaled_width);
|
||
|
json_object_set_int_member (hdr, "frame width", frame_header->frame_width);
|
||
|
json_object_set_int_member (hdr, "frame height", frame_header->frame_height);
|
||
|
json_object_set_int_member (hdr, "render width", frame_header->render_width);
|
||
|
json_object_set_int_member (hdr, "render height",
|
||
|
frame_header->render_height);
|
||
|
json_object_set_int_member (hdr, "tx mode", frame_header->tx_mode);
|
||
|
|
||
|
skip_mode_frame = json_array_new ();
|
||
|
json_array_add_int_element (skip_mode_frame,
|
||
|
frame_header->skip_mode_frame[0]);
|
||
|
json_array_add_int_element (skip_mode_frame,
|
||
|
frame_header->skip_mode_frame[1]);
|
||
|
json_object_set_array_member (hdr, "skip mode frame", skip_mode_frame);
|
||
|
|
||
|
json_object_set_object_member (json, "frame header", hdr);
|
||
|
}
|
||
|
|
||
|
static GstAV1ParserResult
|
||
|
gst_av1_2_json_handle_one_obu (GstAV12json * self, GstAV1OBU * obu)
|
||
|
{
|
||
|
GstAV1ParserResult pres = GST_AV1_PARSER_OK;
|
||
|
GstAV1FrameHeaderOBU frame_header;
|
||
|
GstAV1FrameOBU frame;
|
||
|
|
||
|
switch (obu->obu_type) {
|
||
|
case GST_AV1_OBU_TEMPORAL_DELIMITER:
|
||
|
pres = gst_av1_parser_parse_temporal_delimiter_obu (self->parser, obu);
|
||
|
break;
|
||
|
|
||
|
case GST_AV1_OBU_SEQUENCE_HEADER:
|
||
|
{
|
||
|
GstAV1SequenceHeaderOBU seq_header;
|
||
|
|
||
|
pres = gst_av1_parser_parse_sequence_header_obu (self->parser,
|
||
|
obu, &seq_header);
|
||
|
|
||
|
if (pres == GST_AV1_PARSER_OK)
|
||
|
gst_av1_2_json_sequence_header (self, &seq_header);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case GST_AV1_OBU_REDUNDANT_FRAME_HEADER:
|
||
|
case GST_AV1_OBU_FRAME_HEADER:
|
||
|
pres = gst_av1_parser_parse_frame_header_obu (self->parser, obu,
|
||
|
&frame_header);
|
||
|
if (pres == GST_AV1_PARSER_OK)
|
||
|
gst_av1_2_json_frame_header (self, &frame_header);
|
||
|
break;
|
||
|
|
||
|
case GST_AV1_OBU_FRAME:
|
||
|
pres = gst_av1_parser_parse_frame_obu (self->parser, obu, &frame);
|
||
|
|
||
|
if (pres == GST_AV1_PARSER_OK)
|
||
|
gst_av1_2_json_frame_header (self, &frame.frame_header);
|
||
|
break;
|
||
|
|
||
|
case GST_AV1_OBU_METADATA:
|
||
|
{
|
||
|
GstAV1MetadataOBU metadata;
|
||
|
|
||
|
pres = gst_av1_parser_parse_metadata_obu (self->parser, obu, &metadata);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case GST_AV1_OBU_TILE_GROUP:
|
||
|
{
|
||
|
GstAV1TileGroupOBU tile_group;
|
||
|
|
||
|
pres =
|
||
|
gst_av1_parser_parse_tile_group_obu (self->parser, obu, &tile_group);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case GST_AV1_OBU_TILE_LIST:
|
||
|
{
|
||
|
GstAV1TileListOBU tile_list;
|
||
|
|
||
|
pres = gst_av1_parser_parse_tile_list_obu (self->parser, obu, &tile_list);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case GST_AV1_OBU_PADDING:
|
||
|
break;
|
||
|
default:
|
||
|
GST_WARNING_OBJECT (self, "an unrecognized obu type %d", obu->obu_type);
|
||
|
pres = GST_AV1_PARSER_BITSTREAM_ERROR;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (obu->obu_type == GST_AV1_OBU_FRAME_HEADER
|
||
|
|| obu->obu_type == GST_AV1_OBU_FRAME
|
||
|
|| obu->obu_type == GST_AV1_OBU_REDUNDANT_FRAME_HEADER) {
|
||
|
GstAV1FrameHeaderOBU *fh = &frame_header;
|
||
|
|
||
|
if (obu->obu_type == GST_AV1_OBU_FRAME)
|
||
|
fh = &frame.frame_header;
|
||
|
|
||
|
if (!fh->show_existing_frame || fh->frame_type == GST_AV1_KEY_FRAME)
|
||
|
pres = gst_av1_parser_reference_frame_update (self->parser, fh);
|
||
|
}
|
||
|
|
||
|
return pres;
|
||
|
}
|
||
|
|
||
|
static GstFlowReturn
|
||
|
gst_av1_2_json_chain (GstPad * sinkpad, GstObject * object, GstBuffer * in_buf)
|
||
|
{
|
||
|
GstAV12json *self = GST_AV1_2_JSON (object);
|
||
|
JsonObject *json = self->json;
|
||
|
GstAV1ParserResult pres;
|
||
|
GstAV1OBU obu;
|
||
|
GstBuffer *out_buf;
|
||
|
gchar *json_string;
|
||
|
guint32 offset = 0, consumed;
|
||
|
guint length;
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
while (offset < in_map.size) {
|
||
|
pres =
|
||
|
gst_av1_parser_identify_one_obu (self->parser, in_map.data + offset,
|
||
|
in_map.size - offset, &obu, &consumed);
|
||
|
|
||
|
if (pres != GST_AV1_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Cannot get OBU");
|
||
|
ret = GST_FLOW_ERROR;
|
||
|
goto unmap;
|
||
|
}
|
||
|
|
||
|
pres = gst_av1_2_json_handle_one_obu (self, &obu);
|
||
|
if (pres != GST_AV1_PARSER_OK) {
|
||
|
GST_WARNING_OBJECT (self, "Cannot parse frame header");
|
||
|
ret = GST_FLOW_ERROR;
|
||
|
goto unmap;
|
||
|
}
|
||
|
offset += consumed;
|
||
|
}
|
||
|
|
||
|
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);
|
||
|
|
||
|
unmap:
|
||
|
switch (pres) {
|
||
|
case GST_AV1_PARSER_BITSTREAM_ERROR:
|
||
|
case GST_AV1_PARSER_MISSING_OBU_REFERENCE:
|
||
|
case GST_AV1_PARSER_NO_MORE_DATA:
|
||
|
if (self->use_annex_b)
|
||
|
gst_av1_parser_reset_annex_b (self->parser);
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
gst_buffer_unmap (in_buf, &in_map);
|
||
|
gst_buffer_unref (in_buf);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_av1_2_json_use_annexb (GstAV12json * self, GstCaps * caps)
|
||
|
{
|
||
|
if (caps && gst_caps_get_size (caps) > 0) {
|
||
|
GstStructure *s = gst_caps_get_structure (caps, 0);
|
||
|
const gchar *str_align = NULL;
|
||
|
const gchar *str_stream = NULL;
|
||
|
|
||
|
str_align = gst_structure_get_string (s, "alignment");
|
||
|
str_stream = gst_structure_get_string (s, "stream-format");
|
||
|
|
||
|
self->use_annex_b = FALSE;
|
||
|
if (str_stream && g_strcmp0 (str_stream, "annexb") == 0)
|
||
|
if (str_align && g_strcmp0 (str_align, "tu") == 0) {
|
||
|
self->use_annex_b = TRUE;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
gst_av1_parser_reset (self->parser, self->use_annex_b);
|
||
|
}
|
||
|
|
||
|
|
||
|
static gboolean
|
||
|
gst_av1_2_json_set_caps (GstAV12json * self, GstCaps * caps)
|
||
|
{
|
||
|
GstCaps *src_caps =
|
||
|
gst_caps_new_simple ("text/x-json", "format", G_TYPE_STRING, "av1", NULL);
|
||
|
GstEvent *event;
|
||
|
|
||
|
event = gst_event_new_caps (src_caps);
|
||
|
gst_caps_unref (src_caps);
|
||
|
|
||
|
gst_av1_2_json_use_annexb (self, caps);
|
||
|
|
||
|
return gst_pad_push_event (self->srcpad, event);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
gst_av1_2_json_sink_event (GstPad * pad, GstObject * parent, GstEvent * event)
|
||
|
{
|
||
|
GstAV12json *self = GST_AV1_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_av1_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_av1_2_json_class_init (GstAV12jsonClass * klass)
|
||
|
{
|
||
|
GObjectClass *gobject_class;
|
||
|
GstElementClass *gstelement_class;
|
||
|
|
||
|
gobject_class = (GObjectClass *) klass;
|
||
|
gstelement_class = (GstElementClass *) klass;
|
||
|
|
||
|
gobject_class->finalize = gst_av1_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, "Av12json",
|
||
|
"Transform",
|
||
|
"AV1 to json element",
|
||
|
"Benjamin Gaignard <benjamin.gaignard@collabora.com>");
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_av1_2_json_init (GstAV12json * self)
|
||
|
{
|
||
|
self->sinkpad = gst_pad_new_from_static_template (&sink_factory, "sink");
|
||
|
gst_pad_set_chain_function (self->sinkpad, gst_av1_2_json_chain);
|
||
|
gst_element_add_pad (GST_ELEMENT (self), self->sinkpad);
|
||
|
gst_pad_set_event_function (self->sinkpad,
|
||
|
GST_DEBUG_FUNCPTR (gst_av1_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_av1_parser_new ();
|
||
|
gst_av1_parser_reset (self->parser, FALSE);
|
||
|
}
|