aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorwm4 <nfxjfg@googlemail.com>2015-05-27 14:48:10 +0200
committerMichael Niedermayer <michaelni@gmx.at>2015-05-27 15:50:04 +0200
commit2b3e9bbfb529e6bde238aeb511b55ebe461664c8 (patch)
tree4e7cfa774904b1cd7cdba91a5326db1ef69c3496
parent2ae03968147b891be23ddb7c7851f1a52bfea24b (diff)
downloadffmpeg-2b3e9bbfb529e6bde238aeb511b55ebe461664c8.tar.gz
avformat/mp3: skip junk at the beginning of mp3 files
Apparently it can happen that a mp3 file has junk data between id3 tag and actual mp3 data. Skip this to avoid outputting nonsense timestamps. (Two packets had the same timestamps, because the mp3 parser failed to compute a frame duration.) In this case, the junk consisted of 1044 bytes of zero, which incidentally is the same size as normal mp3 frames in this stream. I suspect the mp3 was edited with some tool which wiped the Xing/LAME headers. Data near the end of the file suggests it was encoded with "LAME3.97", but the normal Xing/LAME headers are missing. So this could be "normal". mpg123 also attempts to skip at least 64KB of junk data by scanning for headers. Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
-rw-r--r--libavformat/mp3dec.c12
1 files changed, 12 insertions, 0 deletions
diff --git a/libavformat/mp3dec.c b/libavformat/mp3dec.c
index abd7fc0bd3..a5374a1037 100644
--- a/libavformat/mp3dec.c
+++ b/libavformat/mp3dec.c
@@ -54,6 +54,8 @@ typedef struct {
int is_cbr;
} MP3DecContext;
+static int check(AVFormatContext *s, int64_t pos);
+
/* mp3 read */
static int mp3_read_probe(AVProbeData *p)
@@ -370,6 +372,16 @@ static int mp3_read_header(AVFormatContext *s)
if (ret < 0)
return ret;
+ off = avio_tell(s->pb);
+ for (i = 0; i < 64 * 1024; i++) {
+ if (check(s, off + i) >= 0) {
+ av_log(s, AV_LOG_INFO, "Skipping %d bytes of junk at %lld.\n", i, (long long)off);
+ avio_seek(s->pb, off + i, SEEK_SET);
+ break;
+ }
+ avio_seek(s->pb, off, SEEK_SET);
+ }
+
// the seek index is relative to the end of the xing vbr headers
for (i = 0; i < st->nb_index_entries; i++)
st->index_entries[i].pos += avio_tell(s->pb);