| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 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 REMOTING_PROTOCOL_MESSAGE_READER_H_ | 5 #ifndef REMOTING_PROTOCOL_MESSAGE_READER_H_ |
| 6 #define REMOTING_PROTOCOL_MESSAGE_READER_H_ | 6 #define REMOTING_PROTOCOL_MESSAGE_READER_H_ |
| 7 | 7 |
| 8 #include "base/callback.h" | 8 #include "base/callback.h" |
| 9 #include "base/ref_counted.h" | 9 #include "base/ref_counted.h" |
| 10 #include "base/scoped_ptr.h" | 10 #include "base/scoped_ptr.h" |
| 11 #include "base/task.h" | 11 #include "base/task.h" |
| 12 #include "net/base/completion_callback.h" | 12 #include "net/base/completion_callback.h" |
| 13 #include "remoting/base/compound_buffer.h" | 13 #include "remoting/base/compound_buffer.h" |
| 14 #include "remoting/protocol/message_decoder.h" | 14 #include "remoting/protocol/message_decoder.h" |
| 15 | 15 |
| 16 class MessageLoop; |
| 17 |
| 16 namespace net { | 18 namespace net { |
| 17 class IOBuffer; | 19 class IOBuffer; |
| 18 class Socket; | 20 class Socket; |
| 19 } // namespace net | 21 } // namespace net |
| 20 | 22 |
| 21 namespace remoting { | 23 namespace remoting { |
| 22 namespace protocol { | 24 namespace protocol { |
| 23 | 25 |
| 24 // MessageReader reads data from the socket asynchronously and calls | 26 // MessageReader reads data from the socket asynchronously and calls |
| 25 // callback for each message it receives | 27 // callback for each message it receives. It stops calling the |
| 26 class MessageReader { | 28 // callback as soon as the socket is closed, so the socket should |
| 29 // always be closed before the callback handler is destroyed. |
| 30 // |
| 31 // In order to throttle the stream, MessageReader doesn't try to read |
| 32 // new data from the socket until all previously received messages are |
| 33 // processed by the receiver (|done_task| is called for each message). |
| 34 // It is still possible that the MessageReceivedCallback is called |
| 35 // twice (so that there is more than one outstanding message), |
| 36 // e.g. when we the sender sends multiple messages in one TCP packet. |
| 37 class MessageReader : public base::RefCountedThreadSafe<MessageReader> { |
| 27 public: | 38 public: |
| 28 typedef Callback1<CompoundBuffer*>::Type MessageReceivedCallback; | 39 // The callback is given ownership of the second argument |
| 40 // (|done_task|). The buffer (first argument) is owned by |
| 41 // MessageReader and is freed when the task specified by the second |
| 42 // argument is called. |
| 43 typedef Callback2<CompoundBuffer*, Task*>::Type MessageReceivedCallback; |
| 29 | 44 |
| 30 MessageReader(); | 45 MessageReader(); |
| 31 virtual ~MessageReader(); | 46 virtual ~MessageReader(); |
| 32 | 47 |
| 33 // Initialize the MessageReader with a socket. If a message is received | 48 // Initialize the MessageReader with a socket. If a message is received |
| 34 // |callback| is called. | 49 // |callback| is called. |
| 35 void Init(net::Socket* socket, MessageReceivedCallback* callback); | 50 void Init(net::Socket* socket, MessageReceivedCallback* callback); |
| 36 | 51 |
| 37 private: | 52 private: |
| 38 void DoRead(); | 53 void DoRead(); |
| 39 void OnRead(int result); | 54 void OnRead(int result); |
| 40 void HandleReadResult(int result); | 55 void HandleReadResult(int result); |
| 41 void OnDataReceived(net::IOBuffer* data, int data_size); | 56 void OnDataReceived(net::IOBuffer* data, int data_size); |
| 57 void OnMessageDone(CompoundBuffer* message); |
| 58 void ProcessDoneEvent(); |
| 42 | 59 |
| 43 net::Socket* socket_; | 60 net::Socket* socket_; |
| 44 | 61 |
| 62 // The network message loop this object runs on. |
| 63 MessageLoop* message_loop_; |
| 64 |
| 65 // Set to true, when we have a socket read pending, and expecting |
| 66 // OnRead() to be called when new data is received. |
| 67 bool read_pending_; |
| 68 |
| 69 // Number of messages that we received, but haven't finished |
| 70 // processing yet, i.e. |done_task| hasn't been called for these |
| 71 // messages. |
| 72 int pending_messages_; |
| 73 |
| 45 bool closed_; | 74 bool closed_; |
| 46 scoped_refptr<net::IOBuffer> read_buffer_; | 75 scoped_refptr<net::IOBuffer> read_buffer_; |
| 47 net::CompletionCallbackImpl<MessageReader> read_callback_; | 76 net::CompletionCallbackImpl<MessageReader> read_callback_; |
| 48 | 77 |
| 49 MessageDecoder message_decoder_; | 78 MessageDecoder message_decoder_; |
| 50 | 79 |
| 51 // Callback is called when a message is received. | 80 // Callback is called when a message is received. |
| 52 scoped_ptr<MessageReceivedCallback> message_received_callback_; | 81 scoped_ptr<MessageReceivedCallback> message_received_callback_; |
| 53 }; | 82 }; |
| 54 | 83 |
| 84 // Version of MessageReader for protocol buffer messages, that parses |
| 85 // each incoming message. |
| 55 template <class T> | 86 template <class T> |
| 56 class ProtobufMessageReader { | 87 class ProtobufMessageReader { |
| 57 public: | 88 public: |
| 58 typedef typename Callback1<T*>::Type MessageReceivedCallback; | 89 typedef typename Callback2<T*, Task*>::Type MessageReceivedCallback; |
| 59 | 90 |
| 60 ProtobufMessageReader() { }; | 91 ProtobufMessageReader() { }; |
| 61 ~ProtobufMessageReader() { }; | 92 ~ProtobufMessageReader() { }; |
| 62 | 93 |
| 63 void Init(net::Socket* socket, MessageReceivedCallback* callback) { | 94 void Init(net::Socket* socket, MessageReceivedCallback* callback) { |
| 64 message_received_callback_.reset(callback); | 95 message_received_callback_.reset(callback); |
| 65 message_reader_.Init( | 96 message_reader_ = new MessageReader(); |
| 97 message_reader_->Init( |
| 66 socket, NewCallback(this, &ProtobufMessageReader<T>::OnNewData)); | 98 socket, NewCallback(this, &ProtobufMessageReader<T>::OnNewData)); |
| 67 } | 99 } |
| 68 | 100 |
| 69 private: | 101 private: |
| 70 void OnNewData(CompoundBuffer* buffer) { | 102 void OnNewData(CompoundBuffer* buffer, Task* done_task) { |
| 71 T* message = new T(); | 103 T* message = new T(); |
| 72 CompoundBufferInputStream stream(buffer); | 104 CompoundBufferInputStream stream(buffer); |
| 73 bool ret = message->ParseFromZeroCopyStream(&stream); | 105 bool ret = message->ParseFromZeroCopyStream(&stream); |
| 74 if (!ret) { | 106 if (!ret) { |
| 107 LOG(WARNING) << "Received message that is not a valid protocol buffer."; |
| 75 delete message; | 108 delete message; |
| 76 } else { | 109 } else { |
| 77 message_received_callback_->Run(message); | 110 DCHECK_EQ(stream.position(), buffer->total_bytes()); |
| 111 message_received_callback_->Run( |
| 112 message, NewRunnableFunction( |
| 113 &ProtobufMessageReader<T>::OnDone, message, done_task)); |
| 78 } | 114 } |
| 79 } | 115 } |
| 80 | 116 |
| 81 MessageReader message_reader_; | 117 static void OnDone(T* message, Task* done_task) { |
| 118 delete message; |
| 119 done_task->Run(); |
| 120 delete done_task; |
| 121 } |
| 122 |
| 123 scoped_refptr<MessageReader> message_reader_; |
| 82 scoped_ptr<MessageReceivedCallback> message_received_callback_; | 124 scoped_ptr<MessageReceivedCallback> message_received_callback_; |
| 83 }; | 125 }; |
| 84 | 126 |
| 85 } // namespace protocol | 127 } // namespace protocol |
| 86 } // namespace remoting | 128 } // namespace remoting |
| 87 | 129 |
| 88 #endif // REMOTING_PROTOCOL_MESSAGE_READER_H_ | 130 #endif // REMOTING_PROTOCOL_MESSAGE_READER_H_ |
| OLD | NEW |