blob: 6ee42564f70c9bd4c132c0badeb720704f5a21c3 [file] [log] [blame]
[email protected]e265ad72012-03-16 17:28:031// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]409ac612011-11-18 04:05:572// 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
avi5a080f012015-12-22 23:15:437#include <stdint.h>
8
[email protected]c6944272012-01-06 22:12:289#include "base/bind_helpers.h"
10#include "base/callback.h"
[email protected]2e8b52c2011-11-22 00:07:1311#include "net/socket/stream_socket.h"
12#include "remoting/base/constants.h"
[email protected]409ac612011-11-18 04:05:5713#include "remoting/proto/control.pb.h"
[email protected]409ac612011-11-18 04:05:5714#include "remoting/proto/internal.pb.h"
15#include "remoting/protocol/client_stub.h"
[email protected]b76a2742014-04-10 05:38:2616#include "remoting/protocol/message_serialization.h"
[email protected]409ac612011-11-18 04:05:5717
18namespace remoting {
19namespace protocol {
20
[email protected]cb959342013-09-12 05:39:0221namespace {
22
23// 32-bit BGRA is 4 bytes per pixel.
24const int kBytesPerPixel = 4;
25
26bool CursorShapeIsValid(const CursorShapeInfo& cursor_shape) {
27 if (!cursor_shape.has_data() ||
28 !cursor_shape.has_width() ||
29 !cursor_shape.has_height() ||
30 !cursor_shape.has_hotspot_x() ||
31 !cursor_shape.has_hotspot_y()) {
32 LOG(ERROR) << "Cursor shape is missing required fields.";
33 return false;
34 }
35
36 int width = cursor_shape.width();
37 int height = cursor_shape.height();
38
39 // Verify that |width| and |height| are within sane limits. Otherwise integer
40 // overflow can occur while calculating |cursor_total_bytes| below.
kelvinpefa63e02015-01-06 23:50:0741 if (width < 0 || width > (SHRT_MAX / 2) ||
42 height < 0 || height > (SHRT_MAX / 2)) {
[email protected]cb959342013-09-12 05:39:0243 LOG(ERROR) << "Cursor dimensions are out of bounds for SetCursor: "
44 << width << "x" << height;
45 return false;
46 }
47
avi5a080f012015-12-22 23:15:4348 uint32_t cursor_total_bytes = width * height * kBytesPerPixel;
[email protected]cb959342013-09-12 05:39:0249 if (cursor_shape.data().size() < cursor_total_bytes) {
50 LOG(ERROR) << "Expected " << cursor_total_bytes << " bytes for a "
51 << width << "x" << height << " cursor. Only received "
52 << cursor_shape.data().size() << " bytes";
53 return false;
54 }
55
56 return true;
57}
58
59} // namespace
60
[email protected]409ac612011-11-18 04:05:5761ClientControlDispatcher::ClientControlDispatcher()
sergeyud8af2ca2016-01-30 03:04:3662 : ChannelDispatcherBase(kControlChannelName) {}
sergeyue61b68d2016-01-30 01:52:3663ClientControlDispatcher::~ClientControlDispatcher() {}
[email protected]409ac612011-11-18 04:05:5764
[email protected]e265ad72012-03-16 17:28:0365void ClientControlDispatcher::InjectClipboardEvent(
66 const ClipboardEvent& event) {
67 ControlMessage message;
68 message.mutable_clipboard_event()->CopyFrom(event);
sergeyu203832d2015-01-10 21:44:0969 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]e265ad72012-03-16 17:28:0370}
71
[email protected]48a8ca32013-02-13 04:31:0172void ClientControlDispatcher::NotifyClientResolution(
73 const ClientResolution& resolution) {
[email protected]f2b9cf32012-04-27 00:13:4374 ControlMessage message;
[email protected]48a8ca32013-02-13 04:31:0175 message.mutable_client_resolution()->CopyFrom(resolution);
sergeyu203832d2015-01-10 21:44:0976 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]f2b9cf32012-04-27 00:13:4377}
78
[email protected]50d71c72012-05-03 01:28:5579void ClientControlDispatcher::ControlVideo(const VideoControl& video_control) {
80 ControlMessage message;
81 message.mutable_video_control()->CopyFrom(video_control);
sergeyu203832d2015-01-10 21:44:0982 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]50d71c72012-05-03 01:28:5583}
84
[email protected]f458bed2012-10-18 03:27:5985void ClientControlDispatcher::ControlAudio(const AudioControl& audio_control) {
86 ControlMessage message;
87 message.mutable_audio_control()->CopyFrom(audio_control);
sergeyu203832d2015-01-10 21:44:0988 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]f458bed2012-10-18 03:27:5989}
90
[email protected]a5d181f2013-04-19 14:55:3791void ClientControlDispatcher::SetCapabilities(
92 const Capabilities& capabilities) {
93 ControlMessage message;
94 message.mutable_capabilities()->CopyFrom(capabilities);
sergeyu203832d2015-01-10 21:44:0995 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]a5d181f2013-04-19 14:55:3796}
97
[email protected]9ffa78a22013-05-10 04:35:1098void ClientControlDispatcher::RequestPairing(
99 const PairingRequest& pairing_request) {
100 ControlMessage message;
101 message.mutable_pairing_request()->CopyFrom(pairing_request);
sergeyu203832d2015-01-10 21:44:09102 writer()->Write(SerializeAndFrameMessage(message), base::Closure());
[email protected]9ffa78a22013-05-10 04:35:10103}
104
[email protected]09eabd65c2013-08-13 00:13:48105void ClientControlDispatcher::DeliverClientMessage(
106 const ExtensionMessage& message) {
107 ControlMessage control_message;
108 control_message.mutable_extension_message()->CopyFrom(message);
sergeyu203832d2015-01-10 21:44:09109 writer()->Write(SerializeAndFrameMessage(control_message), base::Closure());
[email protected]09eabd65c2013-08-13 00:13:48110}
111
sergeyud8af2ca2016-01-30 03:04:36112void ClientControlDispatcher::OnIncomingMessage(
113 scoped_ptr<CompoundBuffer> buffer) {
[email protected]409ac612011-11-18 04:05:57114 DCHECK(client_stub_);
[email protected]ba6d1c2d2012-03-31 01:28:38115 DCHECK(clipboard_stub_);
[email protected]ba6d1c2d2012-03-31 01:28:38116
sergeyud8af2ca2016-01-30 03:04:36117 scoped_ptr<ControlMessage> message =
118 ParseMessage<ControlMessage>(buffer.get());
119 if (!message)
120 return;
121
[email protected]ba6d1c2d2012-03-31 01:28:38122 if (message->has_clipboard_event()) {
123 clipboard_stub_->InjectClipboardEvent(message->clipboard_event());
[email protected]a5d181f2013-04-19 14:55:37124 } else if (message->has_capabilities()) {
125 client_stub_->SetCapabilities(message->capabilities());
[email protected]d65e15bd2012-06-02 22:16:41126 } else if (message->has_cursor_shape()) {
[email protected]cb959342013-09-12 05:39:02127 if (CursorShapeIsValid(message->cursor_shape()))
128 client_stub_->SetCursorShape(message->cursor_shape());
[email protected]9ffa78a22013-05-10 04:35:10129 } else if (message->has_pairing_response()) {
130 client_stub_->SetPairingResponse(message->pairing_response());
[email protected]09eabd65c2013-08-13 00:13:48131 } else if (message->has_extension_message()) {
132 client_stub_->DeliverHostMessage(message->extension_message());
[email protected]ba6d1c2d2012-03-31 01:28:38133 } else {
134 LOG(WARNING) << "Unknown control message received.";
135 }
[email protected]409ac612011-11-18 04:05:57136}
137
138} // namespace protocol
139} // namespace remoting