VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxDispIf.cpp@ 34575

Last change on this file since 34575 was 34130, checked in by vboxsync, 14 years ago

wddm: autoresize fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 25.1 KB
Line 
1/** @file
2 * VBoxTray - Display Settings Interface abstraction for XPDM & WDDM
3 */
4
5/*
6 * Copyright (C) 2006-2010 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#include "VBoxDispIf.h"
18
19#include <iprt/log.h>
20#include <iprt/err.h>
21#include <iprt/assert.h>
22
23#include <malloc.h>
24
25/* display driver interface abstraction for XPDM & WDDM
26 * with WDDM we can not use ExtEscape to communicate with our driver
27 * because we do not have XPDM display driver any more, i.e. escape requests are handled by cdd
28 * that knows nothing about us */
29DWORD VBoxDispIfInit(PVBOXDISPIF pIf)
30{
31 pIf->enmMode = VBOXDISPIF_MODE_XPDM;
32 return NO_ERROR;
33}
34
35DWORD VBoxDispIfTerm(PVBOXDISPIF pIf)
36{
37 pIf->enmMode = VBOXDISPIF_MODE_UNKNOWN;
38 return NO_ERROR;
39}
40
41static DWORD vboxDispIfEscapeXPDM(PCVBOXDISPIF pIf, PVBOXDISPIFESCAPE pEscape, int cbData)
42{
43 HDC hdc = GetDC(HWND_DESKTOP);
44 VOID *pvData = cbData ? VBOXDISPIFESCAPE_DATA(pEscape, VOID) : NULL;
45 int iRet = ExtEscape(hdc, pEscape->escapeCode, cbData, (LPCSTR)pvData, 0, NULL);
46 ReleaseDC(HWND_DESKTOP, hdc);
47 if (iRet > 0)
48 return VINF_SUCCESS;
49 else if (iRet == 0)
50 return ERROR_NOT_SUPPORTED;
51 /* else */
52 return ERROR_GEN_FAILURE;
53}
54
55#ifdef VBOX_WITH_WDDM
56static DWORD vboxDispIfSwitchToWDDM(PVBOXDISPIF pIf)
57{
58 DWORD err = NO_ERROR;
59 OSVERSIONINFO OSinfo;
60 OSinfo.dwOSVersionInfoSize = sizeof (OSinfo);
61 GetVersionEx (&OSinfo);
62 bool bSupported = true;
63
64 if (OSinfo.dwMajorVersion >= 6)
65 {
66 Log((__FUNCTION__": this is vista and up\n"));
67 HMODULE hUser = GetModuleHandle("USER32");
68 if (hUser)
69 {
70 *(uintptr_t *)&pIf->modeData.wddm.pfnChangeDisplaySettingsEx = (uintptr_t)GetProcAddress(hUser, "ChangeDisplaySettingsExA");
71 Log((__FUNCTION__": VBoxDisplayInit: pfnChangeDisplaySettingsEx = %p\n", pIf->modeData.wddm.pfnChangeDisplaySettingsEx));
72 bSupported &= !!(pIf->modeData.wddm.pfnChangeDisplaySettingsEx);
73
74 *(uintptr_t *)&pIf->modeData.wddm.pfnEnumDisplayDevices = (uintptr_t)GetProcAddress(hUser, "EnumDisplayDevicesA");
75 Log((__FUNCTION__": VBoxDisplayInit: pfnEnumDisplayDevices = %p\n", pIf->modeData.wddm.pfnEnumDisplayDevices));
76 bSupported &= !!(pIf->modeData.wddm.pfnEnumDisplayDevices);
77
78 /* this is vista and up */
79 HMODULE hGdi32 = GetModuleHandle("gdi32");
80 if (hGdi32 != NULL)
81 {
82 pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromHdc = (PFND3DKMT_OPENADAPTERFROMHDC)GetProcAddress(hGdi32, "D3DKMTOpenAdapterFromHdc");
83 Log((__FUNCTION__"pfnD3DKMTOpenAdapterFromHdc = %p\n", pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromHdc));
84 bSupported &= !!(pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromHdc);
85
86 pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromGdiDisplayName = (PFND3DKMT_OPENADAPTERFROMGDIDISPLAYNAME)GetProcAddress(hGdi32, "D3DKMTOpenAdapterFromGdiDisplayName");
87 Log((__FUNCTION__": pfnD3DKMTOpenAdapterFromGdiDisplayName = %p\n", pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromGdiDisplayName));
88 bSupported &= !!(pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromGdiDisplayName);
89
90 pIf->modeData.wddm.pfnD3DKMTCloseAdapter = (PFND3DKMT_CLOSEADAPTER)GetProcAddress(hGdi32, "D3DKMTCloseAdapter");
91 Log((__FUNCTION__": pfnD3DKMTCloseAdapter = %p\n", pIf->modeData.wddm.pfnD3DKMTCloseAdapter));
92 bSupported &= !!(pIf->modeData.wddm.pfnD3DKMTCloseAdapter);
93
94 pIf->modeData.wddm.pfnD3DKMTEscape = (PFND3DKMT_ESCAPE)GetProcAddress(hGdi32, "D3DKMTEscape");
95 Log((__FUNCTION__": pfnD3DKMTEscape = %p\n", pIf->modeData.wddm.pfnD3DKMTEscape));
96 bSupported &= !!(pIf->modeData.wddm.pfnD3DKMTCloseAdapter);
97
98 pIf->modeData.wddm.pfnD3DKMTInvalidateActiveVidPn = (PFND3DKMT_INVALIDATEACTIVEVIDPN)GetProcAddress(hGdi32, "D3DKMTInvalidateActiveVidPn");
99 Log((__FUNCTION__": pfnD3DKMTInvalidateActiveVidPn = %p\n", pIf->modeData.wddm.pfnD3DKMTInvalidateActiveVidPn));
100 bSupported &= !!(pIf->modeData.wddm.pfnD3DKMTInvalidateActiveVidPn);
101
102 if (!bSupported)
103 {
104 Log((__FUNCTION__": one of pfnD3DKMT function pointers failed to initialize\n"));
105 err = ERROR_NOT_SUPPORTED;
106 }
107 }
108 else
109 {
110 Log((__FUNCTION__": GetModuleHandle(gdi32) failed, err(%d)\n", GetLastError()));
111 err = ERROR_NOT_SUPPORTED;
112 }
113
114 }
115 else
116 {
117 Log((__FUNCTION__": GetModuleHandle(USER32) failed, err(%d)\n", GetLastError()));
118 err = ERROR_NOT_SUPPORTED;
119 }
120 }
121 else
122 {
123 Log((__FUNCTION__": can not switch to VBOXDISPIF_MODE_WDDM, because os is not Vista or upper\n"));
124 err = ERROR_NOT_SUPPORTED;
125 }
126
127 return err;
128}
129
130static DWORD vboxDispIfWDDMAdpHdcCreate(int iDisplay, HDC *phDc, DISPLAY_DEVICE *pDev)
131{
132 DWORD winEr = ERROR_INVALID_STATE;
133 memset(pDev, 0, sizeof (*pDev));
134 pDev->cb = sizeof (*pDev);
135
136 for (int i = 0; ; ++i)
137 {
138 if (EnumDisplayDevices(NULL, /* LPCTSTR lpDevice */ i, /* DWORD iDevNum */
139 pDev, 0 /* DWORD dwFlags*/))
140 {
141 if (i == iDisplay || (iDisplay < 0 && pDev->StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE))
142 {
143 HDC hDc = CreateDC(NULL, pDev->DeviceName, NULL, NULL);
144 if (hDc)
145 {
146 *phDc = hDc;
147 return NO_ERROR;
148 }
149 else
150 {
151 winEr = GetLastError();
152 Assert(0);
153 break;
154 }
155 }
156 }
157 else
158 {
159 winEr = GetLastError();
160 Assert(0);
161 break;
162 }
163 }
164
165 return winEr;
166}
167
168
169typedef DECLCALLBACK(BOOLEAN) FNVBOXDISPIFWDDM_ADAPTEROP(PCVBOXDISPIF pIf, D3DKMT_HANDLE hAdapter, DISPLAY_DEVICE *pDev, PVOID pContext);
170typedef FNVBOXDISPIFWDDM_ADAPTEROP *PFNVBOXDISPIFWDDM_ADAPTEROP;
171static DWORD vboxDispIfWDDMAdapterOp(PCVBOXDISPIF pIf, int iDisplay, PFNVBOXDISPIFWDDM_ADAPTEROP pfnOp, PVOID pContext)
172{
173 D3DKMT_OPENADAPTERFROMHDC OpenAdapterData = {0};
174 DISPLAY_DEVICE DDev;
175 DWORD err = vboxDispIfWDDMAdpHdcCreate(iDisplay, &OpenAdapterData.hDc, &DDev);
176 Assert(err == NO_ERROR);
177 if (err == NO_ERROR)
178 {
179 NTSTATUS Status = pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromHdc(&OpenAdapterData);
180 Assert(!Status);
181 if (!Status)
182 {
183 BOOLEAN bCloseAdapter = pfnOp(pIf, OpenAdapterData.hAdapter, &DDev, pContext);
184
185 if (bCloseAdapter)
186 {
187 D3DKMT_CLOSEADAPTER ClosaAdapterData = {0};
188 ClosaAdapterData.hAdapter = OpenAdapterData.hAdapter;
189 Status = pIf->modeData.wddm.pfnD3DKMTCloseAdapter(&ClosaAdapterData);
190 if (Status)
191 {
192 Log((__FUNCTION__": pfnD3DKMTCloseAdapter failed, Status (0x%x)\n", Status));
193 }
194 }
195 }
196 else
197 {
198 Log((__FUNCTION__": pfnD3DKMTOpenAdapterFromGdiDisplayName failed, Status (0x%x)\n", Status));
199 err = ERROR_GEN_FAILURE;
200 }
201
202 DeleteDC(OpenAdapterData.hDc);
203 }
204
205 return err;
206}
207
208typedef struct
209{
210 NTSTATUS Status;
211 PVBOXDISPIFESCAPE pEscape;
212 int cbData;
213} VBOXDISPIFWDDM_ESCAPEOP_CONTEXT, *PVBOXDISPIFWDDM_ESCAPEOP_CONTEXT;
214
215DECLCALLBACK(BOOLEAN) vboxDispIfEscapeWDDMOp(PCVBOXDISPIF pIf, D3DKMT_HANDLE hAdapter, DISPLAY_DEVICE *pDev, PVOID pContext)
216{
217 PVBOXDISPIFWDDM_ESCAPEOP_CONTEXT pCtx = (PVBOXDISPIFWDDM_ESCAPEOP_CONTEXT)pContext;
218
219 D3DKMT_ESCAPE EscapeData = {0};
220 EscapeData.hAdapter = hAdapter;
221 //EscapeData.hDevice = NULL;
222 EscapeData.Type = D3DKMT_ESCAPE_DRIVERPRIVATE;
223 EscapeData.Flags.HardwareAccess = 1;
224 EscapeData.pPrivateDriverData = pCtx->pEscape;
225 EscapeData.PrivateDriverDataSize = VBOXDISPIFESCAPE_SIZE(pCtx->cbData);
226 //EscapeData.hContext = NULL;
227
228 pCtx->Status = pIf->modeData.wddm.pfnD3DKMTEscape(&EscapeData);
229
230 return TRUE;
231}
232
233static DWORD vboxDispIfEscapeWDDM(PCVBOXDISPIF pIf, PVBOXDISPIFESCAPE pEscape, int cbData)
234{
235 VBOXDISPIFWDDM_ESCAPEOP_CONTEXT Ctx = {0};
236 Ctx.pEscape = pEscape;
237 Ctx.cbData = cbData;
238 DWORD err = vboxDispIfWDDMAdapterOp(pIf, -1 /* iDisplay, -1 means primary */, vboxDispIfEscapeWDDMOp, &Ctx);
239 if (err == NO_ERROR)
240 {
241 if (!Ctx.Status)
242 err = NO_ERROR;
243 else
244 {
245 if (Ctx.Status == 0xC00000BBL) /* not supported */
246 err = ERROR_NOT_SUPPORTED;
247 else
248 err = ERROR_GEN_FAILURE;
249 Log((__FUNCTION__": pfnD3DKMTEscape failed, Status (0x%x)\n", Ctx.Status));
250 }
251 }
252 else
253 Log((__FUNCTION__": vboxDispIfWDDMAdapterOp failed, err (%d)\n", err));
254
255 return err;
256}
257
258typedef struct
259{
260 NTSTATUS Status;
261 VBOXWDDM_RECOMMENDVIDPN_SCREEN_INFO Info;
262} VBOXDISPIFWDDM_RESIZEOP_CONTEXT, *PVBOXDISPIFWDDM_RESIZEOP_CONTEXT;
263
264DECLCALLBACK(BOOLEAN) vboxDispIfResizeWDDMOp(PCVBOXDISPIF pIf, D3DKMT_HANDLE hAdapter, DISPLAY_DEVICE *pDev, PVOID pContext)
265{
266 PVBOXDISPIFWDDM_RESIZEOP_CONTEXT pCtx = (PVBOXDISPIFWDDM_RESIZEOP_CONTEXT)pContext;
267
268 D3DKMT_INVALIDATEACTIVEVIDPN IAVidPnData = {0};
269 uint32_t cbData = VBOXWDDM_RECOMMENDVIDPN_SIZE(1);
270 PVBOXWDDM_RECOMMENDVIDPN pData = (PVBOXWDDM_RECOMMENDVIDPN)malloc(cbData);
271 if (pData)
272 {
273 memset(pData, 0, cbData);
274 pData->cScreenInfos = 1;
275 memcpy(&pData->aScreenInfos[0], &pCtx->Info, sizeof (VBOXWDDM_RECOMMENDVIDPN_SCREEN_INFO));
276
277 IAVidPnData.hAdapter = hAdapter;
278 IAVidPnData.pPrivateDriverData = pData;
279 IAVidPnData.PrivateDriverDataSize = cbData;
280
281 pCtx->Status = pIf->modeData.wddm.pfnD3DKMTInvalidateActiveVidPn(&IAVidPnData);
282 Assert(!pCtx->Status);
283 if (pCtx->Status)
284 Log((__FUNCTION__": pfnD3DKMTInvalidateActiveVidPn failed, Status (0x%x)\n", pCtx->Status));
285
286 free(pData);
287 }
288 else
289 {
290 Log((__FUNCTION__": malloc failed\n"));
291 pCtx->Status = -1;
292 }
293
294 return TRUE;
295}
296
297static DWORD vboxDispIfResizeWDDM(PCVBOXDISPIF const pIf, ULONG Id, DWORD Width, DWORD Height, DWORD BitsPerPixel)
298{
299 VBOXDISPIFWDDM_RESIZEOP_CONTEXT Ctx = {0};
300 Ctx.Info.Id = Id;
301 Ctx.Info.Width = Width;
302 Ctx.Info.Height = Height;
303 Ctx.Info.BitsPerPixel = BitsPerPixel;
304 DWORD err = vboxDispIfWDDMAdapterOp(pIf, -1, /* (int)Id - always say -1 to use primary display since the display does not really matter here */
305 vboxDispIfResizeWDDMOp, &Ctx);
306 if (err == NO_ERROR)
307 {
308 if (!Ctx.Status)
309 err = NO_ERROR;
310 else
311 {
312 if (Ctx.Status == 0xC00000BBL) /* not supported */
313 err = ERROR_NOT_SUPPORTED;
314 else
315 err = ERROR_GEN_FAILURE;
316 Log((__FUNCTION__": vboxDispIfResizeWDDMOp failed, Status (0x%x)\n", Ctx.Status));
317 }
318 }
319 else
320 Log((__FUNCTION__": vboxDispIfWDDMAdapterOp failed, err (%d)\n", err));
321
322 return err;
323}
324#endif
325
326DWORD VBoxDispIfEscape(PCVBOXDISPIF pIf, PVBOXDISPIFESCAPE pEscape, int cbData)
327{
328 switch (pIf->enmMode)
329 {
330 case VBOXDISPIF_MODE_XPDM_NT4:
331 case VBOXDISPIF_MODE_XPDM:
332 return vboxDispIfEscapeXPDM(pIf, pEscape, cbData);
333#ifdef VBOX_WITH_WDDM
334 case VBOXDISPIF_MODE_WDDM:
335 return vboxDispIfEscapeWDDM(pIf, pEscape, cbData);
336#endif
337 default:
338 Log((__FUNCTION__": unknown mode (%d)\n", pIf->enmMode));
339 return ERROR_INVALID_PARAMETER;
340 }
341}
342
343static DWORD vboxDispIfResizeXPDM(PCVBOXDISPIF const pIf, ULONG Id, DWORD Width, DWORD Height, DWORD BitsPerPixel)
344{
345 return ERROR_NOT_SUPPORTED;
346}
347
348DWORD VBoxDispIfResize(PCVBOXDISPIF const pIf, ULONG Id, DWORD Width, DWORD Height, DWORD BitsPerPixel)
349{
350 switch (pIf->enmMode)
351 {
352 case VBOXDISPIF_MODE_XPDM_NT4:
353 return ERROR_NOT_SUPPORTED;
354 case VBOXDISPIF_MODE_XPDM:
355 return vboxDispIfResizeXPDM(pIf, Id, Width, Height, BitsPerPixel);
356#ifdef VBOX_WITH_WDDM
357 case VBOXDISPIF_MODE_WDDM:
358 return vboxDispIfResizeWDDM(pIf, Id, Width, Height, BitsPerPixel);
359#endif
360 default:
361 Log((__FUNCTION__": unknown mode (%d)\n", pIf->enmMode));
362 return ERROR_INVALID_PARAMETER;
363 }
364}
365
366static BOOL vboxDispIfValidateResize(DISPLAY_DEVICE *paDisplayDevices, DEVMODE *paDeviceModes, UINT cDevModes)
367{
368 DISPLAY_DEVICE DisplayDevice;
369 int i = 0;
370 UINT cMatched = 0;
371 DEVMODE DeviceMode;
372 for (int i = 0; ; ++i)
373 {
374 ZeroMemory(&DisplayDevice, sizeof(DISPLAY_DEVICE));
375 DisplayDevice.cb = sizeof(DISPLAY_DEVICE);
376
377 if (!EnumDisplayDevices (NULL, i, &DisplayDevice, 0))
378 break;
379
380 Log(("VBoxTray: vboxDispIfValidateResize: [%d(%d)] %s\n", i, cMatched, DisplayDevice.DeviceName));
381
382 BOOL bFetchDevice = FALSE;
383
384 if (DisplayDevice.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE)
385 {
386 Log(("VBoxTray: vboxDispIfValidateResize: Found primary device. err %d\n", GetLastError ()));
387 bFetchDevice = TRUE;
388 }
389 else if (!(DisplayDevice.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER))
390 {
391
392 Log(("VBoxTray: vboxDispIfValidateResize: Found secondary device. err %d\n", GetLastError ()));
393 bFetchDevice = TRUE;
394 }
395
396 if (bFetchDevice)
397 {
398 if (cMatched >= cDevModes)
399 {
400 Log(("VBoxTray: vboxDispIfValidateResize: %d >= %d\n", cDevModes, cMatched));
401 return FALSE;
402 }
403
404 /* First try to get the video mode stored in registry (ENUM_REGISTRY_SETTINGS).
405 * A secondary display could be not active at the moment and would not have
406 * a current video mode (ENUM_CURRENT_SETTINGS).
407 */
408 ZeroMemory(&DeviceMode, sizeof(DeviceMode));
409 DeviceMode.dmSize = sizeof(DEVMODE);
410 if (!EnumDisplaySettings((LPSTR)DisplayDevice.DeviceName,
411 ENUM_REGISTRY_SETTINGS, &DeviceMode))
412 {
413 Log(("VBoxTray: vboxDispIfValidateResize: EnumDisplaySettings error %d\n", GetLastError ()));
414 return FALSE;
415 }
416
417 if ( DeviceMode.dmPelsWidth == 0
418 || DeviceMode.dmPelsHeight == 0)
419 {
420 /* No ENUM_REGISTRY_SETTINGS yet. Seen on Vista after installation.
421 * Get the current video mode then.
422 */
423 ZeroMemory(&DeviceMode, sizeof(DeviceMode));
424 DeviceMode.dmSize = sizeof(DeviceMode);
425 if (!EnumDisplaySettings((LPSTR)DisplayDevice.DeviceName,
426 ENUM_CURRENT_SETTINGS, &DeviceMode))
427 {
428 /* ENUM_CURRENT_SETTINGS returns FALSE when the display is not active:
429 * for example a disabled secondary display */
430 Log(("VBoxTray: vboxDispIfValidateResize: EnumDisplaySettings(ENUM_CURRENT_SETTINGS) error %d\n", GetLastError ()));
431 return FALSE;
432 }
433 }
434
435 UINT j = 0;
436 for (; j < cDevModes; ++j)
437 {
438 if (!strncmp(DisplayDevice.DeviceName, paDisplayDevices[j].DeviceName, RT_ELEMENTS(DeviceMode.dmDeviceName)))
439 {
440 if (paDeviceModes[j].dmBitsPerPel != DeviceMode.dmBitsPerPel
441 || (paDeviceModes[j].dmPelsWidth & 0xfff8) != (DeviceMode.dmPelsWidth & 0xfff8)
442 || (paDeviceModes[j].dmPelsHeight & 0xfff8) != (DeviceMode.dmPelsHeight & 0xfff8)
443 || (paDeviceModes[j].dmPosition.x & 0xfff8) != (DeviceMode.dmPosition.x & 0xfff8)
444 || (paDeviceModes[j].dmPosition.y & 0xfff8) != (DeviceMode.dmPosition.y & 0xfff8)
445 || (paDisplayDevices[j].StateFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP) == (DisplayDevice.StateFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP))
446 {
447 return FALSE;
448 }
449 break;
450 }
451 }
452
453 if (j == cDevModes)
454 return FALSE;
455
456 ++cMatched;
457 }
458 }
459
460 return cMatched == cDevModes;
461}
462
463DWORD vboxDispIfResizeModesWDDM(PCVBOXDISPIF const pIf, DISPLAY_DEVICE *paDisplayDevices, DEVMODE *paDeviceModes, UINT cDevModes)
464{
465 UINT cbVidPnInfo = VBOXWDDM_RECOMMENDVIDPN_SIZE(cDevModes);
466 PVBOXWDDM_RECOMMENDVIDPN pVidPnInfo = (PVBOXWDDM_RECOMMENDVIDPN)alloca(cbVidPnInfo);
467 pVidPnInfo->cScreenInfos = cDevModes;
468 D3DKMT_HANDLE hAdapter = NULL;
469 NTSTATUS Status;
470 DWORD winEr = NO_ERROR;
471 UINT i = 0;
472
473 for (; i < cDevModes; i++)
474 {
475 PVBOXWDDM_RECOMMENDVIDPN_SCREEN_INFO pInfo = &pVidPnInfo->aScreenInfos[i];
476 D3DKMT_OPENADAPTERFROMHDC OpenAdapterData = {0};
477 OpenAdapterData.hDc = CreateDC(NULL, paDisplayDevices[i].DeviceName, NULL, NULL);
478 if (!OpenAdapterData.hDc)
479 {
480 winEr = GetLastError();
481 Assert(0);
482 break;
483 }
484
485 Status = pIf->modeData.wddm.pfnD3DKMTOpenAdapterFromHdc(&OpenAdapterData);
486 Assert(!Status);
487 if (Status)
488 {
489 winEr = ERROR_GEN_FAILURE;
490 Assert(0);
491 break;
492 }
493
494 pInfo->Id = OpenAdapterData.VidPnSourceId;
495 pInfo->Width = paDeviceModes[i].dmPelsWidth;
496 pInfo->Height = paDeviceModes[i].dmPelsHeight;
497 pInfo->BitsPerPixel = paDeviceModes[i].dmBitsPerPel;
498
499 if (!hAdapter)
500 {
501 hAdapter = OpenAdapterData.hAdapter;
502 }
503 else
504 {
505 D3DKMT_CLOSEADAPTER ClosaAdapterData = {0};
506 ClosaAdapterData.hAdapter = OpenAdapterData.hAdapter;
507 Status = pIf->modeData.wddm.pfnD3DKMTCloseAdapter(&ClosaAdapterData);
508 Assert(!Status);
509 }
510 }
511
512 if (winEr == NO_ERROR)
513 {
514 Assert(hAdapter);
515
516 D3DKMT_INVALIDATEACTIVEVIDPN IAVidPnData = {0};
517 IAVidPnData.hAdapter = hAdapter;
518 IAVidPnData.pPrivateDriverData = pVidPnInfo;
519 IAVidPnData.PrivateDriverDataSize = cbVidPnInfo;
520
521 DWORD winEr = NO_ERROR;
522 Status = pIf->modeData.wddm.pfnD3DKMTInvalidateActiveVidPn(&IAVidPnData);
523 Assert(!Status);
524 if (Status)
525 {
526 Log((__FUNCTION__": pfnD3DKMTInvalidateActiveVidPn failed, Status (0x%x)\n", Status));
527 winEr = ERROR_GEN_FAILURE;
528 }
529 }
530
531 if (hAdapter)
532 {
533 D3DKMT_CLOSEADAPTER ClosaAdapterData = {0};
534 ClosaAdapterData.hAdapter = hAdapter;
535 Status = pIf->modeData.wddm.pfnD3DKMTCloseAdapter(&ClosaAdapterData);
536 Assert(!Status);
537 }
538
539 /* ignore any prev errors and just check if resize is OK */
540 if (!vboxDispIfValidateResize(paDisplayDevices, paDeviceModes, cDevModes))
541 {
542 /* now try to resize in a "regular" way */
543 /* Assign the new rectangles to displays. */
544 for (i = 0; i < cDevModes; i++)
545 {
546 /* On Vista one must specify DM_BITSPERPEL.
547 * Note that the current mode dmBitsPerPel is already in the DEVMODE structure.
548 */
549 paDeviceModes[i].dmFields = DM_POSITION | DM_PELSHEIGHT | DM_PELSWIDTH | DM_BITSPERPEL;
550
551 Log(("VBoxTray: ResizeDisplayDevice: pfnChangeDisplaySettingsEx %x: %dx%dx%d at %d,%d\n",
552 pIf->modeData.wddm.pfnChangeDisplaySettingsEx,
553 paDeviceModes[i].dmPelsWidth,
554 paDeviceModes[i].dmPelsHeight,
555 paDeviceModes[i].dmBitsPerPel,
556 paDeviceModes[i].dmPosition.x,
557 paDeviceModes[i].dmPosition.y));
558
559 LONG status = pIf->modeData.wddm.pfnChangeDisplaySettingsEx((LPSTR)paDisplayDevices[i].DeviceName,
560 &paDeviceModes[i], NULL, CDS_NORESET | CDS_UPDATEREGISTRY, NULL);
561 Log(("VBoxTray: ResizeDisplayDevice: ChangeDisplaySettingsEx position status %d, err %d\n", status, GetLastError ()));
562 }
563
564 /* A second call to ChangeDisplaySettings updates the monitor. */
565 LONG status = pIf->modeData.wddm.pfnChangeDisplaySettingsEx(NULL, NULL, NULL, 0, NULL);
566 Log(("VBoxTray: ResizeDisplayDevice: ChangeDisplaySettings update status %d\n", status));
567 if (status == DISP_CHANGE_SUCCESSFUL)
568 {
569 winEr = NO_ERROR;
570 }
571 else if (status == DISP_CHANGE_BADMODE)
572 {
573 /* Successfully set new video mode or our driver can not set the requested mode. Stop trying. */
574 winEr = ERROR_RETRY;
575 }
576 else
577 {
578 winEr = ERROR_GEN_FAILURE;
579 }
580 }
581 else
582 {
583 winEr = NO_ERROR;
584 }
585
586 return winEr;
587}
588
589DWORD VBoxDispIfResizeModes(PCVBOXDISPIF const pIf, DISPLAY_DEVICE *paDisplayDevices, DEVMODE *paDeviceModes, UINT cDevModes)
590{
591 switch (pIf->enmMode)
592 {
593 case VBOXDISPIF_MODE_XPDM_NT4:
594 return ERROR_NOT_SUPPORTED;
595 case VBOXDISPIF_MODE_XPDM:
596 return ERROR_NOT_SUPPORTED;
597#ifdef VBOX_WITH_WDDM
598 case VBOXDISPIF_MODE_WDDM:
599 return vboxDispIfResizeModesWDDM(pIf, paDisplayDevices, paDeviceModes, cDevModes);
600#endif
601 default:
602 Log((__FUNCTION__": unknown mode (%d)\n", pIf->enmMode));
603 return ERROR_INVALID_PARAMETER;
604 }
605}
606
607static DWORD vboxDispIfSwitchToXPDM_NT4(PVBOXDISPIF pIf)
608{
609 return NO_ERROR;
610}
611
612static DWORD vboxDispIfSwitchToXPDM(PVBOXDISPIF pIf)
613{
614 DWORD err = NO_ERROR;
615 AssertBreakpoint();
616 OSVERSIONINFO OSinfo;
617 OSinfo.dwOSVersionInfoSize = sizeof (OSinfo);
618 GetVersionEx (&OSinfo);
619 if (OSinfo.dwMajorVersion >= 5)
620 {
621 HMODULE hUser = GetModuleHandle("USER32");
622 if (NULL != hUser)
623 {
624 bool bSupported = true;
625 *(uintptr_t *)&pIf->modeData.xpdm.pfnChangeDisplaySettingsEx = (uintptr_t)GetProcAddress(hUser, "ChangeDisplaySettingsExA");
626 Log((__FUNCTION__": pfnChangeDisplaySettingsEx = %p\n", pIf->modeData.xpdm.pfnChangeDisplaySettingsEx));
627 bSupported &= !!(pIf->modeData.xpdm.pfnChangeDisplaySettingsEx);
628
629 if (!bSupported)
630 {
631 Log((__FUNCTION__": pfnChangeDisplaySettingsEx function pointer failed to initialize\n"));
632 err = ERROR_NOT_SUPPORTED;
633 }
634 }
635 else
636 {
637 Log((__FUNCTION__": failed to get USER32 handle, err (%d)\n", GetLastError()));
638 err = ERROR_NOT_SUPPORTED;
639 }
640 }
641 else
642 {
643 Log((__FUNCTION__": can not switch to VBOXDISPIF_MODE_XPDM, because os is not >= w2k\n"));
644 err = ERROR_NOT_SUPPORTED;
645 }
646
647 return err;
648}
649
650DWORD VBoxDispIfSwitchMode(PVBOXDISPIF pIf, VBOXDISPIF_MODE enmMode, VBOXDISPIF_MODE *penmOldMode)
651{
652 /* @todo: may need to addd synchronization in case we want to change modes dynamically
653 * i.e. currently the mode is supposed to be initialized once on service initialization */
654 if (penmOldMode)
655 *penmOldMode = pIf->enmMode;
656
657 if (enmMode == pIf->enmMode)
658 return VINF_ALREADY_INITIALIZED;
659
660 DWORD err = NO_ERROR;
661 switch (enmMode)
662 {
663 case VBOXDISPIF_MODE_XPDM_NT4:
664 Log((__FUNCTION__": request to switch to VBOXDISPIF_MODE_XPDM_NT4\n"));
665 err = vboxDispIfSwitchToXPDM_NT4(pIf);
666 if (err == NO_ERROR)
667 {
668 Log((__FUNCTION__": successfully switched to XPDM_NT4 mode\n"));
669 pIf->enmMode = VBOXDISPIF_MODE_XPDM_NT4;
670 }
671 else
672 Log((__FUNCTION__": failed to switch to XPDM_NT4 mode, err (%d)\n", err));
673 break;
674 case VBOXDISPIF_MODE_XPDM:
675 Log((__FUNCTION__": request to switch to VBOXDISPIF_MODE_XPDM\n"));
676 err = vboxDispIfSwitchToXPDM(pIf);
677 if (err == NO_ERROR)
678 {
679 Log((__FUNCTION__": successfully switched to XPDM mode\n"));
680 pIf->enmMode = VBOXDISPIF_MODE_XPDM;
681 }
682 else
683 Log((__FUNCTION__": failed to switch to XPDM mode, err (%d)\n", err));
684 break;
685#ifdef VBOX_WITH_WDDM
686 case VBOXDISPIF_MODE_WDDM:
687 {
688 Log((__FUNCTION__": request to switch to VBOXDISPIF_MODE_WDDM\n"));
689 err = vboxDispIfSwitchToWDDM(pIf);
690 if (err == NO_ERROR)
691 {
692 Log((__FUNCTION__": successfully switched to WDDM mode\n"));
693 pIf->enmMode = VBOXDISPIF_MODE_WDDM;
694 }
695 else
696 Log((__FUNCTION__": failed to switch to WDDM mode, err (%d)\n", err));
697 break;
698 }
699#endif
700 default:
701 err = ERROR_INVALID_PARAMETER;
702 break;
703 }
704 return err;
705}
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