Chromium Code Reviews| Index: content/browser/shared_worker/shared_worker_service_impl.cc |
| diff --git a/content/browser/shared_worker/shared_worker_service_impl.cc b/content/browser/shared_worker/shared_worker_service_impl.cc |
| index 03b279369e10145bb8b9bf8ecce92152fd5b7b18..4404a28dce0dd3aa6a1e1efbd655ce9a15fd2d78 100644 |
| --- a/content/browser/shared_worker/shared_worker_service_impl.cc |
| +++ b/content/browser/shared_worker/shared_worker_service_impl.cc |
| @@ -14,15 +14,20 @@ |
| #include "base/memory/ptr_util.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/stl_util.h" |
| +#include "base/threading/thread_task_runner_handle.h" |
| #include "content/browser/devtools/shared_worker_devtools_manager.h" |
| +#include "content/browser/frame_host/render_frame_host_impl.h" |
| #include "content/browser/renderer_host/render_process_host_impl.h" |
| #include "content/browser/shared_worker/shared_worker_host.h" |
| #include "content/browser/shared_worker/shared_worker_instance.h" |
| #include "content/browser/shared_worker/shared_worker_message_filter.h" |
| #include "content/browser/shared_worker/worker_document_set.h" |
| +#include "content/browser/web_contents/web_contents_impl.h" |
| #include "content/common/view_messages.h" |
| #include "content/common/worker_messages.h" |
| #include "content/public/browser/browser_thread.h" |
| +#include "content/public/browser/web_contents.h" |
| +#include "content/public/browser/web_contents_observer.h" |
| #include "content/public/browser/worker_service_observer.h" |
| namespace content { |
| @@ -101,6 +106,56 @@ bool TryIncrementWorkerRefCount(int worker_process_id) { |
| return true; |
| } |
| +void CallRenderFrameDetachedOnIO(int render_process_id, int render_frame_id) { |
| + DCHECK_CURRENTLY_ON(BrowserThread::IO); |
| + SharedWorkerServiceImpl::GetInstance()->RenderFrameDetached(render_process_id, |
| + render_frame_id); |
| +} |
| + |
| +class WorkerDocumentSetWebContentsObserver : public WebContentsObserver { |
| + public: |
| + WorkerDocumentSetWebContentsObserver(WebContents* web_contents, |
| + int render_process_id, |
| + int render_frame_id, |
| + RenderFrameHost* render_frame_host) |
| + : WebContentsObserver(web_contents), |
| + render_process_id_(render_process_id), |
| + render_frame_id_(render_frame_id), |
| + render_frame_host_(render_frame_host) { |
| + DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| + } |
| + |
| + private: |
| + // content::WebContentsObserver overrides. |
|
no sievers
2016/08/23 20:32:42
I wonder if this still needs to also listen to Web
horo
2016/08/24 01:15:49
Done.
|
| + void RenderFrameDeleted(RenderFrameHost* render_frame_host) override { |
| + DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| + if (render_frame_host != render_frame_host_) |
| + return; |
| + Observe(nullptr); |
| + BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, |
| + base::Bind(&CallRenderFrameDetachedOnIO, |
| + render_process_id_, render_frame_id_)); |
| + base::ThreadTaskRunnerHandle::Get()->DeleteSoon(FROM_HERE, this); |
| + } |
| + |
| + const int render_process_id_; |
| + const int render_frame_id_; |
| + const RenderFrameHost* render_frame_host_; |
| +}; |
| + |
| +void RegisterWebContentsObserverOnUI(int render_process_id, |
| + int render_frame_id) { |
| + DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| + RenderFrameHostImpl* render_frame_host = |
| + RenderFrameHostImpl::FromID(render_process_id, render_frame_id); |
| + if (!render_frame_host) |
| + return; |
| + WebContentsImpl* web_contents = static_cast<WebContentsImpl*>( |
| + WebContents::FromRenderFrameHost(render_frame_host)); |
| + new WorkerDocumentSetWebContentsObserver(web_contents, render_process_id, |
| + render_frame_id, render_frame_host); |
| +} |
| + |
| } // namespace |
| class SharedWorkerServiceImpl::SharedWorkerPendingInstance { |
| @@ -157,6 +212,17 @@ class SharedWorkerServiceImpl::SharedWorkerPendingInstance { |
| requests_.erase(to_remove, requests_.end()); |
| } |
| + void RemoveRequestFromFrame(int process_id, int frame_id) { |
| + auto to_remove = std::remove_if( |
| + requests_.begin(), requests_.end(), |
| + [process_id, |
| + frame_id](const std::unique_ptr<SharedWorkerPendingRequest>& r) { |
| + return r->render_process_id == process_id && |
| + r->render_frame_route_id == frame_id; |
| + }); |
| + requests_.erase(to_remove, requests_.end()); |
| + } |
| + |
| void RegisterToSharedWorkerHost(SharedWorkerHost* host) { |
| for (const auto& request : requests_) { |
| host->AddFilter(request->filter, request->route_id); |
| @@ -294,6 +360,11 @@ blink::WebWorkerCreationError SharedWorkerServiceImpl::CreateWorker( |
| request(new SharedWorkerPendingInstance::SharedWorkerPendingRequest( |
| filter, route_id, params.document_id, filter->render_process_id(), |
| params.render_frame_route_id)); |
| + |
| + BrowserThread::PostTask( |
|
no sievers
2016/08/23 20:32:42
Is it possible that there is a pending RenderFrame
horo
2016/08/24 01:15:49
Yes.
So I check "if (!render_frame_host)" in Regis
|
| + BrowserThread::UI, FROM_HERE, |
| + base::Bind(&RegisterWebContentsObserverOnUI, filter->render_process_id(), |
| + params.render_frame_route_id)); |
| if (SharedWorkerPendingInstance* pending = FindPendingInstance(*instance)) { |
| if (params.url != pending->instance()->url()) |
| return blink::WebWorkerCreationErrorURLMismatch; |
| @@ -439,6 +510,23 @@ void SharedWorkerServiceImpl::OnSharedWorkerMessageFilterClosing( |
| pending_instances_.erase(to_remove); |
| } |
| +void SharedWorkerServiceImpl::RenderFrameDetached(int render_process_id, |
| + int render_frame_id) { |
| + ScopedWorkerDependencyChecker checker(this); |
| + for (const auto& it : worker_hosts_) { |
| + it.second->RenderFrameDetached(render_process_id, render_frame_id); |
| + } |
| + |
| + std::vector<int> remove_pending_instance_list; |
| + for (const auto& it : pending_instances_) { |
| + it.second->RemoveRequestFromFrame(render_process_id, render_frame_id); |
| + if (it.second->requests()->empty()) |
| + remove_pending_instance_list.push_back(it.first); |
|
no sievers
2016/08/23 20:32:42
why not erase immediately (while post-incrementing
horo
2016/08/24 01:15:49
Done.
|
| + } |
| + for (int to_remove : remove_pending_instance_list) |
| + pending_instances_.erase(to_remove); |
| +} |
| + |
| void SharedWorkerServiceImpl::NotifyWorkerDestroyed(int worker_process_id, |
| int worker_route_id) { |
| FOR_EACH_OBSERVER(WorkerServiceObserver, |