X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=libavformat%2Fapngdec.c;h=07a21e6d43669f9f9caddbdba5b56c6078f2181f;hb=444e65299ba5cbf390439e27ce2cb91dff0e5aa7;hp=bb17896ee50ef73f0050659e6c7c4508ae20fc83;hpb=ecf65c30cf7545c72c0e27b32283fc9c55b03544;p=ffmpeg diff --git a/libavformat/apngdec.c b/libavformat/apngdec.c index bb17896ee50..07a21e6d436 100644 --- a/libavformat/apngdec.c +++ b/libavformat/apngdec.c @@ -49,6 +49,10 @@ typedef struct APNGDemuxContext { int is_key_frame; + uint8_t *extra_data; + int extra_data_size; + int extra_data_updated; + /* * loop options */ @@ -122,9 +126,9 @@ end: return AVPROBE_SCORE_MAX; } -static int append_extradata(AVCodecParameters *par, AVIOContext *pb, int len) +static int append_extradata(APNGDemuxContext *ctx, AVIOContext *pb, int len) { - int previous_size = par->extradata_size; + int previous_size = ctx->extra_data_size; int new_size, ret; uint8_t *new_extradata; @@ -132,18 +136,30 @@ static int append_extradata(AVCodecParameters *par, AVIOContext *pb, int len) return AVERROR_INVALIDDATA; new_size = previous_size + len; - new_extradata = av_realloc(par->extradata, new_size + AV_INPUT_BUFFER_PADDING_SIZE); + new_extradata = av_realloc(ctx->extra_data, new_size + AV_INPUT_BUFFER_PADDING_SIZE); if (!new_extradata) return AVERROR(ENOMEM); - par->extradata = new_extradata; - par->extradata_size = new_size; + ctx->extra_data = new_extradata; + ctx->extra_data_size = new_size; - if ((ret = avio_read(pb, par->extradata + previous_size, len)) < 0) + if ((ret = avio_read(pb, ctx->extra_data + previous_size, len)) < 0) return ret; return previous_size; } +static int send_extradata(APNGDemuxContext *ctx, AVPacket *pkt) +{ + if (!ctx->extra_data_updated) { + uint8_t *side_data = av_packet_new_side_data(pkt, AV_PKT_DATA_NEW_EXTRADATA, ctx->extra_data_size); + if (!side_data) + return AVERROR(ENOMEM); + memcpy(side_data, ctx->extra_data, ctx->extra_data_size); + ctx->extra_data_updated = 1; + } + return 0; +} + static int apng_read_header(AVFormatContext *s) { APNGDemuxContext *ctx = s->priv_data; @@ -178,15 +194,15 @@ static int apng_read_header(AVFormatContext *s) return ret; /* extradata will contain every chunk up to the first fcTL (excluded) */ - st->codecpar->extradata = av_malloc(len + 12 + AV_INPUT_BUFFER_PADDING_SIZE); - if (!st->codecpar->extradata) + ctx->extra_data = av_malloc(len + 12 + AV_INPUT_BUFFER_PADDING_SIZE); + if (!ctx->extra_data) return AVERROR(ENOMEM); - st->codecpar->extradata_size = len + 12; - AV_WB32(st->codecpar->extradata, len); - AV_WL32(st->codecpar->extradata+4, tag); - AV_WB32(st->codecpar->extradata+8, st->codecpar->width); - AV_WB32(st->codecpar->extradata+12, st->codecpar->height); - if ((ret = avio_read(pb, st->codecpar->extradata+16, 9)) < 0) + ctx->extra_data_size = len + 12; + AV_WB32(ctx->extra_data, len); + AV_WL32(ctx->extra_data+4, tag); + AV_WB32(ctx->extra_data+8, st->codecpar->width); + AV_WB32(ctx->extra_data+12, st->codecpar->height); + if ((ret = avio_read(pb, ctx->extra_data+16, 9)) < 0) goto fail; while (!avio_feof(pb)) { @@ -218,11 +234,11 @@ static int apng_read_header(AVFormatContext *s) switch (tag) { case MKTAG('a', 'c', 'T', 'L'): if ((ret = avio_seek(pb, -8, SEEK_CUR)) < 0 || - (ret = append_extradata(st->codecpar, pb, len + 12)) < 0) + (ret = append_extradata(ctx, pb, len + 12)) < 0) goto fail; acTL_found = 1; - ctx->num_frames = AV_RB32(st->codecpar->extradata + ret + 8); - ctx->num_play = AV_RB32(st->codecpar->extradata + ret + 12); + ctx->num_frames = AV_RB32(ctx->extra_data + ret + 8); + ctx->num_play = AV_RB32(ctx->extra_data + ret + 12); av_log(s, AV_LOG_DEBUG, "num_frames: %"PRIu32", num_play: %"PRIu32"\n", ctx->num_frames, ctx->num_play); break; @@ -236,15 +252,15 @@ static int apng_read_header(AVFormatContext *s) return 0; default: if ((ret = avio_seek(pb, -8, SEEK_CUR)) < 0 || - (ret = append_extradata(st->codecpar, pb, len + 12)) < 0) + (ret = append_extradata(ctx, pb, len + 12)) < 0) goto fail; } } fail: - if (st->codecpar->extradata_size) { - av_freep(&st->codecpar->extradata); - st->codecpar->extradata_size = 0; + if (ctx->extra_data_size) { + av_freep(&ctx->extra_data); + ctx->extra_data_size = 0; } return ret; } @@ -393,16 +409,16 @@ static int apng_read_packet(AVFormatContext *s, AVPacket *pkt) pkt->pts = ctx->pkt_pts; pkt->duration = ctx->pkt_duration; ctx->pkt_pts += ctx->pkt_duration; - return ret; + return send_extradata(ctx, pkt); case MKTAG('I', 'E', 'N', 'D'): ctx->cur_loop++; if (ctx->ignore_loop || ctx->num_play >= 1 && ctx->cur_loop == ctx->num_play) { avio_seek(pb, -8, SEEK_CUR); return AVERROR_EOF; } - if ((ret = avio_seek(pb, s->streams[0]->codecpar->extradata_size + 8, SEEK_SET)) < 0) + if ((ret = avio_seek(pb, ctx->extra_data_size + 8, SEEK_SET)) < 0) return ret; - return 0; + return send_extradata(ctx, pkt); default: { char tag_buf[32]; @@ -417,6 +433,14 @@ static int apng_read_packet(AVFormatContext *s, AVPacket *pkt) return AVERROR_PATCHWELCOME; } +static int apng_read_close(AVFormatContext *s) +{ + APNGDemuxContext *ctx = s->priv_data; + av_freep(&ctx->extra_data); + ctx->extra_data_size = 0; + return 0; +} + static const AVOption options[] = { { "ignore_loop", "ignore loop setting" , offsetof(APNGDemuxContext, ignore_loop), AV_OPT_TYPE_BOOL, { .i64 = 1 } , 0, 1 , AV_OPT_FLAG_DECODING_PARAM }, @@ -442,6 +466,7 @@ AVInputFormat ff_apng_demuxer = { .read_probe = apng_probe, .read_header = apng_read_header, .read_packet = apng_read_packet, + .read_close = apng_read_close, .flags = AVFMT_GENERIC_INDEX, .priv_class = &demuxer_class, };