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_window.h"
11 #include "base/lazy_instance.h"
12 #include "base/logging.h"
13 #include "base/strings/string16.h"
14 #include "base/strings/utf_string_conversions.h"
15 #include "base/threading/thread_local.h"
16 #include "base/win/scoped_bstr.h"
22 // RDP connection disconnect reasons codes that should not be interpreted as
24 const long kDisconnectReasonNoInfo
= 0;
25 const long kDisconnectReasonLocalNotError
= 1;
26 const long kDisconnectReasonRemoteByUser
= 2;
27 const long kDisconnectReasonByServer
= 3;
29 // Maximum length of a window class name including the terminating NULL.
30 const int kMaxWindowClassLength
= 256;
32 // Each member of the array returned by GetKeyboardState() contains status data
33 // for a virtual key. If the high-order bit is 1, the key is down; otherwise, it
35 const BYTE kKeyPressedFlag
= 0x80;
37 const int kKeyboardStateLength
= 256;
39 // The RDP control creates 'IHWindowClass' window to handle keyboard input.
40 const wchar_t kRdpInputWindowClass
[] = L
"IHWindowClass";
43 // Redirect sounds to the client. This is the default value.
44 kRdpAudioModeRedirect
= 0,
46 // Play sounds at the remote computer. Equivalent to |kRdpAudioModeNone| if
47 // the remote computer is running a server SKU.
48 kRdpAudioModePlayOnServer
= 1,
50 // Disable sound redirection; do not play sounds at the remote computer.
54 // Points to a per-thread instance of the window activation hook handle.
55 base::LazyInstance
<base::ThreadLocalPointer
<RdpClientWindow::WindowHook
> >
56 g_window_hook
= LAZY_INSTANCE_INITIALIZER
;
58 // Finds a child window with the class name matching |class_name|. Unlike
59 // FindWindowEx() this function walks the tree of windows recursively. The walk
60 // is done in breadth-first order. The function returns NULL if the child window
61 // could not be found.
62 HWND
FindWindowRecursively(HWND parent
, const base::string16
& class_name
) {
63 std::list
<HWND
> windows
;
64 windows
.push_back(parent
);
66 while (!windows
.empty()) {
67 HWND child
= FindWindowEx(windows
.front(), NULL
, NULL
, NULL
);
68 while (child
!= NULL
) {
69 // See if the window class name matches |class_name|.
70 WCHAR name
[kMaxWindowClassLength
];
71 int length
= GetClassName(child
, name
, arraysize(name
));
72 if (base::string16(name
, length
) == class_name
)
75 // Remember the window to look through its children.
76 windows
.push_back(child
);
78 // Go to the next child.
79 child
= FindWindowEx(windows
.front(), child
, NULL
, NULL
);
90 // Used to close any windows activated on a particular thread. It installs
91 // a WH_CBT window hook to track window activations and close all activated
92 // windows. There should be only one instance of |WindowHook| per thread
93 // at any given moment.
94 class RdpClientWindow::WindowHook
95 : public base::RefCounted
<WindowHook
> {
97 static scoped_refptr
<WindowHook
> Create();
100 friend class base::RefCounted
<WindowHook
>;
103 virtual ~WindowHook();
105 static LRESULT CALLBACK
CloseWindowOnActivation(
106 int code
, WPARAM wparam
, LPARAM lparam
);
110 DISALLOW_COPY_AND_ASSIGN(WindowHook
);
113 RdpClientWindow::RdpClientWindow(const net::IPEndPoint
& server_endpoint
,
114 const std::string
& terminal_id
,
115 EventHandler
* event_handler
)
116 : event_handler_(event_handler
),
117 server_endpoint_(server_endpoint
),
118 terminal_id_(terminal_id
) {
121 RdpClientWindow::~RdpClientWindow() {
126 DCHECK(!client_settings_
);
129 bool RdpClientWindow::Connect(const webrtc::DesktopSize
& screen_size
) {
132 screen_size_
= screen_size
;
133 RECT rect
= { 0, 0, screen_size_
.width(), screen_size_
.height() };
134 bool result
= Create(NULL
, rect
, NULL
) != NULL
;
136 // Hide the window since this class is about establishing a connection, not
137 // about showing a UI to the user.
144 void RdpClientWindow::Disconnect() {
146 SendMessage(WM_CLOSE
);
149 void RdpClientWindow::InjectSas() {
153 // Fins the window handling the keyboard input.
154 HWND input_window
= FindWindowRecursively(m_hWnd
, kRdpInputWindowClass
);
156 LOG(ERROR
) << "Failed to find the window handling the keyboard input.";
160 VLOG(3) << "Injecting Ctrl+Alt+End to emulate SAS.";
162 BYTE keyboard_state
[kKeyboardStateLength
];
163 if (!GetKeyboardState(keyboard_state
)) {
164 PLOG(ERROR
) << "Failed to get the keyboard state.";
168 // This code is running in Session 0, so we expect no keys to be pressed.
169 DCHECK(!(keyboard_state
[VK_CONTROL
] & kKeyPressedFlag
));
170 DCHECK(!(keyboard_state
[VK_MENU
] & kKeyPressedFlag
));
171 DCHECK(!(keyboard_state
[VK_END
] & kKeyPressedFlag
));
173 // Map virtual key codes to scan codes.
174 UINT control
= MapVirtualKey(VK_CONTROL
, MAPVK_VK_TO_VSC
);
175 UINT alt
= MapVirtualKey(VK_MENU
, MAPVK_VK_TO_VSC
);
176 UINT end
= MapVirtualKey(VK_END
, MAPVK_VK_TO_VSC
) | KF_EXTENDED
;
177 UINT up
= KF_UP
| KF_REPEAT
;
180 keyboard_state
[VK_CONTROL
] |= kKeyPressedFlag
;
181 keyboard_state
[VK_LCONTROL
] |= kKeyPressedFlag
;
182 CHECK(SetKeyboardState(keyboard_state
));
183 SendMessage(input_window
, WM_KEYDOWN
, VK_CONTROL
, MAKELPARAM(1, control
));
186 keyboard_state
[VK_MENU
] |= kKeyPressedFlag
;
187 keyboard_state
[VK_LMENU
] |= kKeyPressedFlag
;
188 CHECK(SetKeyboardState(keyboard_state
));
189 SendMessage(input_window
, WM_KEYDOWN
, VK_MENU
,
190 MAKELPARAM(1, alt
| KF_ALTDOWN
));
192 // Press and release 'End'.
193 SendMessage(input_window
, WM_KEYDOWN
, VK_END
,
194 MAKELPARAM(1, end
| KF_ALTDOWN
));
195 SendMessage(input_window
, WM_KEYUP
, VK_END
,
196 MAKELPARAM(1, end
| up
| KF_ALTDOWN
));
199 keyboard_state
[VK_MENU
] &= ~kKeyPressedFlag
;
200 keyboard_state
[VK_LMENU
] &= ~kKeyPressedFlag
;
201 CHECK(SetKeyboardState(keyboard_state
));
202 SendMessage(input_window
, WM_KEYUP
, VK_MENU
, MAKELPARAM(1, alt
| up
));
205 keyboard_state
[VK_CONTROL
] &= ~kKeyPressedFlag
;
206 keyboard_state
[VK_LCONTROL
] &= ~kKeyPressedFlag
;
207 CHECK(SetKeyboardState(keyboard_state
));
208 SendMessage(input_window
, WM_KEYUP
, VK_CONTROL
, MAKELPARAM(1, control
| up
));
211 void RdpClientWindow::OnClose() {
213 NotifyDisconnected();
217 // Request a graceful shutdown.
218 mstsc::ControlCloseStatus close_status
;
219 HRESULT result
= client_
->RequestClose(&close_status
);
220 if (FAILED(result
)) {
221 LOG(ERROR
) << "Failed to request a graceful shutdown of an RDP connection"
222 << ", result=0x" << std::hex
<< result
<< std::dec
;
223 NotifyDisconnected();
227 if (close_status
!= mstsc::controlCloseWaitForEvents
) {
228 NotifyDisconnected();
232 // Expect IMsTscAxEvents::OnConfirmClose() or IMsTscAxEvents::OnDisconnect()
233 // to be called if mstsc::controlCloseWaitForEvents was returned.
236 LRESULT
RdpClientWindow::OnCreate(CREATESTRUCT
* create_struct
) {
237 CAxWindow2 activex_window
;
238 base::win::ScopedComPtr
<IUnknown
> control
;
239 HRESULT result
= E_FAIL
;
240 base::win::ScopedComPtr
<mstsc::IMsTscSecuredSettings
> secured_settings
;
241 base::win::ScopedComPtr
<mstsc::IMsRdpClientSecuredSettings
> secured_settings2
;
242 base::win::ScopedBstr
server_name(
243 base::UTF8ToUTF16(server_endpoint_
.ToStringWithoutPort()).c_str());
244 base::win::ScopedBstr
terminal_id(base::UTF8ToUTF16(terminal_id_
).c_str());
246 // Create the child window that actually hosts the ActiveX control.
247 RECT rect
= { 0, 0, screen_size_
.width(), screen_size_
.height() };
248 activex_window
.Create(m_hWnd
, rect
, NULL
, WS_CHILD
| WS_VISIBLE
| WS_BORDER
);
249 if (activex_window
.m_hWnd
== NULL
) {
250 result
= HRESULT_FROM_WIN32(GetLastError());
254 // Instantiate the RDP ActiveX control.
255 result
= activex_window
.CreateControlEx(
256 OLESTR("MsTscAx.MsTscAx"),
260 __uuidof(mstsc::IMsTscAxEvents
),
261 reinterpret_cast<IUnknown
*>(static_cast<RdpEventsSink
*>(this)));
265 result
= control
.QueryInterface(client_
.Receive());
270 result
= client_
->put_ColorDepth(32);
274 // Set dimensions of the remote desktop.
275 result
= client_
->put_DesktopWidth(screen_size_
.width());
278 result
= client_
->put_DesktopHeight(screen_size_
.height());
282 // Set the server name to connect to.
283 result
= client_
->put_Server(server_name
);
287 // Fetch IMsRdpClientAdvancedSettings interface for the client.
288 result
= client_
->get_AdvancedSettings2(client_settings_
.Receive());
292 // Disable background input mode.
293 result
= client_settings_
->put_allowBackgroundInput(0);
297 // Do not use bitmap cache.
298 result
= client_settings_
->put_BitmapPersistence(0);
299 if (SUCCEEDED(result
))
300 result
= client_settings_
->put_CachePersistenceActive(0);
304 // Do not use compression.
305 result
= client_settings_
->put_Compress(0);
309 // Enable the Ctrl+Alt+Del screen.
310 result
= client_settings_
->put_DisableCtrlAltDel(0);
314 // Disable printer and clipboard redirection.
315 result
= client_settings_
->put_DisableRdpdr(FALSE
);
319 // Do not display the connection bar.
320 result
= client_settings_
->put_DisplayConnectionBar(VARIANT_FALSE
);
324 // Do not grab focus on connect.
325 result
= client_settings_
->put_GrabFocusOnConnect(VARIANT_FALSE
);
329 // Enable enhanced graphics, font smoothing and desktop composition.
330 const LONG kDesiredFlags
= WTS_PERF_ENABLE_ENHANCED_GRAPHICS
|
331 WTS_PERF_ENABLE_FONT_SMOOTHING
|
332 WTS_PERF_ENABLE_DESKTOP_COMPOSITION
;
333 result
= client_settings_
->put_PerformanceFlags(kDesiredFlags
);
337 // Set the port to connect to.
338 result
= client_settings_
->put_RDPPort(server_endpoint_
.port());
342 // Disable audio in the session.
343 // TODO(alexeypa): re-enable audio redirection when http://crbug.com/242312 is
345 result
= client_
->get_SecuredSettings2(secured_settings2
.Receive());
346 if (SUCCEEDED(result
)) {
347 result
= secured_settings2
->put_AudioRedirectionMode(kRdpAudioModeNone
);
352 result
= client_
->get_SecuredSettings(secured_settings
.Receive());
356 // Set the terminal ID as the working directory for the initial program. It is
357 // observed that |WorkDir| is used only if an initial program is also
358 // specified, but is still passed to the RDP server and can then be read back
359 // from the session parameters. This makes it possible to use |WorkDir| to
360 // match the RDP connection with the session it is attached to.
362 // This code should be in sync with WtsTerminalMonitor::LookupTerminalId().
363 result
= secured_settings
->put_WorkDir(terminal_id
);
367 result
= client_
->Connect();
372 if (FAILED(result
)) {
373 LOG(ERROR
) << "RDP: failed to initiate a connection to "
374 << server_endpoint_
.ToString() << ": error="
375 << std::hex
<< result
<< std::dec
;
377 client_settings_
.Release();
384 void RdpClientWindow::OnDestroy() {
386 client_settings_
.Release();
389 HRESULT
RdpClientWindow::OnAuthenticationWarningDisplayed() {
390 LOG(WARNING
) << "RDP: authentication warning is about to be shown.";
392 // Hook window activation to cancel any modal UI shown by the RDP control.
393 // This does not affect creation of other instances of the RDP control on this
394 // thread because the RDP control's window is hidden and is not activated.
395 window_activate_hook_
= WindowHook::Create();
399 HRESULT
RdpClientWindow::OnAuthenticationWarningDismissed() {
400 LOG(WARNING
) << "RDP: authentication warning has been dismissed.";
402 window_activate_hook_
= NULL
;
406 HRESULT
RdpClientWindow::OnConnected() {
407 VLOG(1) << "RDP: successfully connected to " << server_endpoint_
.ToString();
413 HRESULT
RdpClientWindow::OnDisconnected(long reason
) {
414 if (reason
== kDisconnectReasonNoInfo
||
415 reason
== kDisconnectReasonLocalNotError
||
416 reason
== kDisconnectReasonRemoteByUser
||
417 reason
== kDisconnectReasonByServer
) {
418 VLOG(1) << "RDP: disconnected from " << server_endpoint_
.ToString()
419 << ", reason=" << reason
;
420 NotifyDisconnected();
424 // Get the extended disconnect reason code.
425 mstsc::ExtendedDisconnectReasonCode extended_code
;
426 HRESULT result
= client_
->get_ExtendedDisconnectReason(&extended_code
);
428 extended_code
= mstsc::exDiscReasonNoInfo
;
430 // Get the error message as well.
431 base::win::ScopedBstr error_message
;
432 base::win::ScopedComPtr
<mstsc::IMsRdpClient5
> client5
;
433 result
= client_
.QueryInterface(client5
.Receive());
434 if (SUCCEEDED(result
)) {
435 result
= client5
->GetErrorDescription(reason
, extended_code
,
436 error_message
.Receive());
438 error_message
.Reset();
441 LOG(ERROR
) << "RDP: disconnected from " << server_endpoint_
.ToString()
442 << ": " << error_message
<< " (reason=" << reason
443 << ", extended_code=" << extended_code
<< ")";
445 NotifyDisconnected();
449 HRESULT
RdpClientWindow::OnFatalError(long error_code
) {
450 LOG(ERROR
) << "RDP: an error occured: error_code="
453 NotifyDisconnected();
457 HRESULT
RdpClientWindow::OnConfirmClose(VARIANT_BOOL
* allow_close
) {
458 *allow_close
= VARIANT_TRUE
;
460 NotifyDisconnected();
464 void RdpClientWindow::NotifyConnected() {
466 event_handler_
->OnConnected();
469 void RdpClientWindow::NotifyDisconnected() {
470 if (event_handler_
) {
471 EventHandler
* event_handler
= event_handler_
;
472 event_handler_
= NULL
;
473 event_handler
->OnDisconnected();
477 scoped_refptr
<RdpClientWindow::WindowHook
>
478 RdpClientWindow::WindowHook::Create() {
479 scoped_refptr
<WindowHook
> window_hook
= g_window_hook
.Pointer()->Get();
482 window_hook
= new WindowHook();
487 RdpClientWindow::WindowHook::WindowHook() : hook_(NULL
) {
488 DCHECK(!g_window_hook
.Pointer()->Get());
490 // Install a window hook to be called on window activation.
491 hook_
= SetWindowsHookEx(WH_CBT
,
492 &WindowHook::CloseWindowOnActivation
,
494 GetCurrentThreadId());
495 // Without the hook installed, RdpClientWindow will not be able to cancel
496 // modal UI windows. This will block the UI message loop so it is better to
497 // terminate the process now.
500 // Let CloseWindowOnActivation() to access the hook handle.
501 g_window_hook
.Pointer()->Set(this);
504 RdpClientWindow::WindowHook::~WindowHook() {
505 DCHECK(g_window_hook
.Pointer()->Get() == this);
507 g_window_hook
.Pointer()->Set(NULL
);
509 BOOL result
= UnhookWindowsHookEx(hook_
);
514 LRESULT CALLBACK
RdpClientWindow::WindowHook::CloseWindowOnActivation(
515 int code
, WPARAM wparam
, LPARAM lparam
) {
516 // Get the hook handle.
517 HHOOK hook
= g_window_hook
.Pointer()->Get()->hook_
;
519 if (code
!= HCBT_ACTIVATE
)
520 return CallNextHookEx(hook
, code
, wparam
, lparam
);
522 // Close the window once all pending window messages are processed.
523 HWND window
= reinterpret_cast<HWND
>(wparam
);
524 LOG(WARNING
) << "RDP: closing a window: " << std::hex
<< window
<< std::dec
;
525 ::PostMessage(window
, WM_CLOSE
, 0, 0);
529 } // namespace remoting