X-Git-Url: http://git.tuebingen.mpg.de/?p=paraslash.git;a=blobdiff_plain;f=mp3dec_filter.c;h=4bdbc6fd5dc84b6fd73e7678dd04affba1eb6c4b;hp=6982f264a8d866970cc80e960d79628a41c74101;hb=4744d937c4160898d1fe151257606430750e580c;hpb=5ff2b0b3a6f9242033c436fb8272245f5594dd8c diff --git a/mp3dec_filter.c b/mp3dec_filter.c index 6982f264..4bdbc6fd 100644 --- a/mp3dec_filter.c +++ b/mp3dec_filter.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2005-2011 Andre Noll + * Copyright (C) 2005-2012 Andre Noll * * Licensed under the GPL v2. For licencing details see COPYING. */ @@ -8,7 +8,6 @@ #include #include -#include #include "para.h" #include "mp3dec_filter.cmdline.h" @@ -50,16 +49,18 @@ static int handle_decode_error(struct private_mp3dec_data *pmd) return 0; } -static size_t used_mad_buffer_bytes(struct mad_stream *s, size_t max) +static void mp3dec_consume(struct btr_node *btrn, struct mad_stream *s, + size_t max) { - size_t rv; + size_t used; if (!s->next_frame) - return max; - /* we still have some data */ - rv = s->next_frame - s->buffer; - assert(rv <= max); - return rv; + used = max; + else { /* we still have some data */ + used = s->next_frame - s->buffer; + assert(used <= max); + } + btr_consume(btrn, used); } static void mp3dec_close(struct filter_node *fn) @@ -74,13 +75,15 @@ static void mp3dec_close(struct filter_node *fn) fn->private_data = NULL; } +#define MP3DEC_MAX_FRAME 8192 + static void mp3dec_post_select(__a_unused struct sched *s, struct task *t) { struct filter_node *fn = container_of(t, struct filter_node, task); int i, ret; struct private_mp3dec_data *pmd = fn->private_data; struct btr_node *btrn = fn->btrn; - size_t loaded = 0, used, len, iqs; + size_t loaded = 0, len, iqs; char *inbuffer, *outbuffer; next_buffer: @@ -99,25 +102,26 @@ next_buffer: * other buffer tree nodes a chance to run. This is necessary to avoid * buffer underruns on slow machines. */ - len = PARA_MIN(len, (size_t)8192); + len = PARA_MIN(len, (size_t)MP3DEC_MAX_FRAME); mad_stream_buffer(&pmd->stream, (unsigned char *)inbuffer, len); next_frame: ret = mad_header_decode(&pmd->frame.header, &pmd->stream); if (ret < 0) { - used = used_mad_buffer_bytes(&pmd->stream, len); - btr_consume(btrn, used); + mp3dec_consume(btrn, &pmd->stream, len); if (pmd->stream.error == MAD_ERROR_BUFLEN) { if (len == iqs && btr_no_parent(btrn)) { ret = -E_MP3DEC_EOF; goto err; } fn->min_iqs += 100; + ret = -E_MP3DEC_CORRUPT; + if (fn->min_iqs > MP3DEC_MAX_FRAME) + goto err; } if (loaded == 0) goto next_buffer; return; } - fn->min_iqs = 0; pmd->sample_rate = pmd->frame.header.samplerate; pmd->channels = MAD_NCHANNELS(&pmd->frame.header); decode: @@ -126,15 +130,24 @@ decode: ret = handle_decode_error(pmd); if (ret < 0) goto err; - mad_stream_sync(&pmd->stream); - if (pmd->stream.error == MAD_ERROR_BUFLEN) + ret = mad_stream_sync(&pmd->stream); + if (pmd->stream.error == MAD_ERROR_BUFLEN) { + ret = -E_MP3DEC_EOF; + if (len == iqs && btr_no_parent(btrn)) + goto err; + fn->min_iqs += 100; + ret = -E_MP3DEC_CORRUPT; + if (fn->min_iqs > MP3DEC_MAX_FRAME) + goto err; + mp3dec_consume(btrn, &pmd->stream, len); return; + } if (pmd->stream.error != MAD_ERROR_BADDATAPTR) goto decode; - used = used_mad_buffer_bytes(&pmd->stream, len); - btr_consume(btrn, used); + mp3dec_consume(btrn, &pmd->stream, len); return; } + fn->min_iqs = 0; mad_synth_frame(&pmd->synth, &pmd->frame); outbuffer = para_malloc(pmd->synth.pcm.length * 2 * pmd->channels); loaded = 0;