From 65851b56ffc8e5415347c89ba09cc1b9f146c369 Mon Sep 17 00:00:00 2001 From: "rvargas@google.com" Date: Wed, 3 Feb 2010 21:41:57 +0000 Subject: Merge 33760 - Http cache: First pass to move the HttpCache::Transaction to a sate machine. This CL only implements the states that are related to current asynchronous operations. BUG=26729 TEST=current unit tests. Review URL: http://codereview.chromium.org/410005 TBR=rvargas@google.com Review URL: http://codereview.chromium.org/563023 git-svn-id: svn://svn.chromium.org/chrome/branches/249/src@38023 0039d316-1c4b-4281-b951-d872f2087c98 --- net/http/http_cache_transaction.cc | 292 ++++++++++++++++++++++--------------- net/http/http_cache_transaction.h | 78 ++++++---- 2 files changed, 222 insertions(+), 148 deletions(-) diff --git a/net/http/http_cache_transaction.cc b/net/http/http_cache_transaction.cc index 830e4c7..fc41c5b 100644 --- a/net/http/http_cache_transaction.cc +++ b/net/http/http_cache_transaction.cc @@ -97,7 +97,8 @@ static bool HeaderMatches(const HttpUtil::HeadersIterator& h, //----------------------------------------------------------------------------- HttpCache::Transaction::Transaction(HttpCache* cache, bool enable_range_support) - : request_(NULL), + : next_state_(STATE_NONE), + request_(NULL), cache_(cache->AsWeakPtr()), entry_(NULL), network_trans_(NULL), @@ -111,18 +112,10 @@ HttpCache::Transaction::Transaction(HttpCache* cache, bool enable_range_support) effective_load_flags_(0), final_upload_progress_(0), ALLOW_THIS_IN_INITIALIZER_LIST( - network_info_callback_(this, &Transaction::OnNetworkInfoAvailable)), + network_callback_(this, &Transaction::OnIOComplete)), ALLOW_THIS_IN_INITIALIZER_LIST( - network_read_callback_(this, &Transaction::OnNetworkReadCompleted)), - ALLOW_THIS_IN_INITIALIZER_LIST( - cache_read_callback_(new CancelableCompletionCallback( - this, &Transaction::OnCacheReadCompleted))), - ALLOW_THIS_IN_INITIALIZER_LIST( - cache_write_callback_(new CancelableCompletionCallback( - this, &Transaction::OnCacheWriteCompleted))), - ALLOW_THIS_IN_INITIALIZER_LIST( - entry_ready_callback_(new CancelableCompletionCallback( - this, &Transaction::OnCacheEntryReady))) { + cache_callback_(new CancelableCompletionCallback( + this, &Transaction::OnIOComplete))) { COMPILE_ASSERT(HttpCache::Transaction::kNumValidationHeaders == ARRAYSIZE_UNSAFE(kValidationHeaders), Invalid_number_of_validation_headers); @@ -148,9 +141,7 @@ HttpCache::Transaction::~Transaction() { // If there is an outstanding callback, mark it as cancelled so running it // does nothing. - cache_read_callback_->Cancel(); - cache_write_callback_->Cancel(); - entry_ready_callback_->Cancel(); + cache_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. @@ -163,7 +154,7 @@ int HttpCache::Transaction::Start(const HttpRequestInfo* request, DCHECK(request); DCHECK(callback); - // ensure that we only have one asynchronous call at a time. + // Ensure that we only have one asynchronous call at a time. DCHECK(!callback_); if (!cache_) @@ -492,6 +483,61 @@ int HttpCache::Transaction::HandleResult(int rv) { return rv; } +int HttpCache::Transaction::DoLoop(int result) { + DCHECK(next_state_ != STATE_NONE); + + int rv = result; + do { + State state = next_state_; + next_state_ = STATE_NONE; + switch (state) { + case STATE_SEND_REQUEST: + DCHECK_EQ(OK, rv); + rv = DoSendRequest(); + break; + case STATE_SEND_REQUEST_COMPLETE: + rv = DoSendRequestComplete(rv); + break; + case STATE_NETWORK_READ: + DCHECK_EQ(OK, rv); + rv = DoNetworkRead(); + break; + case STATE_NETWORK_READ_COMPLETE: + rv = DoNetworkReadComplete(rv); + break; + case STATE_CACHE_QUERY_DATA: + DCHECK_EQ(OK, rv); + rv = DoCacheQueryData(); + break; + case STATE_CACHE_QUERY_DATA_COMPLETE: + rv = DoCacheQueryDataComplete(rv); + break; + case STATE_CACHE_READ_DATA: + DCHECK_EQ(OK, rv); + rv = DoCacheReadData(); + break; + case STATE_CACHE_READ_DATA_COMPLETE: + rv = DoCacheReadDataComplete(rv); + break; + case STATE_CACHE_WRITE_DATA: + rv = DoCacheWriteData(rv); + break; + case STATE_CACHE_WRITE_DATA_COMPLETE: + rv = DoCacheWriteDataComplete(rv); + break; + default: + NOTREACHED() << "bad state"; + rv = ERR_FAILED; + break; + } + } while (rv != ERR_IO_PENDING && next_state_ != STATE_NONE); + + if (rv != ERR_IO_PENDING) + HandleResult(rv); + + return rv; +} + void HttpCache::Transaction::SetRequest(LoadLog* load_log, const HttpRequestInfo* request) { load_log_ = load_log; @@ -692,34 +738,41 @@ int HttpCache::Transaction::BeginPartialCacheValidation() { bool byte_range_requested = partial_.get() != NULL; if (byte_range_requested) { - // Balanced in ValidateEntryHeadersAndContinue. - entry_ready_callback_->AddRef(); - if (OK != entry_->disk_entry->ReadyForSparseIO(entry_ready_callback_)) - return ERR_IO_PENDING; - } else { - // The request is not for a range, but we have stored just ranges. - partial_.reset(new PartialData()); - partial_->SetHeaders(request_->extra_headers); - if (!custom_request_.get()) { - custom_request_.reset(new HttpRequestInfo(*request_)); - request_ = custom_request_.get(); - } + next_state_ = STATE_CACHE_QUERY_DATA; + return DoLoop(OK); + } + // The request is not for a range, but we have stored just ranges. + partial_.reset(new PartialData()); + partial_->SetHeaders(request_->extra_headers); + if (!custom_request_.get()) { + custom_request_.reset(new HttpRequestInfo(*request_)); + request_ = custom_request_.get(); } - return ValidateEntryHeadersAndContinue(byte_range_requested); + return ValidateEntryHeadersAndContinue(false); } -int HttpCache::Transaction::ValidateEntryHeadersAndContinue( - bool byte_range_requested) { - DCHECK(mode_ == READ_WRITE); +int HttpCache::Transaction::DoCacheQueryData() { + next_state_ = STATE_CACHE_QUERY_DATA_COMPLETE; - if (byte_range_requested) { - // Balance the AddRef from BeginPartialCacheValidation. - entry_ready_callback_->Release(); - } + // Balanced in ValidateEntryHeadersAndContinue. + cache_callback_->AddRef(); + return entry_->disk_entry->ReadyForSparseIO(cache_callback_); +} +int HttpCache::Transaction::DoCacheQueryDataComplete(int result) { + DCHECK_EQ(OK, result); + // Balance the AddRef from BeginPartialCacheValidation. + cache_callback_->Release(); if (!cache_) - return HandleResult(ERR_UNEXPECTED); + return ERR_UNEXPECTED; + + return ValidateEntryHeadersAndContinue(true); +} + +int HttpCache::Transaction::ValidateEntryHeadersAndContinue( + bool byte_range_requested) { + DCHECK(mode_ == READ_WRITE); if (!partial_->UpdateFromStoredHeaders(response_.headers, entry_->disk_entry, truncated_)) { @@ -799,6 +852,11 @@ int HttpCache::Transaction::BeginExternallyConditionalizedRequest() { } int HttpCache::Transaction::BeginNetworkRequest() { + next_state_ = STATE_SEND_REQUEST; + return DoLoop(OK); +} + +int HttpCache::Transaction::DoSendRequest() { DCHECK(mode_ & WRITE || mode_ == NONE); DCHECK(!network_trans_.get()); @@ -807,19 +865,20 @@ int HttpCache::Transaction::BeginNetworkRequest() { if (rv != OK) return rv; - rv = network_trans_->Start(request_, &network_info_callback_, load_log_); - if (rv != ERR_IO_PENDING) - OnNetworkInfoAvailable(rv); + next_state_ = STATE_SEND_REQUEST_COMPLETE; + rv = network_trans_->Start(request_, &network_callback_, load_log_); return rv; } int HttpCache::Transaction::RestartNetworkRequest() { DCHECK(mode_ & WRITE || mode_ == NONE); DCHECK(network_trans_.get()); + DCHECK_EQ(STATE_NONE, next_state_); - int rv = network_trans_->RestartIgnoringLastError(&network_info_callback_); + next_state_ = STATE_SEND_REQUEST_COMPLETE; + int rv = network_trans_->RestartIgnoringLastError(&network_callback_); if (rv != ERR_IO_PENDING) - OnNetworkInfoAvailable(rv); + return DoLoop(rv); return rv; } @@ -827,11 +886,13 @@ int HttpCache::Transaction::RestartNetworkRequestWithCertificate( X509Certificate* client_cert) { DCHECK(mode_ & WRITE || mode_ == NONE); DCHECK(network_trans_.get()); + DCHECK_EQ(STATE_NONE, next_state_); + next_state_ = STATE_SEND_REQUEST_COMPLETE; int rv = network_trans_->RestartWithCertificate(client_cert, - &network_info_callback_); + &network_callback_); if (rv != ERR_IO_PENDING) - OnNetworkInfoAvailable(rv); + return DoLoop(rv); return rv; } @@ -840,11 +901,13 @@ int HttpCache::Transaction::RestartNetworkRequestWithAuth( const std::wstring& password) { DCHECK(mode_ & WRITE || mode_ == NONE); DCHECK(network_trans_.get()); + DCHECK_EQ(STATE_NONE, next_state_); + next_state_ = STATE_SEND_REQUEST_COMPLETE; int rv = network_trans_->RestartWithAuth(username, password, - &network_info_callback_); + &network_callback_); if (rv != ERR_IO_PENDING) - OnNetworkInfoAvailable(rv); + return DoLoop(rv); return rv; } @@ -1045,34 +1108,36 @@ void HttpCache::Transaction::IgnoreRangeRequest() { } int HttpCache::Transaction::ReadFromNetwork(IOBuffer* data, int data_len) { - int rv = network_trans_->Read(data, data_len, &network_read_callback_); read_buf_ = data; read_buf_len_ = data_len; - if (rv >= 0) - rv = DoNetworkReadCompleted(rv); - return rv; + 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_); } int HttpCache::Transaction::ReadFromEntry(IOBuffer* data, int data_len) { - DCHECK(entry_); - int rv; - cache_read_callback_->AddRef(); // Balanced in OnCacheReadCompleted. - if (partial_.get()) { - rv = partial_->CacheRead(entry_->disk_entry, data, data_len, - cache_read_callback_); - } else { - rv = entry_->disk_entry->ReadData(kResponseContentIndex, read_offset_, - data, data_len, cache_read_callback_); - } read_buf_ = data; read_buf_len_ = data_len; - if (rv != ERR_IO_PENDING) - cache_read_callback_->Release(); + next_state_ = STATE_CACHE_READ_DATA; + return DoLoop(OK); +} - if (rv >= 0) - rv = DoCacheReadCompleted(rv); +int HttpCache::Transaction::DoCacheReadData() { + DCHECK(entry_); + 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_, + cache_callback_); + } - return rv; + return entry_->disk_entry->ReadData(kResponseContentIndex, read_offset_, + read_buf_, read_buf_len_, + cache_callback_); } int HttpCache::Transaction::ReadResponseInfoFromEntry() { @@ -1186,19 +1251,21 @@ void HttpCache::Transaction::DoomPartialEntry(bool delete_object) { partial_.reset(NULL); } -int HttpCache::Transaction::DoNetworkReadCompleted(int result) { +int HttpCache::Transaction::DoNetworkReadComplete(int result) { DCHECK(mode_ & WRITE || mode_ == NONE); if (!cache_) - return HandleResult(ERR_UNEXPECTED); + return ERR_UNEXPECTED; - cache_write_callback_->AddRef(); // Balanced in DoCacheWriteCompleted. + next_state_ = STATE_CACHE_WRITE_DATA; + return result; +} - result = AppendResponseDataToEntry(read_buf_, result, cache_write_callback_); - if (result == ERR_IO_PENDING) - return result; +int HttpCache::Transaction::DoCacheWriteData(int num_bytes) { + next_state_ = STATE_CACHE_WRITE_DATA_COMPLETE; + cache_callback_->AddRef(); // Balanced in DoCacheWriteDataComplete. - return DoCacheWriteCompleted(result); + return AppendResponseDataToEntry(read_buf_, num_bytes, cache_callback_); } int HttpCache::Transaction::DoPartialNetworkReadCompleted(int result) { @@ -1215,14 +1282,14 @@ int HttpCache::Transaction::DoPartialNetworkReadCompleted(int result) { } DoneWritingToEntry(true); } - return HandleResult(result); + return result; } -int HttpCache::Transaction::DoCacheReadCompleted(int result) { - DCHECK(cache_); +int HttpCache::Transaction::DoCacheReadDataComplete(int result) { + cache_callback_->Release(); // Balance the AddRef from DoCacheReadData. if (!cache_) - return HandleResult(ERR_UNEXPECTED); + return ERR_UNEXPECTED; if (partial_.get()) return DoPartialCacheReadCompleted(result); @@ -1233,7 +1300,7 @@ int HttpCache::Transaction::DoCacheReadCompleted(int result) { cache_->DoneReadingFromEntry(entry_, this); entry_ = NULL; } - return HandleResult(result); + return result; } int HttpCache::Transaction::DoPartialCacheReadCompleted(int result) { @@ -1252,18 +1319,17 @@ int HttpCache::Transaction::DoPartialCacheReadCompleted(int result) { cache_->DoneReadingFromEntry(entry_, this); entry_ = NULL; } - return HandleResult(result); + return result; } -int HttpCache::Transaction::DoCacheWriteCompleted(int result) { - DCHECK(cache_); - // Balance the AddRef from DoNetworkReadCompleted. - cache_write_callback_->Release(); +int HttpCache::Transaction::DoCacheWriteDataComplete(int result) { + // Balance the AddRef from DoCacheWriteData. + cache_callback_->Release(); if (!cache_) - return HandleResult(ERR_UNEXPECTED); + return ERR_UNEXPECTED; if (result < 0) - return HandleResult(result); + return result; if (partial_.get()) return DoPartialNetworkReadCompleted(result); @@ -1271,16 +1337,12 @@ int HttpCache::Transaction::DoCacheWriteCompleted(int result) { if (result == 0) // End of file. DoneWritingToEntry(true); - return HandleResult(result); + return result; } -void HttpCache::Transaction::OnNetworkInfoAvailable(int result) { - DCHECK(result != ERR_IO_PENDING); - - if (!cache_) { - HandleResult(ERR_UNEXPECTED); - return; - } +int HttpCache::Transaction::DoSendRequestComplete(int result) { + if (!cache_) + return ERR_UNEXPECTED; if (result == OK) { const HttpResponseInfo* new_response = network_trans_->GetResponseInfo(); @@ -1290,16 +1352,15 @@ void HttpCache::Transaction::OnNetworkInfoAvailable(int result) { } else { bool partial_content; if (!ValidatePartialResponse(new_response->headers, &partial_content) && - !auth_response_.headers && callback_) { + !auth_response_.headers) { // Something went wrong with this request and we have to restart it. - // If there is no callback we'll return OK to the caller so we cannot - // restart the request. If we have an authentication response, we are - // exposed to weird things hapenning if the user cancels the - // authentication before we receive the new response. - network_trans_.reset(); + // If we have an authentication response, we are exposed to weird things + // hapenning if the user cancels the authentication before we receive + // the new response. response_ = HttpResponseInfo(); - BeginNetworkRequest(); - return; + network_trans_.reset(); + next_state_ = STATE_SEND_REQUEST; + return OK; } if (partial_content && mode_ == READ_WRITE && !truncated_ && response_.headers->response_code() == 200) { @@ -1369,12 +1430,19 @@ void HttpCache::Transaction::OnNetworkInfoAvailable(int result) { } if (reading_ && partial_.get()) { if (network_trans_.get()) { + next_state_ = STATE_NETWORK_READ_COMPLETE; result = ReadFromNetwork(read_buf_, read_buf_len_); } else { + next_state_ = STATE_CACHE_READ_DATA_COMPLETE; result = ReadFromEntry(read_buf_, read_buf_len_); } - if (result >= 0 || result == net::ERR_IO_PENDING) - return; + if (result >= 0 || result == ERR_IO_PENDING) { + // Keep looping. + return result; + } else { + // Don't keep looping when we return. + next_state_ = STATE_NONE; + } } else if (mode_ != NONE && partial_.get()) { // We are about to return the headers for a byte-range request to the // user, so let's fix them. @@ -1392,25 +1460,11 @@ void HttpCache::Transaction::OnNetworkInfoAvailable(int result) { DCHECK(response); response_.cert_request_info = response->cert_request_info; } - HandleResult(result); -} - -void HttpCache::Transaction::OnNetworkReadCompleted(int result) { - DoNetworkReadCompleted(result); + return result; } -void HttpCache::Transaction::OnCacheReadCompleted(int result) { - cache_read_callback_->Release(); // Balance the AddRef from ReadFromEntry. - DoCacheReadCompleted(result); -} - -void HttpCache::Transaction::OnCacheWriteCompleted(int result) { - DoCacheWriteCompleted(result); -} - -void HttpCache::Transaction::OnCacheEntryReady(int result) { - DCHECK_EQ(OK, result); - ValidateEntryHeadersAndContinue(true); +void HttpCache::Transaction::OnIOComplete(int result) { + DoLoop(result); } } // namespace net diff --git a/net/http/http_cache_transaction.h b/net/http/http_cache_transaction.h index 69b785c..0b7e517 100644 --- a/net/http/http_cache_transaction.h +++ b/net/http/http_cache_transaction.h @@ -94,6 +94,34 @@ class HttpCache::Transaction : public HttpTransaction { bool initialized; }; + enum State { + STATE_NONE, + STATE_START_REQUEST, + STATE_SEND_REQUEST, + STATE_SEND_REQUEST_COMPLETE, + STATE_NETWORK_READ, + STATE_NETWORK_READ_COMPLETE, + STATE_OPEN_ENTRY, + STATE_OPEN_ENTRY_COMPLETE, + STATE_CREATE_ENTRY, + STATE_CREATE_ENTRY_COMPLETE, + STATE_DOOM_ENTRY, + STATE_DOOM_ENTRY_COMPLETE, + STATE_ADD_TO_ENTRY, + STATE_ADD_TO_ENTRY_COMPLETE, + STATE_ENTRY_AVAILABLE, + STATE_CACHE_READ_RESPONSE, + STATE_CACHE_READ_RESPONSE_COMPLETE, + STATE_CACHE_WRITE_RESPONSE, + STATE_CACHE_WRITE_RESPONSE_COMPLETE, + STATE_CACHE_QUERY_DATA, + STATE_CACHE_QUERY_DATA_COMPLETE, + STATE_CACHE_READ_DATA, + STATE_CACHE_READ_DATA_COMPLETE, + STATE_CACHE_WRITE_DATA, + STATE_CACHE_WRITE_DATA_COMPLETE + }; + // This is a helper function used to trigger a completion callback. It may // only be called if callback_ is non-null. void DoCallback(int rv); @@ -101,6 +129,21 @@ class HttpCache::Transaction : public HttpTransaction { // This will trigger the completion callback if appropriate. int HandleResult(int rv); + // Runs the state transition loop. + int DoLoop(int result); + + // Each of these methods corresponds to a State value. + int DoSendRequest(); + int DoSendRequestComplete(int result); + int DoNetworkRead(); + int DoNetworkReadComplete(int result); + int DoCacheReadData(); + int DoCacheReadDataComplete(int result); + int DoCacheQueryData(); + int DoCacheQueryDataComplete(int result); + int DoCacheWriteData(int num_bytes); + int DoCacheWriteDataComplete(int result); + // Sets request_ and fields derived from it. void SetRequest(LoadLog* load_log, const HttpRequestInfo* request); @@ -199,16 +242,10 @@ class HttpCache::Transaction : public HttpTransaction { // the control object (partial_). void DoomPartialEntry(bool delete_object); - // Performs the needed work after receiving data from the network. - int DoNetworkReadCompleted(int result); - // Performs the needed work after receiving data from the network, when // working with range requests. int DoPartialNetworkReadCompleted(int result); - // Performs the needed work after receiving data from the cache. - int DoCacheReadCompleted(int result); - // Performs the needed work after receiving data from the cache, when // working with range requests. int DoPartialCacheReadCompleted(int result); @@ -216,23 +253,12 @@ class HttpCache::Transaction : public HttpTransaction { // Performs the needed work after writing data to the cache. int DoCacheWriteCompleted(int result); - // Called to signal completion of the network transaction's Start method: - void OnNetworkInfoAvailable(int result); - - // Called to signal completion of the network transaction's Read method: - void OnNetworkReadCompleted(int result); + // Called to signal completion of asynchronous IO. + void OnIOComplete(int result); - // Called to signal completion of the cache's ReadData method: - void OnCacheReadCompleted(int result); - - // Called to signal completion of the cache's WriteData method: - void OnCacheWriteCompleted(int result); - - // Called to signal completion of the cache entry's ReadyForSparseIO method: - void OnCacheEntryReady(int result); - - scoped_refptr load_log_; + State next_state_; const HttpRequestInfo* request_; + scoped_refptr load_log_; scoped_ptr custom_request_; // If extra_headers specified a "if-modified-since" or "if-none-match", // |external_validation_| contains the value of those headers. @@ -255,14 +281,8 @@ class HttpCache::Transaction : public HttpTransaction { int effective_load_flags_; scoped_ptr partial_; // We are dealing with range requests. uint64 final_upload_progress_; - CompletionCallbackImpl network_info_callback_; - CompletionCallbackImpl network_read_callback_; - scoped_refptr > - cache_read_callback_; - scoped_refptr > - cache_write_callback_; - scoped_refptr > - entry_ready_callback_; + CompletionCallbackImpl network_callback_; + scoped_refptr > cache_callback_; }; } // namespace net -- cgit v1.1