You've already forked FFmpeg
mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-08-10 06:10:52 +02:00
libavcodec/wmadec: Return AVERROR_INVALIDDATA on decoding errors
WMA files that fail to decode due to incoherent block lengths and frame lengths currently result in a "Operation not permitted". After this change, they will instead result in "Invalid data found when processing input". Several other error cases are also changed from returning -1. As we change the error propagation logic in wma_decode_frame and wma_decode_superframe, previous occurrences of returning AVERROR_INVALIDDATA are also affected by this. This includes "total_gain overread" and a "channel exponents_initialized" check. --- Tomas: changed some -1's to AVERROR_INVALIDDATA
This commit is contained in:
committed by
Tomas Härdin
parent
1306e5c7d3
commit
399234ee2a
@@ -368,7 +368,7 @@ static int decode_exp_vlc(WMACodecContext *s, int ch)
|
|||||||
if ((unsigned) last_exp + 60 >= FF_ARRAY_ELEMS(pow_tab)) {
|
if ((unsigned) last_exp + 60 >= FF_ARRAY_ELEMS(pow_tab)) {
|
||||||
av_log(s->avctx, AV_LOG_ERROR, "Exponent out of range: %d\n",
|
av_log(s->avctx, AV_LOG_ERROR, "Exponent out of range: %d\n",
|
||||||
last_exp);
|
last_exp);
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
v = ptab[last_exp];
|
v = ptab[last_exp];
|
||||||
iv = iptab[last_exp];
|
iv = iptab[last_exp];
|
||||||
@@ -439,8 +439,10 @@ static void wma_window(WMACodecContext *s, float *out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return 0 if OK. 1 if last block of frame. return -1 if
|
* @return
|
||||||
* unrecoverable error.
|
* 0 if OK.
|
||||||
|
* 1 if last block of frame.
|
||||||
|
* AVERROR if unrecoverable error.
|
||||||
*/
|
*/
|
||||||
static int wma_decode_block(WMACodecContext *s)
|
static int wma_decode_block(WMACodecContext *s)
|
||||||
{
|
{
|
||||||
@@ -468,7 +470,7 @@ static int wma_decode_block(WMACodecContext *s)
|
|||||||
av_log(s->avctx, AV_LOG_ERROR,
|
av_log(s->avctx, AV_LOG_ERROR,
|
||||||
"prev_block_len_bits %d out of range\n",
|
"prev_block_len_bits %d out of range\n",
|
||||||
s->frame_len_bits - v);
|
s->frame_len_bits - v);
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
s->prev_block_len_bits = s->frame_len_bits - v;
|
s->prev_block_len_bits = s->frame_len_bits - v;
|
||||||
v = get_bits(&s->gb, n);
|
v = get_bits(&s->gb, n);
|
||||||
@@ -476,7 +478,7 @@ static int wma_decode_block(WMACodecContext *s)
|
|||||||
av_log(s->avctx, AV_LOG_ERROR,
|
av_log(s->avctx, AV_LOG_ERROR,
|
||||||
"block_len_bits %d out of range\n",
|
"block_len_bits %d out of range\n",
|
||||||
s->frame_len_bits - v);
|
s->frame_len_bits - v);
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
s->block_len_bits = s->frame_len_bits - v;
|
s->block_len_bits = s->frame_len_bits - v;
|
||||||
} else {
|
} else {
|
||||||
@@ -489,7 +491,7 @@ static int wma_decode_block(WMACodecContext *s)
|
|||||||
av_log(s->avctx, AV_LOG_ERROR,
|
av_log(s->avctx, AV_LOG_ERROR,
|
||||||
"next_block_len_bits %d out of range\n",
|
"next_block_len_bits %d out of range\n",
|
||||||
s->frame_len_bits - v);
|
s->frame_len_bits - v);
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
s->next_block_len_bits = s->frame_len_bits - v;
|
s->next_block_len_bits = s->frame_len_bits - v;
|
||||||
} else {
|
} else {
|
||||||
@@ -501,14 +503,14 @@ static int wma_decode_block(WMACodecContext *s)
|
|||||||
|
|
||||||
if (s->frame_len_bits - s->block_len_bits >= s->nb_block_sizes){
|
if (s->frame_len_bits - s->block_len_bits >= s->nb_block_sizes){
|
||||||
av_log(s->avctx, AV_LOG_ERROR, "block_len_bits not initialized to a valid value\n");
|
av_log(s->avctx, AV_LOG_ERROR, "block_len_bits not initialized to a valid value\n");
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* now check if the block length is coherent with the frame length */
|
/* now check if the block length is coherent with the frame length */
|
||||||
s->block_len = 1 << s->block_len_bits;
|
s->block_len = 1 << s->block_len_bits;
|
||||||
if ((s->block_pos + s->block_len) > s->frame_len) {
|
if ((s->block_pos + s->block_len) > s->frame_len) {
|
||||||
av_log(s->avctx, AV_LOG_ERROR, "frame_len overflow\n");
|
av_log(s->avctx, AV_LOG_ERROR, "frame_len overflow\n");
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (channels == 2)
|
if (channels == 2)
|
||||||
@@ -590,7 +592,7 @@ static int wma_decode_block(WMACodecContext *s)
|
|||||||
if (s->channel_coded[ch]) {
|
if (s->channel_coded[ch]) {
|
||||||
if (s->use_exp_vlc) {
|
if (s->use_exp_vlc) {
|
||||||
if (decode_exp_vlc(s, ch) < 0)
|
if (decode_exp_vlc(s, ch) < 0)
|
||||||
return -1;
|
return AVERROR_INVALIDDATA;
|
||||||
} else {
|
} else {
|
||||||
decode_exp_lsp(s, ch);
|
decode_exp_lsp(s, ch);
|
||||||
}
|
}
|
||||||
@@ -802,7 +804,7 @@ static int wma_decode_frame(WMACodecContext *s, float **samples,
|
|||||||
for (;;) {
|
for (;;) {
|
||||||
ret = wma_decode_block(s);
|
ret = wma_decode_block(s);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return -1;
|
return ret;
|
||||||
if (ret)
|
if (ret)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -879,8 +881,10 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
return AVERROR_INVALIDDATA;
|
return AVERROR_INVALIDDATA;
|
||||||
|
|
||||||
if ((s->last_superframe_len + buf_size - 1) >
|
if ((s->last_superframe_len + buf_size - 1) >
|
||||||
MAX_CODED_SUPERFRAME_SIZE)
|
MAX_CODED_SUPERFRAME_SIZE) {
|
||||||
|
ret = AVERROR_INVALIDDATA;
|
||||||
goto fail;
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
q = s->last_superframe + s->last_superframe_len;
|
q = s->last_superframe + s->last_superframe_len;
|
||||||
len = buf_size - 1;
|
len = buf_size - 1;
|
||||||
@@ -911,14 +915,17 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
av_log(avctx, AV_LOG_ERROR,
|
av_log(avctx, AV_LOG_ERROR,
|
||||||
"Invalid last frame bit offset %d > buf size %d (%d)\n",
|
"Invalid last frame bit offset %d > buf size %d (%d)\n",
|
||||||
bit_offset, get_bits_left(&s->gb), buf_size);
|
bit_offset, get_bits_left(&s->gb), buf_size);
|
||||||
|
ret = AVERROR_INVALIDDATA;
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (s->last_superframe_len > 0) {
|
if (s->last_superframe_len > 0) {
|
||||||
/* add bit_offset bits to last frame */
|
/* add bit_offset bits to last frame */
|
||||||
if ((s->last_superframe_len + ((bit_offset + 7) >> 3)) >
|
if ((s->last_superframe_len + ((bit_offset + 7) >> 3)) >
|
||||||
MAX_CODED_SUPERFRAME_SIZE)
|
MAX_CODED_SUPERFRAME_SIZE) {
|
||||||
|
ret = AVERROR_INVALIDDATA;
|
||||||
goto fail;
|
goto fail;
|
||||||
|
}
|
||||||
q = s->last_superframe + s->last_superframe_len;
|
q = s->last_superframe + s->last_superframe_len;
|
||||||
len = bit_offset;
|
len = bit_offset;
|
||||||
while (len > 7) {
|
while (len > 7) {
|
||||||
@@ -937,7 +944,7 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
skip_bits(&s->gb, s->last_bitoffset);
|
skip_bits(&s->gb, s->last_bitoffset);
|
||||||
/* this frame is stored in the last superframe and in the
|
/* this frame is stored in the last superframe and in the
|
||||||
* current one */
|
* current one */
|
||||||
if (wma_decode_frame(s, samples, samples_offset) < 0)
|
if ((ret = wma_decode_frame(s, samples, samples_offset)) < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
samples_offset += s->frame_len;
|
samples_offset += s->frame_len;
|
||||||
nb_frames--;
|
nb_frames--;
|
||||||
@@ -954,7 +961,7 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
|
|
||||||
s->reset_block_lengths = 1;
|
s->reset_block_lengths = 1;
|
||||||
for (i = 0; i < nb_frames; i++) {
|
for (i = 0; i < nb_frames; i++) {
|
||||||
if (wma_decode_frame(s, samples, samples_offset) < 0)
|
if ((ret = wma_decode_frame(s, samples, samples_offset)) < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
samples_offset += s->frame_len;
|
samples_offset += s->frame_len;
|
||||||
}
|
}
|
||||||
@@ -967,13 +974,14 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
len = buf_size - pos;
|
len = buf_size - pos;
|
||||||
if (len > MAX_CODED_SUPERFRAME_SIZE || len < 0) {
|
if (len > MAX_CODED_SUPERFRAME_SIZE || len < 0) {
|
||||||
av_log(s->avctx, AV_LOG_ERROR, "len %d invalid\n", len);
|
av_log(s->avctx, AV_LOG_ERROR, "len %d invalid\n", len);
|
||||||
|
ret = AVERROR_INVALIDDATA;
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
s->last_superframe_len = len;
|
s->last_superframe_len = len;
|
||||||
memcpy(s->last_superframe, buf + pos, len);
|
memcpy(s->last_superframe, buf + pos, len);
|
||||||
} else {
|
} else {
|
||||||
/* single frame decode */
|
/* single frame decode */
|
||||||
if (wma_decode_frame(s, samples, samples_offset) < 0)
|
if ((ret = wma_decode_frame(s, samples, samples_offset)) < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
samples_offset += s->frame_len;
|
samples_offset += s->frame_len;
|
||||||
}
|
}
|
||||||
@@ -989,7 +997,7 @@ static int wma_decode_superframe(AVCodecContext *avctx, AVFrame *frame,
|
|||||||
fail:
|
fail:
|
||||||
/* when error, we reset the bit reservoir */
|
/* when error, we reset the bit reservoir */
|
||||||
s->last_superframe_len = 0;
|
s->last_superframe_len = 0;
|
||||||
return -1;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static av_cold void flush(AVCodecContext *avctx)
|
static av_cold void flush(AVCodecContext *avctx)
|
||||||
|
Reference in New Issue
Block a user