blob: 52daac22a311a856f94b1e3077648d2490ead9b1 [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/metrics/histogram_macros.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/browser/service_worker/service_worker_dispatcher_host.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_);
return registry;
}
std::unique_ptr<EmbeddedWorkerInstance> EmbeddedWorkerRegistry::CreateWorker() {
std::unique_ptr<EmbeddedWorkerInstance> worker(
new EmbeddedWorkerInstance(context_, next_embedded_worker_id_));
worker_map_[next_embedded_worker_id_++] = worker.get();
return worker;
}
bool EmbeddedWorkerRegistry::OnMessageReceived(const IPC::Message& message,
int process_id) {
// TODO(kinuko): Move all EmbeddedWorker message handling from
// ServiceWorkerDispatcherHost.
EmbeddedWorkerInstance* worker =
GetWorkerForMessage(process_id, message.routing_id());
if (!worker) {
// Assume this is from a detached worker, return true to indicate we're
// purposely handling the message as no-op.
return true;
}
bool handled = worker->OnMessageReceived(message);
// Assume an unhandled message for a stopping worker is because the message
// was timed out and its handler removed prior to stopping.
// We might be more precise and record timed out request ids, but some
// cumbersome bookkeeping is needed and the IPC messaging will soon migrate
// to Mojo anyway.
return handled || worker->status() == EmbeddedWorkerStatus::STOPPING;
}
void EmbeddedWorkerRegistry::Shutdown() {
for (WorkerInstanceMap::iterator it = worker_map_.begin();
it != worker_map_.end();
++it) {
it->second->Stop();
}
}
bool EmbeddedWorkerRegistry::OnWorkerStarted(int process_id,
int embedded_worker_id) {
if (!base::ContainsKey(worker_process_map_, process_id) ||
!base::ContainsKey(worker_process_map_[process_id], embedded_worker_id)) {
return false;
}
lifetime_tracker_.StartTiming(embedded_worker_id);
return true;
}
void EmbeddedWorkerRegistry::OnWorkerStopped(int process_id,
int embedded_worker_id) {
worker_process_map_[process_id].erase(embedded_worker_id);
lifetime_tracker_.StopTiming(embedded_worker_id);
}
void EmbeddedWorkerRegistry::OnDevToolsAttached(int embedded_worker_id) {
lifetime_tracker_.AbortTiming(embedded_worker_id);
}
EmbeddedWorkerInstance* EmbeddedWorkerRegistry::GetWorker(
int embedded_worker_id) {
WorkerInstanceMap::iterator found = worker_map_.find(embedded_worker_id);
if (found == worker_map_.end())
return nullptr;
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();
}
void EmbeddedWorkerRegistry::BindWorkerToProcess(int process_id,
int embedded_worker_id) {
DCHECK(GetWorker(embedded_worker_id));
DCHECK_EQ(GetWorker(embedded_worker_id)->process_id(), process_id);
DCHECK(
!base::ContainsKey(worker_process_map_, process_id) ||
!base::ContainsKey(worker_process_map_[process_id], embedded_worker_id));
worker_process_map_[process_id].insert(embedded_worker_id);
}
ServiceWorkerStatusCode EmbeddedWorkerRegistry::Send(
int process_id, IPC::Message* message_ptr) {
std::unique_ptr<IPC::Message> message(message_ptr);
if (!context_)
return SERVICE_WORKER_ERROR_ABORT;
IPC::Sender* sender = context_->GetDispatcherHost(process_id);
if (!sender)
return SERVICE_WORKER_ERROR_PROCESS_NOT_FOUND;
if (!sender->Send(message.release()))
return SERVICE_WORKER_ERROR_IPC_FAILED;
return SERVICE_WORKER_OK;
}
void EmbeddedWorkerRegistry::RemoveWorker(int process_id,
int embedded_worker_id) {
DCHECK(base::ContainsKey(worker_map_, embedded_worker_id));
DetachWorker(process_id, embedded_worker_id);
worker_map_.erase(embedded_worker_id);
}
void EmbeddedWorkerRegistry::DetachWorker(int process_id,
int embedded_worker_id) {
DCHECK(base::ContainsKey(worker_map_, embedded_worker_id));
if (!base::ContainsKey(worker_process_map_, process_id))
return;
worker_process_map_[process_id].erase(embedded_worker_id);
if (worker_process_map_[process_id].empty())
worker_process_map_.erase(process_id);
lifetime_tracker_.StopTiming(embedded_worker_id);
}
EmbeddedWorkerInstance* EmbeddedWorkerRegistry::GetWorkerForMessage(
int process_id,
int embedded_worker_id) {
EmbeddedWorkerInstance* worker = GetWorker(embedded_worker_id);
if (!worker || worker->process_id() != process_id) {
UMA_HISTOGRAM_BOOLEAN("ServiceWorker.WorkerForMessageFound", false);
return nullptr;
}
UMA_HISTOGRAM_BOOLEAN("ServiceWorker.WorkerForMessageFound", true);
return worker;
}
} // namespace content