summaryrefslogtreecommitdiffstats
path: root/content/renderer/websharedworker_proxy.cc
diff options
context:
space:
mode:
Diffstat (limited to 'content/renderer/websharedworker_proxy.cc')
-rw-r--r--content/renderer/websharedworker_proxy.cc104
1 files changed, 95 insertions, 9 deletions
diff --git a/content/renderer/websharedworker_proxy.cc b/content/renderer/websharedworker_proxy.cc
index 203ec6b..d3b74d5 100644
--- a/content/renderer/websharedworker_proxy.cc
+++ b/content/renderer/websharedworker_proxy.cc
@@ -1,26 +1,112 @@
-// Copyright (c) 2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2011 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/renderer/websharedworker_proxy.h"
-
+#include "content/common/child_thread.h"
#include "content/common/view_messages.h"
#include "content/common/webmessageportchannel_impl.h"
#include "content/common/worker_messages.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebURL.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebWorkerClient.h"
WebSharedWorkerProxy::WebSharedWorkerProxy(ChildThread* child_thread,
unsigned long long document_id,
bool exists,
int route_id,
int render_view_route_id)
- : WebWorkerBase(child_thread,
- document_id,
- exists ? route_id : MSG_ROUTING_NONE,
- render_view_route_id,
- 0),
+ : route_id_(exists ? route_id : MSG_ROUTING_NONE),
+ render_view_route_id_(render_view_route_id),
+ child_thread_(child_thread),
+ document_id_(document_id),
pending_route_id_(route_id),
connect_listener_(NULL) {
+ if (route_id_ != MSG_ROUTING_NONE)
+ child_thread_->AddRoute(route_id_, this);
+}
+
+WebSharedWorkerProxy::~WebSharedWorkerProxy() {
+ Disconnect();
+
+ // Free up any unsent queued messages.
+ for (size_t i = 0; i < queued_messages_.size(); ++i)
+ delete queued_messages_[i];
+}
+
+void WebSharedWorkerProxy::Disconnect() {
+ if (route_id_ == MSG_ROUTING_NONE)
+ return;
+
+ // So the messages from WorkerContext (like WorkerContextDestroyed) do not
+ // come after nobody is listening. Since Worker and WorkerContext can
+ // terminate independently, already sent messages may still be in the pipe.
+ child_thread_->RemoveRoute(route_id_);
+
+ route_id_ = MSG_ROUTING_NONE;
+}
+
+void WebSharedWorkerProxy::CreateWorkerContext(const GURL& script_url,
+ bool is_shared,
+ const string16& name,
+ const string16& user_agent,
+ const string16& source_code,
+ int pending_route_id,
+ int64 script_resource_appcache_id) {
+ DCHECK(route_id_ == MSG_ROUTING_NONE);
+ ViewHostMsg_CreateWorker_Params params;
+ params.url = script_url;
+ params.name = name;
+ params.document_id = document_id_;
+ params.render_view_route_id = render_view_route_id_;
+ params.route_id = pending_route_id;
+ params.script_resource_appcache_id = script_resource_appcache_id;
+ IPC::Message* create_message = new ViewHostMsg_CreateWorker(
+ params, &route_id_);
+ child_thread_->Send(create_message);
+ if (route_id_ == MSG_ROUTING_NONE)
+ return;
+
+ child_thread_->AddRoute(route_id_, this);
+
+ // We make sure that the start message is the first, since postMessage or
+ // connect might have already been called.
+ queued_messages_.insert(queued_messages_.begin(),
+ new WorkerMsg_StartWorkerContext(
+ route_id_, script_url, user_agent, source_code));
+}
+
+bool WebSharedWorkerProxy::IsStarted() {
+ // Worker is started if we have a route ID and there are no queued messages
+ // (meaning we've sent the WorkerMsg_StartWorkerContext already).
+ return (route_id_ != MSG_ROUTING_NONE && queued_messages_.empty());
+}
+
+bool WebSharedWorkerProxy::Send(IPC::Message* message) {
+ // It's possible that messages will be sent before the worker is created, in
+ // which case route_id_ will be none. Or the worker object can be interacted
+ // with before the browser process told us that it started, in which case we
+ // also want to queue the message.
+ if (!IsStarted()) {
+ queued_messages_.push_back(message);
+ return true;
+ }
+
+ // For now we proxy all messages to the worker process through the browser.
+ // Revisit if we find this slow.
+ // TODO(jabdelmalek): handle sync messages if we need them.
+ IPC::Message* wrapped_msg = new ViewHostMsg_ForwardToWorker(*message);
+ delete message;
+ return child_thread_->Send(wrapped_msg);
+}
+
+void WebSharedWorkerProxy::SendQueuedMessages() {
+ DCHECK(queued_messages_.size());
+ std::vector<IPC::Message*> queued_messages = queued_messages_;
+ queued_messages_.clear();
+ for (size_t i = 0; i < queued_messages.size(); ++i) {
+ queued_messages[i]->set_routing_id(route_id_);
+ Send(queued_messages[i]);
+ }
}
bool WebSharedWorkerProxy::isStarted() {
@@ -34,8 +120,8 @@ void WebSharedWorkerProxy::startWorkerContext(
const WebKit::WebString& source_code,
long long script_resource_appcache_id) {
DCHECK(!isStarted());
- CreateSharedWorkerContext(script_url, name, user_agent, source_code,
- pending_route_id_, script_resource_appcache_id);
+ CreateWorkerContext(script_url, true, name, user_agent, source_code,
+ pending_route_id_, script_resource_appcache_id);
}
void WebSharedWorkerProxy::terminateWorkerContext() {