Index: content/child/service_worker/service_worker_dispatcher.cc |
diff --git a/content/child/service_worker/service_worker_dispatcher.cc b/content/child/service_worker/service_worker_dispatcher.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..9af20ba4830f8beee399ff8e091adc90dd98305e |
--- /dev/null |
+++ b/content/child/service_worker/service_worker_dispatcher.cc |
@@ -0,0 +1,128 @@ |
+// Copyright 2013 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "content/child/service_worker/service_worker_dispatcher.h" |
+ |
+#include "base/lazy_instance.h" |
+#include "base/threading/thread_local.h" |
+#include "content/child/service_worker/web_service_worker_impl.h" |
+#include "content/child/thread_safe_sender.h" |
+#include "content/common/service_worker_messages.h" |
+#include "third_party/WebKit/public/web/WebSecurityOrigin.h" |
+ |
+using WebKit::WebServiceWorkerProvider; |
+using base::ThreadLocalPointer; |
+using webkit_glue::WorkerTaskRunner; |
+ |
+namespace content { |
+ |
+static base::LazyInstance<ThreadLocalPointer<ServiceWorkerDispatcher> >::Leaky |
+ g_dispatcher_tls = LAZY_INSTANCE_INITIALIZER; |
+ |
+ServiceWorkerDispatcher* const kHasBeenDeleted = |
michaeln
2013/10/24 21:57:39
this wants to be either static or in an anon names
alecflett
2013/10/24 22:41:22
Done.
|
+ reinterpret_cast<ServiceWorkerDispatcher*>(0x1); |
+ |
+ServiceWorkerDispatcher::ServiceWorkerDispatcher( |
+ ThreadSafeSender* thread_safe_sender) |
+ : thread_safe_sender_(thread_safe_sender) { |
+ g_dispatcher_tls.Pointer()->Set(this); |
+} |
+ |
+ServiceWorkerDispatcher::~ServiceWorkerDispatcher() { |
+ g_dispatcher_tls.Pointer()->Set(kHasBeenDeleted); |
+} |
+ |
+void ServiceWorkerDispatcher::OnMessageReceived(const IPC::Message& msg) { |
+ bool handled = true; |
+ IPC_BEGIN_MESSAGE_MAP(ServiceWorkerDispatcher, msg) |
+ IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerRegistered, |
+ OnServiceWorkerRegistered) |
+ IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerUnregistered, |
+ OnServiceWorkerUnregistered) |
+ IPC_MESSAGE_UNHANDLED(handled = false) |
+ IPC_END_MESSAGE_MAP() |
+ DCHECK(handled) << "Unhandled message:" << msg.type(); |
+} |
+ |
+bool ServiceWorkerDispatcher::Send(IPC::Message* msg) { |
+ return thread_safe_sender_->Send(msg); |
+} |
+ |
+static int CurrentWorkerId() { |
michaeln
2013/10/24 21:57:39
please move this helper up to where the other non-
alecflett
2013/10/24 22:41:22
Done.
|
+ return WorkerTaskRunner::Instance()->CurrentWorkerId(); |
+} |
+ |
+void ServiceWorkerDispatcher::RegisterServiceWorker( |
+ const GURL& pattern, |
+ const GURL& scriptUrl, |
+ WebServiceWorkerProvider::WebServiceWorkerCallbacks* callbacks) { |
+ DCHECK(callbacks); |
+ int request_id = pending_callbacks_.Add(callbacks); |
+ thread_safe_sender_->Send(new ServiceWorkerHostMsg_RegisterServiceWorker( |
+ CurrentWorkerId(), request_id, pattern, scriptUrl)); |
+} |
+ |
+void ServiceWorkerDispatcher::UnregisterServiceWorker( |
+ const GURL& pattern, |
+ WebServiceWorkerProvider::WebServiceWorkerCallbacks* callbacks) { |
+ DCHECK(callbacks); |
+ int request_id = pending_callbacks_.Add(callbacks); |
+ thread_safe_sender_->Send(new ServiceWorkerHostMsg_UnregisterServiceWorker( |
+ CurrentWorkerId(), request_id, pattern)); |
+} |
+ |
+ServiceWorkerDispatcher* ServiceWorkerDispatcher::ThreadSpecificInstance( |
+ ThreadSafeSender* thread_safe_sender) { |
+ if (g_dispatcher_tls.Pointer()->Get() == kHasBeenDeleted) { |
+ NOTREACHED() << "Re-instantiating TLS ServiceWorkerDispatcher."; |
+ g_dispatcher_tls.Pointer()->Set(NULL); |
+ } |
+ if (g_dispatcher_tls.Pointer()->Get()) |
+ return g_dispatcher_tls.Pointer()->Get(); |
+ |
+ ServiceWorkerDispatcher* dispatcher = |
+ new ServiceWorkerDispatcher(thread_safe_sender); |
+ if (WorkerTaskRunner::Instance()->CurrentWorkerId()) |
+ webkit_glue::WorkerTaskRunner::Instance()->AddStopObserver(dispatcher); |
+ return dispatcher; |
+} |
+ |
+void ServiceWorkerDispatcher::OnServiceWorkerRegistered( |
+ int32 thread_id, |
+ int32 request_id, |
+ int64 service_worker_id) { |
+ WebServiceWorkerProvider::WebServiceWorkerCallbacks* callbacks = |
+ pending_callbacks_.Lookup(request_id); |
+ DCHECK(callbacks); |
+ if (!callbacks) |
+ return; |
+ |
+ // the browser has to generate the service_worker_id so the same |
+ // worker can be called from different renderer contexts. However, |
+ // the impl object doesn't have to be the same instance across calls |
+ // unless we require the DOM objects to be identical when there's a |
+ // duplicate registration. So for now we mint a new object each |
+ // time. |
+ WebServiceWorkerImpl* worker = |
+ new WebServiceWorkerImpl(service_worker_id, true); |
+ callbacks->onSuccess(worker); |
+ pending_callbacks_.Remove(request_id); |
+} |
+ |
+void ServiceWorkerDispatcher::OnServiceWorkerUnregistered( |
+ int32 thread_id, |
+ int32 request_id) { |
+ WebServiceWorkerProvider::WebServiceWorkerCallbacks* callbacks = |
+ pending_callbacks_.Lookup(request_id); |
+ DCHECK(callbacks); |
+ if (!callbacks) |
+ return; |
+ |
+ callbacks->onSuccess(nullptr); |
michaeln
2013/10/24 21:57:39
use NULL here
alecflett
2013/10/24 22:41:22
Done.
|
+ pending_callbacks_.Remove(request_id); |
+} |
+ |
+void ServiceWorkerDispatcher::OnWorkerRunLoopStopped() { delete this; } |
+ |
+} // namespace content |