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.
5 #include "content/browser/service_worker/embedded_worker_registry.h"
7 #include "base/stl_util.h"
8 #include "content/browser/service_worker/embedded_worker_instance.h"
9 #include "content/browser/service_worker/service_worker_context_core.h"
10 #include "content/common/service_worker/embedded_worker_messages.h"
11 #include "content/common/service_worker/service_worker_messages.h"
12 #include "ipc/ipc_message.h"
13 #include "ipc/ipc_sender.h"
17 EmbeddedWorkerRegistry::EmbeddedWorkerRegistry(
18 base::WeakPtr<ServiceWorkerContextCore> context)
20 next_embedded_worker_id_(0) {}
22 scoped_ptr<EmbeddedWorkerInstance> EmbeddedWorkerRegistry::CreateWorker() {
23 scoped_ptr<EmbeddedWorkerInstance> worker(
24 new EmbeddedWorkerInstance(this, next_embedded_worker_id_));
25 worker_map_[next_embedded_worker_id_++] = worker.get();
29 ServiceWorkerStatusCode EmbeddedWorkerRegistry::StartWorker(
31 int embedded_worker_id,
32 int64 service_worker_version_id,
33 const GURL& script_url) {
34 return Send(process_id,
35 new EmbeddedWorkerMsg_StartWorker(embedded_worker_id,
36 service_worker_version_id,
40 ServiceWorkerStatusCode EmbeddedWorkerRegistry::StopWorker(
41 int process_id, int embedded_worker_id) {
42 return Send(process_id,
43 new EmbeddedWorkerMsg_StopWorker(embedded_worker_id));
46 void EmbeddedWorkerRegistry::OnWorkerStarted(
47 int process_id, int thread_id, int embedded_worker_id) {
48 DCHECK(!ContainsKey(worker_process_map_, process_id) ||
49 worker_process_map_[process_id].count(embedded_worker_id) == 0);
50 WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
51 if (found == worker_map_.end()) {
52 LOG(ERROR) << "Worker " << embedded_worker_id << " not registered";
55 worker_process_map_[process_id].insert(embedded_worker_id);
56 DCHECK_EQ(found->second->process_id(), process_id);
57 found->second->OnStarted(thread_id);
60 void EmbeddedWorkerRegistry::OnWorkerStopped(
61 int process_id, int embedded_worker_id) {
62 WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
63 if (found == worker_map_.end()) {
64 LOG(ERROR) << "Worker " << embedded_worker_id << " not registered";
67 DCHECK_EQ(found->second->process_id(), process_id);
68 worker_process_map_[process_id].erase(embedded_worker_id);
69 found->second->OnStopped();
72 void EmbeddedWorkerRegistry::OnSendMessageToBrowser(
73 int embedded_worker_id, int request_id, const IPC::Message& message) {
74 WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
75 if (found == worker_map_.end()) {
76 LOG(ERROR) << "Worker " << embedded_worker_id << " not registered";
79 // Perform security check to filter out any unexpected (and non-test)
80 // messages. This must list up all message types that can go through here.
81 if (message.type() == ServiceWorkerHostMsg_InstallEventFinished::ID ||
82 IPC_MESSAGE_CLASS(message) == TestMsgStart) {
83 found->second->OnMessageReceived(request_id, message);
86 NOTREACHED() << "Got unexpected message: " << message.type();
89 void EmbeddedWorkerRegistry::AddChildProcessSender(
90 int process_id, IPC::Sender* sender) {
91 process_sender_map_[process_id] = sender;
92 DCHECK(!ContainsKey(worker_process_map_, process_id));
95 void EmbeddedWorkerRegistry::RemoveChildProcessSender(int process_id) {
96 process_sender_map_.erase(process_id);
97 std::map<int, std::set<int> >::iterator found =
98 worker_process_map_.find(process_id);
99 if (found != worker_process_map_.end()) {
100 const std::set<int>& worker_set = worker_process_map_[process_id];
101 for (std::set<int>::const_iterator it = worker_set.begin();
102 it != worker_set.end();
104 int embedded_worker_id = *it;
105 DCHECK(ContainsKey(worker_map_, embedded_worker_id));
106 worker_map_[embedded_worker_id]->OnStopped();
108 worker_process_map_.erase(found);
112 EmbeddedWorkerInstance* EmbeddedWorkerRegistry::GetWorker(
113 int embedded_worker_id) {
114 WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
115 if (found == worker_map_.end())
117 return found->second;
120 EmbeddedWorkerRegistry::~EmbeddedWorkerRegistry() {}
122 ServiceWorkerStatusCode EmbeddedWorkerRegistry::Send(
123 int process_id, IPC::Message* message) {
125 return SERVICE_WORKER_ERROR_ABORT;
126 ProcessToSenderMap::iterator found = process_sender_map_.find(process_id);
127 if (found == process_sender_map_.end())
128 return SERVICE_WORKER_ERROR_PROCESS_NOT_FOUND;
129 if (!found->second->Send(message))
130 return SERVICE_WORKER_ERROR_IPC_FAILED;
131 return SERVICE_WORKER_OK;
134 void EmbeddedWorkerRegistry::RemoveWorker(int process_id,
135 int embedded_worker_id) {
136 DCHECK(ContainsKey(worker_map_, embedded_worker_id));
137 worker_map_.erase(embedded_worker_id);
138 worker_process_map_.erase(process_id);
141 } // namespace content