OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "content/browser/service_worker/embedded_worker_instance.h" | 5 #include "content/browser/service_worker/embedded_worker_instance.h" |
6 | 6 |
7 #include <utility> | 7 #include <utility> |
8 | 8 |
9 #include "base/bind_helpers.h" | 9 #include "base/bind_helpers.h" |
10 #include "base/macros.h" | 10 #include "base/macros.h" |
11 #include "base/memory/ptr_util.h" | 11 #include "base/memory/ptr_util.h" |
12 #include "base/metrics/histogram_macros.h" | 12 #include "base/metrics/histogram_macros.h" |
13 #include "base/threading/non_thread_safe.h" | 13 #include "base/threading/non_thread_safe.h" |
14 #include "base/trace_event/trace_event.h" | 14 #include "base/trace_event/trace_event.h" |
15 #include "content/browser/devtools/service_worker_devtools_manager.h" | 15 #include "content/browser/devtools/service_worker_devtools_manager.h" |
16 #include "content/browser/service_worker/embedded_worker_registry.h" | 16 #include "content/browser/service_worker/embedded_worker_registry.h" |
17 #include "content/browser/service_worker/embedded_worker_status.h" | 17 #include "content/browser/service_worker/embedded_worker_status.h" |
18 #include "content/browser/service_worker/service_worker_context_core.h" | 18 #include "content/browser/service_worker/service_worker_context_core.h" |
19 #include "content/common/content_switches_internal.h" | 19 #include "content/common/content_switches_internal.h" |
20 #include "content/common/mojo/service_registry_impl.h" | |
21 #include "content/common/service_worker/embedded_worker_messages.h" | 20 #include "content/common/service_worker/embedded_worker_messages.h" |
22 #include "content/common/service_worker/embedded_worker_settings.h" | 21 #include "content/common/service_worker/embedded_worker_settings.h" |
23 #include "content/common/service_worker/embedded_worker_setup.mojom.h" | 22 #include "content/common/service_worker/embedded_worker_setup.mojom.h" |
24 #include "content/common/service_worker/service_worker_types.h" | 23 #include "content/common/service_worker/service_worker_types.h" |
25 #include "content/public/browser/browser_thread.h" | 24 #include "content/public/browser/browser_thread.h" |
26 #include "content/public/browser/content_browser_client.h" | 25 #include "content/public/browser/content_browser_client.h" |
27 #include "content/public/browser/render_process_host.h" | 26 #include "content/public/browser/render_process_host.h" |
28 #include "content/public/common/child_process_host.h" | 27 #include "content/public/common/child_process_host.h" |
29 #include "ipc/ipc_message.h" | 28 #include "ipc/ipc_message.h" |
30 #include "services/shell/public/cpp/interface_provider.h" | 29 #include "services/shell/public/cpp/interface_provider.h" |
| 30 #include "services/shell/public/cpp/interface_registry.h" |
31 #include "url/gurl.h" | 31 #include "url/gurl.h" |
32 | 32 |
33 namespace content { | 33 namespace content { |
34 | 34 |
35 namespace { | 35 namespace { |
36 | 36 |
37 // When a service worker version's failure count exceeds | 37 // When a service worker version's failure count exceeds |
38 // |kMaxSameProcessFailureCount|, the embedded worker is forced to start in a | 38 // |kMaxSameProcessFailureCount|, the embedded worker is forced to start in a |
39 // new process. | 39 // new process. |
40 const int kMaxSameProcessFailureCount = 2; | 40 const int kMaxSameProcessFailureCount = 2; |
(...skipping 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
97 } | 97 } |
98 BrowserThread::PostTask( | 98 BrowserThread::PostTask( |
99 BrowserThread::IO, | 99 BrowserThread::IO, |
100 FROM_HERE, | 100 FROM_HERE, |
101 base::Bind(callback, worker_devtools_agent_route_id, wait_for_debugger)); | 101 base::Bind(callback, worker_devtools_agent_route_id, wait_for_debugger)); |
102 } | 102 } |
103 | 103 |
104 void SetupMojoOnUIThread( | 104 void SetupMojoOnUIThread( |
105 int process_id, | 105 int process_id, |
106 int thread_id, | 106 int thread_id, |
107 shell::mojom::InterfaceProviderRequest services, | 107 shell::mojom::InterfaceProviderRequest remote_interfaces, |
108 shell::mojom::InterfaceProviderPtrInfo exposed_services) { | 108 shell::mojom::InterfaceProviderPtrInfo exposed_interfaces) { |
109 RenderProcessHost* rph = RenderProcessHost::FromID(process_id); | 109 RenderProcessHost* rph = RenderProcessHost::FromID(process_id); |
110 // |rph| or its InterfaceProvider may be NULL in unit tests. | 110 // |rph| or its InterfaceProvider may be NULL in unit tests. |
111 if (!rph || !rph->GetRemoteInterfaces()) | 111 if (!rph || !rph->GetRemoteInterfaces()) |
112 return; | 112 return; |
113 mojom::EmbeddedWorkerSetupPtr setup; | 113 mojom::EmbeddedWorkerSetupPtr setup; |
114 rph->GetRemoteInterfaces()->GetInterface(&setup); | 114 rph->GetRemoteInterfaces()->GetInterface(&setup); |
115 setup->ExchangeInterfaceProviders( | 115 setup->ExchangeInterfaceProviders( |
116 thread_id, std::move(services), | 116 thread_id, std::move(remote_interfaces), |
117 mojo::MakeProxy(std::move(exposed_services))); | 117 mojo::MakeProxy(std::move(exposed_interfaces))); |
118 } | 118 } |
119 | 119 |
120 } // namespace | 120 } // namespace |
121 | 121 |
122 // Lives on IO thread, proxies notifications to DevToolsManager that lives on | 122 // Lives on IO thread, proxies notifications to DevToolsManager that lives on |
123 // UI thread. Owned by EmbeddedWorkerInstance. | 123 // UI thread. Owned by EmbeddedWorkerInstance. |
124 class EmbeddedWorkerInstance::DevToolsProxy : public base::NonThreadSafe { | 124 class EmbeddedWorkerInstance::DevToolsProxy : public base::NonThreadSafe { |
125 public: | 125 public: |
126 DevToolsProxy(int process_id, int agent_route_id) | 126 DevToolsProxy(int process_id, int agent_route_id) |
127 : process_id_(process_id), | 127 : process_id_(process_id), |
(...skipping 304 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
432 return; | 432 return; |
433 } | 433 } |
434 DCHECK(status_ == EmbeddedWorkerStatus::STOPPED); | 434 DCHECK(status_ == EmbeddedWorkerStatus::STOPPED); |
435 | 435 |
436 DCHECK(!params->pause_after_download || !params->is_installed); | 436 DCHECK(!params->pause_after_download || !params->is_installed); |
437 DCHECK_NE(kInvalidServiceWorkerVersionId, params->service_worker_version_id); | 437 DCHECK_NE(kInvalidServiceWorkerVersionId, params->service_worker_version_id); |
438 step_time_ = base::TimeTicks::Now(); | 438 step_time_ = base::TimeTicks::Now(); |
439 status_ = EmbeddedWorkerStatus::STARTING; | 439 status_ = EmbeddedWorkerStatus::STARTING; |
440 starting_phase_ = ALLOCATING_PROCESS; | 440 starting_phase_ = ALLOCATING_PROCESS; |
441 network_accessed_for_script_ = false; | 441 network_accessed_for_script_ = false; |
442 service_registry_.reset(new ServiceRegistryImpl()); | 442 interface_registry_.reset(new shell::InterfaceRegistry(nullptr)); |
| 443 remote_interfaces_.reset(new shell::InterfaceProvider); |
443 FOR_EACH_OBSERVER(Listener, listener_list_, OnStarting()); | 444 FOR_EACH_OBSERVER(Listener, listener_list_, OnStarting()); |
444 | 445 |
445 params->embedded_worker_id = embedded_worker_id_; | 446 params->embedded_worker_id = embedded_worker_id_; |
446 params->worker_devtools_agent_route_id = MSG_ROUTING_NONE; | 447 params->worker_devtools_agent_route_id = MSG_ROUTING_NONE; |
447 params->wait_for_debugger = false; | 448 params->wait_for_debugger = false; |
448 params->settings.v8_cache_options = GetV8CacheOptions(); | 449 params->settings.v8_cache_options = GetV8CacheOptions(); |
449 | 450 |
450 inflight_start_task_.reset(new StartTask(this, params->script_url)); | 451 inflight_start_task_.reset(new StartTask(this, params->script_url)); |
451 inflight_start_task_->Start(std::move(params), callback); | 452 inflight_start_task_->Start(std::move(params), callback); |
452 } | 453 } |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
498 | 499 |
499 void EmbeddedWorkerInstance::ResumeAfterDownload() { | 500 void EmbeddedWorkerInstance::ResumeAfterDownload() { |
500 if (process_id() == ChildProcessHost::kInvalidUniqueID || | 501 if (process_id() == ChildProcessHost::kInvalidUniqueID || |
501 status_ != EmbeddedWorkerStatus::STARTING) { | 502 status_ != EmbeddedWorkerStatus::STARTING) { |
502 return; | 503 return; |
503 } | 504 } |
504 registry_->Send(process_id(), new EmbeddedWorkerMsg_ResumeAfterDownload( | 505 registry_->Send(process_id(), new EmbeddedWorkerMsg_ResumeAfterDownload( |
505 embedded_worker_id_)); | 506 embedded_worker_id_)); |
506 } | 507 } |
507 | 508 |
508 ServiceRegistry* EmbeddedWorkerInstance::GetServiceRegistry() { | 509 shell::InterfaceRegistry* EmbeddedWorkerInstance::GetInterfaceRegistry() { |
509 DCHECK(status_ == EmbeddedWorkerStatus::STARTING || | 510 DCHECK(status_ == EmbeddedWorkerStatus::STARTING || |
510 status_ == EmbeddedWorkerStatus::RUNNING) | 511 status_ == EmbeddedWorkerStatus::RUNNING) |
511 << static_cast<int>(status_); | 512 << static_cast<int>(status_); |
512 return service_registry_.get(); | 513 return interface_registry_.get(); |
| 514 } |
| 515 |
| 516 shell::InterfaceProvider* EmbeddedWorkerInstance::GetRemoteInterfaces() { |
| 517 DCHECK(status_ == EmbeddedWorkerStatus::STARTING || |
| 518 status_ == EmbeddedWorkerStatus::RUNNING) |
| 519 << static_cast<int>(status_); |
| 520 return remote_interfaces_.get(); |
513 } | 521 } |
514 | 522 |
515 EmbeddedWorkerInstance::EmbeddedWorkerInstance( | 523 EmbeddedWorkerInstance::EmbeddedWorkerInstance( |
516 base::WeakPtr<ServiceWorkerContextCore> context, | 524 base::WeakPtr<ServiceWorkerContextCore> context, |
517 int embedded_worker_id) | 525 int embedded_worker_id) |
518 : context_(context), | 526 : context_(context), |
519 registry_(context->embedded_worker_registry()), | 527 registry_(context->embedded_worker_registry()), |
520 embedded_worker_id_(embedded_worker_id), | 528 embedded_worker_id_(embedded_worker_id), |
521 status_(EmbeddedWorkerStatus::STOPPED), | 529 status_(EmbeddedWorkerStatus::STOPPED), |
522 starting_phase_(NOT_STARTING), | 530 starting_phase_(NOT_STARTING), |
(...skipping 116 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
639 starting_phase_ = THREAD_STARTED; | 647 starting_phase_ = THREAD_STARTED; |
640 if (!step_time_.is_null()) { | 648 if (!step_time_.is_null()) { |
641 base::TimeDelta duration = UpdateStepTime(); | 649 base::TimeDelta duration = UpdateStepTime(); |
642 if (inflight_start_task_->is_installed()) | 650 if (inflight_start_task_->is_installed()) |
643 ServiceWorkerMetrics::RecordTimeToStartThread(duration, start_situation_); | 651 ServiceWorkerMetrics::RecordTimeToStartThread(duration, start_situation_); |
644 } | 652 } |
645 | 653 |
646 thread_id_ = thread_id; | 654 thread_id_ = thread_id; |
647 FOR_EACH_OBSERVER(Listener, listener_list_, OnThreadStarted()); | 655 FOR_EACH_OBSERVER(Listener, listener_list_, OnThreadStarted()); |
648 | 656 |
649 shell::mojom::InterfaceProviderPtr exposed_services; | 657 shell::mojom::InterfaceProviderPtr exposed_interfaces; |
650 service_registry_->Bind(GetProxy(&exposed_services)); | 658 interface_registry_->Bind(GetProxy(&exposed_interfaces)); |
651 shell::mojom::InterfaceProviderRequest request = | 659 shell::mojom::InterfaceProviderPtr remote_interfaces; |
652 service_registry_->TakeRemoteRequest(); | 660 shell::mojom::InterfaceProviderRequest request = GetProxy(&remote_interfaces); |
| 661 remote_interfaces_->Bind(std::move(remote_interfaces)); |
653 BrowserThread::PostTask( | 662 BrowserThread::PostTask( |
654 BrowserThread::UI, FROM_HERE, | 663 BrowserThread::UI, FROM_HERE, |
655 base::Bind(SetupMojoOnUIThread, process_id(), thread_id_, | 664 base::Bind(SetupMojoOnUIThread, process_id(), thread_id_, |
656 base::Passed(&request), | 665 base::Passed(&request), |
657 base::Passed(exposed_services.PassInterface()))); | 666 base::Passed(exposed_interfaces.PassInterface()))); |
658 } | 667 } |
659 | 668 |
660 void EmbeddedWorkerInstance::OnScriptLoadFailed() { | 669 void EmbeddedWorkerInstance::OnScriptLoadFailed() { |
661 if (!inflight_start_task_) | 670 if (!inflight_start_task_) |
662 return; | 671 return; |
663 TRACE_EVENT_ASYNC_STEP_PAST0("ServiceWorker", "EmbeddedWorkerInstance::Start", | 672 TRACE_EVENT_ASYNC_STEP_PAST0("ServiceWorker", "EmbeddedWorkerInstance::Start", |
664 inflight_start_task_.get(), | 673 inflight_start_task_.get(), |
665 "OnScriptLoadFailed"); | 674 "OnScriptLoadFailed"); |
666 FOR_EACH_OBSERVER(Listener, listener_list_, OnScriptLoadFailed()); | 675 FOR_EACH_OBSERVER(Listener, listener_list_, OnScriptLoadFailed()); |
667 } | 676 } |
(...skipping 122 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
790 } | 799 } |
791 | 800 |
792 void EmbeddedWorkerInstance::ReleaseProcess() { | 801 void EmbeddedWorkerInstance::ReleaseProcess() { |
793 // Abort an inflight start task. | 802 // Abort an inflight start task. |
794 inflight_start_task_.reset(); | 803 inflight_start_task_.reset(); |
795 | 804 |
796 devtools_proxy_.reset(); | 805 devtools_proxy_.reset(); |
797 process_handle_.reset(); | 806 process_handle_.reset(); |
798 status_ = EmbeddedWorkerStatus::STOPPED; | 807 status_ = EmbeddedWorkerStatus::STOPPED; |
799 thread_id_ = kInvalidEmbeddedWorkerThreadId; | 808 thread_id_ = kInvalidEmbeddedWorkerThreadId; |
800 service_registry_.reset(); | 809 interface_registry_.reset(); |
| 810 remote_interfaces_.reset(); |
801 } | 811 } |
802 | 812 |
803 void EmbeddedWorkerInstance::OnStartFailed(const StatusCallback& callback, | 813 void EmbeddedWorkerInstance::OnStartFailed(const StatusCallback& callback, |
804 ServiceWorkerStatusCode status) { | 814 ServiceWorkerStatusCode status) { |
805 EmbeddedWorkerStatus old_status = status_; | 815 EmbeddedWorkerStatus old_status = status_; |
806 ReleaseProcess(); | 816 ReleaseProcess(); |
807 base::WeakPtr<EmbeddedWorkerInstance> weak_this = weak_factory_.GetWeakPtr(); | 817 base::WeakPtr<EmbeddedWorkerInstance> weak_this = weak_factory_.GetWeakPtr(); |
808 callback.Run(status); | 818 callback.Run(status); |
809 if (weak_this && old_status != EmbeddedWorkerStatus::STOPPED) | 819 if (weak_this && old_status != EmbeddedWorkerStatus::STOPPED) |
810 FOR_EACH_OBSERVER(Listener, weak_this->listener_list_, | 820 FOR_EACH_OBSERVER(Listener, weak_this->listener_list_, |
(...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
871 case SCRIPT_READ_FINISHED: | 881 case SCRIPT_READ_FINISHED: |
872 return "Script read finished"; | 882 return "Script read finished"; |
873 case STARTING_PHASE_MAX_VALUE: | 883 case STARTING_PHASE_MAX_VALUE: |
874 NOTREACHED(); | 884 NOTREACHED(); |
875 } | 885 } |
876 NOTREACHED() << phase; | 886 NOTREACHED() << phase; |
877 return std::string(); | 887 return std::string(); |
878 } | 888 } |
879 | 889 |
880 } // namespace content | 890 } // namespace content |
OLD | NEW |