@@ -212,6 +212,8 @@ typedef struct MOVStreamContext {
unsigned drefs_count;
MOVDref *drefs;
int dref_id;
+ unsigned tref_flags;
+ int tref_id;
int timecode_track;
int width; ///< tkhd width
int height; ///< tkhd height
@@ -408,6 +410,7 @@ void ff_mp4_parse_es_descr(AVIOContext *pb, int *es_id);
#define MOV_SAMPLE_DEPENDENCY_YES 0x1
#define MOV_SAMPLE_DEPENDENCY_NO 0x2
+#define MOV_TREF_FLAG_ENHANCEMENT 0x1
#define TAG_IS_AVCI(tag) \
((tag) == MKTAG('a', 'i', '5', 'p') || \
@@ -290,6 +290,8 @@ const AVCodecTag ff_codec_movvideo_tags[] = {
{ AV_CODEC_ID_CFHD, MKTAG('C', 'F', 'H', 'D') },
+ { AV_CODEC_ID_LCEVC, MKTAG('l', 'v', 'c', '1') }, /* LCEVC raw payload */
+
{ AV_CODEC_ID_NONE, 0 },
};
@@ -2434,6 +2434,30 @@ static int mov_read_dvc1(MOVContext *c, AVIOContext *pb, MOVAtom atom)
return 0;
}
+static int mov_read_sbas(MOVContext* c, AVIOContext* pb, MOVAtom atom)
+{
+ AVStream* st;
+ MOVStreamContext* sc;
+
+ if (c->fc->nb_streams < 1)
+ return 0;
+
+ /* For SBAS this should be fine - though beware if someone implements a
+ * tref atom processor that doesn't drop down to default then this may
+ * be lost. */
+ if (atom.size > 4) {
+ av_log(c->fc, AV_LOG_ERROR, "Only a single tref of type sbas is supported\n");
+ return AVERROR_PATCHWELCOME;
+ }
+
+ st = c->fc->streams[c->fc->nb_streams - 1];
+ sc = st->priv_data;
+ sc->tref_id = avio_rb32(pb);
+ sc->tref_flags |= MOV_TREF_FLAG_ENHANCEMENT;
+
+ return 0;
+}
+
/**
* An strf atom is a BITMAPINFOHEADER struct. This struct is 40 bytes itself,
* but can have extradata appended at the end after the 40 bytes belonging
@@ -4995,6 +5019,8 @@ static int mov_read_trak(MOVContext *c, AVIOContext *pb, MOVAtom atom)
st->codecpar->codec_type = AVMEDIA_TYPE_DATA;
sc->ffindex = st->index;
c->trak_index = st->index;
+ sc->tref_flags = 0;
+ sc->tref_id = -1;
sc->refcount = 1;
if ((ret = mov_read_default(c, pb, atom)) < 0)
@@ -9052,6 +9078,7 @@ static const MOVParseTableEntry mov_default_parse_table[] = {
{ MKTAG('a','v','c','C'), mov_read_glbl },
{ MKTAG('p','a','s','p'), mov_read_pasp },
{ MKTAG('c','l','a','p'), mov_read_clap },
+{ MKTAG('s','b','a','s'), mov_read_sbas },
{ MKTAG('s','i','d','x'), mov_read_sidx },
{ MKTAG('s','t','b','l'), mov_read_default },
{ MKTAG('s','t','c','o'), mov_read_stco },
@@ -9132,6 +9159,7 @@ static const MOVParseTableEntry mov_default_parse_table[] = {
{ MKTAG('i','i','n','f'), mov_read_iinf },
{ MKTAG('a','m','v','e'), mov_read_amve }, /* ambient viewing environment box */
{ MKTAG('l','h','v','C'), mov_read_lhvc },
+{ MKTAG('l','v','c','C'), mov_read_glbl },
#if CONFIG_IAMFDEC
{ MKTAG('i','a','c','b'), mov_read_iacb },
#endif
@@ -10029,6 +10057,20 @@ static int mov_parse_tiles(AVFormatContext *s)
return 0;
}
+static AVStream *mov_find_reference_track(AVFormatContext *s, AVStream *st)
+{
+ MOVStreamContext *sc = st->priv_data;
+
+ if (sc->tref_id < 0)
+ return NULL;
+
+ for (int i = 0; i < s->nb_streams; i++)
+ if (s->streams[i]->id == sc->tref_id)
+ return s->streams[i];
+
+ return NULL;
+}
+
static int mov_read_header(AVFormatContext *s)
{
MOVContext *mov = s->priv_data;
@@ -10154,6 +10196,50 @@ static int mov_read_header(AVFormatContext *s)
}
export_orphan_timecode(s);
+ /* Update the LCEVC stream as a data stream. */
+ for (i = 0; i < s->nb_streams; i++) {
+ AVStreamGroup *stg;
+ AVStream *st = s->streams[i];
+ AVStream *st_base;
+ MOVStreamContext *sc = st->priv_data;
+ MOVStreamContext *sc_base;
+
+ /* Find the enhancement stream. */
+ if (!(sc->tref_flags & MOV_TREF_FLAG_ENHANCEMENT))
+ continue;
+
+ st_base = mov_find_reference_track(s, st);
+ if (!st_base) {
+ av_log(s, AV_LOG_ERROR, "Failed to find base stream for enhancement stream\n");
+ return AVERROR_INVALIDDATA;
+ }
+ sc_base = st_base->priv_data;
+
+ /* Make reference bidirectional. */
+ if (sc_base->tref_id >= 0) {
+ av_log(s, AV_LOG_ERROR, "Base stream must not already have a stream reference\n");
+ return AVERROR_INVALIDDATA;
+ }
+
+ st->codecpar->codec_type = AVMEDIA_TYPE_DATA;
+ sc_base->tref_id = st->id;
+
+ stg = avformat_stream_group_create(s, AV_STREAM_GROUP_PARAMS_LCEVC, NULL);
+ if (!stg)
+ return AVERROR(ENOMEM);
+
+ stg->id = st->id;
+ stg->params.lcevc->width = st->codecpar->width;
+ stg->params.lcevc->height = st->codecpar->height;
+
+ err = avformat_stream_group_add_stream(stg, st_base);
+ if (err < 0)
+ return err;
+ err = avformat_stream_group_add_stream(stg, st);
+ if (err < 0)
+ return err;
+ }
+
for (i = 0; i < s->nb_streams; i++) {
AVStream *st = s->streams[i];
FFStream *const sti = ffstream(st);
Co-authored-by: V-Nova Team <systems@v-nova.com> Signed-off-by: James Almer <jamrial@gmail.com> --- libavformat/isom.h | 3 ++ libavformat/isom_tags.c | 2 + libavformat/mov.c | 86 +++++++++++++++++++++++++++++++++++++++++ 3 files changed, 91 insertions(+)