diff options
author | rvargas@google.com <rvargas@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-12-21 18:57:01 +0000 |
---|---|---|
committer | rvargas@google.com <rvargas@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-12-21 18:57:01 +0000 |
commit | 504829400b156f8ab93afabeed5cb5837902a5c4 (patch) | |
tree | 32be186199ebcbb23e544be40315b46eeee07b44 /net | |
parent | cef3362fd03475dc6e47637a823b086e8cf13b95 (diff) | |
download | chromium_src-504829400b156f8ab93afabeed5cb5837902a5c4.zip chromium_src-504829400b156f8ab93afabeed5cb5837902a5c4.tar.gz chromium_src-504829400b156f8ab93afabeed5cb5837902a5c4.tar.bz2 |
Http cache: Use asynchronous IO to read an write the
response headers from the disk cache.
BUG=26729
TEST=current unit tests.
Review URL: http://codereview.chromium.org/506081
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@35094 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'net')
-rw-r--r-- | net/base/io_buffer.h | 21 | ||||
-rw-r--r-- | net/http/http_cache.h | 2 | ||||
-rw-r--r-- | net/http/http_cache_transaction.cc | 84 | ||||
-rw-r--r-- | net/http/http_cache_transaction.h | 6 |
4 files changed, 78 insertions, 35 deletions
diff --git a/net/base/io_buffer.h b/net/base/io_buffer.h index 671b263..d908eec 100644 --- a/net/base/io_buffer.h +++ b/net/base/io_buffer.h @@ -7,6 +7,7 @@ #include <string> +#include "base/pickle.h" #include "base/ref_counted.h" #include "base/scoped_ptr.h" @@ -131,6 +132,26 @@ class GrowableIOBuffer : public IOBuffer { int offset_; }; +// This versions allows a pickle to be used as the storage for a write-style +// operation, avoiding an extra data copy. +class PickledIOBuffer : public IOBuffer { + public: + PickledIOBuffer() : IOBuffer() {} + + Pickle* pickle() { return &pickle_; } + + // Signals that we are done writing to the picke and we can use it for a + // write-style IO operation. + void Done() { + data_ = const_cast<char*>(static_cast<const char*>(pickle_.data())); + } + + private: + ~PickledIOBuffer() { data_ = NULL; } + + Pickle pickle_; +}; + // This class allows the creation of a temporary IOBuffer that doesn't really // own the underlying buffer. Please use this class only as a last resort. // A good example is the buffer for a synchronous operation, where we can be diff --git a/net/http/http_cache.h b/net/http/http_cache.h index 7926d5d..959b76c 100644 --- a/net/http/http_cache.h +++ b/net/http/http_cache.h @@ -107,12 +107,14 @@ class HttpCache : public HttpTransactionFactory, // Helper function for reading response info from the disk cache. If the // cache doesn't have the whole resource *|request_truncated| is set to true. + // Avoid this function for performance critical paths as it uses blocking IO. static bool ReadResponseInfo(disk_cache::Entry* disk_entry, HttpResponseInfo* response_info, bool* response_truncated); // Helper function for writing response info into the disk cache. If the // cache doesn't have the whole resource |request_truncated| should be true. + // Avoid this function for performance critical paths as it uses blocking IO. static bool WriteResponseInfo(disk_cache::Entry* disk_entry, const HttpResponseInfo* response_info, bool skip_transient_headers, diff --git a/net/http/http_cache_transaction.cc b/net/http/http_cache_transaction.cc index 918ebad..b829623 100644 --- a/net/http/http_cache_transaction.cc +++ b/net/http/http_cache_transaction.cc @@ -119,6 +119,9 @@ HttpCache::Transaction::Transaction(HttpCache* cache, bool enable_range_support) network_callback_(this, &Transaction::OnIOComplete)), ALLOW_THIS_IN_INITIALIZER_LIST( cache_callback_(new CancelableCompletionCallback<Transaction>( + this, &Transaction::OnIOComplete))), + ALLOW_THIS_IN_INITIALIZER_LIST( + write_headers_callback_(new CancelableCompletionCallback<Transaction>( this, &Transaction::OnIOComplete))) { COMPILE_ASSERT(HttpCache::Transaction::kNumValidationHeaders == ARRAYSIZE_UNSAFE(kValidationHeaders), @@ -146,6 +149,7 @@ HttpCache::Transaction::~Transaction() { // If there is an outstanding callback, mark it as cancelled so running it // does nothing. cache_callback_->Cancel(); + write_headers_callback_->Cancel(); // We could still have a cache read or write in progress, so we just null the // cache_ pointer to signal that we are dead. See DoCacheReadCompleted. @@ -492,7 +496,16 @@ int HttpCache::Transaction::DoEntryAvailable() { return OK; } -int HttpCache::Transaction::DoCacheReadResponseComplete() { +int HttpCache::Transaction::DoCacheReadResponseComplete(int result) { + cache_callback_->Release(); // Balance the AddRef from DoCacheReadResponse. + LoadLog::EndEvent(load_log_, LoadLog::TYPE_HTTP_CACHE_READ_INFO); + if (result != io_buf_len_ || + !HttpCache::ParseResponseInfo(read_buf_->data(), io_buf_len_, + &response_, &truncated_)) { + DLOG(ERROR) << "ReadData failed: " << result; + return ERR_CACHE_READ_FAILURE; + } + // We now have access to the cache entry. // // o if we are a reader for the transaction, then we can start reading the @@ -506,23 +519,22 @@ int HttpCache::Transaction::DoCacheReadResponseComplete() { // conditionalized request (if-modified-since / if-none-match). We check // if the request headers define a validation request. // - int rv = OK; switch (mode_) { case READ: - rv = BeginCacheRead(); + result = BeginCacheRead(); break; case READ_WRITE: - rv = BeginPartialCacheValidation(); + result = BeginPartialCacheValidation(); break; case UPDATE: - rv = BeginExternallyConditionalizedRequest(); + result = BeginExternallyConditionalizedRequest(); break; case WRITE: default: NOTREACHED(); - rv = ERR_FAILED; + result = ERR_FAILED; } - return rv; + return result; } bool HttpCache::Transaction::AddTruncatedFlag() { @@ -536,16 +548,10 @@ bool HttpCache::Transaction::AddTruncatedFlag() { if (!entry_->disk_entry->GetDataSize(kResponseContentIndex)) return false; - // We have a serious problem here: We are inside the object destructor and - // we need to write to the cache. We could even have a pending operation in - // progress already so it's not that we can just issue another operation and - // cancel it immediately. So, for now, just verify that the operation - // completes synchronously. truncated_ = true; target_state_ = STATE_NONE; next_state_ = STATE_CACHE_WRITE_TRUNCATED_RESPONSE; - int rv = DoLoop(OK); - DCHECK_EQ(OK, rv); + DoLoop(OK); return true; } @@ -659,8 +665,7 @@ int HttpCache::Transaction::DoLoop(int result) { rv = DoCacheReadResponse(); break; case STATE_CACHE_READ_RESPONSE_COMPLETE: - DCHECK_EQ(OK, rv); - rv = DoCacheReadResponseComplete(); + rv = DoCacheReadResponseComplete(rv); break; case STATE_CACHE_WRITE_RESPONSE: DCHECK_EQ(OK, rv); @@ -1261,19 +1266,19 @@ void HttpCache::Transaction::IgnoreRangeRequest() { int HttpCache::Transaction::ReadFromNetwork(IOBuffer* data, int data_len) { read_buf_ = data; - read_buf_len_ = data_len; + io_buf_len_ = data_len; next_state_ = STATE_NETWORK_READ; return DoLoop(OK); } int HttpCache::Transaction::DoNetworkRead() { next_state_ = STATE_NETWORK_READ_COMPLETE; - return network_trans_->Read(read_buf_, read_buf_len_, &network_callback_); + return network_trans_->Read(read_buf_, io_buf_len_, &network_callback_); } int HttpCache::Transaction::ReadFromEntry(IOBuffer* data, int data_len) { read_buf_ = data; - read_buf_len_ = data_len; + io_buf_len_ = data_len; next_state_ = STATE_CACHE_READ_DATA; return DoLoop(OK); } @@ -1283,25 +1288,25 @@ int HttpCache::Transaction::DoCacheReadData() { next_state_ = STATE_CACHE_READ_DATA_COMPLETE; cache_callback_->AddRef(); // Balanced in DoCacheReadDataComplete. if (partial_.get()) { - return partial_->CacheRead(entry_->disk_entry, read_buf_, read_buf_len_, + return partial_->CacheRead(entry_->disk_entry, read_buf_, io_buf_len_, cache_callback_); } return entry_->disk_entry->ReadData(kResponseContentIndex, read_offset_, - read_buf_, read_buf_len_, - cache_callback_); + read_buf_, io_buf_len_, cache_callback_); } int HttpCache::Transaction::DoCacheReadResponse() { DCHECK(entry_); next_state_ = STATE_CACHE_READ_RESPONSE_COMPLETE; - LoadLog::BeginEvent(load_log_, LoadLog::TYPE_HTTP_CACHE_READ_INFO); - bool read_ok = - HttpCache::ReadResponseInfo(entry_->disk_entry, &response_, &truncated_); - LoadLog::EndEvent(load_log_, LoadLog::TYPE_HTTP_CACHE_READ_INFO); + io_buf_len_ = entry_->disk_entry->GetDataSize(kResponseInfoIndex); + read_buf_ = new IOBuffer(io_buf_len_); - return read_ok ? OK : ERR_CACHE_READ_FAILURE; + LoadLog::BeginEvent(load_log_, LoadLog::TYPE_HTTP_CACHE_READ_INFO); + cache_callback_->AddRef(); // Balanced in DoCacheReadResponseComplete. + return entry_->disk_entry->ReadData(kResponseInfoIndex, 0, read_buf_, + io_buf_len_, cache_callback_); } int HttpCache::Transaction::WriteToEntry(int index, int offset, @@ -1366,17 +1371,30 @@ int HttpCache::Transaction::WriteResponseInfoToEntry(bool truncated) { DCHECK_EQ(200, response_.headers->response_code()); } - if (!HttpCache::WriteResponseInfo(entry_->disk_entry, &response_, - skip_transient_headers, truncated)) { - DLOG(ERROR) << "failed to write response info to cache"; - DoneWritingToEntry(false); - } - return OK; + scoped_refptr<PickledIOBuffer> data = new PickledIOBuffer(); + response_.Persist(data->pickle(), skip_transient_headers, truncated); + data->Done(); + + // Balanced in DoCacheWriteResponseComplete. We may be running from the + // destructor of this object so cache_callback_ may be currently in use. + write_headers_callback_->AddRef(); + io_buf_len_ = data->pickle()->size(); + return entry_->disk_entry->WriteData(kResponseInfoIndex, 0, data, io_buf_len_, + write_headers_callback_, true); } int HttpCache::Transaction::DoCacheWriteResponseComplete(int result) { next_state_ = target_state_; target_state_ = STATE_NONE; + if (!entry_) + return OK; + + // Balance the AddRef from WriteResponseInfoToEntry. + write_headers_callback_->Release(); + if (result != io_buf_len_) { + DLOG(ERROR) << "failed to write response info to cache"; + DoneWritingToEntry(false); + } return OK; } diff --git a/net/http/http_cache_transaction.h b/net/http/http_cache_transaction.h index 0e66f4d..14e39c7 100644 --- a/net/http/http_cache_transaction.h +++ b/net/http/http_cache_transaction.h @@ -165,7 +165,7 @@ class HttpCache::Transaction : public HttpTransaction { int DoTruncateCachedDataComplete(int result); int DoPartialHeadersReceived(); int DoCacheReadResponse(); - int DoCacheReadResponseComplete(); + int DoCacheReadResponseComplete(int result); int DoCacheWriteResponse(); int DoCacheWriteTruncatedResponse(); int DoCacheWriteResponseComplete(int result); @@ -301,13 +301,15 @@ class HttpCache::Transaction : public HttpTransaction { bool truncated_; // We don't have all the response data. bool server_responded_206_; scoped_refptr<IOBuffer> read_buf_; - int read_buf_len_; + int io_buf_len_; int read_offset_; int effective_load_flags_; scoped_ptr<PartialData> partial_; // We are dealing with range requests. uint64 final_upload_progress_; CompletionCallbackImpl<Transaction> network_callback_; scoped_refptr<CancelableCompletionCallback<Transaction> > cache_callback_; + scoped_refptr<CancelableCompletionCallback<Transaction> > + write_headers_callback_; }; } // namespace net |