summaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
authormattm@chromium.org <mattm@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-08-08 00:49:14 +0000
committermattm@chromium.org <mattm@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-08-08 00:49:14 +0000
commit21fc9dc4cbe2c771e357f4dbed0d2b5131599f8a (patch)
tree48106c981d524a2c14557d3cb764789996af65c7 /net
parent11da7028425a758f1b30d19c802410747b4a98ad (diff)
downloadchromium_src-21fc9dc4cbe2c771e357f4dbed0d2b5131599f8a.zip
chromium_src-21fc9dc4cbe2c771e357f4dbed0d2b5131599f8a.tar.gz
chromium_src-21fc9dc4cbe2c771e357f4dbed0d2b5131599f8a.tar.bz2
Try again: Check blockfile size before attempting to read the header.
Reading past the last page of a mmapped file will SIGBUS. BUG=18174 TEST=truncate Default/Cache/data_* to zero bytes, launch chrome, try to load a website. It shouldn't crash. Review URL: http://codereview.chromium.org/165174 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@22843 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'net')
-rw-r--r--net/disk_cache/block_files.cc6
-rw-r--r--net/disk_cache/block_files.h3
-rw-r--r--net/disk_cache/block_files_unittest.cc62
3 files changed, 53 insertions, 18 deletions
diff --git a/net/disk_cache/block_files.cc b/net/disk_cache/block_files.cc
index cd7e5cf..0e9eb04 100644
--- a/net/disk_cache/block_files.cc
+++ b/net/disk_cache/block_files.cc
@@ -247,6 +247,12 @@ bool BlockFiles::OpenBlockFile(int index) {
return false;
}
+ if (file->GetLength() < static_cast<size_t>(kBlockHeaderSize)) {
+ LOG(ERROR) << "File too small " << name;
+ file->Release();
+ return false;
+ }
+
block_files_[index] = file;
BlockFileHeader* header = reinterpret_cast<BlockFileHeader*>(file->buffer());
diff --git a/net/disk_cache/block_files.h b/net/disk_cache/block_files.h
index 790cc7d..e86546e 100644
--- a/net/disk_cache/block_files.h
+++ b/net/disk_cache/block_files.h
@@ -11,6 +11,7 @@
#include "net/disk_cache/addr.h"
#include "net/disk_cache/mapped_file.h"
+#include "testing/gtest/include/gtest/gtest_prod.h"
namespace disk_cache {
@@ -75,6 +76,8 @@ class BlockFiles {
std::wstring path_; // Path to the backing folder.
std::vector<MappedFile*> block_files_; // The actual files.
+ FRIEND_TEST(DiskCacheTest, BlockFiles_ZeroSizeFile);
+
DISALLOW_EVIL_CONSTRUCTORS(BlockFiles);
};
diff --git a/net/disk_cache/block_files_unittest.cc b/net/disk_cache/block_files_unittest.cc
index b4ce107..e7647d7 100644
--- a/net/disk_cache/block_files_unittest.cc
+++ b/net/disk_cache/block_files_unittest.cc
@@ -26,20 +26,22 @@ int NumberOfFiles(const std::wstring& path) {
} // namespace;
+namespace disk_cache {
+
TEST_F(DiskCacheTest, BlockFiles_Grow) {
std::wstring path = GetCachePath();
ASSERT_TRUE(DeleteCache(path.c_str()));
ASSERT_TRUE(file_util::CreateDirectory(path));
- disk_cache::BlockFiles files(path);
+ BlockFiles files(path);
ASSERT_TRUE(files.Init(true));
const int kMaxSize = 35000;
- disk_cache::Addr address[kMaxSize];
+ Addr address[kMaxSize];
// Fill up the 32-byte block file (use three files).
for (int i = 0; i < kMaxSize; i++) {
- EXPECT_TRUE(files.CreateBlock(disk_cache::RANKINGS, 4, &address[i]));
+ EXPECT_TRUE(files.CreateBlock(RANKINGS, 4, &address[i]));
}
EXPECT_EQ(6, NumberOfFiles(path));
@@ -47,7 +49,7 @@ TEST_F(DiskCacheTest, BlockFiles_Grow) {
for (int i = 0; i < kMaxSize * 4; i += 2) {
int target = i % kMaxSize;
files.DeleteBlock(address[target], false);
- EXPECT_TRUE(files.CreateBlock(disk_cache::RANKINGS, 4, &address[target]));
+ EXPECT_TRUE(files.CreateBlock(RANKINGS, 4, &address[target]));
}
EXPECT_EQ(6, NumberOfFiles(path));
}
@@ -58,15 +60,15 @@ TEST_F(DiskCacheTest, BlockFiles_Shrink) {
ASSERT_TRUE(DeleteCache(path.c_str()));
ASSERT_TRUE(file_util::CreateDirectory(path));
- disk_cache::BlockFiles files(path);
+ BlockFiles files(path);
ASSERT_TRUE(files.Init(true));
const int kMaxSize = 35000;
- disk_cache::Addr address[kMaxSize];
+ Addr address[kMaxSize];
// Fill up the 32-byte block file (use three files).
for (int i = 0; i < kMaxSize; i++) {
- EXPECT_TRUE(files.CreateBlock(disk_cache::RANKINGS, 4, &address[i]));
+ EXPECT_TRUE(files.CreateBlock(RANKINGS, 4, &address[i]));
}
// Now delete all the blocks, so that we can delete the two extra files.
@@ -82,43 +84,43 @@ TEST_F(DiskCacheTest, BlockFiles_Recover) {
ASSERT_TRUE(DeleteCache(path.c_str()));
ASSERT_TRUE(file_util::CreateDirectory(path));
- disk_cache::BlockFiles files(path);
+ BlockFiles files(path);
ASSERT_TRUE(files.Init(true));
const int kNumEntries = 2000;
- disk_cache::CacheAddr entries[kNumEntries];
+ CacheAddr entries[kNumEntries];
int seed = static_cast<int>(Time::Now().ToInternalValue());
srand(seed);
for (int i = 0; i < kNumEntries; i++) {
- disk_cache::Addr address(0);
+ Addr address(0);
int size = (rand() % 4) + 1;
- EXPECT_TRUE(files.CreateBlock(disk_cache::RANKINGS, size, &address));
+ EXPECT_TRUE(files.CreateBlock(RANKINGS, size, &address));
entries[i] = address.value();
}
for (int i = 0; i < kNumEntries; i++) {
int source1 = rand() % kNumEntries;
int source2 = rand() % kNumEntries;
- disk_cache::CacheAddr temp = entries[source1];
+ CacheAddr temp = entries[source1];
entries[source1] = entries[source2];
entries[source2] = temp;
}
for (int i = 0; i < kNumEntries / 2; i++) {
- disk_cache::Addr address(entries[i]);
+ Addr address(entries[i]);
files.DeleteBlock(address, false);
}
// At this point, there are kNumEntries / 2 entries on the file, randomly
// distributed both on location and size.
- disk_cache::Addr address(entries[kNumEntries / 2]);
- disk_cache::MappedFile* file = files.GetFile(address);
+ Addr address(entries[kNumEntries / 2]);
+ MappedFile* file = files.GetFile(address);
ASSERT_TRUE(NULL != file);
- disk_cache::BlockFileHeader* header =
- reinterpret_cast<disk_cache::BlockFileHeader*>(file->buffer());
+ BlockFileHeader* header =
+ reinterpret_cast<BlockFileHeader*>(file->buffer());
ASSERT_TRUE(NULL != header);
ASSERT_EQ(0, header->updating);
@@ -142,7 +144,7 @@ TEST_F(DiskCacheTest, BlockFiles_Recover) {
file = files.GetFile(address);
ASSERT_TRUE(NULL != file);
- header = reinterpret_cast<disk_cache::BlockFileHeader*>(file->buffer());
+ header = reinterpret_cast<BlockFileHeader*>(file->buffer());
ASSERT_TRUE(NULL != header);
ASSERT_EQ(0, header->updating);
@@ -153,3 +155,27 @@ TEST_F(DiskCacheTest, BlockFiles_Recover) {
EXPECT_EQ(empty_3, header->empty[2]);
EXPECT_EQ(empty_4, header->empty[3]);
}
+
+// Handling of truncated files.
+TEST_F(DiskCacheTest, BlockFiles_ZeroSizeFile) {
+ std::wstring path = GetCachePath();
+ ASSERT_TRUE(DeleteCache(path.c_str()));
+ ASSERT_TRUE(file_util::CreateDirectory(path));
+
+ BlockFiles files(path);
+ ASSERT_TRUE(files.Init(true));
+
+ std::wstring filename = files.Name(0);
+ files.CloseFiles();
+ // Truncate one of the files.
+ {
+ scoped_refptr<File> file(new File);
+ ASSERT_TRUE(file->Init(filename));
+ EXPECT_TRUE(file->SetLength(0));
+ }
+
+ // Initializing should fail, not crash.
+ ASSERT_FALSE(files.Init(false));
+}
+
+} // namespace disk_cache