blob: adf2579d733fb27a59a944fb3a29d033db6e9da9 [file] [log] [blame]
// 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/browser/service_worker/embedded_worker_registry.h"
#include "base/bind_helpers.h"
#include "base/stl_util.h"
#include "content/browser/renderer_host/render_widget_helper.h"
#include "content/browser/service_worker/embedded_worker_instance.h"
#include "content/browser/service_worker/service_worker_context_core.h"
#include "content/browser/service_worker/service_worker_context_wrapper.h"
#include "content/common/service_worker/embedded_worker_messages.h"
#include "content/public/browser/browser_thread.h"
#include "ipc/ipc_message.h"
#include "ipc/ipc_sender.h"
namespace content {
// static
scoped_refptr<EmbeddedWorkerRegistry> EmbeddedWorkerRegistry::Create(
const base::WeakPtr<ServiceWorkerContextCore>& context) {
return make_scoped_refptr(new EmbeddedWorkerRegistry(context, 0));
}
// static
scoped_refptr<EmbeddedWorkerRegistry> EmbeddedWorkerRegistry::Create(
const base::WeakPtr<ServiceWorkerContextCore>& context,
EmbeddedWorkerRegistry* old_registry) {
scoped_refptr<EmbeddedWorkerRegistry> registry =
new EmbeddedWorkerRegistry(
context,
old_registry->next_embedded_worker_id_);
registry->process_sender_map_.swap(old_registry->process_sender_map_);
return registry;
}
scoped_ptr<EmbeddedWorkerInstance> EmbeddedWorkerRegistry::CreateWorker() {
scoped_ptr<EmbeddedWorkerInstance> worker(
new EmbeddedWorkerInstance(context_, next_embedded_worker_id_));
worker_map_[next_embedded_worker_id_++] = worker.get();
return worker.Pass();
}
ServiceWorkerStatusCode EmbeddedWorkerRegistry::StopWorker(
int process_id, int embedded_worker_id) {
return Send(process_id,
new EmbeddedWorkerMsg_StopWorker(embedded_worker_id));
}
bool EmbeddedWorkerRegistry::OnMessageReceived(const IPC::Message& message) {
// TODO(kinuko): Move all EmbeddedWorker message handling from
// ServiceWorkerDispatcherHost.
WorkerInstanceMap::iterator found = worker_map_.find(message.routing_id());
DCHECK(found != worker_map_.end());
if (found == worker_map_.end())
return false;
return found->second->OnMessageReceived(message);
}
void EmbeddedWorkerRegistry::Shutdown() {
for (WorkerInstanceMap::iterator it = worker_map_.begin();
it != worker_map_.end();
++it) {
it->second->Stop();
}
}
void EmbeddedWorkerRegistry::OnWorkerReadyForInspection(
int process_id,
int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
found->second->OnReadyForInspection();
}
void EmbeddedWorkerRegistry::OnWorkerScriptLoaded(
int process_id,
int thread_id,
int embedded_worker_id ) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
found->second->OnScriptLoaded(thread_id);
}
void EmbeddedWorkerRegistry::OnWorkerScriptLoadFailed(int process_id,
int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
found->second->OnScriptLoadFailed();
}
void EmbeddedWorkerRegistry::OnWorkerScriptEvaluated(int process_id,
int embedded_worker_id,
bool success) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
found->second->OnScriptEvaluated(success);
}
void EmbeddedWorkerRegistry::OnWorkerStarted(
int process_id, int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
// TODO(falken): Instead of DCHECK, we should terminate the process on
// unexpected message. Same with most of the DCHECKs in this file.
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
DCHECK(ContainsKey(worker_process_map_, process_id) &&
worker_process_map_[process_id].count(embedded_worker_id) == 1);
if (!ContainsKey(worker_process_map_, process_id) ||
worker_process_map_[process_id].count(embedded_worker_id) == 0) {
return;
}
found->second->OnStarted();
}
void EmbeddedWorkerRegistry::OnWorkerStopped(
int process_id, int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
worker_process_map_[process_id].erase(embedded_worker_id);
found->second->OnStopped();
}
void EmbeddedWorkerRegistry::OnPausedAfterDownload(
int process_id, int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
if (found == worker_map_.end() || found->second->process_id() != process_id)
return;
found->second->OnPausedAfterDownload();
}
void EmbeddedWorkerRegistry::OnReportException(
int embedded_worker_id,
const base::string16& error_message,
int line_number,
int column_number,
const GURL& source_url) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
if (found == worker_map_.end())
return;
found->second->OnReportException(
error_message, line_number, column_number, source_url);
}
void EmbeddedWorkerRegistry::OnReportConsoleMessage(
int embedded_worker_id,
int source_identifier,
int message_level,
const base::string16& message,
int line_number,
const GURL& source_url) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
if (found == worker_map_.end())
return;
found->second->OnReportConsoleMessage(
source_identifier, message_level, message, line_number, source_url);
}
void EmbeddedWorkerRegistry::AddChildProcessSender(
int process_id, IPC::Sender* sender) {
process_sender_map_[process_id] = sender;
DCHECK(!ContainsKey(worker_process_map_, process_id));
}
void EmbeddedWorkerRegistry::RemoveChildProcessSender(int process_id) {
process_sender_map_.erase(process_id);
std::map<int, std::set<int> >::iterator found =
worker_process_map_.find(process_id);
if (found != worker_process_map_.end()) {
const std::set<int>& worker_set = worker_process_map_[process_id];
for (std::set<int>::const_iterator it = worker_set.begin();
it != worker_set.end();
++it) {
int embedded_worker_id = *it;
DCHECK(ContainsKey(worker_map_, embedded_worker_id));
worker_map_[embedded_worker_id]->OnStopped();
}
worker_process_map_.erase(found);
}
}
EmbeddedWorkerInstance* EmbeddedWorkerRegistry::GetWorker(
int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
if (found == worker_map_.end())
return NULL;
return found->second;
}
bool EmbeddedWorkerRegistry::CanHandle(int embedded_worker_id) const {
if (embedded_worker_id < initial_embedded_worker_id_ ||
next_embedded_worker_id_ <= embedded_worker_id) {
return false;
}
return true;
}
EmbeddedWorkerRegistry::EmbeddedWorkerRegistry(
const base::WeakPtr<ServiceWorkerContextCore>& context,
int initial_embedded_worker_id)
: context_(context),
next_embedded_worker_id_(initial_embedded_worker_id),
initial_embedded_worker_id_(initial_embedded_worker_id) {
}
EmbeddedWorkerRegistry::~EmbeddedWorkerRegistry() {
Shutdown();
}
ServiceWorkerStatusCode EmbeddedWorkerRegistry::SendStartWorker(
scoped_ptr<EmbeddedWorkerMsg_StartWorker_Params> params,
int process_id) {
// The ServiceWorkerDispatcherHost is supposed to be created when the process
// is created, and keep an entry in process_sender_map_ for its whole
// lifetime.
DCHECK(ContainsKey(process_sender_map_, process_id));
int embedded_worker_id = params->embedded_worker_id;
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
DCHECK(found != worker_map_.end());
DCHECK_EQ(found->second->process_id(), process_id);
DCHECK(!ContainsKey(worker_process_map_, process_id) ||
worker_process_map_[process_id].count(embedded_worker_id) == 0);
ServiceWorkerStatusCode status =
Send(process_id, new EmbeddedWorkerMsg_StartWorker(*params));
if (status == SERVICE_WORKER_OK)
worker_process_map_[process_id].insert(embedded_worker_id);
return status;
}
ServiceWorkerStatusCode EmbeddedWorkerRegistry::Send(
int process_id, IPC::Message* message_ptr) {
scoped_ptr<IPC::Message> message(message_ptr);
if (!context_)
return SERVICE_WORKER_ERROR_ABORT;
ProcessToSenderMap::iterator found = process_sender_map_.find(process_id);
if (found == process_sender_map_.end())
return SERVICE_WORKER_ERROR_PROCESS_NOT_FOUND;
if (!found->second->Send(message.release()))
return SERVICE_WORKER_ERROR_IPC_FAILED;
return SERVICE_WORKER_OK;
}
void EmbeddedWorkerRegistry::RemoveWorker(int process_id,
int embedded_worker_id) {
DCHECK(ContainsKey(worker_map_, embedded_worker_id));
worker_map_.erase(embedded_worker_id);
worker_process_map_.erase(process_id);
}
} // namespace content