• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "content/child/service_worker/service_worker_dispatcher.h"
6 
7 #include "base/debug/trace_event.h"
8 #include "base/lazy_instance.h"
9 #include "base/stl_util.h"
10 #include "base/threading/thread_local.h"
11 #include "content/child/child_thread.h"
12 #include "content/child/service_worker/service_worker_handle_reference.h"
13 #include "content/child/service_worker/service_worker_provider_context.h"
14 #include "content/child/service_worker/service_worker_registration_handle_reference.h"
15 #include "content/child/service_worker/web_service_worker_impl.h"
16 #include "content/child/service_worker/web_service_worker_registration_impl.h"
17 #include "content/child/thread_safe_sender.h"
18 #include "content/child/webmessageportchannel_impl.h"
19 #include "content/common/service_worker/service_worker_messages.h"
20 #include "content/public/common/url_utils.h"
21 #include "third_party/WebKit/public/platform/WebServiceWorkerProviderClient.h"
22 #include "third_party/WebKit/public/web/WebSecurityOrigin.h"
23 
24 using blink::WebServiceWorkerError;
25 using blink::WebServiceWorkerProvider;
26 using base::ThreadLocalPointer;
27 
28 namespace content {
29 
30 namespace {
31 
32 base::LazyInstance<ThreadLocalPointer<ServiceWorkerDispatcher> >::Leaky
33     g_dispatcher_tls = LAZY_INSTANCE_INITIALIZER;
34 
35 ServiceWorkerDispatcher* const kHasBeenDeleted =
36     reinterpret_cast<ServiceWorkerDispatcher*>(0x1);
37 
CurrentWorkerId()38 int CurrentWorkerId() {
39   return WorkerTaskRunner::Instance()->CurrentWorkerId();
40 }
41 
42 }  // namespace
43 
ServiceWorkerDispatcher(ThreadSafeSender * thread_safe_sender)44 ServiceWorkerDispatcher::ServiceWorkerDispatcher(
45     ThreadSafeSender* thread_safe_sender)
46     : thread_safe_sender_(thread_safe_sender) {
47   g_dispatcher_tls.Pointer()->Set(this);
48 }
49 
~ServiceWorkerDispatcher()50 ServiceWorkerDispatcher::~ServiceWorkerDispatcher() {
51   g_dispatcher_tls.Pointer()->Set(kHasBeenDeleted);
52 }
53 
OnMessageReceived(const IPC::Message & msg)54 void ServiceWorkerDispatcher::OnMessageReceived(const IPC::Message& msg) {
55   bool handled = true;
56   IPC_BEGIN_MESSAGE_MAP(ServiceWorkerDispatcher, msg)
57     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_AssociateRegistration,
58                         OnAssociateRegistration)
59     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_DisassociateRegistration,
60                         OnDisassociateRegistration)
61     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerRegistered, OnRegistered)
62     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerUnregistered,
63                         OnUnregistered)
64     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_DidGetRegistration,
65                         OnDidGetRegistration)
66     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerRegistrationError,
67                         OnRegistrationError)
68     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerUnregistrationError,
69                         OnUnregistrationError)
70     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerGetRegistrationError,
71                         OnGetRegistrationError)
72     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_ServiceWorkerStateChanged,
73                         OnServiceWorkerStateChanged)
74     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_SetVersionAttributes,
75                         OnSetVersionAttributes)
76     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_UpdateFound,
77                         OnUpdateFound)
78     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_SetControllerServiceWorker,
79                         OnSetControllerServiceWorker)
80     IPC_MESSAGE_HANDLER(ServiceWorkerMsg_MessageToDocument,
81                         OnPostMessage)
82     IPC_MESSAGE_UNHANDLED(handled = false)
83   IPC_END_MESSAGE_MAP()
84   DCHECK(handled) << "Unhandled message:" << msg.type();
85 }
86 
Send(IPC::Message * msg)87 bool ServiceWorkerDispatcher::Send(IPC::Message* msg) {
88   return thread_safe_sender_->Send(msg);
89 }
90 
RegisterServiceWorker(int provider_id,const GURL & pattern,const GURL & script_url,WebServiceWorkerRegistrationCallbacks * callbacks)91 void ServiceWorkerDispatcher::RegisterServiceWorker(
92     int provider_id,
93     const GURL& pattern,
94     const GURL& script_url,
95     WebServiceWorkerRegistrationCallbacks* callbacks) {
96   DCHECK(callbacks);
97 
98   if (pattern.possibly_invalid_spec().size() > GetMaxURLChars() ||
99       script_url.possibly_invalid_spec().size() > GetMaxURLChars()) {
100     scoped_ptr<WebServiceWorkerRegistrationCallbacks>
101         owned_callbacks(callbacks);
102     scoped_ptr<WebServiceWorkerError> error(new WebServiceWorkerError(
103         WebServiceWorkerError::ErrorTypeSecurity, "URL too long"));
104     callbacks->onError(error.release());
105     return;
106   }
107 
108   int request_id = pending_registration_callbacks_.Add(callbacks);
109   TRACE_EVENT_ASYNC_BEGIN2("ServiceWorker",
110                            "ServiceWorkerDispatcher::RegisterServiceWorker",
111                            request_id,
112                            "Scope", pattern.spec(),
113                            "Script URL", script_url.spec());
114   thread_safe_sender_->Send(new ServiceWorkerHostMsg_RegisterServiceWorker(
115       CurrentWorkerId(), request_id, provider_id, pattern, script_url));
116 }
117 
UnregisterServiceWorker(int provider_id,const GURL & pattern,WebServiceWorkerUnregistrationCallbacks * callbacks)118 void ServiceWorkerDispatcher::UnregisterServiceWorker(
119     int provider_id,
120     const GURL& pattern,
121     WebServiceWorkerUnregistrationCallbacks* callbacks) {
122   DCHECK(callbacks);
123 
124   if (pattern.possibly_invalid_spec().size() > GetMaxURLChars()) {
125     scoped_ptr<WebServiceWorkerUnregistrationCallbacks>
126         owned_callbacks(callbacks);
127     scoped_ptr<WebServiceWorkerError> error(new WebServiceWorkerError(
128         WebServiceWorkerError::ErrorTypeSecurity, "URL too long"));
129     callbacks->onError(error.release());
130     return;
131   }
132 
133   int request_id = pending_unregistration_callbacks_.Add(callbacks);
134   TRACE_EVENT_ASYNC_BEGIN1("ServiceWorker",
135                            "ServiceWorkerDispatcher::UnregisterServiceWorker",
136                            request_id,
137                            "Scope", pattern.spec());
138   thread_safe_sender_->Send(new ServiceWorkerHostMsg_UnregisterServiceWorker(
139       CurrentWorkerId(), request_id, provider_id, pattern));
140 }
141 
GetRegistration(int provider_id,const GURL & document_url,WebServiceWorkerRegistrationCallbacks * callbacks)142 void ServiceWorkerDispatcher::GetRegistration(
143     int provider_id,
144     const GURL& document_url,
145     WebServiceWorkerRegistrationCallbacks* callbacks) {
146   DCHECK(callbacks);
147 
148   if (document_url.possibly_invalid_spec().size() > GetMaxURLChars()) {
149     scoped_ptr<WebServiceWorkerRegistrationCallbacks>
150         owned_callbacks(callbacks);
151     scoped_ptr<WebServiceWorkerError> error(new WebServiceWorkerError(
152         WebServiceWorkerError::ErrorTypeSecurity, "URL too long"));
153     callbacks->onError(error.release());
154     return;
155   }
156 
157   int request_id = pending_get_registration_callbacks_.Add(callbacks);
158   TRACE_EVENT_ASYNC_BEGIN1("ServiceWorker",
159                            "ServiceWorkerDispatcher::GetRegistration",
160                            request_id,
161                            "Document URL", document_url.spec());
162   thread_safe_sender_->Send(new ServiceWorkerHostMsg_GetRegistration(
163       CurrentWorkerId(), request_id, provider_id, document_url));
164 }
165 
AddProviderContext(ServiceWorkerProviderContext * provider_context)166 void ServiceWorkerDispatcher::AddProviderContext(
167     ServiceWorkerProviderContext* provider_context) {
168   DCHECK(provider_context);
169   int provider_id = provider_context->provider_id();
170   DCHECK(!ContainsKey(provider_contexts_, provider_id));
171   provider_contexts_[provider_id] = provider_context;
172 }
173 
RemoveProviderContext(ServiceWorkerProviderContext * provider_context)174 void ServiceWorkerDispatcher::RemoveProviderContext(
175     ServiceWorkerProviderContext* provider_context) {
176   DCHECK(provider_context);
177   DCHECK(ContainsKey(provider_contexts_, provider_context->provider_id()));
178   provider_contexts_.erase(provider_context->provider_id());
179   worker_to_provider_.erase(provider_context->installing_handle_id());
180   worker_to_provider_.erase(provider_context->waiting_handle_id());
181   worker_to_provider_.erase(provider_context->active_handle_id());
182   worker_to_provider_.erase(provider_context->controller_handle_id());
183 }
184 
AddScriptClient(int provider_id,blink::WebServiceWorkerProviderClient * client)185 void ServiceWorkerDispatcher::AddScriptClient(
186     int provider_id,
187     blink::WebServiceWorkerProviderClient* client) {
188   DCHECK(client);
189   DCHECK(!ContainsKey(script_clients_, provider_id));
190   script_clients_[provider_id] = client;
191 }
192 
RemoveScriptClient(int provider_id)193 void ServiceWorkerDispatcher::RemoveScriptClient(int provider_id) {
194   // This could be possibly called multiple times to ensure termination.
195   if (ContainsKey(script_clients_, provider_id))
196     script_clients_.erase(provider_id);
197 }
198 
199 ServiceWorkerDispatcher*
GetOrCreateThreadSpecificInstance(ThreadSafeSender * thread_safe_sender)200 ServiceWorkerDispatcher::GetOrCreateThreadSpecificInstance(
201     ThreadSafeSender* thread_safe_sender) {
202   if (g_dispatcher_tls.Pointer()->Get() == kHasBeenDeleted) {
203     NOTREACHED() << "Re-instantiating TLS ServiceWorkerDispatcher.";
204     g_dispatcher_tls.Pointer()->Set(NULL);
205   }
206   if (g_dispatcher_tls.Pointer()->Get())
207     return g_dispatcher_tls.Pointer()->Get();
208 
209   ServiceWorkerDispatcher* dispatcher =
210       new ServiceWorkerDispatcher(thread_safe_sender);
211   if (WorkerTaskRunner::Instance()->CurrentWorkerId())
212     WorkerTaskRunner::Instance()->AddStopObserver(dispatcher);
213   return dispatcher;
214 }
215 
GetThreadSpecificInstance()216 ServiceWorkerDispatcher* ServiceWorkerDispatcher::GetThreadSpecificInstance() {
217   if (g_dispatcher_tls.Pointer()->Get() == kHasBeenDeleted)
218     return NULL;
219   return g_dispatcher_tls.Pointer()->Get();
220 }
221 
OnWorkerRunLoopStopped()222 void ServiceWorkerDispatcher::OnWorkerRunLoopStopped() {
223   delete this;
224 }
225 
GetServiceWorker(const ServiceWorkerObjectInfo & info,bool adopt_handle)226 WebServiceWorkerImpl* ServiceWorkerDispatcher::GetServiceWorker(
227     const ServiceWorkerObjectInfo& info,
228     bool adopt_handle) {
229   if (info.handle_id == kInvalidServiceWorkerHandleId)
230     return NULL;
231 
232   WorkerObjectMap::iterator existing_worker =
233       service_workers_.find(info.handle_id);
234 
235   if (existing_worker != service_workers_.end()) {
236     if (adopt_handle) {
237       // We are instructed to adopt a handle but we already have one, so
238       // adopt and destroy a handle ref.
239       ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get());
240     }
241     return existing_worker->second;
242   }
243 
244   scoped_ptr<ServiceWorkerHandleReference> handle_ref =
245       adopt_handle
246           ? ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get())
247           : ServiceWorkerHandleReference::Create(info,
248                                                  thread_safe_sender_.get());
249   // WebServiceWorkerImpl constructor calls AddServiceWorker.
250   return new WebServiceWorkerImpl(handle_ref.Pass(), thread_safe_sender_.get());
251 }
252 
253 WebServiceWorkerRegistrationImpl*
FindServiceWorkerRegistration(const ServiceWorkerRegistrationObjectInfo & info,bool adopt_handle)254 ServiceWorkerDispatcher::FindServiceWorkerRegistration(
255     const ServiceWorkerRegistrationObjectInfo& info,
256     bool adopt_handle) {
257   RegistrationObjectMap::iterator registration =
258       registrations_.find(info.handle_id);
259   if (registration == registrations_.end())
260     return NULL;
261   if (adopt_handle) {
262     // We are instructed to adopt a handle but we already have one, so
263     // adopt and destroy a handle ref.
264     ServiceWorkerRegistrationHandleReference::Adopt(
265         info, thread_safe_sender_.get());
266   }
267   return registration->second;
268 }
269 
270 WebServiceWorkerRegistrationImpl*
CreateServiceWorkerRegistration(const ServiceWorkerRegistrationObjectInfo & info,bool adopt_handle)271 ServiceWorkerDispatcher::CreateServiceWorkerRegistration(
272     const ServiceWorkerRegistrationObjectInfo& info,
273     bool adopt_handle) {
274   DCHECK(!FindServiceWorkerRegistration(info, adopt_handle));
275   if (info.handle_id == kInvalidServiceWorkerRegistrationHandleId)
276     return NULL;
277 
278   scoped_ptr<ServiceWorkerRegistrationHandleReference> handle_ref =
279       adopt_handle ? ServiceWorkerRegistrationHandleReference::Adopt(
280                          info, thread_safe_sender_.get())
281                    : ServiceWorkerRegistrationHandleReference::Create(
282                          info, thread_safe_sender_.get());
283 
284   // WebServiceWorkerRegistrationImpl constructor calls
285   // AddServiceWorkerRegistration.
286   return new WebServiceWorkerRegistrationImpl(handle_ref.Pass());
287 }
288 
OnAssociateRegistration(int thread_id,int provider_id,const ServiceWorkerRegistrationObjectInfo & info,const ServiceWorkerVersionAttributes & attrs)289 void ServiceWorkerDispatcher::OnAssociateRegistration(
290     int thread_id,
291     int provider_id,
292     const ServiceWorkerRegistrationObjectInfo& info,
293     const ServiceWorkerVersionAttributes& attrs) {
294   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
295   if (provider == provider_contexts_.end())
296     return;
297   provider->second->OnAssociateRegistration(info, attrs);
298   if (attrs.installing.handle_id != kInvalidServiceWorkerHandleId)
299     worker_to_provider_[attrs.installing.handle_id] = provider->second;
300   if (attrs.waiting.handle_id != kInvalidServiceWorkerHandleId)
301     worker_to_provider_[attrs.waiting.handle_id] = provider->second;
302   if (attrs.active.handle_id != kInvalidServiceWorkerHandleId)
303     worker_to_provider_[attrs.active.handle_id] = provider->second;
304 }
305 
OnDisassociateRegistration(int thread_id,int provider_id)306 void ServiceWorkerDispatcher::OnDisassociateRegistration(
307     int thread_id,
308     int provider_id) {
309   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
310   if (provider == provider_contexts_.end())
311     return;
312   provider->second->OnDisassociateRegistration();
313   worker_to_provider_.erase(provider->second->installing_handle_id());
314   worker_to_provider_.erase(provider->second->waiting_handle_id());
315   worker_to_provider_.erase(provider->second->active_handle_id());
316   worker_to_provider_.erase(provider->second->controller_handle_id());
317 }
318 
OnRegistered(int thread_id,int request_id,const ServiceWorkerRegistrationObjectInfo & info,const ServiceWorkerVersionAttributes & attrs)319 void ServiceWorkerDispatcher::OnRegistered(
320     int thread_id,
321     int request_id,
322     const ServiceWorkerRegistrationObjectInfo& info,
323     const ServiceWorkerVersionAttributes& attrs) {
324   TRACE_EVENT_ASYNC_STEP_INTO0("ServiceWorker",
325                                "ServiceWorkerDispatcher::RegisterServiceWorker",
326                                request_id,
327                                "OnRegistered");
328   TRACE_EVENT_ASYNC_END0("ServiceWorker",
329                          "ServiceWorkerDispatcher::RegisterServiceWorker",
330                          request_id);
331   WebServiceWorkerRegistrationCallbacks* callbacks =
332       pending_registration_callbacks_.Lookup(request_id);
333   DCHECK(callbacks);
334   if (!callbacks)
335     return;
336 
337   callbacks->onSuccess(FindOrCreateRegistration(info, attrs));
338   pending_registration_callbacks_.Remove(request_id);
339 }
340 
OnUnregistered(int thread_id,int request_id,bool is_success)341 void ServiceWorkerDispatcher::OnUnregistered(int thread_id,
342                                              int request_id,
343                                              bool is_success) {
344   TRACE_EVENT_ASYNC_STEP_INTO0(
345       "ServiceWorker",
346       "ServiceWorkerDispatcher::UnregisterServiceWorker",
347       request_id,
348       "OnUnregistered");
349   TRACE_EVENT_ASYNC_END0("ServiceWorker",
350                          "ServiceWorkerDispatcher::UnregisterServiceWorker",
351                          request_id);
352   WebServiceWorkerUnregistrationCallbacks* callbacks =
353       pending_unregistration_callbacks_.Lookup(request_id);
354   DCHECK(callbacks);
355   if (!callbacks)
356     return;
357   callbacks->onSuccess(&is_success);
358   pending_unregistration_callbacks_.Remove(request_id);
359 }
360 
OnDidGetRegistration(int thread_id,int request_id,const ServiceWorkerRegistrationObjectInfo & info,const ServiceWorkerVersionAttributes & attrs)361 void ServiceWorkerDispatcher::OnDidGetRegistration(
362     int thread_id,
363     int request_id,
364     const ServiceWorkerRegistrationObjectInfo& info,
365     const ServiceWorkerVersionAttributes& attrs) {
366   TRACE_EVENT_ASYNC_STEP_INTO0(
367       "ServiceWorker",
368       "ServiceWorkerDispatcher::GetRegistration",
369       request_id,
370       "OnDidGetRegistration");
371   TRACE_EVENT_ASYNC_END0("ServiceWorker",
372                          "ServiceWorkerDispatcher::GetRegistration",
373                          request_id);
374   WebServiceWorkerRegistrationCallbacks* callbacks =
375       pending_get_registration_callbacks_.Lookup(request_id);
376   DCHECK(callbacks);
377   if (!callbacks)
378     return;
379 
380   WebServiceWorkerRegistrationImpl* registration = NULL;
381   if (info.handle_id != kInvalidServiceWorkerHandleId)
382     registration = FindOrCreateRegistration(info, attrs);
383 
384   callbacks->onSuccess(registration);
385   pending_get_registration_callbacks_.Remove(request_id);
386 }
387 
OnRegistrationError(int thread_id,int request_id,WebServiceWorkerError::ErrorType error_type,const base::string16 & message)388 void ServiceWorkerDispatcher::OnRegistrationError(
389     int thread_id,
390     int request_id,
391     WebServiceWorkerError::ErrorType error_type,
392     const base::string16& message) {
393   TRACE_EVENT_ASYNC_STEP_INTO0("ServiceWorker",
394                                "ServiceWorkerDispatcher::RegisterServiceWorker",
395                                request_id,
396                                "OnRegistrationError");
397   TRACE_EVENT_ASYNC_END0("ServiceWorker",
398                          "ServiceWorkerDispatcher::RegisterServiceWorker",
399                          request_id);
400   WebServiceWorkerRegistrationCallbacks* callbacks =
401       pending_registration_callbacks_.Lookup(request_id);
402   DCHECK(callbacks);
403   if (!callbacks)
404     return;
405 
406   scoped_ptr<WebServiceWorkerError> error(
407       new WebServiceWorkerError(error_type, message));
408   callbacks->onError(error.release());
409   pending_registration_callbacks_.Remove(request_id);
410 }
411 
OnUnregistrationError(int thread_id,int request_id,WebServiceWorkerError::ErrorType error_type,const base::string16 & message)412 void ServiceWorkerDispatcher::OnUnregistrationError(
413     int thread_id,
414     int request_id,
415     WebServiceWorkerError::ErrorType error_type,
416     const base::string16& message) {
417   TRACE_EVENT_ASYNC_STEP_INTO0(
418       "ServiceWorker",
419       "ServiceWorkerDispatcher::UnregisterServiceWorker",
420       request_id,
421       "OnUnregistrationError");
422   TRACE_EVENT_ASYNC_END0("ServiceWorker",
423                          "ServiceWorkerDispatcher::UnregisterServiceWorker",
424                          request_id);
425   WebServiceWorkerUnregistrationCallbacks* callbacks =
426       pending_unregistration_callbacks_.Lookup(request_id);
427   DCHECK(callbacks);
428   if (!callbacks)
429     return;
430 
431   scoped_ptr<WebServiceWorkerError> error(
432       new WebServiceWorkerError(error_type, message));
433   callbacks->onError(error.release());
434   pending_unregistration_callbacks_.Remove(request_id);
435 }
436 
OnGetRegistrationError(int thread_id,int request_id,WebServiceWorkerError::ErrorType error_type,const base::string16 & message)437 void ServiceWorkerDispatcher::OnGetRegistrationError(
438     int thread_id,
439     int request_id,
440     WebServiceWorkerError::ErrorType error_type,
441     const base::string16& message) {
442   TRACE_EVENT_ASYNC_STEP_INTO0(
443       "ServiceWorker",
444       "ServiceWorkerDispatcher::GetRegistration",
445       request_id,
446       "OnGetRegistrationError");
447   TRACE_EVENT_ASYNC_END0("ServiceWorker",
448                          "ServiceWorkerDispatcher::GetRegistration",
449                          request_id);
450   WebServiceWorkerGetRegistrationCallbacks* callbacks =
451       pending_get_registration_callbacks_.Lookup(request_id);
452   DCHECK(callbacks);
453   if (!callbacks)
454     return;
455 
456   scoped_ptr<WebServiceWorkerError> error(
457       new WebServiceWorkerError(error_type, message));
458   callbacks->onError(error.release());
459   pending_get_registration_callbacks_.Remove(request_id);
460 }
461 
OnServiceWorkerStateChanged(int thread_id,int handle_id,blink::WebServiceWorkerState state)462 void ServiceWorkerDispatcher::OnServiceWorkerStateChanged(
463     int thread_id,
464     int handle_id,
465     blink::WebServiceWorkerState state) {
466   TRACE_EVENT2("ServiceWorker",
467                "ServiceWorkerDispatcher::OnServiceWorkerStateChanged",
468                "Thread ID", thread_id,
469                "State", state);
470   WorkerObjectMap::iterator worker = service_workers_.find(handle_id);
471   if (worker != service_workers_.end())
472     worker->second->OnStateChanged(state);
473 
474   WorkerToProviderMap::iterator provider = worker_to_provider_.find(handle_id);
475   if (provider != worker_to_provider_.end())
476     provider->second->OnServiceWorkerStateChanged(handle_id, state);
477 }
478 
OnSetVersionAttributes(int thread_id,int provider_id,int registration_handle_id,int changed_mask,const ServiceWorkerVersionAttributes & attributes)479 void ServiceWorkerDispatcher::OnSetVersionAttributes(
480     int thread_id,
481     int provider_id,
482     int registration_handle_id,
483     int changed_mask,
484     const ServiceWorkerVersionAttributes& attributes) {
485   TRACE_EVENT1("ServiceWorker",
486                "ServiceWorkerDispatcher::OnSetVersionAttributes",
487                "Thread ID", thread_id);
488   ChangedVersionAttributesMask mask(changed_mask);
489   if (mask.installing_changed()) {
490     SetInstallingServiceWorker(provider_id,
491                                registration_handle_id,
492                                attributes.installing);
493   }
494   if (mask.waiting_changed()) {
495     SetWaitingServiceWorker(provider_id,
496                             registration_handle_id,
497                             attributes.waiting);
498   }
499   if (mask.active_changed()) {
500     SetActiveServiceWorker(provider_id,
501                            registration_handle_id,
502                            attributes.active);
503     SetReadyRegistration(provider_id, registration_handle_id);
504   }
505 }
506 
OnUpdateFound(int thread_id,const ServiceWorkerRegistrationObjectInfo & info)507 void ServiceWorkerDispatcher::OnUpdateFound(
508     int thread_id,
509     const ServiceWorkerRegistrationObjectInfo& info) {
510   TRACE_EVENT0("ServiceWorker",
511                "ServiceWorkerDispatcher::OnUpdateFound");
512   RegistrationObjectMap::iterator found = registrations_.find(info.handle_id);
513   if (found != registrations_.end())
514     found->second->OnUpdateFound();
515 }
516 
SetInstallingServiceWorker(int provider_id,int registration_handle_id,const ServiceWorkerObjectInfo & info)517 void ServiceWorkerDispatcher::SetInstallingServiceWorker(
518     int provider_id,
519     int registration_handle_id,
520     const ServiceWorkerObjectInfo& info) {
521   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
522   if (provider != provider_contexts_.end() &&
523       provider->second->registration_handle_id() == registration_handle_id) {
524     int existing_installing_id = provider->second->installing_handle_id();
525     if (existing_installing_id != info.handle_id &&
526         existing_installing_id != kInvalidServiceWorkerHandleId) {
527       WorkerToProviderMap::iterator associated_provider =
528           worker_to_provider_.find(existing_installing_id);
529       DCHECK(associated_provider != worker_to_provider_.end());
530       DCHECK(associated_provider->second->provider_id() == provider_id);
531       worker_to_provider_.erase(associated_provider);
532     }
533     provider->second->OnSetInstallingServiceWorker(
534         registration_handle_id, info);
535     if (info.handle_id != kInvalidServiceWorkerHandleId)
536       worker_to_provider_[info.handle_id] = provider->second;
537   }
538 
539   RegistrationObjectMap::iterator found =
540       registrations_.find(registration_handle_id);
541   if (found != registrations_.end()) {
542     // Populate the .installing field with the new worker object.
543     found->second->SetInstalling(GetServiceWorker(info, false));
544   }
545 }
546 
SetWaitingServiceWorker(int provider_id,int registration_handle_id,const ServiceWorkerObjectInfo & info)547 void ServiceWorkerDispatcher::SetWaitingServiceWorker(
548     int provider_id,
549     int registration_handle_id,
550     const ServiceWorkerObjectInfo& info) {
551   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
552   if (provider != provider_contexts_.end() &&
553       provider->second->registration_handle_id() == registration_handle_id) {
554     int existing_waiting_id = provider->second->waiting_handle_id();
555     if (existing_waiting_id != info.handle_id &&
556         existing_waiting_id != kInvalidServiceWorkerHandleId) {
557       WorkerToProviderMap::iterator associated_provider =
558           worker_to_provider_.find(existing_waiting_id);
559       DCHECK(associated_provider != worker_to_provider_.end());
560       DCHECK(associated_provider->second->provider_id() == provider_id);
561       worker_to_provider_.erase(associated_provider);
562     }
563     provider->second->OnSetWaitingServiceWorker(registration_handle_id, info);
564     if (info.handle_id != kInvalidServiceWorkerHandleId)
565       worker_to_provider_[info.handle_id] = provider->second;
566   }
567 
568   RegistrationObjectMap::iterator found =
569       registrations_.find(registration_handle_id);
570   if (found != registrations_.end()) {
571     // Populate the .waiting field with the new worker object.
572     found->second->SetWaiting(GetServiceWorker(info, false));
573   }
574 }
575 
SetActiveServiceWorker(int provider_id,int registration_handle_id,const ServiceWorkerObjectInfo & info)576 void ServiceWorkerDispatcher::SetActiveServiceWorker(
577     int provider_id,
578     int registration_handle_id,
579     const ServiceWorkerObjectInfo& info) {
580   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
581   if (provider != provider_contexts_.end() &&
582       provider->second->registration_handle_id() == registration_handle_id) {
583     int existing_active_id = provider->second->active_handle_id();
584     if (existing_active_id != info.handle_id &&
585         existing_active_id != kInvalidServiceWorkerHandleId) {
586       WorkerToProviderMap::iterator associated_provider =
587           worker_to_provider_.find(existing_active_id);
588       DCHECK(associated_provider != worker_to_provider_.end());
589       DCHECK(associated_provider->second->provider_id() == provider_id);
590       worker_to_provider_.erase(associated_provider);
591     }
592     provider->second->OnSetActiveServiceWorker(registration_handle_id, info);
593     if (info.handle_id != kInvalidServiceWorkerHandleId)
594       worker_to_provider_[info.handle_id] = provider->second;
595   }
596 
597   RegistrationObjectMap::iterator found =
598       registrations_.find(registration_handle_id);
599   if (found != registrations_.end()) {
600     // Populate the .active field with the new worker object.
601     found->second->SetActive(GetServiceWorker(info, false));
602   }
603 }
604 
SetReadyRegistration(int provider_id,int registration_handle_id)605 void ServiceWorkerDispatcher::SetReadyRegistration(
606     int provider_id,
607     int registration_handle_id) {
608   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
609   if (provider == provider_contexts_.end() ||
610       provider->second->registration_handle_id() != registration_handle_id ||
611       provider->second->active_handle_id() == kInvalidServiceWorkerHandleId) {
612     return;
613   }
614 
615   ScriptClientMap::iterator client = script_clients_.find(provider_id);
616   if (client == script_clients_.end())
617     return;
618 
619   ServiceWorkerRegistrationObjectInfo info =
620       provider->second->registration()->info();
621   WebServiceWorkerRegistrationImpl* registration =
622       FindServiceWorkerRegistration(info, false);
623   if (!registration) {
624     registration = CreateServiceWorkerRegistration(info, false);
625     ServiceWorkerVersionAttributes attrs =
626         provider->second->GetVersionAttributes();
627     registration->SetInstalling(GetServiceWorker(attrs.installing, false));
628     registration->SetWaiting(GetServiceWorker(attrs.waiting, false));
629     registration->SetActive(GetServiceWorker(attrs.active, false));
630   }
631 
632   // Resolve the .ready promise with the registration object.
633   client->second->setReadyRegistration(registration);
634 }
635 
OnSetControllerServiceWorker(int thread_id,int provider_id,const ServiceWorkerObjectInfo & info)636 void ServiceWorkerDispatcher::OnSetControllerServiceWorker(
637     int thread_id,
638     int provider_id,
639     const ServiceWorkerObjectInfo& info) {
640   TRACE_EVENT2("ServiceWorker",
641                "ServiceWorkerDispatcher::OnSetControllerServiceWorker",
642                "Thread ID", thread_id,
643                "Provider ID", provider_id);
644   ProviderContextMap::iterator provider = provider_contexts_.find(provider_id);
645   if (provider != provider_contexts_.end()) {
646     provider->second->OnSetControllerServiceWorker(
647         provider->second->registration_handle_id(), info);
648     worker_to_provider_[info.handle_id] = provider->second;
649   }
650 
651   ScriptClientMap::iterator found = script_clients_.find(provider_id);
652   if (found != script_clients_.end()) {
653     // Populate the .controller field with the new worker object.
654     found->second->setController(GetServiceWorker(info, false));
655   }
656 }
657 
OnPostMessage(int thread_id,int provider_id,const base::string16 & message,const std::vector<int> & sent_message_port_ids,const std::vector<int> & new_routing_ids)658 void ServiceWorkerDispatcher::OnPostMessage(
659     int thread_id,
660     int provider_id,
661     const base::string16& message,
662     const std::vector<int>& sent_message_port_ids,
663     const std::vector<int>& new_routing_ids) {
664   // Make sure we're on the main document thread. (That must be the only
665   // thread we get this message)
666   DCHECK(ChildThread::current());
667   TRACE_EVENT1("ServiceWorker",
668                "ServiceWorkerDispatcher::OnPostMessage",
669                "Thread ID", thread_id);
670 
671   ScriptClientMap::iterator found = script_clients_.find(provider_id);
672   if (found == script_clients_.end()) {
673     // For now we do no queueing for messages sent to nonexistent / unattached
674     // client.
675     return;
676   }
677 
678   std::vector<WebMessagePortChannelImpl*> ports;
679   if (!sent_message_port_ids.empty()) {
680     ports.resize(sent_message_port_ids.size());
681     for (size_t i = 0; i < sent_message_port_ids.size(); ++i) {
682       ports[i] = new WebMessagePortChannelImpl(
683           new_routing_ids[i], sent_message_port_ids[i],
684           base::MessageLoopProxy::current());
685     }
686   }
687 
688   found->second->dispatchMessageEvent(message, ports);
689 }
690 
AddServiceWorker(int handle_id,WebServiceWorkerImpl * worker)691 void ServiceWorkerDispatcher::AddServiceWorker(
692     int handle_id, WebServiceWorkerImpl* worker) {
693   DCHECK(!ContainsKey(service_workers_, handle_id));
694   service_workers_[handle_id] = worker;
695 }
696 
RemoveServiceWorker(int handle_id)697 void ServiceWorkerDispatcher::RemoveServiceWorker(int handle_id) {
698   DCHECK(ContainsKey(service_workers_, handle_id));
699   service_workers_.erase(handle_id);
700 }
701 
AddServiceWorkerRegistration(int registration_handle_id,WebServiceWorkerRegistrationImpl * registration)702 void ServiceWorkerDispatcher::AddServiceWorkerRegistration(
703     int registration_handle_id,
704     WebServiceWorkerRegistrationImpl* registration) {
705   DCHECK(!ContainsKey(registrations_, registration_handle_id));
706   registrations_[registration_handle_id] = registration;
707 }
708 
RemoveServiceWorkerRegistration(int registration_handle_id)709 void ServiceWorkerDispatcher::RemoveServiceWorkerRegistration(
710     int registration_handle_id) {
711   DCHECK(ContainsKey(registrations_, registration_handle_id));
712   registrations_.erase(registration_handle_id);
713 }
714 
715 WebServiceWorkerRegistrationImpl*
FindOrCreateRegistration(const ServiceWorkerRegistrationObjectInfo & info,const ServiceWorkerVersionAttributes & attrs)716 ServiceWorkerDispatcher::FindOrCreateRegistration(
717     const ServiceWorkerRegistrationObjectInfo& info,
718     const ServiceWorkerVersionAttributes& attrs) {
719   WebServiceWorkerRegistrationImpl* registration =
720       FindServiceWorkerRegistration(info, true);
721   if (!registration) {
722     registration = CreateServiceWorkerRegistration(info, true);
723     registration->SetInstalling(GetServiceWorker(attrs.installing, true));
724     registration->SetWaiting(GetServiceWorker(attrs.waiting, true));
725     registration->SetActive(GetServiceWorker(attrs.active, true));
726   } else {
727     // |registration| must already have version attributes, so adopt and destroy
728     // handle refs for them.
729     ServiceWorkerHandleReference::Adopt(
730         attrs.installing, thread_safe_sender_.get());
731     ServiceWorkerHandleReference::Adopt(
732         attrs.waiting, thread_safe_sender_.get());
733     ServiceWorkerHandleReference::Adopt(
734         attrs.active, thread_safe_sender_.get());
735   }
736   return registration;
737 }
738 
739 }  // namespace content
740