Also enable MPEG-2 support in the trace_headers filter.tags/n4.0
@@ -1742,6 +1742,7 @@ CONFIG_EXTRA=" | |||
cbs | |||
cbs_h264 | |||
cbs_h265 | |||
cbs_mpeg2 | |||
dirac_parse | |||
dvprofile | |||
faandct | |||
@@ -1970,6 +1971,7 @@ threads_if_any="$THREADS_LIST" | |||
# subsystems | |||
cbs_h264_select="cbs golomb" | |||
cbs_h265_select="cbs golomb" | |||
cbs_mpeg2_select="cbs" | |||
dct_select="rdft" | |||
dirac_parse_select="golomb" | |||
error_resilience_select="me_cmp" | |||
@@ -2327,7 +2329,7 @@ h264_metadata_bsf_select="cbs_h264" | |||
h264_redundant_pps_bsf_select="cbs_h264" | |||
hevc_metadata_bsf_select="cbs_h265" | |||
mjpeg2jpeg_bsf_select="jpegtables" | |||
trace_headers_bsf_select="cbs_h264 cbs_h265" | |||
trace_headers_bsf_select="cbs_h264 cbs_h265 cbs_mpeg2" | |||
# external libraries | |||
avisynth_deps="LoadLibrary" | |||
@@ -227,7 +227,7 @@ Log trace output containing all syntax elements in the coded stream | |||
headers (everything above the level of individual coded blocks). | |||
This can be useful for debugging low-level stream issues. | |||
Supports H.264 and H.265. | |||
Supports H.264, H.265 and MPEG-2. | |||
@section vp9_superframe | |||
@@ -56,6 +56,7 @@ OBJS-$(CONFIG_CABAC) += cabac.o | |||
OBJS-$(CONFIG_CBS) += cbs.o | |||
OBJS-$(CONFIG_CBS_H264) += cbs_h2645.o h2645_parse.o | |||
OBJS-$(CONFIG_CBS_H265) += cbs_h2645.o h2645_parse.o | |||
OBJS-$(CONFIG_CBS_MPEG2) += cbs_mpeg2.o | |||
OBJS-$(CONFIG_DCT) += dct.o dct32_fixed.o dct32_float.o | |||
OBJS-$(CONFIG_ERROR_RESILIENCE) += error_resilience.o | |||
OBJS-$(CONFIG_FAANDCT) += faandct.o | |||
@@ -34,6 +34,9 @@ static const CodedBitstreamType *cbs_type_table[] = { | |||
#if CONFIG_CBS_H265 | |||
&ff_cbs_type_h265, | |||
#endif | |||
#if CONFIG_CBS_MPEG2 | |||
&ff_cbs_type_mpeg2, | |||
#endif | |||
}; | |||
int ff_cbs_init(CodedBitstreamContext *ctx, | |||
@@ -82,6 +82,7 @@ int ff_cbs_write_unsigned(CodedBitstreamContext *ctx, PutBitContext *pbc, | |||
extern const CodedBitstreamType ff_cbs_type_h264; | |||
extern const CodedBitstreamType ff_cbs_type_h265; | |||
extern const CodedBitstreamType ff_cbs_type_mpeg2; | |||
#endif /* AVCODEC_CBS_INTERNAL_H */ |
@@ -0,0 +1,409 @@ | |||
/* | |||
* This file is part of Libav. | |||
* | |||
* Libav is free software; you can redistribute it and/or | |||
* modify it under the terms of the GNU Lesser General Public | |||
* License as published by the Free Software Foundation; either | |||
* version 2.1 of the License, or (at your option) any later version. | |||
* | |||
* Libav 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 | |||
* Lesser General Public License for more details. | |||
* | |||
* You should have received a copy of the GNU Lesser General Public | |||
* License along with Libav; if not, write to the Free Software | |||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||
*/ | |||
#include "libavutil/avassert.h" | |||
#include "cbs.h" | |||
#include "cbs_internal.h" | |||
#include "cbs_mpeg2.h" | |||
#include "internal.h" | |||
#define HEADER(name) do { \ | |||
ff_cbs_trace_header(ctx, name); \ | |||
} while (0) | |||
#define CHECK(call) do { \ | |||
err = (call); \ | |||
if (err < 0) \ | |||
return err; \ | |||
} while (0) | |||
#define FUNC_NAME(rw, codec, name) cbs_ ## codec ## _ ## rw ## _ ## name | |||
#define FUNC_MPEG2(rw, name) FUNC_NAME(rw, mpeg2, name) | |||
#define FUNC(name) FUNC_MPEG2(READWRITE, name) | |||
#define READ | |||
#define READWRITE read | |||
#define RWContext BitstreamContext | |||
#define xui(width, name, var) do { \ | |||
uint32_t value = 0; \ | |||
CHECK(ff_cbs_read_unsigned(ctx, rw, width, #name, \ | |||
&value, 0, (1 << width) - 1)); \ | |||
var = value; \ | |||
} while (0) | |||
#define ui(width, name) \ | |||
xui(width, name, current->name) | |||
#define marker_bit() do { \ | |||
av_unused int one = 1; \ | |||
CHECK(ff_cbs_read_unsigned(ctx, rw, 1, "marker_bit", &one, 1, 1)); \ | |||
} while (0) | |||
#define nextbits(width, compare, var) (var = bitstream_peek(rw, width), \ | |||
var == (compare)) | |||
#include "cbs_mpeg2_syntax_template.c" | |||
#undef READ | |||
#undef READWRITE | |||
#undef RWContext | |||
#undef xui | |||
#undef ui | |||
#undef marker_bit | |||
#undef nextbits | |||
#define WRITE | |||
#define READWRITE write | |||
#define RWContext PutBitContext | |||
#define xui(width, name, var) do { \ | |||
CHECK(ff_cbs_write_unsigned(ctx, rw, width, #name, \ | |||
var, 0, (1 << width) - 1)); \ | |||
} while (0) | |||
#define ui(width, name) \ | |||
xui(width, name, current->name) | |||
#define marker_bit() do { \ | |||
CHECK(ff_cbs_write_unsigned(ctx, rw, 1, "marker_bit", 1, 1, 1)); \ | |||
} while (0) | |||
#define nextbits(width, compare, var) (var) | |||
#include "cbs_mpeg2_syntax_template.c" | |||
#undef READ | |||
#undef READWRITE | |||
#undef RWContext | |||
#undef xui | |||
#undef ui | |||
#undef marker_bit | |||
#undef nextbits | |||
static int cbs_mpeg2_split_fragment(CodedBitstreamContext *ctx, | |||
CodedBitstreamFragment *frag, | |||
int header) | |||
{ | |||
const uint8_t *start, *end; | |||
uint8_t *unit_data; | |||
uint32_t start_code = -1, next_start_code = -1; | |||
size_t unit_size; | |||
int err, i, unit_type; | |||
start = avpriv_find_start_code(frag->data, frag->data + frag->data_size, | |||
&start_code); | |||
for (i = 0;; i++) { | |||
end = avpriv_find_start_code(start, frag->data + frag->data_size, | |||
&next_start_code); | |||
unit_type = start_code & 0xff; | |||
// The start and end pointers point at to the byte following the | |||
// start_code_identifier in the start code that they found. | |||
if (end == frag->data + frag->data_size) { | |||
// We didn't find a start code, so this is the final unit. | |||
unit_size = end - (start - 1); | |||
} else { | |||
// Unit runs from start to the beginning of the start code | |||
// pointed to by end (including any padding zeroes). | |||
unit_size = (end - 4) - (start - 1); | |||
} | |||
unit_data = av_malloc(unit_size); | |||
if (!unit_data) | |||
return AVERROR(ENOMEM); | |||
memcpy(unit_data, start - 1, unit_size); | |||
err = ff_cbs_insert_unit_data(ctx, frag, i, unit_type, | |||
unit_data, unit_size); | |||
if (err < 0) { | |||
av_freep(&unit_data); | |||
return err; | |||
} | |||
if (end == frag->data + frag->data_size) | |||
break; | |||
start_code = next_start_code; | |||
start = end; | |||
} | |||
return 0; | |||
} | |||
static int cbs_mpeg2_read_unit(CodedBitstreamContext *ctx, | |||
CodedBitstreamUnit *unit) | |||
{ | |||
BitstreamContext bc; | |||
int err; | |||
err = bitstream_init(&bc, unit->data, 8 * unit->data_size); | |||
if (err < 0) | |||
return err; | |||
if (MPEG2_START_IS_SLICE(unit->type)) { | |||
MPEG2RawSlice *slice; | |||
int pos, len; | |||
slice = av_mallocz(sizeof(*slice)); | |||
if (!slice) | |||
return AVERROR(ENOMEM); | |||
err = cbs_mpeg2_read_slice_header(ctx, &bc, &slice->header); | |||
if (err < 0) { | |||
av_free(slice); | |||
return err; | |||
} | |||
pos = bitstream_tell(&bc); | |||
len = unit->data_size; | |||
slice->data_size = len - pos / 8; | |||
slice->data = av_malloc(slice->data_size); | |||
if (!slice->data) { | |||
av_free(slice); | |||
return AVERROR(ENOMEM); | |||
} | |||
memcpy(slice->data, | |||
unit->data + pos / 8, slice->data_size); | |||
slice->data_bit_start = pos % 8; | |||
unit->content = slice; | |||
} else { | |||
switch (unit->type) { | |||
#define START(start_code, type, func) \ | |||
case start_code: \ | |||
{ \ | |||
type *header; \ | |||
header = av_mallocz(sizeof(*header)); \ | |||
if (!header) \ | |||
return AVERROR(ENOMEM); \ | |||
err = cbs_mpeg2_read_ ## func(ctx, &bc, header); \ | |||
if (err < 0) { \ | |||
av_free(header); \ | |||
return err; \ | |||
} \ | |||
unit->content = header; \ | |||
} \ | |||
break; | |||
START(0x00, MPEG2RawPictureHeader, picture_header); | |||
START(0xb2, MPEG2RawUserData, user_data); | |||
START(0xb3, MPEG2RawSequenceHeader, sequence_header); | |||
START(0xb5, MPEG2RawExtensionData, extension_data); | |||
START(0xb8, MPEG2RawGroupOfPicturesHeader, group_of_pictures_header); | |||
#undef START | |||
default: | |||
av_log(ctx->log_ctx, AV_LOG_ERROR, "Unknown start code %02x.\n", | |||
unit->type); | |||
return AVERROR_INVALIDDATA; | |||
} | |||
} | |||
return 0; | |||
} | |||
static int cbs_mpeg2_write_header(CodedBitstreamContext *ctx, | |||
CodedBitstreamUnit *unit, | |||
PutBitContext *pbc) | |||
{ | |||
int err; | |||
switch (unit->type) { | |||
#define START(start_code, type, func) \ | |||
case start_code: \ | |||
err = cbs_mpeg2_write_ ## func(ctx, pbc, unit->content); \ | |||
break; | |||
START(0x00, MPEG2RawPictureHeader, picture_header); | |||
START(0xb2, MPEG2RawUserData, user_data); | |||
START(0xb3, MPEG2RawSequenceHeader, sequence_header); | |||
START(0xb5, MPEG2RawExtensionData, extension_data); | |||
START(0xb8, MPEG2RawGroupOfPicturesHeader, group_of_pictures_header); | |||
#undef START | |||
default: | |||
av_log(ctx->log_ctx, AV_LOG_ERROR, "Write unimplemented for start " | |||
"code %02x.\n", unit->type); | |||
return AVERROR_PATCHWELCOME; | |||
} | |||
return err; | |||
} | |||
static int cbs_mpeg2_write_slice(CodedBitstreamContext *ctx, | |||
CodedBitstreamUnit *unit, | |||
PutBitContext *pbc) | |||
{ | |||
MPEG2RawSlice *slice = unit->content; | |||
BitstreamContext bc; | |||
size_t bits_left; | |||
int err; | |||
err = cbs_mpeg2_write_slice_header(ctx, pbc, &slice->header); | |||
if (err < 0) | |||
return err; | |||
if (slice->data) { | |||
if (slice->data_size * 8 + 8 > put_bits_left(pbc)) | |||
return AVERROR(ENOSPC); | |||
bitstream_init(&bc, slice->data, slice->data_size * 8); | |||
bitstream_skip(&bc, slice->data_bit_start); | |||
while (bitstream_bits_left(&bc) > 15) | |||
put_bits(pbc, 16, bitstream_read(&bc, 16)); | |||
bits_left = bitstream_bits_left(&bc); | |||
put_bits(pbc, bits_left, bitstream_read(&bc, bits_left)); | |||
// Align with zeroes. | |||
while (put_bits_count(pbc) % 8 != 0) | |||
put_bits(pbc, 1, 0); | |||
} | |||
return 0; | |||
} | |||
static int cbs_mpeg2_write_unit(CodedBitstreamContext *ctx, | |||
CodedBitstreamUnit *unit) | |||
{ | |||
CodedBitstreamMPEG2Context *priv = ctx->priv_data; | |||
PutBitContext pbc; | |||
int err; | |||
if (!priv->write_buffer) { | |||
// Initial write buffer size is 1MB. | |||
priv->write_buffer_size = 1024 * 1024; | |||
reallocate_and_try_again: | |||
err = av_reallocp(&priv->write_buffer, priv->write_buffer_size); | |||
if (err < 0) { | |||
av_log(ctx->log_ctx, AV_LOG_ERROR, "Unable to allocate a " | |||
"sufficiently large write buffer (last attempt " | |||
"%zu bytes).\n", priv->write_buffer_size); | |||
return err; | |||
} | |||
} | |||
init_put_bits(&pbc, priv->write_buffer, priv->write_buffer_size); | |||
if (unit->type >= 0x01 && unit->type <= 0xaf) | |||
err = cbs_mpeg2_write_slice(ctx, unit, &pbc); | |||
else | |||
err = cbs_mpeg2_write_header(ctx, unit, &pbc); | |||
if (err == AVERROR(ENOSPC)) { | |||
// Overflow. | |||
priv->write_buffer_size *= 2; | |||
goto reallocate_and_try_again; | |||
} | |||
if (err < 0) { | |||
// Write failed for some other reason. | |||
return err; | |||
} | |||
if (put_bits_count(&pbc) % 8) | |||
unit->data_bit_padding = 8 - put_bits_count(&pbc) % 8; | |||
else | |||
unit->data_bit_padding = 0; | |||
unit->data_size = (put_bits_count(&pbc) + 7) / 8; | |||
flush_put_bits(&pbc); | |||
err = av_reallocp(&unit->data, unit->data_size); | |||
if (err < 0) | |||
return err; | |||
memcpy(unit->data, priv->write_buffer, unit->data_size); | |||
return 0; | |||
} | |||
static int cbs_mpeg2_assemble_fragment(CodedBitstreamContext *ctx, | |||
CodedBitstreamFragment *frag) | |||
{ | |||
uint8_t *data; | |||
size_t size, dp, sp; | |||
int i; | |||
size = 0; | |||
for (i = 0; i < frag->nb_units; i++) | |||
size += 3 + frag->units[i].data_size; | |||
data = av_malloc(size); | |||
if (!data) | |||
return AVERROR(ENOMEM); | |||
dp = 0; | |||
for (i = 0; i < frag->nb_units; i++) { | |||
CodedBitstreamUnit *unit = &frag->units[i]; | |||
data[dp++] = 0; | |||
data[dp++] = 0; | |||
data[dp++] = 1; | |||
for (sp = 0; sp < unit->data_size; sp++) | |||
data[dp++] = unit->data[sp]; | |||
} | |||
av_assert0(dp == size); | |||
frag->data = data; | |||
frag->data_size = size; | |||
return 0; | |||
} | |||
static void cbs_mpeg2_free_unit(CodedBitstreamUnit *unit) | |||
{ | |||
if (MPEG2_START_IS_SLICE(unit->type)) { | |||
MPEG2RawSlice *slice = unit->content; | |||
av_freep(&slice->data); | |||
av_freep(&slice->header.extra_information); | |||
} else if (unit->type == MPEG2_START_USER_DATA) { | |||
MPEG2RawUserData *user = unit->content; | |||
av_freep(&user->user_data); | |||
} | |||
av_freep(&unit->content); | |||
} | |||
static void cbs_mpeg2_close(CodedBitstreamContext *ctx) | |||
{ | |||
CodedBitstreamMPEG2Context *priv = ctx->priv_data; | |||
av_freep(&priv->write_buffer); | |||
} | |||
const CodedBitstreamType ff_cbs_type_mpeg2 = { | |||
.codec_id = AV_CODEC_ID_MPEG2VIDEO, | |||
.priv_data_size = sizeof(CodedBitstreamMPEG2Context), | |||
.split_fragment = &cbs_mpeg2_split_fragment, | |||
.read_unit = &cbs_mpeg2_read_unit, | |||
.write_unit = &cbs_mpeg2_write_unit, | |||
.assemble_fragment = &cbs_mpeg2_assemble_fragment, | |||
.free_unit = &cbs_mpeg2_free_unit, | |||
.close = &cbs_mpeg2_close, | |||
}; |
@@ -0,0 +1,216 @@ | |||
/* | |||
* This file is part of Libav. | |||
* | |||
* Libav is free software; you can redistribute it and/or | |||
* modify it under the terms of the GNU Lesser General Public | |||
* License as published by the Free Software Foundation; either | |||
* version 2.1 of the License, or (at your option) any later version. | |||
* | |||
* Libav 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 | |||
* Lesser General Public License for more details. | |||
* | |||
* You should have received a copy of the GNU Lesser General Public | |||
* License along with Libav; if not, write to the Free Software | |||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||
*/ | |||
#ifndef AVCODEC_CBS_MPEG2_H | |||
#define AVCODEC_CBS_MPEG2_H | |||
#include <stddef.h> | |||
#include <stdint.h> | |||
enum { | |||
MPEG2_START_PICTURE = 0x00, | |||
MPEG2_START_SLICE_MIN = 0x01, | |||
MPEG2_START_SLICE_MAX = 0xaf, | |||
MPEG2_START_USER_DATA = 0xb2, | |||
MPEG2_START_SEQUENCE_HEADER = 0xb3, | |||
MPEG2_START_SEQUENCE_ERROR = 0xb4, | |||
MPEG2_START_EXTENSION = 0xb5, | |||
MPEG2_START_SEQUENCE_END = 0xb7, | |||
MPEG2_START_GROUP = 0xb8, | |||
}; | |||
#define MPEG2_START_IS_SLICE(type) \ | |||
((type) >= MPEG2_START_SLICE_MIN && \ | |||
(type) <= MPEG2_START_SLICE_MAX) | |||
enum { | |||
MPEG2_EXTENSION_SEQUENCE = 0x1, | |||
MPEG2_EXTENSION_SEQUENCE_DISPLAY = 0x2, | |||
MPEG2_EXTENSION_QUANT_MATRIX = 0x3, | |||
MPEG2_EXTENSION_COPYRIGHT = 0x4, | |||
MPEG2_EXTENSION_SEQUENCE_SCALABLE = 0x5, | |||
MPEG2_EXTENSION_PICTURE_DISPLAY = 0x7, | |||
MPEG2_EXTENSION_PICTURE_CODING = 0x8, | |||
MPEG2_EXTENSION_PICTURE_SPATIAL_SCALABLE = 0x9, | |||
MPEG2_EXTENSION_PICTURE_TEMPORAL_SCALABLE = 0xa, | |||
MPEG2_EXTENSION_CAMAERA_PARAMETERS = 0xb, | |||
MPEG2_EXTENSION_ITU_T = 0xc, | |||
}; | |||
typedef struct MPEG2RawSequenceHeader { | |||
uint8_t sequence_header_code; | |||
uint16_t horizontal_size_value; | |||
uint16_t vertical_size_value; | |||
uint8_t aspect_ratio_information; | |||
uint8_t frame_rate_code; | |||
uint32_t bit_rate_value; | |||
uint16_t vbv_buffer_size_value; | |||
uint8_t constrained_parameters_flag; | |||
uint8_t load_intra_quantiser_matrix; | |||
uint8_t intra_quantiser_matrix[64]; | |||
uint8_t load_non_intra_quantiser_matrix; | |||
uint8_t non_intra_quantiser_matrix[64]; | |||
} MPEG2RawSequenceHeader; | |||
typedef struct MPEG2RawUserData { | |||
uint8_t user_data_start_code; | |||
uint8_t *user_data; | |||
size_t user_data_length; | |||
} MPEG2RawUserData; | |||
typedef struct MPEG2RawSequenceExtension { | |||
uint8_t profile_and_level_indication; | |||
uint8_t progressive_sequence; | |||
uint8_t chroma_format; | |||
uint8_t horizontal_size_extension; | |||
uint8_t vertical_size_extension; | |||
uint16_t bit_rate_extension; | |||
uint8_t vbv_buffer_size_extension; | |||
uint8_t low_delay; | |||
uint8_t frame_rate_extension_n; | |||
uint8_t frame_rate_extension_d; | |||
} MPEG2RawSequenceExtension; | |||
typedef struct MPEG2RawSequenceDisplayExtension { | |||
uint8_t video_format; | |||
uint8_t colour_description; | |||
uint8_t colour_primaries; | |||
uint8_t transfer_characteristics; | |||
uint8_t matrix_coefficients; | |||
uint16_t display_horizontal_size; | |||
uint16_t display_vertical_size; | |||
} MPEG2RawSequenceDisplayExtension; | |||
typedef struct MPEG2RawGroupOfPicturesHeader { | |||
uint8_t group_start_code; | |||
uint32_t time_code; | |||
uint8_t closed_gop; | |||
uint8_t broken_link; | |||
} MPEG2RawGroupOfPicturesHeader; | |||
typedef struct MPEG2RawPictureHeader { | |||
uint8_t picture_start_code; | |||
uint16_t temporal_reference; | |||
uint8_t picture_coding_type; | |||
uint16_t vbv_delay; | |||
uint8_t full_pel_forward_vector; | |||
uint8_t forward_f_code; | |||
uint8_t full_pel_backward_vector; | |||
uint8_t backward_f_code; | |||
uint8_t extra_bit_picture; | |||
} MPEG2RawPictureHeader; | |||
typedef struct MPEG2RawPictureCodingExtension { | |||
uint8_t f_code[2][2]; | |||
uint8_t intra_dc_precision; | |||
uint8_t picture_structure; | |||
uint8_t top_field_first; | |||
uint8_t frame_pred_frame_dct; | |||
uint8_t concealment_motion_vectors; | |||
uint8_t q_scale_type; | |||
uint8_t intra_vlc_format; | |||
uint8_t alternate_scan; | |||
uint8_t repeat_first_field; | |||
uint8_t chroma_420_type; | |||
uint8_t progressive_frame; | |||
uint8_t composite_display_flag; | |||
uint8_t v_axis; | |||
uint8_t field_sequence; | |||
uint8_t sub_carrier; | |||
uint8_t burst_amplitude; | |||
uint8_t sub_carrier_phase; | |||
} MPEG2RawPictureCodingExtension; | |||
typedef struct MPEG2RawQuantMatrixExtension { | |||
uint8_t load_intra_quantiser_matrix; | |||
uint8_t intra_quantiser_matrix[64]; | |||
uint8_t load_non_intra_quantiser_matrix; | |||
uint8_t non_intra_quantiser_matrix[64]; | |||
uint8_t load_chroma_intra_quantiser_matrix; | |||
uint8_t chroma_intra_quantiser_matrix[64]; | |||
uint8_t load_chroma_non_intra_quantiser_matrix; | |||
uint8_t chroma_non_intra_quantiser_matrix[64]; | |||
} MPEG2RawQuantMatrixExtension; | |||
typedef struct MPEG2RawExtensionData { | |||
uint8_t extension_start_code; | |||
uint8_t extension_start_code_identifier; | |||
union { | |||
MPEG2RawSequenceExtension sequence; | |||
MPEG2RawSequenceDisplayExtension sequence_display; | |||
MPEG2RawQuantMatrixExtension quant_matrix; | |||
MPEG2RawPictureCodingExtension picture_coding; | |||
} data; | |||
} MPEG2RawExtensionData; | |||
typedef struct MPEG2RawSliceHeader { | |||
uint8_t slice_vertical_position; | |||
uint8_t slice_vertical_position_extension; | |||
uint8_t priority_breakpoint; | |||
uint8_t quantiser_scale_code; | |||
uint8_t slice_extension_flag; | |||
uint8_t intra_slice; | |||
uint8_t slice_picture_id_enable; | |||
uint8_t slice_picture_id; | |||
uint8_t extra_bit_slice; | |||
size_t extra_information_length; | |||
uint8_t *extra_information; | |||
} MPEG2RawSliceHeader; | |||
typedef struct MPEG2RawSlice { | |||
MPEG2RawSliceHeader header; | |||
uint8_t *data; | |||
size_t data_size; | |||
int data_bit_start; | |||
} MPEG2RawSlice; | |||
typedef struct CodedBitstreamMPEG2Context { | |||
// Elements stored in headers which are required for other decoding. | |||
uint16_t horizontal_size; | |||
uint16_t vertical_size; | |||
uint8_t scalable; | |||
uint8_t scalable_mode; | |||
// Write buffer. | |||
uint8_t *write_buffer; | |||
size_t write_buffer_size; | |||
} CodedBitstreamMPEG2Context; | |||
#endif /* AVCODEC_CBS_MPEG2_H */ |
@@ -0,0 +1,340 @@ | |||
/* | |||
* This file is part of Libav. | |||
* | |||
* Libav is free software; you can redistribute it and/or | |||
* modify it under the terms of the GNU Lesser General Public | |||
* License as published by the Free Software Foundation; either | |||
* version 2.1 of the License, or (at your option) any later version. | |||
* | |||
* Libav 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 | |||
* Lesser General Public License for more details. | |||
* | |||
* You should have received a copy of the GNU Lesser General Public | |||
* License along with Libav; if not, write to the Free Software | |||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||
*/ | |||
static int FUNC(sequence_header)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawSequenceHeader *current) | |||
{ | |||
CodedBitstreamMPEG2Context *mpeg2 = ctx->priv_data; | |||
int err, i; | |||
HEADER("Sequence Header"); | |||
ui(8, sequence_header_code); | |||
ui(12, horizontal_size_value); | |||
ui(12, vertical_size_value); | |||
mpeg2->horizontal_size = current->horizontal_size_value; | |||
mpeg2->vertical_size = current->vertical_size_value; | |||
ui(4, aspect_ratio_information); | |||
ui(4, frame_rate_code); | |||
ui(18, bit_rate_value); | |||
marker_bit(); | |||
ui(10, vbv_buffer_size_value); | |||
ui(1, constrained_parameters_flag); | |||
ui(1, load_intra_quantiser_matrix); | |||
if (current->load_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, intra_quantiser_matrix[i]); | |||
} | |||
ui(1, load_non_intra_quantiser_matrix); | |||
if (current->load_non_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, non_intra_quantiser_matrix[i]); | |||
} | |||
return 0; | |||
} | |||
static int FUNC(user_data)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawUserData *current) | |||
{ | |||
size_t k; | |||
int err; | |||
HEADER("User Data"); | |||
ui(8, user_data_start_code); | |||
#ifdef READ | |||
k = bitstream_bits_left(rw); | |||
av_assert0(k % 8 == 0); | |||
current->user_data_length = k /= 8; | |||
if (k > 0) { | |||
current->user_data = av_malloc(k); | |||
if (!current->user_data) | |||
return AVERROR(ENOMEM); | |||
} | |||
#endif | |||
for (k = 0; k < current->user_data_length; k++) | |||
xui(8, user_data, current->user_data[k]); | |||
return 0; | |||
} | |||
static int FUNC(sequence_extension)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawSequenceExtension *current) | |||
{ | |||
CodedBitstreamMPEG2Context *mpeg2 = ctx->priv_data; | |||
int err; | |||
HEADER("Sequence Extension"); | |||
ui(8, profile_and_level_indication); | |||
ui(1, progressive_sequence); | |||
ui(2, chroma_format); | |||
ui(2, horizontal_size_extension); | |||
ui(2, vertical_size_extension); | |||
mpeg2->horizontal_size = (mpeg2->horizontal_size & 0xfff) | | |||
current->horizontal_size_extension << 12; | |||
mpeg2->vertical_size = (mpeg2->vertical_size & 0xfff) | | |||
current->vertical_size_extension << 12; | |||
ui(12, bit_rate_extension); | |||
marker_bit(); | |||
ui(8, vbv_buffer_size_extension); | |||
ui(1, low_delay); | |||
ui(2, frame_rate_extension_n); | |||
ui(5, frame_rate_extension_d); | |||
return 0; | |||
} | |||
static int FUNC(sequence_display_extension)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawSequenceDisplayExtension *current) | |||
{ | |||
int err; | |||
HEADER("Sequence Display Extension"); | |||
ui(3, video_format); | |||
ui(1, colour_description); | |||
if (current->colour_description) { | |||
ui(8, colour_primaries); | |||
ui(8, transfer_characteristics); | |||
ui(8, matrix_coefficients); | |||
} | |||
ui(14, display_horizontal_size); | |||
marker_bit(); | |||
ui(14, display_vertical_size); | |||
return 0; | |||
} | |||
static int FUNC(group_of_pictures_header)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawGroupOfPicturesHeader *current) | |||
{ | |||
int err; | |||
HEADER("Group of Pictures Header"); | |||
ui(8, group_start_code); | |||
ui(25, time_code); | |||
ui(1, closed_gop); | |||
ui(1, broken_link); | |||
return 0; | |||
} | |||
static int FUNC(picture_header)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawPictureHeader *current) | |||
{ | |||
int err; | |||
HEADER("Picture Header"); | |||
ui(8, picture_start_code); | |||
ui(10, temporal_reference); | |||
ui(3, picture_coding_type); | |||
ui(16, vbv_delay); | |||
if (current->picture_coding_type == 2 || | |||
current->picture_coding_type == 3) { | |||
ui(1, full_pel_forward_vector); | |||
ui(3, forward_f_code); | |||
} | |||
if (current->picture_coding_type == 3) { | |||
ui(1, full_pel_backward_vector); | |||
ui(3, backward_f_code); | |||
} | |||
ui(1, extra_bit_picture); | |||
return 0; | |||
} | |||
static int FUNC(picture_coding_extension)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawPictureCodingExtension *current) | |||
{ | |||
int err; | |||
HEADER("Picture Coding Extension"); | |||
ui(4, f_code[0][0]); | |||
ui(4, f_code[0][1]); | |||
ui(4, f_code[1][0]); | |||
ui(4, f_code[1][1]); | |||
ui(2, intra_dc_precision); | |||
ui(2, picture_structure); | |||
ui(1, top_field_first); | |||
ui(1, frame_pred_frame_dct); | |||
ui(1, concealment_motion_vectors); | |||
ui(1, q_scale_type); | |||
ui(1, intra_vlc_format); | |||
ui(1, alternate_scan); | |||
ui(1, repeat_first_field); | |||
ui(1, chroma_420_type); | |||
ui(1, progressive_frame); | |||
ui(1, composite_display_flag); | |||
if (current->composite_display_flag) { | |||
ui(1, v_axis); | |||
ui(3, field_sequence); | |||
ui(1, sub_carrier); | |||
ui(7, burst_amplitude); | |||
ui(8, sub_carrier_phase); | |||
} | |||
return 0; | |||
} | |||
static int FUNC(quant_matrix_extension)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawQuantMatrixExtension *current) | |||
{ | |||
int err, i; | |||
HEADER("Quant Matrix Extension"); | |||
ui(1, load_intra_quantiser_matrix); | |||
if (current->load_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, intra_quantiser_matrix[i]); | |||
} | |||
ui(1, load_non_intra_quantiser_matrix); | |||
if (current->load_non_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, non_intra_quantiser_matrix[i]); | |||
} | |||
ui(1, load_chroma_intra_quantiser_matrix); | |||
if (current->load_chroma_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, intra_quantiser_matrix[i]); | |||
} | |||
ui(1, load_chroma_non_intra_quantiser_matrix); | |||
if (current->load_chroma_non_intra_quantiser_matrix) { | |||
for (i = 0; i < 64; i++) | |||
ui(8, chroma_non_intra_quantiser_matrix[i]); | |||
} | |||
return 0; | |||
} | |||
static int FUNC(extension_data)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawExtensionData *current) | |||
{ | |||
int err; | |||
HEADER("Extension Data"); | |||
ui(8, extension_start_code); | |||
ui(4, extension_start_code_identifier); | |||
switch (current->extension_start_code_identifier) { | |||
case 1: | |||
return FUNC(sequence_extension) | |||
(ctx, rw, ¤t->data.sequence); | |||
case 2: | |||
return FUNC(sequence_display_extension) | |||
(ctx, rw, ¤t->data.sequence_display); | |||
case 3: | |||
return FUNC(quant_matrix_extension) | |||
(ctx, rw, ¤t->data.quant_matrix); | |||
case 8: | |||
return FUNC(picture_coding_extension) | |||
(ctx, rw, ¤t->data.picture_coding); | |||
default: | |||
av_log(ctx->log_ctx, AV_LOG_ERROR, "Invalid extension ID %d.\n", | |||
current->extension_start_code_identifier); | |||
return AVERROR_INVALIDDATA; | |||
} | |||
} | |||
static int FUNC(slice_header)(CodedBitstreamContext *ctx, RWContext *rw, | |||
MPEG2RawSliceHeader *current) | |||
{ | |||
CodedBitstreamMPEG2Context *mpeg2 = ctx->priv_data; | |||
int err; | |||
HEADER("Slice Header"); | |||
ui(8, slice_vertical_position); | |||
if (mpeg2->vertical_size > 2800) | |||
ui(3, slice_vertical_position_extension); | |||
if (mpeg2->scalable) { | |||
if (mpeg2->scalable_mode == 0) | |||
ui(7, priority_breakpoint); | |||
} | |||
ui(5, quantiser_scale_code); | |||
if (nextbits(1, 1, current->slice_extension_flag)) { | |||
ui(1, slice_extension_flag); | |||
ui(1, intra_slice); | |||
ui(1, slice_picture_id_enable); | |||
ui(6, slice_picture_id); | |||
{ | |||
size_t k; | |||
#ifdef READ | |||
BitstreamContext start; | |||
uint8_t bit; | |||
start = *rw; | |||
for (k = 0; nextbits(1, 1, bit); k++) | |||
bitstream_skip(rw, 8); | |||
current->extra_information_length = k; | |||
if (k > 0) { | |||
*rw = start; | |||
current->extra_information = | |||
av_malloc(current->extra_information_length); | |||
if (!current->extra_information) | |||
return AVERROR(ENOMEM); | |||
for (k = 0; k < current->extra_information_length; k++) { | |||
xui(1, extra_bit_slice, bit); | |||
xui(8, extra_information_slice, | |||
current->extra_information[k]); | |||
} | |||
} | |||
#else | |||
for (k = 0; k < current->extra_information_length; k++) { | |||
xui(1, extra_bit_slice, 1); | |||
xui(8, extra_information_slice, current->extra_information[k]); | |||
} | |||
#endif | |||
} | |||
} | |||
ui(1, extra_bit_slice); | |||
return 0; | |||
} |
@@ -112,6 +112,7 @@ static int trace_headers(AVBSFContext *bsf, AVPacket *out) | |||
static const enum AVCodecID trace_headers_codec_ids[] = { | |||
AV_CODEC_ID_H264, | |||
AV_CODEC_ID_HEVC, | |||
AV_CODEC_ID_MPEG2VIDEO, | |||
AV_CODEC_ID_NONE, | |||
}; | |||