blob: 78a580fb4c679c419c10d92fcef8b110a965bdef [file] [log] [blame]
// Copyright (c) 2012 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 <queue>
#include <vector>
#include "base/basictypes.h"
#include "base/memory/ref_counted.h"
#include "base/memory/scoped_ptr.h"
#include "base/strings/string16.h"
#include "base/synchronization/lock.h"
#include "content/public/common/message_port_types.h"
#include "ipc/ipc_listener.h"
#include "third_party/WebKit/public/platform/WebMessagePortChannel.h"
namespace base {
class SingleThreadTaskRunner;
class Value;
namespace content {
class ChildThread;
// This is thread safe.
class WebMessagePortChannelImpl
: public blink::WebMessagePortChannel,
public IPC::Listener,
public base::RefCountedThreadSafe<WebMessagePortChannelImpl> {
explicit WebMessagePortChannelImpl(
const scoped_refptr<base::SingleThreadTaskRunner>&
int route_id,
const TransferredMessagePort& port,
const scoped_refptr<base::SingleThreadTaskRunner>&
static void CreatePair(
const scoped_refptr<base::SingleThreadTaskRunner>&
blink::WebMessagePortChannel** channel1,
blink::WebMessagePortChannel** channel2);
// Extracts port IDs for passing on to the browser process, and queues any
// received messages.
static std::vector<TransferredMessagePort> ExtractMessagePortIDs(
scoped_ptr<blink::WebMessagePortChannelArray> channels);
// Extracts port IDs for passing on to the browser process, and queues any
// received messages.
static std::vector<TransferredMessagePort> ExtractMessagePortIDs(
const blink::WebMessagePortChannelArray& channels);
// Extracts port IDs for passing on to the browser process, but doesn't
// send a separate IPC to the browser to initiate queueing messages. Instead
// calling code is responsible for initiating the queueing in the browser
// process. This is useful when transfering ports over an IPC channel that
// does not share ordering guarentees with regular IPC.
static std::vector<TransferredMessagePort>
scoped_ptr<blink::WebMessagePortChannelArray> channels);
// Creates WebMessagePortChannelImpl instances for port IDs passed in from the
// browser process.
static blink::WebMessagePortChannelArray CreatePorts(
const std::vector<TransferredMessagePort>& message_ports,
const std::vector<int>& new_routing_ids,
const scoped_refptr<base::SingleThreadTaskRunner>&
// Queues received and incoming messages until there are no more in-flight
// messages, then sends all of them to the browser process.
void QueueMessages();
int message_port_id() const { return message_port_id_; }
friend class base::RefCountedThreadSafe<WebMessagePortChannelImpl>;
~WebMessagePortChannelImpl() override;
// WebMessagePortChannel implementation.
virtual void setClient(blink::WebMessagePortChannelClient* client);
virtual void destroy();
virtual void postMessage(const blink::WebString& message,
blink::WebMessagePortChannelArray* channels_ptr);
virtual bool tryGetMessage(blink::WebString* message,
blink::WebMessagePortChannelArray& channels);
void Init();
void Entangle(scoped_refptr<WebMessagePortChannelImpl> channel);
void Send(IPC::Message* message);
void PostMessage(const MessagePortMessage& message,
scoped_ptr<blink::WebMessagePortChannelArray> channels);
// IPC::Listener implementation.
bool OnMessageReceived(const IPC::Message& message) override;
void OnMessage(const MessagePortMessage& message,
const std::vector<TransferredMessagePort>& sent_message_ports,
const std::vector<int>& new_routing_ids);
void OnMessagesQueued();
struct Message {
MessagePortMessage message;
blink::WebMessagePortChannelArray ports;
typedef std::queue<Message> MessageQueue;
MessageQueue message_queue_;
blink::WebMessagePortChannelClient* client_;
base::Lock lock_; // Locks access to above.
int route_id_; // The routing id for this object.
int message_port_id_; // A globally unique identifier for this message port.
// Flag to indicate if messages should be sent to the browser process as
// base::Value instances as opposed to being serialized using the default
// blink::WebSerializedScriptValue.
bool send_messages_as_values_;
scoped_refptr<base::SingleThreadTaskRunner> main_thread_task_runner_;
} // namespace content