Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #ifndef IPC_IPC_CHANNEL_NACL_H_ | 5 #ifndef IPC_IPC_CHANNEL_NACL_H_ |
| 6 #define IPC_IPC_CHANNEL_NACL_H_ | 6 #define IPC_IPC_CHANNEL_NACL_H_ |
| 7 #pragma once | 7 #pragma once |
| 8 | 8 |
| 9 #include <deque> | |
| 10 #include <string> | |
| 11 | |
| 12 #include "base/memory/linked_ptr.h" | |
| 13 #include "base/memory/scoped_ptr.h" | |
| 14 #include "base/memory/weak_ptr.h" | |
| 15 #include "base/threading/simple_thread.h" | |
| 9 #include "ipc/ipc_channel.h" | 16 #include "ipc/ipc_channel.h" |
| 10 #include "ipc/ipc_channel_reader.h" | 17 #include "ipc/ipc_channel_reader.h" |
| 11 | 18 |
| 12 namespace IPC { | 19 namespace IPC { |
| 13 | 20 |
| 14 // Similar to the Posix version of ChannelImpl but for Native Client code. | 21 // Similar to the Posix version of ChannelImpl but for Native Client code. |
| 15 // This is somewhat different because NaCl's send/recvmsg is slightly different | 22 // This is somewhat different because NaCl's send/recvmsg is slightly different |
| 16 // and we don't need to worry about complicated set up and READWRITE mode for | 23 // and we don't need to worry about complicated set up and READWRITE mode for |
| 17 // sharing handles. | 24 // sharing handles. We also currently do not support passing file descriptors or |
| 25 // named pipes, and we use background threads to emulate signaling when we can | |
| 26 // read or write without blocking. | |
| 18 class Channel::ChannelImpl : public internal::ChannelReader { | 27 class Channel::ChannelImpl : public internal::ChannelReader { |
| 19 public: | 28 public: |
| 29 // Mirror methods of Channel, see ipc_channel.h for description. | |
| 20 ChannelImpl(const IPC::ChannelHandle& channel_handle, | 30 ChannelImpl(const IPC::ChannelHandle& channel_handle, |
| 21 Mode mode, | 31 Mode mode, |
| 22 Listener* listener); | 32 Listener* listener); |
| 23 virtual ~ChannelImpl(); | 33 virtual ~ChannelImpl(); |
| 24 | 34 |
| 25 // Channel implementation. | 35 // Channel implementation. |
| 26 bool Connect(); | 36 bool Connect(); |
| 27 void Close(); | 37 void Close(); |
| 28 bool Send(Message* message); | 38 bool Send(Message* message); |
| 29 int GetClientFileDescriptor() const; | |
| 30 int TakeClientFileDescriptor(); | |
| 31 bool AcceptsConnections() const; | |
| 32 bool HasAcceptedConnection() const; | |
| 33 bool GetClientEuid(uid_t* client_euid) const; | |
| 34 void ResetToAcceptingConnectionState(); | |
| 35 static bool IsNamedServerInitialized(const std::string& channel_id); | |
| 36 | 39 |
| 40 // Posted to the main thread by ReaderThreadRunner. | |
| 41 void DidRecvMsg(scoped_ptr<std::vector<char> > buffer); | |
| 42 void ReadDidFail(); | |
| 43 | |
| 44 private: | |
| 45 class ReaderThreadRunner; | |
| 46 | |
| 47 bool CreatePipe(const IPC::ChannelHandle& channel_handle); | |
| 48 bool ProcessOutgoingMessages(); | |
| 49 int GetHelloMessageProcId(); | |
| 50 void QueueHelloMessage(); | |
| 51 | |
| 52 // ChannelReader implementation. | |
| 37 virtual ReadState ReadData(char* buffer, | 53 virtual ReadState ReadData(char* buffer, |
| 38 int buffer_len, | 54 int buffer_len, |
| 39 int* bytes_read) OVERRIDE; | 55 int* bytes_read) OVERRIDE; |
| 40 virtual bool WillDispatchInputMessage(Message* msg) OVERRIDE; | 56 virtual bool WillDispatchInputMessage(Message* msg) OVERRIDE; |
| 41 virtual bool DidEmptyInputBuffers() OVERRIDE; | 57 virtual bool DidEmptyInputBuffers() OVERRIDE; |
| 42 virtual void HandleHelloMessage(const Message& msg) OVERRIDE; | 58 virtual void HandleHelloMessage(const Message& msg) OVERRIDE; |
| 43 | 59 |
| 44 private: | 60 Mode mode_; |
| 61 base::ProcessId peer_pid_; | |
| 62 bool waiting_connect_; | |
| 63 | |
| 64 // The pipe used for communication. | |
| 65 int pipe_; | |
| 66 | |
| 67 // The "name" of our pipe. On Windows this is the global identifier for | |
| 68 // the pipe. On POSIX it's used as a key in a local map of file descriptors. | |
| 69 // For NaCl, we don't actually support looking up file descriptors by name, | |
| 70 // and it's only used for debug information. | |
| 71 std::string pipe_name_; | |
| 72 | |
| 73 // We use a thread for reading, so that we can simply block on reading and | |
| 74 // post the received data back to the main thread to be properly interleaved | |
| 75 // with other tasks in the MessagePump. | |
| 76 // | |
| 77 // imc_recvmsg supports non-blocking reads, but there's no easy way to be | |
| 78 // informed when a write or read can be done without blocking (this is handled | |
| 79 // by libevent in Posix). | |
| 80 scoped_ptr<ReaderThreadRunner> reader_thread_runner_; | |
| 81 scoped_ptr<base::DelegateSimpleThread> reader_thread_; | |
| 82 // IPC::ChannelReader expects to be able to call ReadData on us to | |
|
brettw
2012/05/01 22:29:48
Blank line before this.
| |
| 83 // synchronously read data waiting in the pipe's buffer without blocking. | |
| 84 // Since we can't do that (see 1 and 2 above), the reader thread does blocking | |
| 85 // reads and posts the data over to the main thread in byte vectors. Each byte | |
| 86 // vector is the result of one call to "recvmsg". When ReadData is called, it | |
| 87 // pulls the bytes out of these vectors in order. | |
| 88 // TODO(dmichael): There's probably a more efficient way to emulate this with | |
| 89 // a circular buffer or something, so we don't have to do so | |
| 90 // many heap allocations. But it maybe isn't worth | |
| 91 // the trouble given that we probably want to implement 1 and | |
| 92 // 2 above in NaCl eventually. | |
| 93 std::deque<linked_ptr<std::vector<char> > > read_queue_; | |
| 94 | |
| 95 // This queue is used when a message is sent prior to Connect having been | |
| 96 // called. Normally after we're connected, the queue is empty. | |
| 97 std::deque<linked_ptr<Message> > output_queue_; | |
| 98 | |
| 99 base::WeakPtrFactory<ChannelImpl> weak_ptr_factory_; | |
| 100 | |
| 45 DISALLOW_IMPLICIT_CONSTRUCTORS(ChannelImpl); | 101 DISALLOW_IMPLICIT_CONSTRUCTORS(ChannelImpl); |
| 46 }; | 102 }; |
| 47 | 103 |
| 48 } // namespace IPC | 104 } // namespace IPC |
| 49 | 105 |
| 50 #endif // IPC_IPC_CHANNEL_NACL_H_ | 106 #endif // IPC_IPC_CHANNEL_NACL_H_ |
| OLD | NEW |