gstreamer/subprojects/gst-plugins-bad/ext/codec2json/gstav12json.c

1036 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 = GST_AV1_PARSER_OK;
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);
}