summaryrefslogtreecommitdiff
path: root/libavcodec/cbs_h2645.c
diff options
context:
space:
mode:
Diffstat (limited to 'libavcodec/cbs_h2645.c')
-rw-r--r--libavcodec/cbs_h2645.c410
1 files changed, 407 insertions, 3 deletions
diff --git a/libavcodec/cbs_h2645.c b/libavcodec/cbs_h2645.c
index 2350f039fa..23556f47f8 100644
--- a/libavcodec/cbs_h2645.c
+++ b/libavcodec/cbs_h2645.c
@@ -23,10 +23,12 @@
#include "cbs.h"
#include "cbs_internal.h"
#include "cbs_h264.h"
+#include "cbs_h265.h"
#include "golomb.h"
#include "h264.h"
#include "h264_sei.h"
#include "h2645_parse.h"
+#include "hevc.h"
static int cbs_read_ue_golomb(CodedBitstreamContext *ctx, BitstreamContext *bc,
@@ -237,6 +239,7 @@ static int cbs_write_se_golomb(CodedBitstreamContext *ctx, PutBitContext *pbc,
#define FUNC_NAME(rw, codec, name) cbs_ ## codec ## _ ## rw ## _ ## name
#define FUNC_H264(rw, name) FUNC_NAME(rw, h264, name)
+#define FUNC_H265(rw, name) FUNC_NAME(rw, h265, name)
#define READ
@@ -299,6 +302,10 @@ static int cbs_h2645_read_more_rbsp_data(BitstreamContext *bc)
#include "cbs_h264_syntax_template.c"
#undef FUNC
+#define FUNC(name) FUNC_H265(READWRITE, name)
+#include "cbs_h265_syntax_template.c"
+#undef FUNC
+
#undef READ
#undef READWRITE
#undef RWContext
@@ -368,6 +375,10 @@ static int cbs_h2645_read_more_rbsp_data(BitstreamContext *bc)
#include "cbs_h264_syntax_template.c"
#undef FUNC
+#define FUNC(name) FUNC_H265(READWRITE, name)
+#include "cbs_h265_syntax_template.c"
+#undef FUNC
+
#undef WRITE
#undef READWRITE
#undef RWContext
@@ -428,6 +439,40 @@ static void cbs_h264_free_nal_unit(CodedBitstreamUnit *unit)
av_freep(&unit->content);
}
+static void cbs_h265_free_nal_unit(CodedBitstreamUnit *unit)
+{
+ switch (unit->type) {
+ case HEVC_NAL_VPS:
+ av_freep(&((H265RawVPS*)unit->content)->extension_data.data);
+ break;
+ case HEVC_NAL_SPS:
+ av_freep(&((H265RawSPS*)unit->content)->extension_data.data);
+ break;
+ case HEVC_NAL_PPS:
+ av_freep(&((H265RawPPS*)unit->content)->extension_data.data);
+ break;
+ case HEVC_NAL_TRAIL_N:
+ case HEVC_NAL_TRAIL_R:
+ case HEVC_NAL_TSA_N:
+ case HEVC_NAL_TSA_R:
+ case HEVC_NAL_STSA_N:
+ case HEVC_NAL_STSA_R:
+ case HEVC_NAL_RADL_N:
+ case HEVC_NAL_RADL_R:
+ case HEVC_NAL_RASL_N:
+ case HEVC_NAL_RASL_R:
+ case HEVC_NAL_BLA_W_LP:
+ case HEVC_NAL_BLA_W_RADL:
+ case HEVC_NAL_BLA_N_LP:
+ case HEVC_NAL_IDR_W_RADL:
+ case HEVC_NAL_IDR_N_LP:
+ case HEVC_NAL_CRA_NUT:
+ av_freep(&((H265RawSlice*)unit->content)->data);
+ break;
+ }
+ av_freep(&unit->content);
+}
+
static int cbs_h2645_fragment_add_nals(CodedBitstreamContext *ctx,
CodedBitstreamFragment *frag,
const H2645Packet *packet)
@@ -542,6 +587,58 @@ static int cbs_h2645_split_fragment(CodedBitstreamContext *ctx,
"header.\n", bytestream2_get_bytes_left(&gbc));
}
+ } else if (header && frag->data[0] && codec_id == AV_CODEC_ID_HEVC) {
+ // HVCC header.
+ size_t size, start, end;
+ int i, j, nb_arrays, nal_unit_type, nb_nals, version;
+
+ priv->mp4 = 1;
+
+ bytestream2_init(&gbc, frag->data, frag->data_size);
+
+ if (bytestream2_get_bytes_left(&gbc) < 23)
+ return AVERROR_INVALIDDATA;
+
+ version = bytestream2_get_byte(&gbc);
+ if (version != 1) {
+ av_log(ctx->log_ctx, AV_LOG_ERROR, "Invalid HVCC header: "
+ "first byte %u.", version);
+ return AVERROR_INVALIDDATA;
+ }
+
+ bytestream2_skip(&gbc, 20);
+ priv->nal_length_size = (bytestream2_get_byte(&gbc) & 3) + 1;
+
+ nb_arrays = bytestream2_get_byte(&gbc);
+ for (i = 0; i < nb_arrays; i++) {
+ nal_unit_type = bytestream2_get_byte(&gbc) & 0x3f;
+ nb_nals = bytestream2_get_be16(&gbc);
+
+ start = bytestream2_tell(&gbc);
+ for (j = 0; j < nb_nals; j++) {
+ if (bytestream2_get_bytes_left(&gbc) < 2)
+ return AVERROR_INVALIDDATA;
+ size = bytestream2_get_be16(&gbc);
+ if (bytestream2_get_bytes_left(&gbc) < size)
+ return AVERROR_INVALIDDATA;
+ bytestream2_skip(&gbc, size);
+ }
+ end = bytestream2_tell(&gbc);
+
+ err = ff_h2645_packet_split(&priv->read_packet,
+ frag->data + start, end - start,
+ ctx->log_ctx, 1, 2, AV_CODEC_ID_HEVC);
+ if (err < 0) {
+ av_log(ctx->log_ctx, AV_LOG_ERROR, "Failed to split "
+ "HVCC array %d (%d NAL units of type %d).\n",
+ i, nb_nals, nal_unit_type);
+ return err;
+ }
+ err = cbs_h2645_fragment_add_nals(ctx, frag, &priv->read_packet);
+ if (err < 0)
+ return err;
+ }
+
} else {
// Annex B, or later MP4 with already-known parameters.
@@ -582,6 +679,9 @@ static int cbs_h26 ## h26n ## _replace_ ## ps_var(CodedBitstreamContext *ctx, \
cbs_h2645_replace_ps(4, SPS, sps, seq_parameter_set_id)
cbs_h2645_replace_ps(4, PPS, pps, pic_parameter_set_id)
+cbs_h2645_replace_ps(5, VPS, vps, vps_video_parameter_set_id)
+cbs_h2645_replace_ps(5, SPS, sps, sps_seq_parameter_set_id)
+cbs_h2645_replace_ps(5, PPS, pps, pps_pic_parameter_set_id)
static int cbs_h264_read_nal_unit(CodedBitstreamContext *ctx,
CodedBitstreamUnit *unit)
@@ -730,6 +830,150 @@ static int cbs_h264_read_nal_unit(CodedBitstreamContext *ctx,
return 0;
}
+static int cbs_h265_read_nal_unit(CodedBitstreamContext *ctx,
+ CodedBitstreamUnit *unit)
+{
+ BitstreamContext bc;
+ int err;
+
+ err = bitstream_init(&bc, unit->data, 8 * unit->data_size);
+ if (err < 0)
+ return err;
+
+ switch (unit->type) {
+ case HEVC_NAL_VPS:
+ {
+ H265RawVPS *vps;
+
+ vps = av_mallocz(sizeof(*vps));
+ if (!vps)
+ return AVERROR(ENOMEM);
+ err = cbs_h265_read_vps(ctx, &bc, vps);
+ if (err >= 0)
+ err = cbs_h265_replace_vps(ctx, vps);
+ if (err < 0) {
+ av_free(vps);
+ return err;
+ }
+
+ unit->content = vps;
+ }
+ break;
+ case HEVC_NAL_SPS:
+ {
+ H265RawSPS *sps;
+
+ sps = av_mallocz(sizeof(*sps));
+ if (!sps)
+ return AVERROR(ENOMEM);
+ err = cbs_h265_read_sps(ctx, &bc, sps);
+ if (err >= 0)
+ err = cbs_h265_replace_sps(ctx, sps);
+ if (err < 0) {
+ av_free(sps);
+ return err;
+ }
+
+ unit->content = sps;
+ }
+ break;
+
+ case HEVC_NAL_PPS:
+ {
+ H265RawPPS *pps;
+
+ pps = av_mallocz(sizeof(*pps));
+ if (!pps)
+ return AVERROR(ENOMEM);
+ err = cbs_h265_read_pps(ctx, &bc, pps);
+ if (err >= 0)
+ err = cbs_h265_replace_pps(ctx, pps);
+ if (err < 0) {
+ av_free(pps);
+ return err;
+ }
+
+ unit->content = pps;
+ }
+ break;
+
+ case HEVC_NAL_TRAIL_N:
+ case HEVC_NAL_TRAIL_R:
+ case HEVC_NAL_TSA_N:
+ case HEVC_NAL_TSA_R:
+ case HEVC_NAL_STSA_N:
+ case HEVC_NAL_STSA_R:
+ case HEVC_NAL_RADL_N:
+ case HEVC_NAL_RADL_R:
+ case HEVC_NAL_RASL_N:
+ case HEVC_NAL_RASL_R:
+ case HEVC_NAL_BLA_W_LP:
+ case HEVC_NAL_BLA_W_RADL:
+ case HEVC_NAL_BLA_N_LP:
+ case HEVC_NAL_IDR_W_RADL:
+ case HEVC_NAL_IDR_N_LP:
+ case HEVC_NAL_CRA_NUT:
+ {
+ H265RawSlice *slice;
+ int pos, len;
+
+ slice = av_mallocz(sizeof(*slice));
+ if (!slice)
+ return AVERROR(ENOMEM);
+ err = cbs_h265_read_slice_segment_header(ctx, &bc, &slice->header);
+ if (err < 0) {
+ av_free(slice);
+ return err;
+ }
+
+ pos = bitstream_tell(&bc);
+ len = unit->data_size;
+ if (!unit->data[len - 1]) {
+ int z;
+ for (z = 0; z < len && !unit->data[len - z - 1]; z++);
+ av_log(ctx->log_ctx, AV_LOG_DEBUG, "Deleted %d trailing zeroes "
+ "from slice data.\n", z);
+ len -= z;
+ }
+
+ 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;
+ }
+ break;
+
+ case HEVC_NAL_AUD:
+ {
+ H265RawAUD *aud;
+
+ aud = av_mallocz(sizeof(*aud));
+ if (!aud)
+ return AVERROR(ENOMEM);
+ err = cbs_h265_read_aud(ctx, &bc, aud);
+ if (err < 0) {
+ av_free(aud);
+ return err;
+ }
+
+ unit->content = aud;
+ }
+ break;
+
+ default:
+ return AVERROR(ENOSYS);
+ }
+
+ return 0;
+}
+
static int cbs_h264_write_nal_unit(CodedBitstreamContext *ctx,
CodedBitstreamUnit *unit,
PutBitContext *pbc)
@@ -842,6 +1086,127 @@ static int cbs_h264_write_nal_unit(CodedBitstreamContext *ctx,
return 0;
}
+static int cbs_h265_write_nal_unit(CodedBitstreamContext *ctx,
+ CodedBitstreamUnit *unit,
+ PutBitContext *pbc)
+{
+ int err;
+
+ switch (unit->type) {
+ case HEVC_NAL_VPS:
+ {
+ H265RawVPS *vps = unit->content;
+
+ err = cbs_h265_write_vps(ctx, pbc, vps);
+ if (err < 0)
+ return err;
+
+ err = cbs_h265_replace_vps(ctx, vps);
+ if (err < 0)
+ return err;
+ }
+ break;
+
+ case HEVC_NAL_SPS:
+ {
+ H265RawSPS *sps = unit->content;
+
+ err = cbs_h265_write_sps(ctx, pbc, sps);
+ if (err < 0)
+ return err;
+
+ err = cbs_h265_replace_sps(ctx, sps);
+ if (err < 0)
+ return err;
+ }
+ break;
+
+ case HEVC_NAL_PPS:
+ {
+ H265RawPPS *pps = unit->content;
+
+ err = cbs_h265_write_pps(ctx, pbc, pps);
+ if (err < 0)
+ return err;
+
+ err = cbs_h265_replace_pps(ctx, pps);
+ if (err < 0)
+ return err;
+ }
+ break;
+
+ case HEVC_NAL_TRAIL_N:
+ case HEVC_NAL_TRAIL_R:
+ case HEVC_NAL_TSA_N:
+ case HEVC_NAL_TSA_R:
+ case HEVC_NAL_STSA_N:
+ case HEVC_NAL_STSA_R:
+ case HEVC_NAL_RADL_N:
+ case HEVC_NAL_RADL_R:
+ case HEVC_NAL_RASL_N:
+ case HEVC_NAL_RASL_R:
+ case HEVC_NAL_BLA_W_LP:
+ case HEVC_NAL_BLA_W_RADL:
+ case HEVC_NAL_BLA_N_LP:
+ case HEVC_NAL_IDR_W_RADL:
+ case HEVC_NAL_IDR_N_LP:
+ case HEVC_NAL_CRA_NUT:
+ {
+ H265RawSlice *slice = unit->content;
+ BitstreamContext bc;
+ int bits_left, end, zeroes;
+
+ err = cbs_h265_write_slice_segment_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);
+
+ // Copy in two-byte blocks, but stop before copying the
+ // rbsp_stop_one_bit in the final byte.
+ while (bitstream_bits_left(&bc) > 23)
+ put_bits(pbc, 16, bitstream_read(&bc, 16));
+
+ bits_left = bitstream_bits_left(&bc);
+ end = bitstream_read(&bc, bits_left);
+
+ // rbsp_stop_one_bit must be present here.
+ av_assert0(end);
+ zeroes = ff_ctz(end);
+ if (bits_left > zeroes + 1)
+ put_bits(pbc, bits_left - zeroes - 1,
+ end >> (zeroes + 1));
+ put_bits(pbc, 1, 1);
+ while (put_bits_count(pbc) % 8 != 0)
+ put_bits(pbc, 1, 0);
+ } else {
+ // No slice data - that was just the header.
+ }
+ }
+ break;
+
+ case HEVC_NAL_AUD:
+ {
+ err = cbs_h265_write_aud(ctx, pbc, unit->content);
+ if (err < 0)
+ return err;
+ }
+ break;
+
+ default:
+ av_log(ctx->log_ctx, AV_LOG_ERROR, "Write unimplemented for "
+ "NAL unit type %d.\n", unit->type);
+ return AVERROR_PATCHWELCOME;
+ }
+
+ return 0;
+}
+
static int cbs_h2645_write_nal_unit(CodedBitstreamContext *ctx,
CodedBitstreamUnit *unit)
{
@@ -866,7 +1231,10 @@ static int cbs_h2645_write_nal_unit(CodedBitstreamContext *ctx,
init_put_bits(&pbc, priv->write_buffer, priv->write_buffer_size);
- err = cbs_h264_write_nal_unit(ctx, unit, &pbc);
+ if (codec_id == AV_CODEC_ID_H264)
+ err = cbs_h264_write_nal_unit(ctx, unit, &pbc);
+ else
+ err = cbs_h265_write_nal_unit(ctx, unit, &pbc);
if (err == AVERROR(ENOSPC)) {
// Overflow.
@@ -927,8 +1295,13 @@ static int cbs_h2645_assemble_fragment(CodedBitstreamContext *ctx,
frag->data_bit_padding = unit->data_bit_padding;
}
- if (unit->type == H264_NAL_SPS ||
- unit->type == H264_NAL_PPS ||
+ if ((ctx->codec->codec_id == AV_CODEC_ID_H264 &&
+ (unit->type == H264_NAL_SPS ||
+ unit->type == H264_NAL_PPS)) ||
+ (ctx->codec->codec_id == AV_CODEC_ID_HEVC &&
+ (unit->type == HEVC_NAL_VPS ||
+ unit->type == HEVC_NAL_SPS ||
+ unit->type == HEVC_NAL_PPS)) ||
i == 0 /* (Assume this is the start of an access unit.) */) {
// zero_byte
data[dp++] = 0;
@@ -982,6 +1355,23 @@ static void cbs_h264_close(CodedBitstreamContext *ctx)
av_freep(&h264->pps[i]);
}
+static void cbs_h265_close(CodedBitstreamContext *ctx)
+{
+ CodedBitstreamH265Context *h265 = ctx->priv_data;
+ int i;
+
+ ff_h2645_packet_uninit(&h265->common.read_packet);
+
+ av_freep(&h265->common.write_buffer);
+
+ for (i = 0; i < FF_ARRAY_ELEMS(h265->vps); i++)
+ av_freep(&h265->vps[i]);
+ for (i = 0; i < FF_ARRAY_ELEMS(h265->sps); i++)
+ av_freep(&h265->sps[i]);
+ for (i = 0; i < FF_ARRAY_ELEMS(h265->pps); i++)
+ av_freep(&h265->pps[i]);
+}
+
const CodedBitstreamType ff_cbs_type_h264 = {
.codec_id = AV_CODEC_ID_H264,
@@ -995,3 +1385,17 @@ const CodedBitstreamType ff_cbs_type_h264 = {
.free_unit = &cbs_h264_free_nal_unit,
.close = &cbs_h264_close,
};
+
+const CodedBitstreamType ff_cbs_type_h265 = {
+ .codec_id = AV_CODEC_ID_HEVC,
+
+ .priv_data_size = sizeof(CodedBitstreamH265Context),
+
+ .split_fragment = &cbs_h2645_split_fragment,
+ .read_unit = &cbs_h265_read_nal_unit,
+ .write_unit = &cbs_h2645_write_nal_unit,
+ .assemble_fragment = &cbs_h2645_assemble_fragment,
+
+ .free_unit = &cbs_h265_free_nal_unit,
+ .close = &cbs_h265_close,
+};