1
0
mirror of https://github.com/FFmpeg/FFmpeg.git synced 2025-01-24 13:56:33 +02:00

avcodec/tiff: Restrict tag order based on specification

"The entries in an IFD must be sorted in ascending order by Tag. Note that this is
 not the order in which the fields are described in this document."

This way various dimensions, sample and bit sizes cannot be changed at
arbitrary times which reduces the potential for bugs.
The tag reading code also on various places assumes that numerically previous
tags have already been parsed, so this needs to be enforced one way or another.

If this commit causes problems with real world files which are not easy to fix
then some other form of checks are needed to ensure the various dependencies
in the tag reading are not violated.

Fixes: out of array access
Fixes: 24825/clusterfuzz-testcase-minimized-ffmpeg_AV_CODEC_ID_TIFF_fuzzer-6326925027704832

Found-by: continuous fuzzing process https://github.com/google/oss-fuzz/tree/master/projects/ffmpeg
Signed-off-by: Michael Niedermayer <michael@niedermayer.cc>
(cherry picked from commit ad29f9e47cb848e11ee1d358d2bae15cd35ef04b)
Signed-off-by: Michael Niedermayer <michael@niedermayer.cc>
This commit is contained in:
Michael Niedermayer 2020-08-20 01:05:35 +02:00
parent ea12dd67ee
commit ea43ce9aa9

View File

@ -64,6 +64,7 @@ typedef struct TiffContext {
int predictor; int predictor;
int fill_order; int fill_order;
uint32_t res[4]; uint32_t res[4];
unsigned last_tag;
int strips, rps, sstype; int strips, rps, sstype;
int sot; int sot;
@ -801,6 +802,12 @@ static int tiff_decode_tag(TiffContext *s, AVFrame *frame)
if (ret < 0) { if (ret < 0) {
goto end; goto end;
} }
if (tag <= s->last_tag)
return AVERROR_INVALIDDATA;
// We ignore TIFF_STRIP_SIZE as it is sometimes in the logic but wrong order around TIFF_STRIP_OFFS
if (tag != TIFF_STRIP_SIZE)
s->last_tag = tag;
off = bytestream2_tell(&s->gb); off = bytestream2_tell(&s->gb);
if (count == 1) { if (count == 1) {
@ -1239,6 +1246,7 @@ static int decode_frame(AVCodecContext *avctx,
s->photometric = TIFF_PHOTOMETRIC_NONE; s->photometric = TIFF_PHOTOMETRIC_NONE;
s->compr = TIFF_RAW; s->compr = TIFF_RAW;
s->fill_order = 0; s->fill_order = 0;
s->last_tag = 0;
free_geotags(s); free_geotags(s);
// Reset these offsets so we can tell if they were set this frame // Reset these offsets so we can tell if they were set this frame