summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authortwiz@google.com <twiz@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2010-10-04 20:08:26 +0000
committertwiz@google.com <twiz@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2010-10-04 20:08:26 +0000
commit7c6536a2757ceb31441f7cd0c4f4db2608b9a855 (patch)
tree645dbd39b03b13619285b7b69823813796e1bd9f
parent2248697e7d6c5d9de6fc31a7261f36dffaa0a5d3 (diff)
downloadchromium_src-7c6536a2757ceb31441f7cd0c4f4db2608b9a855.zip
chromium_src-7c6536a2757ceb31441f7cd0c4f4db2608b9a855.tar.gz
chromium_src-7c6536a2757ceb31441f7cd0c4f4db2608b9a855.tar.bz2
Simple fix correcting the lifetime scope of automated url request jobs during chrome-initiated xmlhttprequest aborts. Two similar problems are fixed by this CL.
I found a race condition during the early-termination processing of the NpapiUrlRequest objects. If NpapiUrlRequest::Stop is invoked before NpapiUrlRequest::OnStreamCreated, then it is possible that a new stream will be created on the Chrome-Frame side, for which an operation that has been officially cancelled within Chrome. The fix is to add a new parameter fo the NPAPIUrlRequest, stop_requested_ that is used to track this behaviour. If a new stream is created on a request that has been stopped, then that stream is immediately destroyed, without notification back to Chrome. If Chrome requests an abort of a url request, then Chrome-Frame should not send any notifications back to Chrome concerning the tear-down of that request. The delegate is revoked in NPAPIUrlRequest::Stop to achieve this behaviour. BUG=None TEST=None Review URL: http://codereview.chromium.org/3530002 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@61410 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--chrome_frame/npapi_url_request.cc68
1 files changed, 49 insertions, 19 deletions
diff --git a/chrome_frame/npapi_url_request.cc b/chrome_frame/npapi_url_request.cc
index 9799a87..ebc4782 100644
--- a/chrome_frame/npapi_url_request.cc
+++ b/chrome_frame/npapi_url_request.cc
@@ -29,6 +29,14 @@ class NPAPIUrlRequest : public PluginUrlRequest {
virtual unsigned long API_CALL AddRef();
virtual unsigned long API_CALL Release();
+ const URLRequestStatus& status() const {
+ return status_;
+ }
+
+ NPP instance() const {
+ return instance_;
+ }
+
private:
unsigned long ref_count_;
NPP instance_;
@@ -113,6 +121,8 @@ bool NPAPIUrlRequest::Start() {
void NPAPIUrlRequest::Stop() {
DLOG(INFO) << "Finished request: Url - " << url() << " result: "
<< static_cast<int>(status_.status());
+
+ status_.set_status(URLRequestStatus::CANCELED);
if (stream_) {
npapi::DestroyStream(instance_, stream_, NPRES_USER_BREAK);
stream_ = NULL;
@@ -137,21 +147,28 @@ NPError NPAPIUrlRequest::OnStreamCreated(const char* mime_type,
}
NPError NPAPIUrlRequest::OnStreamDestroyed(NPReason reason) {
- URLRequestStatus::Status status = URLRequestStatus::FAILED;
- switch (reason) {
- case NPRES_DONE:
- status_.set_status(URLRequestStatus::SUCCESS);
- status_.set_os_error(0);
- break;
- case NPRES_USER_BREAK:
- status_.set_status(URLRequestStatus::CANCELED);
- status_.set_os_error(net::ERR_ABORTED);
- break;
- case NPRES_NETWORK_ERR:
- default:
- status_.set_status(URLRequestStatus::FAILED);
- status_.set_os_error(net::ERR_CONNECTION_CLOSED);
- break;
+ // If the request has been aborted, then ignore the |reason| argument.
+ // Normally the execution flow is such than NPRES_USER_BREAK will be passed
+ // when the stream is aborted, but sometimes NPRES_NETWORK_ERROR is passed
+ // instead. To prevent Chrome from receiving a notification of a failed
+ // network connection, when Chrome actually canceled the request, we ignore
+ // the status here.
+ if (URLRequestStatus::CANCELED != status_.status()) {
+ switch (reason) {
+ case NPRES_DONE:
+ status_.set_status(URLRequestStatus::SUCCESS);
+ status_.set_os_error(0);
+ break;
+ case NPRES_USER_BREAK:
+ status_.set_status(URLRequestStatus::CANCELED);
+ status_.set_os_error(net::ERR_ABORTED);
+ break;
+ case NPRES_NETWORK_ERR:
+ default:
+ status_.set_status(URLRequestStatus::FAILED);
+ status_.set_os_error(net::ERR_CONNECTION_CLOSED);
+ break;
+ }
}
delegate_->OnResponseEnd(id(), status_);
@@ -207,10 +224,9 @@ void NPAPIUrlRequestManager::StartRequest(int request_id,
request_info.upload_data,
static_cast<ResourceType::Type>(request_info.resource_type),
enable_frame_busting_)) {
- // Add to map.
DCHECK(request_map_.find(request_id) == request_map_.end());
- request_map_[request_id] = new_request;
if (new_request->Start()) {
+ request_map_[request_id] = new_request;
// Keep additional reference on request for NPSTREAM
// This will be released in NPP_UrlNotify
new_request->AddRef();
@@ -343,6 +359,13 @@ NPError NPAPIUrlRequestManager::NewStream(NPMIMEType type,
if (!request)
return NPERR_NO_ERROR;
+ // This stream is being constructed for a request that has already been
+ // canceled. Signal its immediate termination.
+ if (URLRequestStatus::CANCELED == request->status().status()) {
+ return npapi::DestroyStream(request->instance(),
+ stream, NPRES_USER_BREAK);
+ }
+
DCHECK(request_map_.find(request->id()) != request_map_.end());
// We need to return the requested stream mode if we are returning a success
// code. If we don't do this it causes Opera to blow up.
@@ -372,8 +395,15 @@ NPError NPAPIUrlRequestManager::DestroyStream(NPStream* stream,
NPAPIUrlRequest* request = RequestFromNotifyData(stream->notifyData);
if (!request)
return NPERR_NO_ERROR;
- DCHECK(request_map_.find(request->id()) != request_map_.end());
- return request->OnStreamDestroyed(reason);
+
+ // It is possible to receive notification of a destroyed stream for a
+ // unregistered request: EndRequest will unregister a request in response
+ // to an AutomationMsg_RequestEnd message. EndRequest will also invoke
+ // npapi::DestroyStream, which will call back to this function.
+ if (request_map_.find(request->id()) != request_map_.end())
+ return request->OnStreamDestroyed(reason);
+
+ return NPERR_NO_ERROR;
}
void NPAPIUrlRequestManager::UrlNotify(const char* url, NPReason reason,