aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAndreas Rheinhardt <andreas.rheinhardt@outlook.com>2021-09-23 02:38:49 +0200
committerAndreas Rheinhardt <andreas.rheinhardt@outlook.com>2022-01-11 15:15:53 +0100
commit4d8beeaa8f5746ce243c3cfa95100289b987307e (patch)
treed38e5c57dfb0d9e637ffd798fb927c4f2d6691a0
parent71eee0d4dc55356ce4127908f0d5f6411dc71c6d (diff)
downloadffmpeg-4d8beeaa8f5746ce243c3cfa95100289b987307e.tar.gz
avformat/cafenc: Don't segfault upon allocation error
If an array for the packet sizes could not be successfully reallocated when writing a packet, the CAF muxer frees said array, but does not reset the number of valid bytes. As a result, when the trailer is written later, avio_write tries to read that many bytes from NULL, which segfaults. Fix this by not freeing the array in case of error; also, postpone writing the packet data after having successfully (re)allocated the array, so that even on allocation error the file can be correctly finalized. Also remove an unnecessary resetting of the number of size entries used at the end. Reviewed-by: Paul B Mahol <onemda@gmail.com> Signed-off-by: Andreas Rheinhardt <andreas.rheinhardt@outlook.com> (cherry picked from commit 19a6b51fe61b915b734319b5d917192108df8188)
-rw-r--r--libavformat/cafenc.c23
1 files changed, 10 insertions, 13 deletions
diff --git a/libavformat/cafenc.c b/libavformat/cafenc.c
index 16abf02162..28ce2c01c8 100644
--- a/libavformat/cafenc.c
+++ b/libavformat/cafenc.c
@@ -210,21 +210,18 @@ static int caf_write_packet(AVFormatContext *s, AVPacket *pkt)
{
CAFContext *caf = s->priv_data;
- avio_write(s->pb, pkt->data, pkt->size);
if (!s->streams[0]->codecpar->block_align) {
- void *pkt_sizes = caf->pkt_sizes;
+ void *pkt_sizes;
int i, alloc_size = caf->size_entries_used + 5U;
- if (alloc_size < 0) {
- caf->pkt_sizes = NULL;
- } else {
- caf->pkt_sizes = av_fast_realloc(caf->pkt_sizes,
- &caf->size_buffer_size,
- alloc_size);
- }
- if (!caf->pkt_sizes) {
- av_free(pkt_sizes);
+ if (alloc_size < 0)
+ return AVERROR(ERANGE);
+
+ pkt_sizes = av_fast_realloc(caf->pkt_sizes,
+ &caf->size_buffer_size,
+ alloc_size);
+ if (!pkt_sizes)
return AVERROR(ENOMEM);
- }
+ caf->pkt_sizes = pkt_sizes;
for (i = 4; i > 0; i--) {
unsigned top = pkt->size >> i * 7;
if (top)
@@ -233,6 +230,7 @@ static int caf_write_packet(AVFormatContext *s, AVPacket *pkt)
caf->pkt_sizes[caf->size_entries_used++] = pkt->size & 127;
caf->packets++;
}
+ avio_write(s->pb, pkt->data, pkt->size);
return 0;
}
@@ -256,7 +254,6 @@ static int caf_write_trailer(AVFormatContext *s)
avio_wb32(pb, 0); ///< mPrimingFrames
avio_wb32(pb, 0); ///< mRemainderFrames
avio_write(pb, caf->pkt_sizes, caf->size_entries_used);
- caf->size_buffer_size = 0;
}
}
av_freep(&caf->pkt_sizes);