diff options
author | stoyan@chromium.org <stoyan@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-02-02 01:43:41 +0000 |
---|---|---|
committer | stoyan@chromium.org <stoyan@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-02-02 01:43:41 +0000 |
commit | c21a502e9a25710f456e6240fc3455d8a98713d0 (patch) | |
tree | 186b2ce6bdbb77fb3fe855e58f4d53064cf4178a | |
parent | f884f2a323456f31486f9dd33d704ee086475355 (diff) | |
download | chromium_src-c21a502e9a25710f456e6240fc3455d8a98713d0.zip chromium_src-c21a502e9a25710f456e6240fc3455d8a98713d0.tar.gz chromium_src-c21a502e9a25710f456e6240fc3455d8a98713d0.tar.bz2 |
Bring the lock in UrlmonUrlRequestManager back. We need it to access worker thread's message loop.
Added the unsafe redirect handling.
Review URL: http://codereview.chromium.org/568004
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@37782 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome_frame/urlmon_url_request.cc | 43 | ||||
-rw-r--r-- | chrome_frame/urlmon_url_request.h | 4 |
2 files changed, 39 insertions, 8 deletions
diff --git a/chrome_frame/urlmon_url_request.cc b/chrome_frame/urlmon_url_request.cc index 35e588e..11f6ff6 100644 --- a/chrome_frame/urlmon_url_request.cc +++ b/chrome_frame/urlmon_url_request.cc @@ -397,6 +397,18 @@ STDMETHODIMP UrlmonUrlRequest::OnStopBinding(HRESULT result, LPCWSTR error) { if (state == Status::WORKING) { status_.set_result(result); + // Special case. If the last request was a redirect and the current OS + // error value is E_ACCESSDENIED, that means an unsafe redirect was + // attempted. In that case, correct the OS error value to be the more + // specific ERR_UNSAFE_REDIRECT error value. + if (result == E_ACCESSDENIED) { + int http_code = GetHttpResponseStatus(); + if (300 <= http_code && http_code < 400) { + status_.set_result(URLRequestStatus::FAILED, + net::ERR_UNSAFE_REDIRECT); + } + } + // The code below seems easy but it is not. :) // we cannot have pending read and data_avail at the same time. DCHECK(!(pending_read_size_ > 0 && cached_data_.is_valid())); @@ -1033,9 +1045,9 @@ void UrlmonUrlRequestManager::StartRequest(int request_id, } } - worker_thread_.message_loop()->PostTask(FROM_HERE, - NewRunnableMethod(this, &UrlmonUrlRequestManager::StartRequestWorker, - request_id, request_info, use_moniker)); + ExecuteInWorkerThread(FROM_HERE, NewRunnableMethod(this, + &UrlmonUrlRequestManager::StartRequestWorker, + request_id, request_info, use_moniker)); } void UrlmonUrlRequestManager::StartRequestWorker(int request_id, @@ -1079,7 +1091,7 @@ void UrlmonUrlRequestManager::ReadRequest(int request_id, int bytes_to_read) { if (stopping_) return; - worker_thread_.message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this, + ExecuteInWorkerThread(FROM_HERE, NewRunnableMethod(this, &UrlmonUrlRequestManager::ReadRequestWorker, request_id, bytes_to_read)); } @@ -1097,7 +1109,7 @@ void UrlmonUrlRequestManager::EndRequest(int request_id) { if (stopping_) return; - worker_thread_.message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this, + ExecuteInWorkerThread(FROM_HERE, NewRunnableMethod(this, &UrlmonUrlRequestManager::EndRequestWorker, request_id)); } @@ -1118,7 +1130,7 @@ void UrlmonUrlRequestManager::StopAll() { if (!worker_thread_.IsRunning()) return; - worker_thread_.message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this, + ExecuteInWorkerThread(FROM_HERE, NewRunnableMethod(this, &UrlmonUrlRequestManager::StopAllWorker)); // Note we may not call worker_thread_.Stop() here. The MessageLoop's quit @@ -1130,7 +1142,10 @@ void UrlmonUrlRequestManager::StopAll() { // The problem is that while waiting for OnStopBinding(), Quit Task may be // picked up and executed, thus exiting the thread. map_empty_.Wait(); + + worker_thread_access_.Acquire(); worker_thread_.Stop(); + worker_thread_access_.Release(); DCHECK_EQ(0, UrlmonUrlRequest::instance_count_); } @@ -1209,10 +1224,10 @@ void UrlmonUrlRequestManager::StealMonikerFromRequest(int request_id, return; base::WaitableEvent done(true, false); - worker_thread_.message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this, + bool posted = ExecuteInWorkerThread(FROM_HERE, NewRunnableMethod(this, &UrlmonUrlRequestManager::StealMonikerFromRequestWorker, request_id, moniker, &done)); - + DCHECK_EQ(true, posted); // Wait until moniker is grabbed from a request in the worker thread. done.Wait(); } @@ -1229,3 +1244,15 @@ void UrlmonUrlRequestManager::StealMonikerFromRequestWorker(int request_id, done->Signal(); } + +bool UrlmonUrlRequestManager::ExecuteInWorkerThread( + const tracked_objects::Location& from_here, Task* task) { + AutoLock lock(worker_thread_access_); + if (worker_thread_.IsRunning()) { + worker_thread_.message_loop()->PostTask(from_here, task); + return true; + } + + delete task; + return false; +} diff --git a/chrome_frame/urlmon_url_request.h b/chrome_frame/urlmon_url_request.h index e47edcc..325edb8 100644 --- a/chrome_frame/urlmon_url_request.h +++ b/chrome_frame/urlmon_url_request.h @@ -10,6 +10,7 @@ #include <atlcom.h> #include <string> +#include "base/lock.h" #include "base/scoped_comptr_win.h" #include "base/thread.h" #include "base/waitable_event.h" @@ -60,6 +61,8 @@ class UrlmonUrlRequestManager : virtual void OnReadComplete(int request_id, const void* buffer, int len); virtual void OnResponseEnd(int request_id, const URLRequestStatus& status); + bool ExecuteInWorkerThread(const tracked_objects::Location& from_here, + Task* task); // Methods executed in worker thread. void StartRequestWorker(int request_id, const IPC::AutomationURLRequest& request_info, @@ -79,6 +82,7 @@ class UrlmonUrlRequestManager : STAThread worker_thread_; base::WaitableEvent map_empty_; bool stopping_; + Lock worker_thread_access_; }; #endif // CHROME_FRAME_URLMON_URL_REQUEST_H_ |