Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1241)

Unified Diff: content/browser/shared_worker/shared_worker_service_impl.cc

Issue 2249173003: Removes the references to shared workers from the all documents in being deleted frames (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: use WebContentsObserver::RenderFrameDeleted Created 4 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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,
« no previous file with comments | « content/browser/shared_worker/shared_worker_service_impl.h ('k') | content/browser/shared_worker/worker_document_set.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698