mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-20 23:36:38 +00:00
cedc7d4def
According to the vp8 spec, the first partition (size can be derived from the frame header) should have all compressed header information and we implemented gst codecparser based on that. But it doesn't seem to be the case with some of the streams (#792773) and libvpx works fine because it uses the whole frame size (not the first partition size) to initialize the bool decoder. https://bugzilla.gnome.org/show_bug.cgi?id=792773
568 lines
16 KiB
C
568 lines
16 KiB
C
/*
|
|
* gstvp8parser.c - VP8 parser
|
|
*
|
|
* Copyright (C) 2013-2014 Intel Corporation
|
|
* Author: Halley Zhao <halley.zhao@intel.com>
|
|
* Author: Gwenole Beauchesne <gwenole.beauchesne@intel.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:gstvp8parser
|
|
* @title: GstVp8Parser
|
|
* @short_description: Convenience library for parsing vp8 video bitstream.
|
|
*
|
|
* For more details about the structures, you can refer to the
|
|
* specifications: VP8-rfc6386.pdf
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
#include <string.h>
|
|
#include <gst/base/gstbytereader.h>
|
|
#include "gstvp8parser.h"
|
|
#include "gstvp8rangedecoder.h"
|
|
#include "vp8utils.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (vp8_parser_debug);
|
|
#define GST_CAT_DEFAULT vp8_parser_debug
|
|
|
|
#define INITIALIZE_DEBUG_CATEGORY ensure_debug_category ()
|
|
static void
|
|
ensure_debug_category (void)
|
|
{
|
|
#ifndef GST_DISABLE_GST_DEBUG
|
|
static gsize is_initialized;
|
|
|
|
if (g_once_init_enter (&is_initialized)) {
|
|
GST_DEBUG_CATEGORY_INIT (vp8_parser_debug, "codecparsers_vp8", 0,
|
|
"vp8 parser library");
|
|
g_once_init_leave (&is_initialized, TRUE);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static GstVp8MvProbs vp8_mv_update_probs;
|
|
static GstVp8TokenProbs vp8_token_update_probs;
|
|
|
|
static void
|
|
ensure_prob_tables (void)
|
|
{
|
|
static gsize is_initialized;
|
|
|
|
if (g_once_init_enter (&is_initialized)) {
|
|
gst_vp8_mv_update_probs_init (&vp8_mv_update_probs);
|
|
gst_vp8_token_update_probs_init (&vp8_token_update_probs);
|
|
g_once_init_leave (&is_initialized, TRUE);
|
|
}
|
|
}
|
|
|
|
#define READ_BOOL(rd, val, field_name) \
|
|
val = vp8_read_bool ((rd))
|
|
#define READ_UINT(rd, val, nbits, field_name) \
|
|
val = vp8_read_uint ((rd), (nbits))
|
|
#define READ_SINT(rd, val, nbits, field_name) \
|
|
val = vp8_read_sint ((rd), (nbits))
|
|
|
|
static inline gboolean
|
|
vp8_read_bool (GstVp8RangeDecoder * rd)
|
|
{
|
|
return (gboolean) gst_vp8_range_decoder_read_literal (rd, 1);
|
|
}
|
|
|
|
static inline guint
|
|
vp8_read_uint (GstVp8RangeDecoder * rd, guint nbits)
|
|
{
|
|
return (guint) gst_vp8_range_decoder_read_literal (rd, nbits);
|
|
}
|
|
|
|
static inline gint
|
|
vp8_read_sint (GstVp8RangeDecoder * rd, guint nbits)
|
|
{
|
|
gint v;
|
|
|
|
v = gst_vp8_range_decoder_read_literal (rd, nbits);
|
|
if (gst_vp8_range_decoder_read_literal (rd, 1))
|
|
v = -v;
|
|
return v;
|
|
}
|
|
|
|
/* Parse update_segmentation() */
|
|
static gboolean
|
|
parse_update_segmentation (GstVp8RangeDecoder * rd, GstVp8Segmentation * seg)
|
|
{
|
|
gboolean update;
|
|
gint i;
|
|
|
|
seg->update_mb_segmentation_map = FALSE;
|
|
seg->update_segment_feature_data = FALSE;
|
|
|
|
READ_BOOL (rd, seg->segmentation_enabled, "segmentation_enabled");
|
|
if (!seg->segmentation_enabled)
|
|
return TRUE;
|
|
|
|
READ_BOOL (rd, seg->update_mb_segmentation_map, "update_mb_segmentation_map");
|
|
READ_BOOL (rd, seg->update_segment_feature_data,
|
|
"update_segment_feature_data");
|
|
|
|
if (seg->update_segment_feature_data) {
|
|
READ_UINT (rd, seg->segment_feature_mode, 1, "segment_feature_mode");
|
|
|
|
/* quantizer_update_value defaults to zero if update flag is zero
|
|
(Section 9.3, 4.b) */
|
|
for (i = 0; i < 4; i++) {
|
|
READ_BOOL (rd, update, "quantizer_update");
|
|
if (update) {
|
|
READ_SINT (rd, seg->quantizer_update_value[i], 7,
|
|
"quantizer_update_value");
|
|
} else
|
|
seg->quantizer_update_value[i] = 0;
|
|
}
|
|
|
|
/* lf_update_value defaults to zero if update flag is zero
|
|
(Section 9.3, 4.b) */
|
|
for (i = 0; i < 4; i++) {
|
|
READ_BOOL (rd, update, "loop_filter_update");
|
|
if (update) {
|
|
READ_SINT (rd, seg->lf_update_value[i], 6, "lf_update_value");
|
|
} else
|
|
seg->lf_update_value[i] = 0;
|
|
}
|
|
}
|
|
|
|
/* segment_prob defaults to 255 if update flag is zero
|
|
(Section 9.3, 5) */
|
|
if (seg->update_mb_segmentation_map) {
|
|
for (i = 0; i < 3; i++) {
|
|
READ_BOOL (rd, update, "segment_prob_update");
|
|
if (update) {
|
|
READ_UINT (rd, seg->segment_prob[i], 8, "segment_prob");
|
|
} else
|
|
seg->segment_prob[i] = 255;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* Parse mb_lf_adjustments() to update loop filter delta adjustments */
|
|
static gboolean
|
|
parse_mb_lf_adjustments (GstVp8RangeDecoder * rd, GstVp8MbLfAdjustments * adj)
|
|
{
|
|
gboolean update;
|
|
gint i;
|
|
|
|
adj->mode_ref_lf_delta_update = FALSE;
|
|
|
|
READ_BOOL (rd, adj->loop_filter_adj_enable, "loop_filter_adj_enable");
|
|
if (!adj->loop_filter_adj_enable)
|
|
return TRUE;
|
|
|
|
READ_BOOL (rd, adj->mode_ref_lf_delta_update, "mode_ref_lf_delta_update");
|
|
if (!adj->mode_ref_lf_delta_update)
|
|
return TRUE;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
READ_BOOL (rd, update, "ref_frame_delta_update_flag");
|
|
if (update) {
|
|
READ_SINT (rd, adj->ref_frame_delta[i], 6, "ref_frame_delta_magniture");
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
READ_BOOL (rd, update, "mb_mode_delta_update_flag");
|
|
if (update) {
|
|
READ_SINT (rd, adj->mb_mode_delta[i], 6, "mb_mode_delta_magnitude");
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* Parse quant_indices() */
|
|
static gboolean
|
|
parse_quant_indices (GstVp8RangeDecoder * rd, GstVp8QuantIndices * qip)
|
|
{
|
|
gboolean update;
|
|
|
|
READ_UINT (rd, qip->y_ac_qi, 7, "y_ac_qi");
|
|
|
|
READ_BOOL (rd, update, "y_dc_delta_present");
|
|
if (update) {
|
|
READ_SINT (rd, qip->y_dc_delta, 4, "y_dc_delta_magnitude");
|
|
} else
|
|
qip->y_dc_delta = 0;
|
|
|
|
READ_BOOL (rd, update, "y2_dc_delta_present");
|
|
if (update) {
|
|
READ_SINT (rd, qip->y2_dc_delta, 4, "y2_dc_delta_magnitude");
|
|
} else
|
|
qip->y2_dc_delta = 0;
|
|
|
|
READ_BOOL (rd, update, "y2_ac_delta_present");
|
|
if (update) {
|
|
READ_SINT (rd, qip->y2_ac_delta, 4, "y2_ac_delta_magnitude");
|
|
} else
|
|
qip->y2_ac_delta = 0;
|
|
|
|
READ_BOOL (rd, update, "uv_dc_delta_present");
|
|
if (update) {
|
|
READ_SINT (rd, qip->uv_dc_delta, 4, "uv_dc_delta_magnitude");
|
|
} else
|
|
qip->uv_dc_delta = 0;
|
|
|
|
READ_BOOL (rd, update, "uv_ac_delta_present");
|
|
if (update) {
|
|
READ_SINT (rd, qip->uv_ac_delta, 4, "uv_ac_delta_magnitude");
|
|
} else
|
|
qip->uv_ac_delta = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Parse token_prob_update() to update persistent token probabilities */
|
|
static gboolean
|
|
parse_token_prob_update (GstVp8RangeDecoder * rd, GstVp8TokenProbs * probs)
|
|
{
|
|
gint i, j, k, l;
|
|
guint8 prob;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
for (j = 0; j < 8; j++) {
|
|
for (k = 0; k < 3; k++) {
|
|
for (l = 0; l < 11; l++) {
|
|
if (gst_vp8_range_decoder_read (rd,
|
|
vp8_token_update_probs.prob[i][j][k][l])) {
|
|
READ_UINT (rd, prob, 8, "token_prob_update");
|
|
probs->prob[i][j][k][l] = prob;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* Parse prob_update() to update probabilities used for MV decoding */
|
|
static gboolean
|
|
parse_mv_prob_update (GstVp8RangeDecoder * rd, GstVp8MvProbs * probs)
|
|
{
|
|
gint i, j;
|
|
guint8 prob;
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
for (j = 0; j < 19; j++) {
|
|
if (gst_vp8_range_decoder_read (rd, vp8_mv_update_probs.prob[i][j])) {
|
|
READ_UINT (rd, prob, 7, "mv_prob_update");
|
|
probs->prob[i][j] = prob ? (prob << 1) : 1;
|
|
}
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* Calculate partition sizes */
|
|
static gboolean
|
|
calc_partition_sizes (GstVp8FrameHdr * frame_hdr, const guint8 * data,
|
|
guint size)
|
|
{
|
|
const guint num_partitions = 1 << frame_hdr->log2_nbr_of_dct_partitions;
|
|
guint i, ofs, part_size, part_size_ofs = frame_hdr->first_part_size;
|
|
|
|
ofs = part_size_ofs + 3 * (num_partitions - 1);
|
|
if (ofs > size) {
|
|
GST_ERROR ("not enough bytes left to parse partition sizes");
|
|
return FALSE;
|
|
}
|
|
|
|
/* The size of the last partition is not specified (9.5) */
|
|
for (i = 0; i < num_partitions - 1; i++) {
|
|
part_size = (guint32) data[part_size_ofs + 0] |
|
|
((guint32) data[part_size_ofs + 1] << 8) |
|
|
((guint32) data[part_size_ofs + 2] << 16);
|
|
part_size_ofs += 3;
|
|
|
|
frame_hdr->partition_size[i] = part_size;
|
|
ofs += part_size;
|
|
}
|
|
|
|
if (ofs > size) {
|
|
GST_ERROR ("not enough bytes left to determine the last partition size");
|
|
return FALSE;
|
|
}
|
|
frame_hdr->partition_size[i] = size - ofs;
|
|
|
|
while (++i < G_N_ELEMENTS (frame_hdr->partition_size))
|
|
frame_hdr->partition_size[i] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
/* Parse uncompressed data chunk (19.1) */
|
|
static GstVp8ParserResult
|
|
parse_uncompressed_data_chunk (GstVp8Parser * parser, GstByteReader * br,
|
|
GstVp8FrameHdr * frame_hdr)
|
|
{
|
|
guint32 frame_tag, start_code;
|
|
guint16 size_code;
|
|
|
|
GST_DEBUG ("parsing \"Uncompressed Data Chunk\"");
|
|
|
|
if (!gst_byte_reader_get_uint24_le (br, &frame_tag))
|
|
goto error;
|
|
|
|
frame_hdr->key_frame = !(frame_tag & 0x01);
|
|
frame_hdr->version = (frame_tag >> 1) & 0x07;
|
|
frame_hdr->show_frame = (frame_tag >> 4) & 0x01;
|
|
frame_hdr->first_part_size = (frame_tag >> 5) & 0x7ffff;
|
|
|
|
if (frame_hdr->key_frame) {
|
|
if (!gst_byte_reader_get_uint24_be (br, &start_code))
|
|
goto error;
|
|
if (start_code != 0x9d012a)
|
|
GST_WARNING ("vp8 parser: invalid start code in frame header");
|
|
|
|
if (!gst_byte_reader_get_uint16_le (br, &size_code))
|
|
goto error;
|
|
frame_hdr->width = size_code & 0x3fff;
|
|
frame_hdr->horiz_scale_code = size_code >> 14;
|
|
|
|
if (!gst_byte_reader_get_uint16_le (br, &size_code)) {
|
|
goto error;
|
|
}
|
|
frame_hdr->height = size_code & 0x3fff;
|
|
frame_hdr->vert_scale_code = (size_code >> 14);
|
|
|
|
/* Reset parser state on key frames */
|
|
gst_vp8_parser_init (parser);
|
|
} else {
|
|
frame_hdr->width = 0;
|
|
frame_hdr->height = 0;
|
|
frame_hdr->horiz_scale_code = 0;
|
|
frame_hdr->vert_scale_code = 0;
|
|
}
|
|
|
|
/* Calculated values */
|
|
frame_hdr->data_chunk_size = gst_byte_reader_get_pos (br);
|
|
return GST_VP8_PARSER_OK;
|
|
|
|
error:
|
|
GST_WARNING ("error parsing \"Uncompressed Data Chunk\"");
|
|
return GST_VP8_PARSER_ERROR;
|
|
}
|
|
|
|
/* Parse Frame Header (19.2) */
|
|
static GstVp8ParserResult
|
|
parse_frame_header (GstVp8Parser * parser, GstVp8RangeDecoder * rd,
|
|
GstVp8FrameHdr * frame_hdr)
|
|
{
|
|
gboolean update;
|
|
guint i;
|
|
|
|
GST_DEBUG ("parsing \"Frame Header\"");
|
|
|
|
if (frame_hdr->key_frame) {
|
|
READ_UINT (rd, frame_hdr->color_space, 1, "color_space");
|
|
READ_UINT (rd, frame_hdr->clamping_type, 1, "clamping_type");
|
|
}
|
|
|
|
if (!parse_update_segmentation (rd, &parser->segmentation))
|
|
goto error;
|
|
|
|
READ_UINT (rd, frame_hdr->filter_type, 1, "filter_type");
|
|
READ_UINT (rd, frame_hdr->loop_filter_level, 6, "loop_filter_level");
|
|
READ_UINT (rd, frame_hdr->sharpness_level, 3, "sharpness_level");
|
|
|
|
if (!parse_mb_lf_adjustments (rd, &parser->mb_lf_adjust))
|
|
goto error;
|
|
|
|
READ_UINT (rd, frame_hdr->log2_nbr_of_dct_partitions, 2,
|
|
"log2_nbr_of_dct_partitions");
|
|
|
|
if (!parse_quant_indices (rd, &frame_hdr->quant_indices))
|
|
goto error;
|
|
|
|
frame_hdr->copy_buffer_to_golden = 0;
|
|
frame_hdr->copy_buffer_to_alternate = 0;
|
|
if (frame_hdr->key_frame) {
|
|
READ_BOOL (rd, frame_hdr->refresh_entropy_probs, "refresh_entropy_probs");
|
|
|
|
frame_hdr->refresh_last = TRUE;
|
|
frame_hdr->refresh_golden_frame = TRUE;
|
|
frame_hdr->refresh_alternate_frame = TRUE;
|
|
|
|
gst_vp8_mode_probs_init_defaults (&frame_hdr->mode_probs, TRUE);
|
|
} else {
|
|
READ_BOOL (rd, frame_hdr->refresh_golden_frame, "refresh_golden_frame");
|
|
READ_BOOL (rd, frame_hdr->refresh_alternate_frame,
|
|
"refresh_alternate_frame");
|
|
|
|
if (!frame_hdr->refresh_golden_frame) {
|
|
READ_UINT (rd, frame_hdr->copy_buffer_to_golden, 2,
|
|
"copy_buffer_to_golden");
|
|
}
|
|
|
|
if (!frame_hdr->refresh_alternate_frame) {
|
|
READ_UINT (rd, frame_hdr->copy_buffer_to_alternate, 2,
|
|
"copy_buffer_to_alternate");
|
|
}
|
|
|
|
READ_UINT (rd, frame_hdr->sign_bias_golden, 1, "sign_bias_golden");
|
|
READ_UINT (rd, frame_hdr->sign_bias_alternate, 1, "sign_bias_alternate");
|
|
READ_BOOL (rd, frame_hdr->refresh_entropy_probs, "refresh_entropy_probs");
|
|
READ_BOOL (rd, frame_hdr->refresh_last, "refresh_last");
|
|
|
|
memcpy (&frame_hdr->mode_probs, &parser->mode_probs,
|
|
sizeof (parser->mode_probs));
|
|
}
|
|
memcpy (&frame_hdr->token_probs, &parser->token_probs,
|
|
sizeof (parser->token_probs));
|
|
memcpy (&frame_hdr->mv_probs, &parser->mv_probs, sizeof (parser->mv_probs));
|
|
|
|
if (!parse_token_prob_update (rd, &frame_hdr->token_probs))
|
|
goto error;
|
|
|
|
READ_BOOL (rd, frame_hdr->mb_no_skip_coeff, "mb_no_skip_coeff");
|
|
if (frame_hdr->mb_no_skip_coeff) {
|
|
READ_UINT (rd, frame_hdr->prob_skip_false, 8, "prob_skip_false");
|
|
}
|
|
|
|
if (!frame_hdr->key_frame) {
|
|
READ_UINT (rd, frame_hdr->prob_intra, 8, "prob_intra");
|
|
READ_UINT (rd, frame_hdr->prob_last, 8, "prob_last");
|
|
READ_UINT (rd, frame_hdr->prob_gf, 8, "prob_gf");
|
|
|
|
READ_BOOL (rd, update, "intra_16x16_prob_update_flag");
|
|
if (update) {
|
|
for (i = 0; i < 4; i++) {
|
|
READ_UINT (rd, frame_hdr->mode_probs.y_prob[i], 8, "intra_16x16_prob");
|
|
}
|
|
}
|
|
|
|
READ_BOOL (rd, update, "intra_chroma_prob_update_flag");
|
|
if (update) {
|
|
for (i = 0; i < 3; i++) {
|
|
READ_UINT (rd, frame_hdr->mode_probs.uv_prob[i], 8,
|
|
"intra_chroma_prob");
|
|
}
|
|
}
|
|
|
|
if (!parse_mv_prob_update (rd, &frame_hdr->mv_probs))
|
|
goto error;
|
|
}
|
|
|
|
/* Refresh entropy probabilities */
|
|
if (frame_hdr->refresh_entropy_probs) {
|
|
memcpy (&parser->token_probs, &frame_hdr->token_probs,
|
|
sizeof (frame_hdr->token_probs));
|
|
memcpy (&parser->mv_probs, &frame_hdr->mv_probs,
|
|
sizeof (frame_hdr->mv_probs));
|
|
if (!frame_hdr->key_frame)
|
|
memcpy (&parser->mode_probs, &frame_hdr->mode_probs,
|
|
sizeof (frame_hdr->mode_probs));
|
|
}
|
|
|
|
/* Calculated values */
|
|
frame_hdr->header_size = gst_vp8_range_decoder_get_pos (rd);
|
|
return GST_VP8_PARSER_OK;
|
|
|
|
error:
|
|
GST_WARNING ("error parsing \"Frame Header\"");
|
|
return GST_VP8_PARSER_ERROR;
|
|
}
|
|
|
|
/**** API ****/
|
|
/**
|
|
* gst_vp8_parser_init:
|
|
* @parser: The #GstVp8Parser to initialize
|
|
*
|
|
* Initializes the supplied @parser structure with its default values.
|
|
*
|
|
* Since: 1.4
|
|
*/
|
|
void
|
|
gst_vp8_parser_init (GstVp8Parser * parser)
|
|
{
|
|
g_return_if_fail (parser != NULL);
|
|
|
|
memset (&parser->segmentation, 0, sizeof (parser->segmentation));
|
|
memset (&parser->mb_lf_adjust, 0, sizeof (parser->mb_lf_adjust));
|
|
gst_vp8_token_probs_init_defaults (&parser->token_probs);
|
|
gst_vp8_mv_probs_init_defaults (&parser->mv_probs);
|
|
gst_vp8_mode_probs_init_defaults (&parser->mode_probs, FALSE);
|
|
}
|
|
|
|
/**
|
|
* gst_vp8_parser_parse_frame_header:
|
|
* @parser: The #GstVp8Parser
|
|
* @frame_hdr: The #GstVp8FrameHdr to fill
|
|
* @data: The data to parse
|
|
* @size: The size of the @data to parse
|
|
*
|
|
* Parses the VP8 bitstream contained in @data, and fills in @frame_hdr
|
|
* with the information. The supplied @data shall point to a complete
|
|
* frame since there is no sync code specified for VP8 bitstreams. Thus,
|
|
* the @size argument shall represent the whole frame size.
|
|
*
|
|
* Returns: a #GstVp8ParserResult
|
|
*
|
|
* Since: 1.4
|
|
*/
|
|
GstVp8ParserResult
|
|
gst_vp8_parser_parse_frame_header (GstVp8Parser * parser,
|
|
GstVp8FrameHdr * frame_hdr, const guint8 * data, gsize size)
|
|
{
|
|
GstByteReader br;
|
|
GstVp8RangeDecoder rd;
|
|
GstVp8RangeDecoderState rd_state;
|
|
GstVp8ParserResult result;
|
|
|
|
ensure_debug_category ();
|
|
ensure_prob_tables ();
|
|
|
|
g_return_val_if_fail (frame_hdr != NULL, GST_VP8_PARSER_ERROR);
|
|
g_return_val_if_fail (parser != NULL, GST_VP8_PARSER_ERROR);
|
|
|
|
/* Uncompressed Data Chunk */
|
|
gst_byte_reader_init (&br, data, size);
|
|
|
|
result = parse_uncompressed_data_chunk (parser, &br, frame_hdr);
|
|
if (result != GST_VP8_PARSER_OK)
|
|
return result;
|
|
|
|
/* Frame Header */
|
|
if (frame_hdr->data_chunk_size + frame_hdr->first_part_size > size)
|
|
return GST_VP8_PARSER_BROKEN_DATA;
|
|
|
|
data += frame_hdr->data_chunk_size;
|
|
size -= frame_hdr->data_chunk_size;
|
|
if (!gst_vp8_range_decoder_init (&rd, data, size))
|
|
return GST_VP8_PARSER_BROKEN_DATA;
|
|
|
|
result = parse_frame_header (parser, &rd, frame_hdr);
|
|
if (result != GST_VP8_PARSER_OK)
|
|
return result;
|
|
|
|
/* Calculate partition sizes */
|
|
if (!calc_partition_sizes (frame_hdr, data, size))
|
|
return GST_VP8_PARSER_BROKEN_DATA;
|
|
|
|
/* Sync range decoder state */
|
|
gst_vp8_range_decoder_get_state (&rd, &rd_state);
|
|
frame_hdr->rd_range = rd_state.range;
|
|
frame_hdr->rd_value = rd_state.value;
|
|
frame_hdr->rd_count = rd_state.count;
|
|
return GST_VP8_PARSER_OK;
|
|
}
|