VirtualBox

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

Last change on this file since 42154 was 42154, checked in by vboxsync, 13 years ago

VS2010 preps.

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