aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLukáš Lalinský <lalinsky@gmail.com>2011-10-08 18:41:15 +0200
committerLukáš Lalinský <lalinsky@gmail.com>2012-03-10 09:06:55 +0100
commitf59c3b67aa4eb4a6a651a70d457211eddbf53b12 (patch)
treede6c8cd4fef9a4f860f6834436820c9e32960f89
parent294cb222414292a03a5dd69931d365f53a0356cf (diff)
Detect RIFF files with invalid chunk sizes
The bug report has a WAVE file with zero-sized 'data' chunk, which causes TagLib to iterate over the file, 8 bytes in each iteration. The new code adds a check for the chunk name, which forces it to mark the file as invalid if the chunk name doesn't contain ASCII characters. https://bugs.kde.org/show_bug.cgi?id=283412
-rw-r--r--taglib/riff/aiff/aifffile.cpp5
-rw-r--r--taglib/riff/rifffile.cpp22
-rw-r--r--taglib/riff/wav/wavfile.cpp5
-rw-r--r--tests/data/zero-size-chunk.wavbin0 -> 1024 bytes
-rw-r--r--tests/test_wav.cpp8
5 files changed, 39 insertions, 1 deletions
diff --git a/taglib/riff/aiff/aifffile.cpp b/taglib/riff/aiff/aifffile.cpp
index 425bfa02..72667f6e 100644
--- a/taglib/riff/aiff/aifffile.cpp
+++ b/taglib/riff/aiff/aifffile.cpp
@@ -87,6 +87,11 @@ bool RIFF::AIFF::File::save()
return false;
}
+ if(!isValid()) {
+ debug("RIFF::AIFF::File::save() -- Trying to save invalid file.");
+ return false;
+ }
+
setChunkData(d->tagChunkID, d->tag->render());
return true;
diff --git a/taglib/riff/rifffile.cpp b/taglib/riff/rifffile.cpp
index 8d23bcd6..a3ca0e3e 100644
--- a/taglib/riff/rifffile.cpp
+++ b/taglib/riff/rifffile.cpp
@@ -194,6 +194,19 @@ void RIFF::File::setChunkData(const ByteVector &name, const ByteVector &data)
// private members
////////////////////////////////////////////////////////////////////////////////
+static bool isValidChunkID(const ByteVector &name)
+{
+ if(name.size() != 4) {
+ return false;
+ }
+ for(int i = 0; i < 4; i++) {
+ if(name[i] < 32 || name[i] > 127) {
+ return false;
+ }
+ }
+ return true;
+}
+
void RIFF::File::read()
{
bool bigEndian = (d->endianness == BigEndian);
@@ -207,8 +220,15 @@ void RIFF::File::read()
ByteVector chunkName = readBlock(4);
uint chunkSize = readBlock(4).toUInt(bigEndian);
+ if(!isValidChunkID(chunkName)) {
+ debug("RIFF::File::read() -- Chunk '" + chunkName + "' has invalid ID");
+ setValid(false);
+ break;
+ }
+
if(tell() + chunkSize > uint(length())) {
- // something wrong
+ debug("RIFF::File::read() -- Chunk '" + chunkName + "' has invalid size (larger than the file size)");
+ setValid(false);
break;
}
diff --git a/taglib/riff/wav/wavfile.cpp b/taglib/riff/wav/wavfile.cpp
index 9ec3b510..37d8a4d2 100644
--- a/taglib/riff/wav/wavfile.cpp
+++ b/taglib/riff/wav/wavfile.cpp
@@ -87,6 +87,11 @@ bool RIFF::WAV::File::save()
return false;
}
+ if(!isValid()) {
+ debug("RIFF::WAV::File::save() -- Trying to save invalid file.");
+ return false;
+ }
+
setChunkData(d->tagChunkID, d->tag->render());
return true;
diff --git a/tests/data/zero-size-chunk.wav b/tests/data/zero-size-chunk.wav
new file mode 100644
index 00000000..8517e797
--- /dev/null
+++ b/tests/data/zero-size-chunk.wav
Binary files differ
diff --git a/tests/test_wav.cpp b/tests/test_wav.cpp
index 38a9a0fb..0bd82018 100644
--- a/tests/test_wav.cpp
+++ b/tests/test_wav.cpp
@@ -13,6 +13,7 @@ class TestWAV : public CppUnit::TestFixture
{
CPPUNIT_TEST_SUITE(TestWAV);
CPPUNIT_TEST(testLength);
+ CPPUNIT_TEST(testZeroSizeDataChunk);
CPPUNIT_TEST_SUITE_END();
public:
@@ -20,9 +21,16 @@ public:
void testLength()
{
RIFF::WAV::File f("data/empty.wav");
+ CPPUNIT_ASSERT_EQUAL(true, f.isValid());
CPPUNIT_ASSERT_EQUAL(3, f.audioProperties()->length());
}
+ void testZeroSizeDataChunk()
+ {
+ RIFF::WAV::File f("data/zero-size-chunk.wav");
+ CPPUNIT_ASSERT_EQUAL(false, f.isValid());
+ }
+
};
CPPUNIT_TEST_SUITE_REGISTRATION(TestWAV);