// 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/child/webmessageportchannel_impl.h" #include "base/bind.h" #include "base/message_loop/message_loop_proxy.h" #include "base/values.h" #include "content/child/child_process.h" #include "content/child/child_thread_impl.h" #include "content/common/message_port_messages.h" #include "content/public/child/v8_value_converter.h" #include "third_party/WebKit/public/platform/WebMessagePortChannelClient.h" #include "third_party/WebKit/public/platform/WebString.h" #include "third_party/WebKit/public/web/WebSerializedScriptValue.h" #include "v8/include/v8.h" using blink::WebMessagePortChannel; using blink::WebMessagePortChannelArray; using blink::WebMessagePortChannelClient; using blink::WebString; namespace content { WebMessagePortChannelImpl::WebMessagePortChannelImpl( const scoped_refptr& main_thread_task_runner) : client_(NULL), route_id_(MSG_ROUTING_NONE), message_port_id_(MSG_ROUTING_NONE), send_messages_as_values_(false), main_thread_task_runner_(main_thread_task_runner) { AddRef(); Init(); } WebMessagePortChannelImpl::WebMessagePortChannelImpl( int route_id, const TransferredMessagePort& port, const scoped_refptr& main_thread_task_runner) : client_(NULL), route_id_(route_id), message_port_id_(port.id), send_messages_as_values_(port.send_messages_as_values), main_thread_task_runner_(main_thread_task_runner) { AddRef(); Init(); } WebMessagePortChannelImpl::~WebMessagePortChannelImpl() { // If we have any queued messages with attached ports, manually destroy them. while (!message_queue_.empty()) { const WebMessagePortChannelArray& channel_array = message_queue_.front().ports; for (size_t i = 0; i < channel_array.size(); i++) { channel_array[i]->destroy(); } message_queue_.pop(); } if (message_port_id_ != MSG_ROUTING_NONE) Send(new MessagePortHostMsg_DestroyMessagePort(message_port_id_)); if (route_id_ != MSG_ROUTING_NONE) ChildThreadImpl::current()->GetRouter()->RemoveRoute(route_id_); } // static void WebMessagePortChannelImpl::CreatePair( const scoped_refptr& main_thread_task_runner, blink::WebMessagePortChannel** channel1, blink::WebMessagePortChannel** channel2) { WebMessagePortChannelImpl* impl1 = new WebMessagePortChannelImpl(main_thread_task_runner); WebMessagePortChannelImpl* impl2 = new WebMessagePortChannelImpl(main_thread_task_runner); impl1->Entangle(impl2); impl2->Entangle(impl1); *channel1 = impl1; *channel2 = impl2; } // static std::vector WebMessagePortChannelImpl::ExtractMessagePortIDs( WebMessagePortChannelArray* channels) { std::vector message_ports; if (channels) { // Extract the port IDs from the source array, then free it. message_ports = ExtractMessagePortIDs(*channels); delete channels; } return message_ports; } // static std::vector WebMessagePortChannelImpl::ExtractMessagePortIDs( const WebMessagePortChannelArray& channels) { std::vector message_ports(channels.size()); for (size_t i = 0; i < channels.size(); ++i) { WebMessagePortChannelImpl* webchannel = static_cast(channels[i]); // The message port ids might not be set up yet if this channel // wasn't created on the main thread. DCHECK(webchannel->main_thread_task_runner_->BelongsToCurrentThread()); message_ports[i].id = webchannel->message_port_id(); message_ports[i].send_messages_as_values = webchannel->send_messages_as_values_; webchannel->QueueMessages(); DCHECK(message_ports[i].id != MSG_ROUTING_NONE); } return message_ports; } // static WebMessagePortChannelArray WebMessagePortChannelImpl::CreatePorts( const std::vector& message_ports, const std::vector& new_routing_ids, const scoped_refptr& main_thread_task_runner) { DCHECK_EQ(message_ports.size(), new_routing_ids.size()); WebMessagePortChannelArray channels(message_ports.size()); for (size_t i = 0; i < message_ports.size() && i < new_routing_ids.size(); ++i) { channels[i] = new WebMessagePortChannelImpl( new_routing_ids[i], message_ports[i], main_thread_task_runner); } return channels; } void WebMessagePortChannelImpl::setClient(WebMessagePortChannelClient* client) { // Must lock here since client_ is called on the main thread. base::AutoLock auto_lock(lock_); client_ = client; } void WebMessagePortChannelImpl::destroy() { setClient(NULL); // Release the object on the main thread, since the destructor might want to // send an IPC, and that has to happen on the main thread. main_thread_task_runner_->ReleaseSoon(FROM_HERE, this); } void WebMessagePortChannelImpl::postMessage( const WebString& message_as_string, WebMessagePortChannelArray* channels) { MessagePortMessage message(message_as_string); if (send_messages_as_values_) { blink::WebSerializedScriptValue serialized_value = blink::WebSerializedScriptValue::fromString(message_as_string); v8::Handle v8_value = serialized_value.deserialize(); scoped_ptr converter(V8ValueConverter::create()); converter->SetDateAllowed(true); converter->SetRegExpAllowed(true); scoped_ptr message_as_value(converter->FromV8Value( v8_value, v8::Isolate::GetCurrent()->GetCurrentContext())); message = MessagePortMessage(message_as_value.Pass()); } if (!main_thread_task_runner_->BelongsToCurrentThread()) { main_thread_task_runner_->PostTask( FROM_HERE, base::Bind(&WebMessagePortChannelImpl::PostMessage, this, message, channels)); } else { PostMessage(message, channels); } } void WebMessagePortChannelImpl::PostMessage( const MessagePortMessage& message, WebMessagePortChannelArray* channels) { IPC::Message* msg = new MessagePortHostMsg_PostMessage( message_port_id_, message, ExtractMessagePortIDs(channels)); Send(msg); } bool WebMessagePortChannelImpl::tryGetMessage( WebString* message, WebMessagePortChannelArray& channels) { base::AutoLock auto_lock(lock_); if (message_queue_.empty()) return false; const MessagePortMessage& data = message_queue_.front().message; DCHECK(data.is_string() != data.is_value()); if (data.is_value()) { v8::HandleScope handle_scope(client_->scriptIsolate()); v8::Context::Scope context_scope( client_->scriptContextForMessageConversion()); scoped_ptr converter(V8ValueConverter::create()); converter->SetDateAllowed(true); converter->SetRegExpAllowed(true); v8::Handle v8_value = converter->ToV8Value( data.as_value(), client_->scriptContextForMessageConversion()); blink::WebSerializedScriptValue serialized_value = blink::WebSerializedScriptValue::serialize(v8_value); *message = serialized_value.toString(); } else { *message = message_queue_.front().message.message_as_string; } channels = message_queue_.front().ports; message_queue_.pop(); return true; } void WebMessagePortChannelImpl::Init() { if (!main_thread_task_runner_->BelongsToCurrentThread()) { main_thread_task_runner_->PostTask( FROM_HERE, base::Bind(&WebMessagePortChannelImpl::Init, this)); return; } if (route_id_ == MSG_ROUTING_NONE) { DCHECK(message_port_id_ == MSG_ROUTING_NONE); Send(new MessagePortHostMsg_CreateMessagePort( &route_id_, &message_port_id_)); } else if (message_port_id_ != MSG_ROUTING_NONE) { Send(new MessagePortHostMsg_ReleaseMessages(message_port_id_)); } ChildThreadImpl::current()->GetRouter()->AddRoute(route_id_, this); } void WebMessagePortChannelImpl::Entangle( scoped_refptr channel) { // The message port ids might not be set up yet, if this channel wasn't // created on the main thread. So need to wait until we're on the main thread // before getting the other message port id. if (!main_thread_task_runner_->BelongsToCurrentThread()) { main_thread_task_runner_->PostTask( FROM_HERE, base::Bind(&WebMessagePortChannelImpl::Entangle, this, channel)); return; } Send(new MessagePortHostMsg_Entangle( message_port_id_, channel->message_port_id())); } void WebMessagePortChannelImpl::QueueMessages() { if (!main_thread_task_runner_->BelongsToCurrentThread()) { main_thread_task_runner_->PostTask( FROM_HERE, base::Bind(&WebMessagePortChannelImpl::QueueMessages, this)); return; } // This message port is being sent elsewhere (perhaps to another process). // The new endpoint needs to receive the queued messages, including ones that // could still be in-flight. So we tell the browser to queue messages, and it // sends us an ack, whose receipt we know means that no more messages are // in-flight. We then send the queued messages to the browser, which prepends // them to the ones it queued and it sends them to the new endpoint. Send(new MessagePortHostMsg_QueueMessages(message_port_id_)); // The process could potentially go away while we're still waiting for // in-flight messages. Ensure it stays alive. ChildProcess::current()->AddRefProcess(); } void WebMessagePortChannelImpl::Send(IPC::Message* message) { if (!main_thread_task_runner_->BelongsToCurrentThread()) { DCHECK(!message->is_sync()); main_thread_task_runner_->PostTask( FROM_HERE, base::Bind(&WebMessagePortChannelImpl::Send, this, message)); return; } ChildThreadImpl::current()->GetRouter()->Send(message); } bool WebMessagePortChannelImpl::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(WebMessagePortChannelImpl, message) IPC_MESSAGE_HANDLER(MessagePortMsg_Message, OnMessage) IPC_MESSAGE_HANDLER(MessagePortMsg_MessagesQueued, OnMessagesQueued) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void WebMessagePortChannelImpl::OnMessage( const MessagePortMessage& message, const std::vector& sent_message_ports, const std::vector& new_routing_ids) { base::AutoLock auto_lock(lock_); Message msg; msg.message = message; msg.ports = CreatePorts(sent_message_ports, new_routing_ids, main_thread_task_runner_.get()); bool was_empty = message_queue_.empty(); message_queue_.push(msg); if (client_ && was_empty) client_->messageAvailable(); } void WebMessagePortChannelImpl::OnMessagesQueued() { std::vector queued_messages; { base::AutoLock auto_lock(lock_); queued_messages.reserve(message_queue_.size()); while (!message_queue_.empty()) { MessagePortMessage message = message_queue_.front().message; std::vector ports = ExtractMessagePortIDs(message_queue_.front().ports); queued_messages.push_back(std::make_pair(message, ports)); message_queue_.pop(); } } Send(new MessagePortHostMsg_SendQueuedMessages( message_port_id_, queued_messages)); message_port_id_ = MSG_ROUTING_NONE; Release(); ChildProcess::current()->ReleaseProcess(); } WebMessagePortChannelImpl::Message::Message() {} WebMessagePortChannelImpl::Message::~Message() {} } // namespace content