VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxSeamless.cpp@ 40866

Last change on this file since 40866 was 35863, checked in by vboxsync, 14 years ago

VBoxTray: New startup code, more detailed VBoxTray status-to-host reporting, cleaned up error handling, some renaming.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 15.0 KB
Line 
1/* $Id: VBoxSeamless.cpp 35863 2011-02-07 10:59:08Z vboxsync $ */
2/** @file
3 * VBoxSeamless - Seamless windows
4 */
5
6/*
7 * Copyright (C) 2006-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#define _WIN32_WINNT 0x0500
18#include <windows.h>
19#include "VBoxTray.h"
20#include "VBoxHelpers.h"
21#include "VBoxSeamless.h"
22#include <VBoxHook.h>
23#include <VBoxDisplay.h>
24#include <VBox/VMMDev.h>
25#include <iprt/assert.h>
26#include <VBoxGuestInternal.h>
27
28typedef struct _VBOXSEAMLESSCONTEXT
29{
30 const VBOXSERVICEENV *pEnv;
31
32 HMODULE hModule;
33
34 BOOL (* pfnVBoxInstallHook)(HMODULE hDll);
35 BOOL (* pfnVBoxRemoveHook)();
36
37 PVBOXDISPIFESCAPE lpEscapeData;
38} VBOXSEAMLESSCONTEXT;
39
40typedef struct
41{
42 HDC hdc;
43 HRGN hrgn;
44} VBOX_ENUM_PARAM, *PVBOX_ENUM_PARAM;
45
46static VBOXSEAMLESSCONTEXT gCtx = {0};
47
48void VBoxLogString(HANDLE hDriver, char *pszStr);
49
50int VBoxSeamlessInit(const VBOXSERVICEENV *pEnv, void **ppInstance, bool *pfStartThread)
51{
52 Log(("VBoxTray: VBoxSeamlessInit\n"));
53
54 *pfStartThread = false;
55 gCtx.pEnv = pEnv;
56
57 OSVERSIONINFO OSinfo;
58 OSinfo.dwOSVersionInfoSize = sizeof (OSinfo);
59 GetVersionEx (&OSinfo);
60
61 int rc = VINF_SUCCESS;
62
63 /* We have to jump out here when using NT4, otherwise it complains about
64 a missing API function "UnhookWinEvent" used by the dynamically loaded VBoxHook.dll below */
65 if (OSinfo.dwMajorVersion <= 4) /* Windows NT 4.0 or older */
66 {
67 Log(("VBoxTray: VBoxSeamlessInit: Windows NT 4.0 or older not supported!\n"));
68 rc = VERR_NOT_SUPPORTED;
69 }
70 else
71 {
72 /* Will fail if SetWinEventHook is not present (version < NT4 SP6 apparently) */
73 gCtx.hModule = LoadLibrary(VBOXHOOK_DLL_NAME);
74 if (gCtx.hModule)
75 {
76 *(uintptr_t *)&gCtx.pfnVBoxInstallHook = (uintptr_t)GetProcAddress(gCtx.hModule, "VBoxInstallHook");
77 *(uintptr_t *)&gCtx.pfnVBoxRemoveHook = (uintptr_t)GetProcAddress(gCtx.hModule, "VBoxRemoveHook");
78
79 /* Inform the host that we support the seamless window mode. */
80 rc = VbglR3SetGuestCaps(VMMDEV_GUEST_SUPPORTS_SEAMLESS, 0);
81 if (RT_SUCCESS(rc))
82 {
83 *pfStartThread = true;
84 *ppInstance = &gCtx;
85 }
86 else
87 Log(("VBoxTray: VBoxSeamlessInit: Failed to set seamless capability\n"));
88 }
89 else
90 {
91 rc = RTErrConvertFromWin32(GetLastError());
92 Log(("VBoxTray: VBoxSeamlessInit: LoadLibrary of \"%s\" failed with rc=%Rrc\n", VBOXHOOK_DLL_NAME, rc));
93 }
94 }
95
96 return rc;
97}
98
99
100void VBoxSeamlessDestroy(const VBOXSERVICEENV *pEnv, void *pInstance)
101{
102 Log(("VBoxTray: VBoxSeamlessDestroy\n"));
103
104 /* Inform the host that we no longer support the seamless window mode. */
105 int rc = VbglR3SetGuestCaps(0, VMMDEV_GUEST_SUPPORTS_SEAMLESS);
106 if (RT_FAILURE(rc))
107 Log(("VBoxTray: VBoxSeamlessDestroy: Failed to unset seamless capability, rc=%Rrc\n", rc));
108
109 if (gCtx.pfnVBoxRemoveHook)
110 gCtx.pfnVBoxRemoveHook();
111 if (gCtx.hModule)
112 FreeLibrary(gCtx.hModule);
113 gCtx.hModule = 0;
114 return;
115}
116
117void VBoxSeamlessInstallHook()
118{
119 if (gCtx.pfnVBoxInstallHook)
120 {
121 /* Check current visible region state */
122 VBoxSeamlessCheckWindows();
123
124 gCtx.pfnVBoxInstallHook(gCtx.hModule);
125 }
126}
127
128void VBoxSeamlessRemoveHook()
129{
130 if (gCtx.pfnVBoxRemoveHook)
131 gCtx.pfnVBoxRemoveHook();
132
133 if (gCtx.lpEscapeData)
134 {
135 free(gCtx.lpEscapeData);
136 gCtx.lpEscapeData = NULL;
137 }
138}
139
140BOOL CALLBACK VBoxEnumFunc(HWND hwnd, LPARAM lParam)
141{
142 PVBOX_ENUM_PARAM lpParam = (PVBOX_ENUM_PARAM)lParam;
143 DWORD dwStyle, dwExStyle;
144 RECT rectWindow, rectVisible;
145
146 dwStyle = GetWindowLong(hwnd, GWL_STYLE);
147 dwExStyle = GetWindowLong(hwnd, GWL_EXSTYLE);
148 if ( !(dwStyle & WS_VISIBLE)
149 || (dwStyle & WS_CHILD))
150 return TRUE;
151
152 Log(("VBoxTray: VBoxEnumFunc %x\n", hwnd));
153 /* Only visible windows that are present on the desktop are interesting here */
154 if (GetWindowRect(hwnd, &rectWindow))
155 {
156 rectVisible = rectWindow;
157
158 char szWindowText[256];
159 szWindowText[0] = 0;
160 GetWindowText(hwnd, szWindowText, sizeof(szWindowText));
161
162 /* Filter out Windows XP shadow windows */
163 /** @todo still shows inside the guest */
164 if ( szWindowText[0] == 0
165 && dwStyle == (WS_POPUP|WS_VISIBLE|WS_CLIPSIBLINGS)
166 && dwExStyle == (WS_EX_LAYERED|WS_EX_TOOLWINDOW|WS_EX_TRANSPARENT|WS_EX_TOPMOST))
167 {
168 Log(("VBoxTray: Filter out shadow window style=%x exstyle=%x\n", dwStyle, dwExStyle));
169 return TRUE;
170 }
171
172 /** @todo will this suffice? The Program Manager window covers the whole screen */
173 if (strcmp(szWindowText, "Program Manager"))
174 {
175 Log(("VBoxTray: Enum hwnd=%x rect (%d,%d) (%d,%d)\n", hwnd, rectWindow.left, rectWindow.top, rectWindow.right, rectWindow.bottom));
176 Log(("VBoxTray: title=%s style=%x exStyle=%x\n", szWindowText, dwStyle, dwExStyle));
177
178 HRGN hrgn = CreateRectRgn(0,0,0,0);
179
180 int ret = GetWindowRgn(hwnd, hrgn);
181
182 if (ret == ERROR)
183 {
184 Log(("VBoxTray: GetWindowRgn failed with rc=%d\n", GetLastError()));
185 SetRectRgn(hrgn, rectVisible.left, rectVisible.top, rectVisible.right, rectVisible.bottom);
186 }
187 else
188 {
189 /* this region is relative to the window origin instead of the desktop origin */
190 OffsetRgn(hrgn, rectWindow.left, rectWindow.top);
191 }
192 if (lpParam->hrgn)
193 {
194 /* create a union of the current visible region and the visible rectangle of this window. */
195 CombineRgn(lpParam->hrgn, lpParam->hrgn, hrgn, RGN_OR);
196 DeleteObject(hrgn);
197 }
198 else
199 lpParam->hrgn = hrgn;
200 }
201 else
202 {
203 Log(("VBoxTray: Enum hwnd=%x rect (%d,%d) (%d,%d) (ignored)\n", hwnd, rectWindow.left, rectWindow.top, rectWindow.right, rectWindow.bottom));
204 Log(("VBoxTray: title=%s style=%x\n", szWindowText, dwStyle));
205 }
206 }
207 return TRUE; /* continue enumeration */
208}
209
210void VBoxSeamlessCheckWindows()
211{
212 VBOX_ENUM_PARAM param;
213
214 param.hdc = GetDC(HWND_DESKTOP);
215 param.hrgn = 0;
216
217 EnumWindows(VBoxEnumFunc, (LPARAM)&param);
218
219 if (param.hrgn)
220 {
221 DWORD cbSize;
222
223 cbSize = GetRegionData(param.hrgn, 0, NULL);
224 if (cbSize)
225 {
226 PVBOXDISPIFESCAPE lpEscapeData = (PVBOXDISPIFESCAPE)malloc(VBOXDISPIFESCAPE_SIZE(cbSize));
227 if (lpEscapeData)
228 {
229 lpEscapeData->escapeCode = VBOXESC_SETVISIBLEREGION;
230 LPRGNDATA lpRgnData = VBOXDISPIFESCAPE_DATA(lpEscapeData, RGNDATA);
231 memset(lpRgnData, 0, cbSize);
232 cbSize = GetRegionData(param.hrgn, cbSize, lpRgnData);
233 if (cbSize)
234 {
235#ifdef DEBUG
236 RECT *lpRect = (RECT *)&lpRgnData->Buffer[0];
237 Log(("VBoxTray: New visible region: \n"));
238
239 for (DWORD i=0;i<lpRgnData->rdh.nCount;i++)
240 {
241 Log(("VBoxTray: visible rect (%d,%d)(%d,%d)\n", lpRect[i].left, lpRect[i].top, lpRect[i].right, lpRect[i].bottom));
242 }
243#endif
244 LPRGNDATA lpCtxRgnData = VBOXDISPIFESCAPE_DATA(gCtx.lpEscapeData, RGNDATA);
245 if ( !gCtx.lpEscapeData
246 || (lpCtxRgnData->rdh.dwSize + lpCtxRgnData->rdh.nRgnSize != cbSize)
247 || memcmp(lpCtxRgnData, lpRgnData, cbSize))
248 {
249 /* send to display driver */
250 VBoxDispIfEscape(&gCtx.pEnv->dispIf, lpEscapeData, cbSize);
251
252 if (gCtx.lpEscapeData)
253 free(gCtx.lpEscapeData);
254 gCtx.lpEscapeData = lpEscapeData;
255 }
256 else
257 Log(("VBoxTray: Visible rectangles haven't changed; ignore\n"));
258 }
259 if (lpEscapeData != gCtx.lpEscapeData)
260 free(lpEscapeData);
261 }
262 }
263
264 DeleteObject(param.hrgn);
265 }
266
267 ReleaseDC(HWND_DESKTOP, param.hdc);
268}
269
270/**
271 * Thread function to wait for and process seamless mode change
272 * requests
273 */
274unsigned __stdcall VBoxSeamlessThread(void *pInstance)
275{
276 VBOXSEAMLESSCONTEXT *pCtx = (VBOXSEAMLESSCONTEXT *)pInstance;
277 HANDLE gVBoxDriver = pCtx->pEnv->hDriver;
278 bool fTerminate = false;
279 VBoxGuestFilterMaskInfo maskInfo;
280 DWORD cbReturned;
281 BOOL fWasScreenSaverActive = FALSE, ret;
282
283 maskInfo.u32OrMask = VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST;
284 maskInfo.u32NotMask = 0;
285 if (DeviceIoControl (gVBoxDriver, VBOXGUEST_IOCTL_CTL_FILTER_MASK, &maskInfo, sizeof (maskInfo), NULL, 0, &cbReturned, NULL))
286 {
287 Log(("VBoxTray: VBoxSeamlessThread: DeviceIOControl(CtlMask - or) succeeded\n"));
288 }
289 else
290 {
291 Log(("VBoxTray: VBoxSeamlessThread: DeviceIOControl(CtlMask) failed, SeamlessChangeThread exited\n"));
292 return 0;
293 }
294
295 do
296 {
297 /* wait for a seamless change event */
298 VBoxGuestWaitEventInfo waitEvent;
299 waitEvent.u32TimeoutIn = 5000;
300 waitEvent.u32EventMaskIn = VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST;
301 if (DeviceIoControl(gVBoxDriver, VBOXGUEST_IOCTL_WAITEVENT, &waitEvent, sizeof(waitEvent), &waitEvent, sizeof(waitEvent), &cbReturned, NULL))
302 {
303 Log(("VBoxTray: VBoxSeamlessThread: DeviceIOControl succeeded\n"));
304
305 /* are we supposed to stop? */
306 if (WaitForSingleObject(pCtx->pEnv->hStopEvent, 0) == WAIT_OBJECT_0)
307 break;
308
309 Log(("VBoxTray: VBoxSeamlessThread: checking event\n"));
310
311 /* did we get the right event? */
312 if (waitEvent.u32EventFlagsOut & VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST)
313 {
314 Log(("VBoxTray: VBoxTray: going to get seamless change information\n"));
315
316 /* We got at least one event. Read the requested resolution
317 * and try to set it until success. New events will not be seen
318 * but a new resolution will be read in this poll loop.
319 */
320 for (;;)
321 {
322 /* get the seamless change request */
323 VMMDevSeamlessChangeRequest seamlessChangeRequest = {0};
324 vmmdevInitRequest((VMMDevRequestHeader*)&seamlessChangeRequest, VMMDevReq_GetSeamlessChangeRequest);
325 seamlessChangeRequest.eventAck = VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST;
326
327 BOOL fSeamlessChangeQueried = DeviceIoControl(gVBoxDriver, VBOXGUEST_IOCTL_VMMREQUEST(sizeof(seamlessChangeRequest)), &seamlessChangeRequest, sizeof(seamlessChangeRequest),
328 &seamlessChangeRequest, sizeof(seamlessChangeRequest), &cbReturned, NULL);
329 if (fSeamlessChangeQueried)
330 {
331 Log(("VBoxTray: VBoxSeamlessThread: mode change to %d\n", seamlessChangeRequest.mode));
332
333 switch(seamlessChangeRequest.mode)
334 {
335 case VMMDev_Seamless_Disabled:
336 if (fWasScreenSaverActive)
337 {
338 Log(("VBoxTray: Re-enabling the screensaver\n"));
339 ret = SystemParametersInfo(SPI_SETSCREENSAVEACTIVE, TRUE, NULL, 0);
340 if (!ret)
341 Log(("VBoxTray: SystemParametersInfo SPI_SETSCREENSAVEACTIVE failed with %d\n", GetLastError()));
342 }
343 PostMessage(ghwndToolWindow, WM_VBOX_REMOVE_SEAMLESS_HOOK, 0, 0);
344 break;
345
346 case VMMDev_Seamless_Visible_Region:
347 ret = SystemParametersInfo(SPI_GETSCREENSAVEACTIVE, 0, &fWasScreenSaverActive, 0);
348 if (!ret)
349 Log(("VBoxTray: SystemParametersInfo SPI_GETSCREENSAVEACTIVE failed with %d\n", GetLastError()));
350
351 if (fWasScreenSaverActive)
352 Log(("VBoxTray: Disabling the screensaver\n"));
353
354 ret = SystemParametersInfo(SPI_SETSCREENSAVEACTIVE, FALSE, NULL, 0);
355 if (!ret)
356 Log(("VBoxTray: SystemParametersInfo SPI_SETSCREENSAVEACTIVE failed with %d\n", GetLastError()));
357 PostMessage(ghwndToolWindow, WM_VBOX_INSTALL_SEAMLESS_HOOK, 0, 0);
358 break;
359
360 case VMMDev_Seamless_Host_Window:
361 break;
362
363 default:
364 AssertFailed();
365 break;
366 }
367 break;
368 }
369 else
370 {
371 Log(("VBoxTray: VBoxSeamlessThread: error from DeviceIoControl VBOXGUEST_IOCTL_VMMREQUEST\n"));
372 }
373 /* sleep a bit to not eat too much CPU while retrying */
374 /* are we supposed to stop? */
375 if (WaitForSingleObject(pCtx->pEnv->hStopEvent, 50) == WAIT_OBJECT_0)
376 {
377 fTerminate = true;
378 break;
379 }
380 }
381 }
382 }
383 else
384 {
385 Log(("VBoxTray: VBoxTray: error 0 from DeviceIoControl VBOXGUEST_IOCTL_WAITEVENT\n"));
386 /* sleep a bit to not eat too much CPU in case the above call always fails */
387 if (WaitForSingleObject(pCtx->pEnv->hStopEvent, 10) == WAIT_OBJECT_0)
388 {
389 fTerminate = true;
390 break;
391 }
392 }
393 }
394 while (!fTerminate);
395
396 maskInfo.u32OrMask = 0;
397 maskInfo.u32NotMask = VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST;
398 if (DeviceIoControl (gVBoxDriver, VBOXGUEST_IOCTL_CTL_FILTER_MASK, &maskInfo, sizeof (maskInfo), NULL, 0, &cbReturned, NULL))
399 {
400 Log(("VBoxTray: VBoxSeamlessThread: DeviceIOControl(CtlMask - not) succeeded\n"));
401 }
402 else
403 {
404 Log(("VBoxTray: VBoxSeamlessThread: DeviceIOControl(CtlMask) failed\n"));
405 }
406
407 Log(("VBoxTray: VBoxSeamlessThread: finished seamless change request thread\n"));
408 return 0;
409}
410
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette