summaryrefslogtreecommitdiffstats
path: root/net/disk_cache/entry_impl.cc
diff options
context:
space:
mode:
authorrvargas@google.com <rvargas@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-16 20:59:29 +0000
committerrvargas@google.com <rvargas@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-16 20:59:29 +0000
commit65188eb51c19157f9b360c84acbfa66543f0d781 (patch)
tree28d1816af4f58cd77fdaec56ab40a3c6c1dc8f95 /net/disk_cache/entry_impl.cc
parentf7fcceefed4e4817f3fca6fdd2156136662ae39b (diff)
downloadchromium_src-65188eb51c19157f9b360c84acbfa66543f0d781.zip
chromium_src-65188eb51c19157f9b360c84acbfa66543f0d781.tar.gz
chromium_src-65188eb51c19157f9b360c84acbfa66543f0d781.tar.bz2
Disk cache: Fix the order in which we delete data
from the block files. Stress testing the cache reveals a problem with the deletion of some data from an entry: it is possible to crash in a way that the block file thinks a block is free and an entry thinks the block is in use. This CL corrects that issue. There is also some new tests and a bunch of DCHECKS added while looking for the problem, as well as adding tests to make sure that a block file is accessed only from one thread (there is no problem with the current code in that regard) BUG=55605 TEST=netunittests Review URL: http://codereview.chromium.org/3430004 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@59711 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'net/disk_cache/entry_impl.cc')
-rw-r--r--net/disk_cache/entry_impl.cc25
1 files changed, 18 insertions, 7 deletions
diff --git a/net/disk_cache/entry_impl.cc b/net/disk_cache/entry_impl.cc
index f152f10..5b65344 100644
--- a/net/disk_cache/entry_impl.cc
+++ b/net/disk_cache/entry_impl.cc
@@ -303,6 +303,9 @@ EntryImpl::EntryImpl(BackendImpl* backend, Addr address, bool read_only)
// data related to a previous cache entry because the range was not fully
// written before).
EntryImpl::~EntryImpl() {
+ Log("~EntryImpl in");
+ backend_->OnEntryDestroyBegin(entry_.address());
+
// Save the sparse info to disk before deleting this entry.
sparse_.reset();
@@ -333,7 +336,8 @@ EntryImpl::~EntryImpl() {
}
}
- backend_->CacheEntryDestroyed(entry_.address());
+ Trace("~EntryImpl out 0x%p", reinterpret_cast<void*>(this));
+ backend_->OnEntryDestroyEnd();
}
void EntryImpl::Doom() {
@@ -578,9 +582,11 @@ int EntryImpl::WriteDataImpl(int index, int offset, net::IOBuffer* buf,
int entry_size = entry_.Data()->data_size[index];
bool extending = entry_size < offset + buf_len;
truncate = truncate && entry_size > offset + buf_len;
+ Trace("To PrepareTarget 0x%x", entry_.address().value());
if (!PrepareTarget(index, offset, buf_len, truncate))
return net::ERR_FAILED;
+ Trace("From PrepareTarget 0x%x", entry_.address().value());
if (extending || truncate)
UpdateSize(index, entry_size, offset + buf_len);
@@ -771,18 +777,17 @@ void EntryImpl::DeleteEntryData(bool everything) {
for (int index = 0; index < kNumStreams; index++) {
Addr address(entry_.Data()->data_addr[index]);
if (address.is_initialized()) {
- DeleteData(address, index);
backend_->ModifyStorageSize(entry_.Data()->data_size[index] -
unreported_size_[index], 0);
entry_.Data()->data_addr[index] = 0;
entry_.Data()->data_size[index] = 0;
+ entry_.Store();
+ DeleteData(address, index);
}
}
- if (!everything) {
- entry_.Store();
+ if (!everything)
return;
- }
// Remove all traces of this entry.
backend_->RemoveEntry(this);
@@ -939,6 +944,12 @@ bool EntryImpl::CreateBlock(int size, Addr* address) {
return true;
}
+// Note that this method may end up modifying a block file so upon return the
+// involved block will be free, and could be reused for something else. If there
+// is a crash after that point (and maybe before returning to the caller), the
+// entry will be left dirty... and at some point it will be discarded; it is
+// important that the entry doesn't keep a reference to this address, or we'll
+// end up deleting the contents of |address| once again.
void EntryImpl::DeleteData(Addr address, int index) {
if (!address.is_initialized())
return;
@@ -1038,12 +1049,12 @@ bool EntryImpl::HandleTruncation(int index, int offset, int buf_len) {
if (!new_size) {
// This is by far the most common scenario.
- DeleteData(address, index);
backend_->ModifyStorageSize(current_size - unreported_size_[index], 0);
entry_.Data()->data_addr[index] = 0;
entry_.Data()->data_size[index] = 0;
unreported_size_[index] = 0;
entry_.Store();
+ DeleteData(address, index);
user_buffers_[index].reset();
return true;
@@ -1115,9 +1126,9 @@ bool EntryImpl::MoveToLocalBuffer(int index) {
return false;
Addr address(entry_.Data()->data_addr[index]);
- DeleteData(address, index);
entry_.Data()->data_addr[index] = 0;
entry_.Store();
+ DeleteData(address, index);
// If we lose this entry we'll see it as zero sized.
int len = entry_.Data()->data_size[index];