raw_channel.cc revision 5f1c94371a64b3196d4be9466099bb892df9b88e
1a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)// Copyright 2014 The Chromium Authors. All rights reserved.
2a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
3a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)// found in the LICENSE file.
4a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
5a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "mojo/system/raw_channel.h"
6a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
7a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include <string.h>
8a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
9a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include <algorithm>
10a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
11a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "base/bind.h"
12a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "base/location.h"
13a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "base/logging.h"
14a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "base/message_loop/message_loop.h"
15a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "base/stl_util.h"
16a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)#include "mojo/system/message_in_transit.h"
17010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)#include "mojo/system/transport_data.h"
18a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
19a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)namespace mojo {
20a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)namespace system {
21a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
22a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)const size_t kReadSize = 4096;
23a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
240de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)// RawChannel::ReadBuffer ------------------------------------------------------
250de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
265f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)RawChannel::ReadBuffer::ReadBuffer() : buffer_(kReadSize), num_valid_bytes_(0) {
27a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
28a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
290de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)RawChannel::ReadBuffer::~ReadBuffer() {
300de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)}
31a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
32a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void RawChannel::ReadBuffer::GetBuffer(char** addr, size_t* size) {
335f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_GE(buffer_.size(), num_valid_bytes_ + kReadSize);
34a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  *addr = &buffer_[0] + num_valid_bytes_;
35a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  *size = kReadSize;
36a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
37a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
380de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)// RawChannel::WriteBuffer -----------------------------------------------------
390de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
400de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)RawChannel::WriteBuffer::WriteBuffer(size_t serialized_platform_handle_size)
410de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    : serialized_platform_handle_size_(serialized_platform_handle_size),
420de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      platform_handles_offset_(0),
430de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      data_offset_(0) {
440de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)}
45a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
46a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)RawChannel::WriteBuffer::~WriteBuffer() {
47a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  STLDeleteElements(&message_queue_);
48a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
49a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
500de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)bool RawChannel::WriteBuffer::HavePlatformHandlesToSend() const {
510de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  if (message_queue_.empty())
520de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    return false;
530de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
540de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  const TransportData* transport_data =
550de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      message_queue_.front()->transport_data();
560de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  if (!transport_data)
570de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    return false;
580de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
59cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  const embedder::PlatformHandleVector* all_platform_handles =
600de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      transport_data->platform_handles();
610de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  if (!all_platform_handles) {
625f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_EQ(platform_handles_offset_, 0u);
630de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    return false;
640de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  }
650de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  if (platform_handles_offset_ >= all_platform_handles->size()) {
665f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_EQ(platform_handles_offset_, all_platform_handles->size());
670de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    return false;
680de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  }
690de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
700de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  return true;
710de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)}
720de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
730de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)void RawChannel::WriteBuffer::GetPlatformHandlesToSend(
740de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    size_t* num_platform_handles,
750de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    embedder::PlatformHandle** platform_handles,
760de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    void** serialization_data) {
775f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(HavePlatformHandlesToSend());
780de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
790de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  TransportData* transport_data = message_queue_.front()->transport_data();
80cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  embedder::PlatformHandleVector* all_platform_handles =
810de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      transport_data->platform_handles();
820de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  *num_platform_handles =
830de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      all_platform_handles->size() - platform_handles_offset_;
840de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  *platform_handles = &(*all_platform_handles)[platform_handles_offset_];
850de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  size_t serialization_data_offset =
860de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      transport_data->platform_handle_table_offset();
875f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_GT(serialization_data_offset, 0u);
880de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  serialization_data_offset +=
890de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      platform_handles_offset_ * serialized_platform_handle_size_;
900de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  *serialization_data =
910de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      static_cast<char*>(transport_data->buffer()) + serialization_data_offset;
920de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)}
930de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
94a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void RawChannel::WriteBuffer::GetBuffers(std::vector<Buffer>* buffers) const {
95a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  buffers->clear();
96a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
97010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)  if (message_queue_.empty())
98a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return;
99a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
100a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  MessageInTransit* message = message_queue_.front();
1015f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_LT(data_offset_, message->total_size());
1020de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  size_t bytes_to_write = message->total_size() - data_offset_;
103010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)
1045f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  size_t transport_data_buffer_size =
1055f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      message->transport_data() ? message->transport_data()->buffer_size() : 0;
106010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)
107010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)  if (!transport_data_buffer_size) {
108a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Only write from the main buffer.
1095f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_LT(data_offset_, message->main_buffer_size());
1105f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_LE(bytes_to_write, message->main_buffer_size());
111a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    Buffer buffer = {
1120de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)        static_cast<const char*>(message->main_buffer()) + data_offset_,
113a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)        bytes_to_write};
114a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    buffers->push_back(buffer);
115a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return;
116a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
117a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
1180de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  if (data_offset_ >= message->main_buffer_size()) {
119010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    // Only write from the transport data buffer.
1205f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_LT(data_offset_ - message->main_buffer_size(),
1215f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)             transport_data_buffer_size);
1225f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_LE(bytes_to_write, transport_data_buffer_size);
123a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    Buffer buffer = {
124010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)        static_cast<const char*>(message->transport_data()->buffer()) +
1250de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)            (data_offset_ - message->main_buffer_size()),
126a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)        bytes_to_write};
127a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    buffers->push_back(buffer);
128a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return;
129a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
130a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
131cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // TODO(vtl): We could actually send out buffers from multiple messages, with
132cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // the "stopping" condition being reaching a message with platform handles
133cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // attached.
134cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
135a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // Write from both buffers.
1365f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(
1375f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      bytes_to_write,
1385f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      message->main_buffer_size() - data_offset_ + transport_data_buffer_size);
139a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  Buffer buffer1 = {
1405f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      static_cast<const char*>(message->main_buffer()) + data_offset_,
1415f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      message->main_buffer_size() - data_offset_};
142a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  buffers->push_back(buffer1);
143a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  Buffer buffer2 = {
1445f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      static_cast<const char*>(message->transport_data()->buffer()),
1455f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      transport_data_buffer_size};
146a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  buffers->push_back(buffer2);
147a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
148a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
1490de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)// RawChannel ------------------------------------------------------------------
1500de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)
151c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen MurdochRawChannel::RawChannel()
1525c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    : message_loop_for_io_(NULL),
1535c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      delegate_(NULL),
154a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_stopped_(false),
155a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      write_stopped_(false),
156a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      weak_ptr_factory_(this) {
157a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
158a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
159a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)RawChannel::~RawChannel() {
1605f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!read_buffer_);
1615f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!write_buffer_);
162a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
163a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // No need to take the |write_lock_| here -- if there are still weak pointers
164a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // outstanding, then we're hosed anyway (since we wouldn't be able to
165a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // invalidate them cleanly, since we might not be on the I/O thread).
1665f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!weak_ptr_factory_.HasWeakPtrs());
167a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
168a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
169c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdochbool RawChannel::Init(Delegate* delegate) {
1705f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(delegate);
171c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch
1725f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!delegate_);
173c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch  delegate_ = delegate;
174c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch
175c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch  CHECK_EQ(base::MessageLoop::current()->type(), base::MessageLoop::TYPE_IO);
1765f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!message_loop_for_io_);
177c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch  message_loop_for_io_ =
178c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch      static_cast<base::MessageLoopForIO*>(base::MessageLoop::current());
179a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
180a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // No need to take the lock. No one should be using us yet.
1815f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!read_buffer_);
182a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  read_buffer_.reset(new ReadBuffer);
1835f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!write_buffer_);
1840de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  write_buffer_.reset(new WriteBuffer(GetSerializedPlatformHandleSize()));
185a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
1865c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (!OnInit()) {
1875c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    delegate_ = NULL;
1885c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    message_loop_for_io_ = NULL;
1895c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    read_buffer_.reset();
1905c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    write_buffer_.reset();
191a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return false;
1925c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  }
193a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
194cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  if (ScheduleRead() != IO_PENDING) {
195cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    // This will notify the delegate about the read failure. Although we're on
196cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    // the I/O thread, don't call it in the nested context.
1975f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    message_loop_for_io_->PostTask(FROM_HERE,
1985f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                   base::Bind(&RawChannel::OnReadCompleted,
1995f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                              weak_ptr_factory_.GetWeakPtr(),
2005f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                              false,
2015f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                              0));
202cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  }
203cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
204cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // ScheduleRead() failure is treated as a read failure (by notifying the
205cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // delegate), not as an init failure.
206cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  return true;
207a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
208a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
209a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void RawChannel::Shutdown() {
2105f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(base::MessageLoop::current(), message_loop_for_io_);
211a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
212a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  base::AutoLock locker(write_lock_);
213a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
214effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch  LOG_IF(WARNING, !write_buffer_->message_queue_.empty())
215effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch      << "Shutting down RawChannel with write buffer nonempty";
216effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch
2175c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  // Reset the delegate so that it won't receive further calls.
2185c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  delegate_ = NULL;
219a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  read_stopped_ = true;
220a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  write_stopped_ = true;
2215c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  weak_ptr_factory_.InvalidateWeakPtrs();
222a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
223a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  OnShutdownNoLock(read_buffer_.Pass(), write_buffer_.Pass());
224a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
225a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
226a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)// Reminder: This must be thread-safe.
227a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)bool RawChannel::WriteMessage(scoped_ptr<MessageInTransit> message) {
2285f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(message);
229c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch
230a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  base::AutoLock locker(write_lock_);
231a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (write_stopped_)
232a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return false;
233a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
234a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (!write_buffer_->message_queue_.empty()) {
235cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    EnqueueMessageNoLock(message.Pass());
236a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return true;
237a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
238a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
239cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  EnqueueMessageNoLock(message.Pass());
2405f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(write_buffer_->data_offset_, 0u);
241a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
2420de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  size_t platform_handles_written = 0;
243a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  size_t bytes_written = 0;
2440de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  IOResult io_result = WriteNoLock(&platform_handles_written, &bytes_written);
245a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (io_result == IO_PENDING)
246a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return true;
247a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
2485f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  bool result = OnWriteCompletedNoLock(
2495f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      io_result == IO_SUCCEEDED, platform_handles_written, bytes_written);
250a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (!result) {
251a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Even if we're on the I/O thread, don't call |OnFatalError()| in the
252a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // nested context.
2535f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    message_loop_for_io_->PostTask(FROM_HERE,
2545f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                   base::Bind(&RawChannel::CallOnFatalError,
2555f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                              weak_ptr_factory_.GetWeakPtr(),
2565f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                              Delegate::FATAL_ERROR_WRITE));
257a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
258a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
259a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  return result;
260a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
261a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
262effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch// Reminder: This must be thread-safe.
263effb81e5f8246d0db0270817048dc992db66e9fbBen Murdochbool RawChannel::IsWriteBufferEmpty() {
264effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch  base::AutoLock locker(write_lock_);
265effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch  return write_buffer_->message_queue_.empty();
266effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch}
267effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch
268a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void RawChannel::OnReadCompleted(bool result, size_t bytes_read) {
2695f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(base::MessageLoop::current(), message_loop_for_io_);
270a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
271a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (read_stopped_) {
272a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    NOTREACHED();
273a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    return;
274a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
275a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
276a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  IOResult io_result = result ? IO_SUCCEEDED : IO_FAILED;
277a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
278a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // Keep reading data in a loop, and dispatch messages if enough data is
279a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // received. Exit the loop if any of the following happens:
280a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  //   - one or more messages were dispatched;
281a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  //   - the last read failed, was a partial read or would block;
282a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  //   - |Shutdown()| was called.
283a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  do {
284a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    if (io_result != IO_SUCCEEDED) {
285a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_stopped_ = true;
28646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      CallOnFatalError(Delegate::FATAL_ERROR_READ);
287a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      return;
288a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    }
289a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
290a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    read_buffer_->num_valid_bytes_ += bytes_read;
291a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
292a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Dispatch all the messages that we can.
293a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    bool did_dispatch_message = false;
294a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Tracks the offset of the first undispatched message in |read_buffer_|.
295a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Currently, we copy data to ensure that this is zero at the beginning.
296a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    size_t read_buffer_start = 0;
297a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    size_t remaining_bytes = read_buffer_->num_valid_bytes_;
298a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    size_t message_size;
299a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Note that we rely on short-circuit evaluation here:
300a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    //   - |read_buffer_start| may be an invalid index into
301a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    //     |read_buffer_->buffer_| if |remaining_bytes| is zero.
302a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    //   - |message_size| is only valid if |GetNextMessageSize()| returns true.
303a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // TODO(vtl): Use |message_size| more intelligently (e.g., to request the
304a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // next read).
305a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // TODO(vtl): Validate that |message_size| is sane.
3065f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    while (remaining_bytes > 0 && MessageInTransit::GetNextMessageSize(
3075f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                      &read_buffer_->buffer_[read_buffer_start],
3085f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                      remaining_bytes,
3095f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                                      &message_size) &&
310a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)           remaining_bytes >= message_size) {
3115f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      MessageInTransit::View message_view(
3125f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)          message_size, &read_buffer_->buffer_[read_buffer_start]);
3135f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      CHECK_EQ(message_view.total_size(), message_size);
314a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
315cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      const char* error_message = NULL;
316cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      if (!message_view.IsValid(GetSerializedPlatformHandleSize(),
317cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)                                &error_message)) {
3185f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)        CHECK(error_message);
319cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        LOG(WARNING) << "Received invalid message: " << error_message;
320cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        read_stopped_ = true;
32146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)        CallOnFatalError(Delegate::FATAL_ERROR_READ);
322a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)        return;
323a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      }
324cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
325cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      if (message_view.type() == MessageInTransit::kTypeRawChannel) {
326cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        if (!OnReadMessageForRawChannel(message_view)) {
327cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          read_stopped_ = true;
32846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)          CallOnFatalError(Delegate::FATAL_ERROR_READ);
329cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          return;
330cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        }
331cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      } else {
332cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        embedder::ScopedPlatformHandleVectorPtr platform_handles;
333cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        if (message_view.transport_data_buffer()) {
334cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          size_t num_platform_handles;
335cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          const void* platform_handle_table;
336cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          TransportData::GetPlatformHandleTable(
337cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              message_view.transport_data_buffer(),
338cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              &num_platform_handles,
339cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              &platform_handle_table);
340cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
341cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          if (num_platform_handles > 0) {
342cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)            platform_handles =
343cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)                GetReadPlatformHandles(num_platform_handles,
344cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)                                       platform_handle_table).Pass();
345cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)            if (!platform_handles) {
346cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              LOG(WARNING) << "Invalid number of platform handles received";
347cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              read_stopped_ = true;
34846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)              CallOnFatalError(Delegate::FATAL_ERROR_READ);
349cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)              return;
350cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)            }
351cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          }
352cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        }
353cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
354cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        // TODO(vtl): In the case that we aren't expecting any platform handles,
355cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        // for the POSIX implementation, we should confirm that none are stored.
356cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
357cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        // Dispatch the message.
3585f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)        CHECK(delegate_);
359cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        delegate_->OnReadMessage(message_view, platform_handles.Pass());
360cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        if (read_stopped_) {
361cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          // |Shutdown()| was called in |OnReadMessage()|.
362cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          // TODO(vtl): Add test for this case.
363cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          return;
364cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        }
365cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      }
366cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
367a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      did_dispatch_message = true;
368a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
369a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // Update our state.
370a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_buffer_start += message_size;
371a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      remaining_bytes -= message_size;
372a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    }
373a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
374a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    if (read_buffer_start > 0) {
375a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // Move data back to start.
376a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_buffer_->num_valid_bytes_ = remaining_bytes;
377a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      if (read_buffer_->num_valid_bytes_ > 0) {
378a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)        memmove(&read_buffer_->buffer_[0],
3795f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                &read_buffer_->buffer_[read_buffer_start],
3805f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                remaining_bytes);
381a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      }
382a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_buffer_start = 0;
383a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    }
384a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
385a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    if (read_buffer_->buffer_.size() - read_buffer_->num_valid_bytes_ <
3865f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)        kReadSize) {
387a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // Use power-of-2 buffer sizes.
388a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // TODO(vtl): Make sure the buffer doesn't get too large (and enforce the
389a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // maximum message size to whatever extent necessary).
390a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // TODO(vtl): We may often be able to peek at the header and get the real
391a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // required extra space (which may be much bigger than |kReadSize|).
392a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      size_t new_size = std::max(read_buffer_->buffer_.size(), kReadSize);
393a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      while (new_size < read_buffer_->num_valid_bytes_ + kReadSize)
394a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)        new_size *= 2;
395a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
396a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // TODO(vtl): It's suboptimal to zero out the fresh memory.
397a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      read_buffer_->buffer_.resize(new_size, 0);
398a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    }
399a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
400a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // (1) If we dispatched any messages, stop reading for now (and let the
401a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // message loop do its thing for another round).
402a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // TODO(vtl): Is this the behavior we want? (Alternatives: i. Dispatch only
403a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // a single message. Risks: slower, more complex if we want to avoid lots of
404a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // copying. ii. Keep reading until there's no more data and dispatch all the
405a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // messages we can. Risks: starvation of other users of the message loop.)
406a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // (2) If we didn't max out |kReadSize|, stop reading for now.
407a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    bool schedule_for_later = did_dispatch_message || bytes_read < kReadSize;
408a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    bytes_read = 0;
409a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    io_result = schedule_for_later ? ScheduleRead() : Read(&bytes_read);
410a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  } while (io_result != IO_PENDING);
411a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
412a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
4130de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)void RawChannel::OnWriteCompleted(bool result,
4140de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)                                  size_t platform_handles_written,
4150de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)                                  size_t bytes_written) {
4165f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(base::MessageLoop::current(), message_loop_for_io_);
417a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
418a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  bool did_fail = false;
419a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  {
420a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    base::AutoLock locker(write_lock_);
4215f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_EQ(write_stopped_, write_buffer_->message_queue_.empty());
422a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
423a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    if (write_stopped_) {
424a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      NOTREACHED();
425a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      return;
426a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    }
427a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
4285f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    did_fail = !OnWriteCompletedNoLock(
4295f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)                   result, platform_handles_written, bytes_written);
430a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
431a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
432a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (did_fail)
43346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    CallOnFatalError(Delegate::FATAL_ERROR_WRITE);
434a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
435a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
436cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)void RawChannel::EnqueueMessageNoLock(scoped_ptr<MessageInTransit> message) {
437cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  write_lock_.AssertAcquired();
438cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  write_buffer_->message_queue_.push_back(message.release());
439cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)}
440cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
441cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)bool RawChannel::OnReadMessageForRawChannel(
442cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    const MessageInTransit::View& message_view) {
443cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  // No non-implementation specific |RawChannel| control messages.
444cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  LOG(ERROR) << "Invalid control message (subtype " << message_view.subtype()
445cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)             << ")";
446cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  return false;
447cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)}
448cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
449a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void RawChannel::CallOnFatalError(Delegate::FatalError fatal_error) {
4505f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK_EQ(base::MessageLoop::current(), message_loop_for_io_);
451a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  // TODO(vtl): Add a "write_lock_.AssertNotAcquired()"?
4525c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (delegate_)
4535c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    delegate_->OnFatalError(fatal_error);
454a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
455a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
4560de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)bool RawChannel::OnWriteCompletedNoLock(bool result,
4570de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)                                        size_t platform_handles_written,
4580de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)                                        size_t bytes_written) {
459a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  write_lock_.AssertAcquired();
460a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
4615f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!write_stopped_);
4625f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  CHECK(!write_buffer_->message_queue_.empty());
463a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
464a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (result) {
4650de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    write_buffer_->platform_handles_offset_ += platform_handles_written;
4660de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    write_buffer_->data_offset_ += bytes_written;
467010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)
468010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    MessageInTransit* message = write_buffer_->message_queue_.front();
4690de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)    if (write_buffer_->data_offset_ >= message->total_size()) {
470a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      // Complete write.
4715f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)      CHECK_EQ(write_buffer_->data_offset_, message->total_size());
472a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      write_buffer_->message_queue_.pop_front();
473010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      delete message;
4740de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      write_buffer_->platform_handles_offset_ = 0;
4750de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)      write_buffer_->data_offset_ = 0;
476a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
477010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      if (write_buffer_->message_queue_.empty())
478010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)        return true;
479010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    }
480a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
481a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    // Schedule the next write.
482010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    IOResult io_result = ScheduleWriteNoLock();
483010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    if (io_result == IO_PENDING)
484a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      return true;
4855f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    CHECK_EQ(io_result, IO_FAILED);
486a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
487a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
488a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  write_stopped_ = true;
489a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  STLDeleteElements(&write_buffer_->message_queue_);
4900de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  write_buffer_->platform_handles_offset_ = 0;
4910de6073388f4e2780db8536178b129cd8f6ab386Torne (Richard Coles)  write_buffer_->data_offset_ = 0;
492a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  return false;
493a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}
494a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
495a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}  // namespace system
496a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)}  // namespace mojo
497