summaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
Diffstat (limited to 'net')
-rw-r--r--net/http/http_cache_transaction.cc292
-rw-r--r--net/http/http_cache_transaction.h78
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<Transaction>(
- this, &Transaction::OnCacheReadCompleted))),
- ALLOW_THIS_IN_INITIALIZER_LIST(
- cache_write_callback_(new CancelableCompletionCallback<Transaction>(
- this, &Transaction::OnCacheWriteCompleted))),
- ALLOW_THIS_IN_INITIALIZER_LIST(
- entry_ready_callback_(new CancelableCompletionCallback<Transaction>(
- this, &Transaction::OnCacheEntryReady))) {
+ cache_callback_(new CancelableCompletionCallback<Transaction>(
+ 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<LoadLog> load_log_;
+ State next_state_;
const HttpRequestInfo* request_;
+ scoped_refptr<LoadLog> load_log_;
scoped_ptr<HttpRequestInfo> 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<PartialData> partial_; // We are dealing with range requests.
uint64 final_upload_progress_;
- CompletionCallbackImpl<Transaction> network_info_callback_;
- CompletionCallbackImpl<Transaction> network_read_callback_;
- scoped_refptr<CancelableCompletionCallback<Transaction> >
- cache_read_callback_;
- scoped_refptr<CancelableCompletionCallback<Transaction> >
- cache_write_callback_;
- scoped_refptr<CancelableCompletionCallback<Transaction> >
- entry_ready_callback_;
+ CompletionCallbackImpl<Transaction> network_callback_;
+ scoped_refptr<CancelableCompletionCallback<Transaction> > cache_callback_;
};
} // namespace net