[email protected] | e265ad7 | 2012-03-16 17:28:03 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "remoting/protocol/client_control_dispatcher.h" |
| 6 | |
avi | 5a080f01 | 2015-12-22 23:15:43 | [diff] [blame^] | 7 | #include <stdint.h> |
| 8 | |
[email protected] | c694427 | 2012-01-06 22:12:28 | [diff] [blame] | 9 | #include "base/bind_helpers.h" |
| 10 | #include "base/callback.h" |
[email protected] | 1c232c2 | 2013-08-30 02:04:04 | [diff] [blame] | 11 | #include "base/callback_helpers.h" |
[email protected] | 2e8b52c | 2011-11-22 00:07:13 | [diff] [blame] | 12 | #include "net/socket/stream_socket.h" |
| 13 | #include "remoting/base/constants.h" |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 14 | #include "remoting/proto/control.pb.h" |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 15 | #include "remoting/proto/internal.pb.h" |
| 16 | #include "remoting/protocol/client_stub.h" |
[email protected] | b76a274 | 2014-04-10 05:38:26 | [diff] [blame] | 17 | #include "remoting/protocol/message_serialization.h" |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 18 | |
| 19 | namespace remoting { |
| 20 | namespace protocol { |
| 21 | |
[email protected] | cb95934 | 2013-09-12 05:39:02 | [diff] [blame] | 22 | namespace { |
| 23 | |
| 24 | // 32-bit BGRA is 4 bytes per pixel. |
| 25 | const int kBytesPerPixel = 4; |
| 26 | |
| 27 | bool CursorShapeIsValid(const CursorShapeInfo& cursor_shape) { |
| 28 | if (!cursor_shape.has_data() || |
| 29 | !cursor_shape.has_width() || |
| 30 | !cursor_shape.has_height() || |
| 31 | !cursor_shape.has_hotspot_x() || |
| 32 | !cursor_shape.has_hotspot_y()) { |
| 33 | LOG(ERROR) << "Cursor shape is missing required fields."; |
| 34 | return false; |
| 35 | } |
| 36 | |
| 37 | int width = cursor_shape.width(); |
| 38 | int height = cursor_shape.height(); |
| 39 | |
| 40 | // Verify that |width| and |height| are within sane limits. Otherwise integer |
| 41 | // overflow can occur while calculating |cursor_total_bytes| below. |
kelvinp | efa63e0 | 2015-01-06 23:50:07 | [diff] [blame] | 42 | if (width < 0 || width > (SHRT_MAX / 2) || |
| 43 | height < 0 || height > (SHRT_MAX / 2)) { |
[email protected] | cb95934 | 2013-09-12 05:39:02 | [diff] [blame] | 44 | LOG(ERROR) << "Cursor dimensions are out of bounds for SetCursor: " |
| 45 | << width << "x" << height; |
| 46 | return false; |
| 47 | } |
| 48 | |
avi | 5a080f01 | 2015-12-22 23:15:43 | [diff] [blame^] | 49 | uint32_t cursor_total_bytes = width * height * kBytesPerPixel; |
[email protected] | cb95934 | 2013-09-12 05:39:02 | [diff] [blame] | 50 | if (cursor_shape.data().size() < cursor_total_bytes) { |
| 51 | LOG(ERROR) << "Expected " << cursor_total_bytes << " bytes for a " |
| 52 | << width << "x" << height << " cursor. Only received " |
| 53 | << cursor_shape.data().size() << " bytes"; |
| 54 | return false; |
| 55 | } |
| 56 | |
| 57 | return true; |
| 58 | } |
| 59 | |
| 60 | } // namespace |
| 61 | |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 62 | ClientControlDispatcher::ClientControlDispatcher() |
[email protected] | 2e8b52c | 2011-11-22 00:07:13 | [diff] [blame] | 63 | : ChannelDispatcherBase(kControlChannelName), |
sergeyu | c5f104b | 2015-01-09 19:33:24 | [diff] [blame] | 64 | client_stub_(nullptr), |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 65 | clipboard_stub_(nullptr), |
| 66 | parser_(base::Bind(&ClientControlDispatcher::OnMessageReceived, |
| 67 | base::Unretained(this)), |
| 68 | reader()) { |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 69 | } |
| 70 | |
| 71 | ClientControlDispatcher::~ClientControlDispatcher() { |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 72 | } |
| 73 | |
[email protected] | e265ad7 | 2012-03-16 17:28:03 | [diff] [blame] | 74 | void ClientControlDispatcher::InjectClipboardEvent( |
| 75 | const ClipboardEvent& event) { |
| 76 | ControlMessage message; |
| 77 | message.mutable_clipboard_event()->CopyFrom(event); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 78 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | e265ad7 | 2012-03-16 17:28:03 | [diff] [blame] | 79 | } |
| 80 | |
[email protected] | 48a8ca3 | 2013-02-13 04:31:01 | [diff] [blame] | 81 | void ClientControlDispatcher::NotifyClientResolution( |
| 82 | const ClientResolution& resolution) { |
[email protected] | f2b9cf3 | 2012-04-27 00:13:43 | [diff] [blame] | 83 | ControlMessage message; |
[email protected] | 48a8ca3 | 2013-02-13 04:31:01 | [diff] [blame] | 84 | message.mutable_client_resolution()->CopyFrom(resolution); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 85 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | f2b9cf3 | 2012-04-27 00:13:43 | [diff] [blame] | 86 | } |
| 87 | |
[email protected] | 50d71c7 | 2012-05-03 01:28:55 | [diff] [blame] | 88 | void ClientControlDispatcher::ControlVideo(const VideoControl& video_control) { |
| 89 | ControlMessage message; |
| 90 | message.mutable_video_control()->CopyFrom(video_control); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 91 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | 50d71c7 | 2012-05-03 01:28:55 | [diff] [blame] | 92 | } |
| 93 | |
[email protected] | f458bed | 2012-10-18 03:27:59 | [diff] [blame] | 94 | void ClientControlDispatcher::ControlAudio(const AudioControl& audio_control) { |
| 95 | ControlMessage message; |
| 96 | message.mutable_audio_control()->CopyFrom(audio_control); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 97 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | f458bed | 2012-10-18 03:27:59 | [diff] [blame] | 98 | } |
| 99 | |
[email protected] | a5d181f | 2013-04-19 14:55:37 | [diff] [blame] | 100 | void ClientControlDispatcher::SetCapabilities( |
| 101 | const Capabilities& capabilities) { |
| 102 | ControlMessage message; |
| 103 | message.mutable_capabilities()->CopyFrom(capabilities); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 104 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | a5d181f | 2013-04-19 14:55:37 | [diff] [blame] | 105 | } |
| 106 | |
[email protected] | 9ffa78a2 | 2013-05-10 04:35:10 | [diff] [blame] | 107 | void ClientControlDispatcher::RequestPairing( |
| 108 | const PairingRequest& pairing_request) { |
| 109 | ControlMessage message; |
| 110 | message.mutable_pairing_request()->CopyFrom(pairing_request); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 111 | writer()->Write(SerializeAndFrameMessage(message), base::Closure()); |
[email protected] | 9ffa78a2 | 2013-05-10 04:35:10 | [diff] [blame] | 112 | } |
| 113 | |
[email protected] | 09eabd65c | 2013-08-13 00:13:48 | [diff] [blame] | 114 | void ClientControlDispatcher::DeliverClientMessage( |
| 115 | const ExtensionMessage& message) { |
| 116 | ControlMessage control_message; |
| 117 | control_message.mutable_extension_message()->CopyFrom(message); |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 118 | writer()->Write(SerializeAndFrameMessage(control_message), base::Closure()); |
[email protected] | 09eabd65c | 2013-08-13 00:13:48 | [diff] [blame] | 119 | } |
| 120 | |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 121 | void ClientControlDispatcher::OnMessageReceived( |
sergeyu | 203832d | 2015-01-10 21:44:09 | [diff] [blame] | 122 | scoped_ptr<ControlMessage> message, |
| 123 | const base::Closure& done_task) { |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 124 | DCHECK(client_stub_); |
[email protected] | ba6d1c2d | 2012-03-31 01:28:38 | [diff] [blame] | 125 | DCHECK(clipboard_stub_); |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 126 | base::ScopedClosureRunner done_runner(done_task); |
[email protected] | ba6d1c2d | 2012-03-31 01:28:38 | [diff] [blame] | 127 | |
| 128 | if (message->has_clipboard_event()) { |
| 129 | clipboard_stub_->InjectClipboardEvent(message->clipboard_event()); |
[email protected] | a5d181f | 2013-04-19 14:55:37 | [diff] [blame] | 130 | } else if (message->has_capabilities()) { |
| 131 | client_stub_->SetCapabilities(message->capabilities()); |
[email protected] | d65e15bd | 2012-06-02 22:16:41 | [diff] [blame] | 132 | } else if (message->has_cursor_shape()) { |
[email protected] | cb95934 | 2013-09-12 05:39:02 | [diff] [blame] | 133 | if (CursorShapeIsValid(message->cursor_shape())) |
| 134 | client_stub_->SetCursorShape(message->cursor_shape()); |
[email protected] | 9ffa78a2 | 2013-05-10 04:35:10 | [diff] [blame] | 135 | } else if (message->has_pairing_response()) { |
| 136 | client_stub_->SetPairingResponse(message->pairing_response()); |
[email protected] | 09eabd65c | 2013-08-13 00:13:48 | [diff] [blame] | 137 | } else if (message->has_extension_message()) { |
| 138 | client_stub_->DeliverHostMessage(message->extension_message()); |
[email protected] | ba6d1c2d | 2012-03-31 01:28:38 | [diff] [blame] | 139 | } else { |
| 140 | LOG(WARNING) << "Unknown control message received."; |
| 141 | } |
[email protected] | 409ac61 | 2011-11-18 04:05:57 | [diff] [blame] | 142 | } |
| 143 | |
| 144 | } // namespace protocol |
| 145 | } // namespace remoting |