1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "remoting/host/win/rdp_client.h"
10 #include "base/bind_helpers.h"
11 #include "base/logging.h"
12 #include "base/single_thread_task_runner.h"
13 #include "base/win/registry.h"
14 #include "net/base/ip_endpoint.h"
15 #include "remoting/base/typed_buffer.h"
16 #include "remoting/host/win/rdp_client_window.h"
17 #include "third_party/webrtc/modules/desktop_capture/desktop_geometry.h"
23 // 127.0.0.1 is explicitly blocked by the RDP ActiveX control, so we use
25 const unsigned char kRdpLoopbackAddress
[] = { 127, 0, 0, 2 };
27 const int kDefaultRdpPort
= 3389;
29 // The port number used by RDP is stored in the registry.
30 const wchar_t kRdpPortKeyName
[] = L
"SYSTEM\\CurrentControlSet\\Control\\"
31 L
"Terminal Server\\WinStations\\RDP-Tcp";
32 const wchar_t kRdpPortValueName
[] = L
"PortNumber";
36 // The core of RdpClient is ref-counted since it services calls and notifies
37 // events on the caller task runner, but runs the ActiveX control on the UI
40 : public base::RefCountedThreadSafe
<Core
>,
41 public RdpClientWindow::EventHandler
{
44 scoped_refptr
<base::SingleThreadTaskRunner
> caller_task_runner
,
45 scoped_refptr
<base::SingleThreadTaskRunner
> ui_task_runner
,
46 RdpClient::EventHandler
* event_handler
);
48 // Initiates a loopback RDP connection.
49 void Connect(const webrtc::DesktopSize
& screen_size
,
50 const std::string
& terminal_id
);
52 // Initiates a graceful shutdown of the RDP connection.
55 // Sends Secure Attention Sequence to the session.
58 // RdpClientWindow::EventHandler interface.
59 virtual void OnConnected() OVERRIDE
;
60 virtual void OnDisconnected() OVERRIDE
;
63 friend class base::RefCountedThreadSafe
<Core
>;
66 // Helpers for the event handler's methods that make sure that OnRdpClosed()
67 // is the last notification delivered and is delevered only once.
68 void NotifyConnected();
71 // Task runner on which the caller expects |event_handler_| to be notified.
72 scoped_refptr
<base::SingleThreadTaskRunner
> caller_task_runner_
;
74 // Task runner on which |rdp_client_window_| is running.
75 scoped_refptr
<base::SingleThreadTaskRunner
> ui_task_runner_
;
77 // Event handler receiving notification about connection state. The pointer is
78 // cleared when Disconnect() methods is called, stopping any further updates.
79 RdpClient::EventHandler
* event_handler_
;
81 // Hosts the RDP ActiveX control.
82 scoped_ptr
<RdpClientWindow
> rdp_client_window_
;
84 // A self-reference to keep the object alive during connection shutdown.
85 scoped_refptr
<Core
> self_
;
87 DISALLOW_COPY_AND_ASSIGN(Core
);
91 scoped_refptr
<base::SingleThreadTaskRunner
> caller_task_runner
,
92 scoped_refptr
<base::SingleThreadTaskRunner
> ui_task_runner
,
93 const webrtc::DesktopSize
& screen_size
,
94 const std::string
& terminal_id
,
95 EventHandler
* event_handler
) {
96 DCHECK(caller_task_runner
->BelongsToCurrentThread());
98 core_
= new Core(caller_task_runner
, ui_task_runner
, event_handler
);
99 core_
->Connect(screen_size
, terminal_id
);
102 RdpClient::~RdpClient() {
103 DCHECK(CalledOnValidThread());
108 void RdpClient::InjectSas() {
109 DCHECK(CalledOnValidThread());
114 RdpClient::Core::Core(
115 scoped_refptr
<base::SingleThreadTaskRunner
> caller_task_runner
,
116 scoped_refptr
<base::SingleThreadTaskRunner
> ui_task_runner
,
117 RdpClient::EventHandler
* event_handler
)
118 : caller_task_runner_(caller_task_runner
),
119 ui_task_runner_(ui_task_runner
),
120 event_handler_(event_handler
) {
123 void RdpClient::Core::Connect(const webrtc::DesktopSize
& screen_size
,
124 const std::string
& terminal_id
) {
125 if (!ui_task_runner_
->BelongsToCurrentThread()) {
126 ui_task_runner_
->PostTask(
127 FROM_HERE
, base::Bind(&Core::Connect
, this, screen_size
, terminal_id
));
131 DCHECK(base::MessageLoopForUI::IsCurrent());
132 DCHECK(!rdp_client_window_
);
135 // Read the port number used by RDP.
137 base::win::RegKey
key(HKEY_LOCAL_MACHINE
, kRdpPortKeyName
, KEY_READ
);
139 (key
.ReadValueDW(kRdpPortValueName
, &server_port
) != ERROR_SUCCESS
)) {
140 server_port
= kDefaultRdpPort
;
143 net::IPAddressNumber
server_address(
145 kRdpLoopbackAddress
+ arraysize(kRdpLoopbackAddress
));
146 net::IPEndPoint
server_endpoint(server_address
, server_port
);
148 // Create the ActiveX control window.
149 rdp_client_window_
.reset(new RdpClientWindow(server_endpoint
, terminal_id
,
151 if (!rdp_client_window_
->Connect(screen_size
)) {
152 rdp_client_window_
.reset();
154 // Notify the caller that connection attempt failed.
159 void RdpClient::Core::Disconnect() {
160 if (!ui_task_runner_
->BelongsToCurrentThread()) {
161 ui_task_runner_
->PostTask(FROM_HERE
, base::Bind(&Core::Disconnect
, this));
165 // The caller does not expect any notifications to be delivered after this
167 event_handler_
= NULL
;
169 // Gracefully shutdown the RDP connection.
170 if (rdp_client_window_
) {
172 rdp_client_window_
->Disconnect();
176 void RdpClient::Core::InjectSas() {
177 if (!ui_task_runner_
->BelongsToCurrentThread()) {
178 ui_task_runner_
->PostTask(FROM_HERE
, base::Bind(&Core::InjectSas
, this));
182 if (rdp_client_window_
)
183 rdp_client_window_
->InjectSas();
186 void RdpClient::Core::OnConnected() {
187 DCHECK(ui_task_runner_
->BelongsToCurrentThread());
188 DCHECK(rdp_client_window_
);
193 void RdpClient::Core::OnDisconnected() {
194 DCHECK(ui_task_runner_
->BelongsToCurrentThread());
195 DCHECK(rdp_client_window_
);
199 // Delay window destruction until no ActiveX control's code is on the stack.
200 ui_task_runner_
->DeleteSoon(FROM_HERE
, rdp_client_window_
.release());
204 RdpClient::Core::~Core() {
205 DCHECK(!event_handler_
);
206 DCHECK(!rdp_client_window_
);
209 void RdpClient::Core::NotifyConnected() {
210 if (!caller_task_runner_
->BelongsToCurrentThread()) {
211 caller_task_runner_
->PostTask(
212 FROM_HERE
, base::Bind(&Core::NotifyConnected
, this));
217 event_handler_
->OnRdpConnected();
220 void RdpClient::Core::NotifyClosed() {
221 if (!caller_task_runner_
->BelongsToCurrentThread()) {
222 caller_task_runner_
->PostTask(
223 FROM_HERE
, base::Bind(&Core::NotifyClosed
, this));
227 if (event_handler_
) {
228 RdpClient::EventHandler
* event_handler
= event_handler_
;
229 event_handler_
= NULL
;
230 event_handler
->OnRdpClosed();
234 } // namespace remoting