xref: /reactos/win32ss/gdi/eng/device.c (revision b3194e32)
1 /*
2  * COPYRIGHT:         See COPYING in the top level directory
3  * PROJECT:           ReactOS kernel
4  * PURPOSE:           GDI Driver Device Functions
5  * FILE:              win32ss/gdi/eng/device.c
6  * PROGRAMER:         Jason Filby
7  *                    Timo Kreuzer
8  */
9 
10 #include <win32k.h>
11 #include <ntddvdeo.h>
12 
13 DBG_DEFAULT_CHANNEL(EngDev);
14 
15 PGRAPHICS_DEVICE gpPrimaryGraphicsDevice;
16 PGRAPHICS_DEVICE gpVgaGraphicsDevice;
17 
18 static PGRAPHICS_DEVICE gpGraphicsDeviceFirst = NULL;
19 static PGRAPHICS_DEVICE gpGraphicsDeviceLast = NULL;
20 static HSEMAPHORE ghsemGraphicsDeviceList;
21 static ULONG giDevNum = 1;
22 
23 CODE_SEG("INIT")
24 NTSTATUS
25 NTAPI
26 InitDeviceImpl(VOID)
27 {
28     ghsemGraphicsDeviceList = EngCreateSemaphore();
29     if (!ghsemGraphicsDeviceList)
30         return STATUS_INSUFFICIENT_RESOURCES;
31 
32     return STATUS_SUCCESS;
33 }
34 
35 static
36 BOOLEAN
37 EngpHasVgaDriver(
38     _In_ PGRAPHICS_DEVICE pGraphicsDevice)
39 {
40     WCHAR awcDeviceKey[256], awcServiceName[100];
41     PWSTR lastBkSlash;
42     NTSTATUS Status;
43     ULONG cbValue;
44     HKEY hkey;
45 
46     /* Open the key for the adapters */
47     Status = RegOpenKey(L"\\Registry\\Machine\\HARDWARE\\DEVICEMAP\\VIDEO", &hkey);
48     if (!NT_SUCCESS(Status))
49     {
50         ERR("Could not open HARDWARE\\DEVICEMAP\\VIDEO registry key: 0x%08lx\n", Status);
51         return FALSE;
52     }
53 
54     /* Read the name of the device key */
55     cbValue = sizeof(awcDeviceKey);
56     Status = RegQueryValue(hkey, pGraphicsDevice->szNtDeviceName, REG_SZ, awcDeviceKey, &cbValue);
57     ZwClose(hkey);
58     if (!NT_SUCCESS(Status))
59     {
60         ERR("Could not read '%S' registry value: 0x%08lx\n", Status);
61         return FALSE;
62     }
63 
64     /* Replace 'DeviceN' by 'Video' */
65     lastBkSlash = wcsrchr(awcDeviceKey, L'\\');
66     if (!lastBkSlash)
67     {
68         ERR("Invalid registry key '%S'\n", lastBkSlash);
69         return FALSE;
70     }
71     if (!NT_SUCCESS(RtlStringCchCopyW(lastBkSlash + 1,
72                                       ARRAYSIZE(awcDeviceKey) - (lastBkSlash + 1 - awcDeviceKey),
73                                       L"Video")))
74     {
75         ERR("Failed to add 'Video' to registry key '%S'\n", awcDeviceKey);
76         return FALSE;
77     }
78 
79     /* Open device key */
80     Status = RegOpenKey(awcDeviceKey, &hkey);
81     if (!NT_SUCCESS(Status))
82     {
83         ERR("Could not open %S registry key: 0x%08lx\n", awcDeviceKey, Status);
84         return FALSE;
85     }
86 
87     /* Read service name */
88     cbValue = sizeof(awcServiceName);
89     Status = RegQueryValue(hkey, L"Service", REG_SZ, awcServiceName, &cbValue);
90     ZwClose(hkey);
91     if (!NT_SUCCESS(Status))
92     {
93         ERR("Could not read Service registry value in %S: 0x%08lx\n", awcDeviceKey, Status);
94         return FALSE;
95     }
96 
97     /* Device is using VGA driver if service name starts with 'VGA' (case insensitive) */
98     return (_wcsnicmp(awcServiceName, L"VGA", 3) == 0);
99 }
100 
101 /*
102  * Add a device to gpGraphicsDeviceFirst/gpGraphicsDeviceLast list (if not already present).
103  */
104 _Requires_lock_held_(ghsemGraphicsDeviceList)
105 static
106 VOID
107 EngpLinkGraphicsDevice(
108     _In_ PGRAPHICS_DEVICE pToAdd)
109 {
110     PGRAPHICS_DEVICE pGraphicsDevice;
111 
112     TRACE("EngLinkGraphicsDevice(%p)\n", pToAdd);
113 
114     /* Search if device is not already linked */
115     for (pGraphicsDevice = gpGraphicsDeviceFirst;
116          pGraphicsDevice;
117          pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice)
118     {
119         if (pGraphicsDevice == pToAdd)
120             return;
121     }
122 
123     pToAdd->pNextGraphicsDevice = NULL;
124     if (gpGraphicsDeviceLast)
125         gpGraphicsDeviceLast->pNextGraphicsDevice = pToAdd;
126     gpGraphicsDeviceLast = pToAdd;
127     if (!gpGraphicsDeviceFirst)
128         gpGraphicsDeviceFirst = pToAdd;
129 }
130 
131 /*
132  * Remove a device from gpGraphicsDeviceFirst/gpGraphicsDeviceLast list.
133  */
134 _Requires_lock_held_(ghsemGraphicsDeviceList)
135 static
136 VOID
137 EngpUnlinkGraphicsDevice(
138     _In_ PGRAPHICS_DEVICE pToDelete)
139 {
140     PGRAPHICS_DEVICE pPrevGraphicsDevice = NULL;
141     PGRAPHICS_DEVICE pGraphicsDevice = gpGraphicsDeviceFirst;
142 
143     TRACE("EngpUnlinkGraphicsDevice('%S')\n", pToDelete->szNtDeviceName);
144 
145     while (pGraphicsDevice)
146     {
147         if (pGraphicsDevice != pToDelete)
148         {
149             /* Keep current device */
150             pPrevGraphicsDevice = pGraphicsDevice;
151             pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice;
152         }
153         else
154         {
155             /* At first, link again associated VGA Device */
156             if (pGraphicsDevice->pVgaDevice)
157                 EngpLinkGraphicsDevice(pGraphicsDevice->pVgaDevice);
158 
159             /* We need to remove current device */
160             pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice;
161 
162             /* Unlink chain */
163             if (!pPrevGraphicsDevice)
164                 gpGraphicsDeviceFirst = pToDelete->pNextGraphicsDevice;
165             else
166                 pPrevGraphicsDevice->pNextGraphicsDevice = pToDelete->pNextGraphicsDevice;
167             if (gpGraphicsDeviceLast == pToDelete)
168                 gpGraphicsDeviceLast = pPrevGraphicsDevice;
169         }
170     }
171 }
172 
173 NTSTATUS
174 EngpUpdateGraphicsDeviceList(VOID)
175 {
176     ULONG iDevNum, iVGACompatible = -1, ulMaxObjectNumber = 0;
177     WCHAR awcDeviceName[20], awcWinDeviceName[20];
178     UNICODE_STRING ustrDeviceName;
179     WCHAR awcBuffer[256];
180     NTSTATUS Status;
181     PGRAPHICS_DEVICE pGraphicsDevice;
182     BOOLEAN bFoundNewDevice = FALSE;
183     ULONG cbValue;
184     HKEY hkey;
185 
186     /* Open the key for the adapters */
187     Status = RegOpenKey(L"\\Registry\\Machine\\HARDWARE\\DEVICEMAP\\VIDEO", &hkey);
188     if (!NT_SUCCESS(Status))
189     {
190         ERR("Could not open HARDWARE\\DEVICEMAP\\VIDEO registry key:0x%lx\n", Status);
191         return Status;
192     }
193 
194     /* Read the name of the VGA adapter */
195     cbValue = sizeof(awcDeviceName);
196     Status = RegQueryValue(hkey, L"VgaCompatible", REG_SZ, awcDeviceName, &cbValue);
197     if (NT_SUCCESS(Status))
198     {
199         iVGACompatible = _wtoi(&awcDeviceName[sizeof("\\Device\\Video")-1]);
200         ERR("VGA adapter = %lu\n", iVGACompatible);
201     }
202 
203     /* Get the maximum number of adapters */
204     if (!RegReadDWORD(hkey, L"MaxObjectNumber", &ulMaxObjectNumber))
205     {
206         ERR("Could not read MaxObjectNumber, defaulting to 0.\n");
207     }
208 
209     TRACE("Found %lu devices\n", ulMaxObjectNumber + 1);
210 
211     /* Loop through all adapters */
212     for (iDevNum = 0; iDevNum <= ulMaxObjectNumber; iDevNum++)
213     {
214         /* Create the adapter's key name */
215         swprintf(awcDeviceName, L"\\Device\\Video%lu", iDevNum);
216 
217         /* Create the display device name */
218         swprintf(awcWinDeviceName, L"\\\\.\\DISPLAY%lu", iDevNum + 1);
219         RtlInitUnicodeString(&ustrDeviceName, awcWinDeviceName);
220 
221         /* Check if the device exists already */
222         pGraphicsDevice = EngpFindGraphicsDevice(&ustrDeviceName, iDevNum);
223         if (pGraphicsDevice != NULL)
224         {
225             continue;
226         }
227 
228         /* Read the reg key name */
229         cbValue = sizeof(awcBuffer);
230         Status = RegQueryValue(hkey, awcDeviceName, REG_SZ, awcBuffer, &cbValue);
231         if (!NT_SUCCESS(Status))
232         {
233             ERR("failed to query the registry path:0x%lx\n", Status);
234             continue;
235         }
236 
237         /* Initialize the driver for this device */
238         pGraphicsDevice = InitDisplayDriver(awcDeviceName, awcBuffer);
239         if (!pGraphicsDevice) continue;
240 
241         /* Check if this is a VGA compatible adapter */
242         if (pGraphicsDevice->StateFlags & DISPLAY_DEVICE_VGA_COMPATIBLE)
243         {
244             /* Save this as the VGA adapter */
245             if (!gpVgaGraphicsDevice)
246             {
247                 gpVgaGraphicsDevice = pGraphicsDevice;
248                 TRACE("gpVgaGraphicsDevice = %p\n", gpVgaGraphicsDevice);
249             }
250         }
251         bFoundNewDevice = TRUE;
252 
253         /* Set the first one as primary device */
254         if (!gpPrimaryGraphicsDevice || EngpHasVgaDriver(gpPrimaryGraphicsDevice))
255         {
256             gpPrimaryGraphicsDevice = pGraphicsDevice;
257             TRACE("gpPrimaryGraphicsDevice = %p\n", gpPrimaryGraphicsDevice);
258         }
259     }
260 
261     /* Close the device map registry key */
262     ZwClose(hkey);
263 
264     /* Can we link VGA device to primary device? */
265     if (gpPrimaryGraphicsDevice &&
266         gpVgaGraphicsDevice &&
267         gpPrimaryGraphicsDevice != gpVgaGraphicsDevice &&
268         !gpPrimaryGraphicsDevice->pVgaDevice)
269     {
270         /* Yes. Remove VGA device from global list, and attach it to primary device */
271         TRACE("Linking VGA device %S to primary device %S\n", gpVgaGraphicsDevice->szNtDeviceName, gpPrimaryGraphicsDevice->szNtDeviceName);
272         EngpUnlinkGraphicsDevice(gpVgaGraphicsDevice);
273         gpPrimaryGraphicsDevice->pVgaDevice = gpVgaGraphicsDevice;
274     }
275 
276     if (bFoundNewDevice && gbBaseVideo)
277     {
278         PGRAPHICS_DEVICE pToDelete;
279 
280         /* Lock list */
281         EngAcquireSemaphore(ghsemGraphicsDeviceList);
282 
283         /* Remove every device from linked list, except base-video one */
284         pGraphicsDevice = gpGraphicsDeviceFirst;
285         while (pGraphicsDevice)
286         {
287             if (!EngpHasVgaDriver(pGraphicsDevice))
288             {
289                 /* Not base-video device. Remove it */
290                 pToDelete = pGraphicsDevice;
291                 TRACE("Removing non-base-video device %S (%S)\n", pToDelete->szWinDeviceName, pToDelete->szNtDeviceName);
292 
293                 EngpUnlinkGraphicsDevice(pGraphicsDevice);
294                 pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice;
295 
296                 /* Free memory */
297                 ExFreePoolWithTag(pToDelete->pDiplayDrivers, GDITAG_DRVSUP);
298                 ExFreePoolWithTag(pToDelete, GDITAG_GDEVICE);
299             }
300             else
301             {
302                 pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice;
303             }
304         }
305 
306         /* Unlock list */
307         EngReleaseSemaphore(ghsemGraphicsDeviceList);
308     }
309 
310     return STATUS_SUCCESS;
311 }
312 
313 /* Open display settings registry key
314  * Returns NULL in case of error. */
315 static HKEY
316 EngpGetRegistryHandleFromDeviceMap(
317     _In_ PGRAPHICS_DEVICE pGraphicsDevice)
318 {
319     static const PWCHAR KEY_VIDEO = L"\\Registry\\Machine\\HARDWARE\\DEVICEMAP\\VIDEO";
320     HKEY hKey;
321     WCHAR szDeviceKey[256];
322     ULONG cbSize;
323     NTSTATUS Status;
324 
325     /* Open the device map registry key */
326     Status = RegOpenKey(KEY_VIDEO, &hKey);
327     if (!NT_SUCCESS(Status))
328     {
329         ERR("Could not open HARDWARE\\DEVICEMAP\\VIDEO registry key: status 0x%08x\n", Status);
330         return NULL;
331     }
332 
333     /* Query the registry path */
334     cbSize = sizeof(szDeviceKey);
335     RegQueryValue(hKey,
336                   pGraphicsDevice->szNtDeviceName,
337                   REG_SZ,
338                   szDeviceKey,
339                   &cbSize);
340     ZwClose(hKey);
341 
342     /* Open the registry key */
343     Status = RegOpenKey(szDeviceKey, &hKey);
344     if (!NT_SUCCESS(Status))
345     {
346         ERR("Could not open registry key '%S': status 0x%08x\n", szDeviceKey, Status);
347         return NULL;
348     }
349 
350     return hKey;
351 }
352 
353 NTSTATUS
354 EngpGetDisplayDriverParameters(
355     _In_ PGRAPHICS_DEVICE pGraphicsDevice,
356     _Out_ PDEVMODEW pdm)
357 {
358     HKEY hKey;
359     NTSTATUS Status;
360     RTL_QUERY_REGISTRY_TABLE DisplaySettingsTable[] =
361     {
362 #define READ(field, str) \
363         { \
364             NULL, \
365             RTL_QUERY_REGISTRY_DIRECT, \
366             L ##str, \
367             &pdm->field, \
368             REG_NONE, NULL, 0 \
369         },
370     READ(dmBitsPerPel, "DefaultSettings.BitsPerPel")
371     READ(dmPelsWidth, "DefaultSettings.XResolution")
372     READ(dmPelsHeight, "DefaultSettings.YResolution")
373     READ(dmDisplayFlags, "DefaultSettings.Flags")
374     READ(dmDisplayFrequency, "DefaultSettings.VRefresh")
375     READ(dmPanningWidth, "DefaultSettings.XPanning")
376     READ(dmPanningHeight, "DefaultSettings.YPanning")
377     READ(dmDisplayOrientation, "DefaultSettings.Orientation")
378     READ(dmDisplayFixedOutput, "DefaultSettings.FixedOutput")
379     READ(dmPosition.x, "Attach.RelativeX")
380     READ(dmPosition.y, "Attach.RelativeY")
381 #undef READ
382         {0}
383     };
384 
385     hKey = EngpGetRegistryHandleFromDeviceMap(pGraphicsDevice);
386     if (!hKey)
387         return STATUS_UNSUCCESSFUL;
388 
389     Status = RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
390                                     (PWSTR)hKey,
391                                     DisplaySettingsTable,
392                                     NULL,
393                                     NULL);
394 
395     ZwClose(hKey);
396     return Status;
397 }
398 
399 DWORD
400 EngpGetDisplayDriverAccelerationLevel(
401     _In_ PGRAPHICS_DEVICE pGraphicsDevice)
402 {
403     HKEY hKey;
404     DWORD dwAccelerationLevel = 0;
405     RTL_QUERY_REGISTRY_TABLE DisplaySettingsTable[] =
406     {
407         {
408             NULL,
409             RTL_QUERY_REGISTRY_DIRECT,
410             L"Acceleration.Level",
411             &dwAccelerationLevel,
412             REG_NONE, NULL, 0
413         },
414         {0}
415     };
416 
417     hKey = EngpGetRegistryHandleFromDeviceMap(pGraphicsDevice);
418     if (!hKey)
419         return 0;
420 
421     RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
422                            (PWSTR)hKey,
423                            DisplaySettingsTable,
424                            NULL,
425                            NULL);
426     ZwClose(hKey);
427 
428     return dwAccelerationLevel;
429 }
430 
431 extern VOID
432 UserRefreshDisplay(IN PPDEVOBJ ppdev);
433 
434 // PVIDEO_WIN32K_CALLOUT
435 VOID
436 NTAPI
437 VideoPortCallout(
438     _In_ PVOID Params)
439 {
440 /*
441  * IMPORTANT NOTICE!! On Windows XP/2003 this function triggers the creation of
442  * a specific VideoPortCalloutThread() system thread using the same mechanism
443  * as the RIT/desktop/Ghost system threads.
444  */
445 
446     PVIDEO_WIN32K_CALLBACKS_PARAMS CallbackParams = (PVIDEO_WIN32K_CALLBACKS_PARAMS)Params;
447 
448     TRACE("VideoPortCallout(0x%p, 0x%x)\n",
449           CallbackParams, CallbackParams ? CallbackParams->CalloutType : -1);
450 
451     if (!CallbackParams)
452         return;
453 
454     switch (CallbackParams->CalloutType)
455     {
456         case VideoFindAdapterCallout:
457         {
458             TRACE("VideoPortCallout: VideoFindAdapterCallout called - Param = %s\n",
459                   CallbackParams->Param ? "TRUE" : "FALSE");
460             if (CallbackParams->Param == TRUE)
461             {
462                 /* Re-enable the display */
463                 UserRefreshDisplay(gpmdev->ppdevGlobal);
464             }
465             else
466             {
467                 /* Disable the display */
468                 NOTHING; // Nothing to do for the moment...
469             }
470 
471             CallbackParams->Status = STATUS_SUCCESS;
472             break;
473         }
474 
475         case VideoPowerNotifyCallout:
476         case VideoDisplaySwitchCallout:
477         case VideoEnumChildPdoNotifyCallout:
478         case VideoWakeupCallout:
479         case VideoChangeDisplaySettingsCallout:
480         case VideoPnpNotifyCallout:
481         case VideoDxgkDisplaySwitchCallout:
482         case VideoDxgkMonitorEventCallout:
483         case VideoDxgkFindAdapterTdrCallout:
484             ERR("VideoPortCallout: CalloutType 0x%x is UNIMPLEMENTED!\n", CallbackParams->CalloutType);
485             CallbackParams->Status = STATUS_NOT_IMPLEMENTED;
486             break;
487 
488         default:
489             ERR("VideoPortCallout: Unknown CalloutType 0x%x\n", CallbackParams->CalloutType);
490             CallbackParams->Status = STATUS_UNSUCCESSFUL;
491             break;
492     }
493 }
494 
495 PGRAPHICS_DEVICE
496 NTAPI
497 EngpRegisterGraphicsDevice(
498     _In_ PUNICODE_STRING pustrDeviceName,
499     _In_ PUNICODE_STRING pustrDiplayDrivers,
500     _In_ PUNICODE_STRING pustrDescription)
501 {
502     PGRAPHICS_DEVICE pGraphicsDevice;
503     PDEVICE_OBJECT pDeviceObject;
504     PFILE_OBJECT pFileObject;
505     NTSTATUS Status;
506     VIDEO_WIN32K_CALLBACKS Win32kCallbacks;
507     ULONG ulReturn;
508     PWSTR pwsz;
509     ULONG cj;
510 
511     TRACE("EngpRegisterGraphicsDevice(%wZ)\n", pustrDeviceName);
512 
513     /* Allocate a GRAPHICS_DEVICE structure */
514     pGraphicsDevice = ExAllocatePoolZero(PagedPool,
515                                          sizeof(GRAPHICS_DEVICE),
516                                          GDITAG_GDEVICE);
517     if (!pGraphicsDevice)
518     {
519         ERR("ExAllocatePoolWithTag failed\n");
520         return NULL;
521     }
522 
523     /* Try to open and enable the device */
524     Status = IoGetDeviceObjectPointer(pustrDeviceName,
525                                       FILE_READ_DATA | FILE_WRITE_DATA,
526                                       &pFileObject,
527                                       &pDeviceObject);
528     if (!NT_SUCCESS(Status))
529     {
530         ERR("Could not open device %wZ, 0x%lx\n", pustrDeviceName, Status);
531         ExFreePoolWithTag(pGraphicsDevice, GDITAG_GDEVICE);
532         return NULL;
533     }
534 
535     /* Copy the device and file object pointers */
536     pGraphicsDevice->DeviceObject = pDeviceObject;
537     pGraphicsDevice->FileObject = pFileObject;
538 
539     /* Initialize and register the device with videoprt for Win32k callbacks */
540     Win32kCallbacks.PhysDisp = pGraphicsDevice;
541     Win32kCallbacks.Callout = VideoPortCallout;
542     // Reset the data being returned prior to the call.
543     Win32kCallbacks.bACPI = FALSE;
544     Win32kCallbacks.pPhysDeviceObject = NULL;
545     Win32kCallbacks.DualviewFlags = 0;
546     Status = (NTSTATUS)EngDeviceIoControl((HANDLE)pDeviceObject,
547                                           IOCTL_VIDEO_INIT_WIN32K_CALLBACKS,
548                                           &Win32kCallbacks,
549                                           sizeof(Win32kCallbacks),
550                                           &Win32kCallbacks,
551                                           sizeof(Win32kCallbacks),
552                                           &ulReturn);
553     if (Status != ERROR_SUCCESS)
554     {
555         ERR("EngDeviceIoControl(0x%p, IOCTL_VIDEO_INIT_WIN32K_CALLBACKS) failed, Status 0x%lx\n",
556             pDeviceObject, Status);
557     }
558     // TODO: Set flags according to the results.
559     // if (Win32kCallbacks.bACPI)
560     // if (Win32kCallbacks.DualviewFlags & ???)
561     pGraphicsDevice->PhysDeviceHandle = Win32kCallbacks.pPhysDeviceObject;
562 
563     /* FIXME: Enumerate children monitor devices for this video adapter
564      *
565      * - Force the adapter to re-enumerate its monitors:
566      *   IoSynchronousInvalidateDeviceRelations(pdo, BusRelations)
567      *
568      * - Retrieve all monitor PDOs from VideoPrt:
569      *   EngDeviceIoControl(0x%p, IOCTL_VIDEO_ENUM_MONITOR_PDO)
570      *
571      * - Initialize these fields and structures accordingly:
572      *   pGraphicsDevice->dwMonCnt
573      *   pGraphicsDevice->pvMonDev[0..dwMonCnt-1]
574      */
575 
576     /* Copy the device name */
577     RtlStringCbCopyNW(pGraphicsDevice->szNtDeviceName,
578                       sizeof(pGraphicsDevice->szNtDeviceName),
579                       pustrDeviceName->Buffer,
580                       pustrDeviceName->Length);
581 
582     /* Create a Win32 device name (FIXME: virtual devices!) */
583     RtlStringCbPrintfW(pGraphicsDevice->szWinDeviceName,
584                        sizeof(pGraphicsDevice->szWinDeviceName),
585                        L"\\\\.\\DISPLAY%d",
586                        (int)giDevNum);
587 
588     /* Allocate a buffer for the strings */
589     cj = pustrDiplayDrivers->Length + pustrDescription->Length + sizeof(WCHAR);
590     pwsz = ExAllocatePoolWithTag(PagedPool, cj, GDITAG_DRVSUP);
591     if (!pwsz)
592     {
593         ERR("Could not allocate string buffer\n");
594         ASSERT(FALSE); // FIXME
595         ExFreePoolWithTag(pGraphicsDevice, GDITAG_GDEVICE);
596         return NULL;
597     }
598 
599     /* Copy the display driver names */
600     pGraphicsDevice->pDiplayDrivers = pwsz;
601     RtlCopyMemory(pGraphicsDevice->pDiplayDrivers,
602                   pustrDiplayDrivers->Buffer,
603                   pustrDiplayDrivers->Length);
604 
605     /* Copy the description */
606     pGraphicsDevice->pwszDescription = pwsz + pustrDiplayDrivers->Length / sizeof(WCHAR);
607     RtlCopyMemory(pGraphicsDevice->pwszDescription,
608                   pustrDescription->Buffer,
609                   pustrDescription->Length);
610     pGraphicsDevice->pwszDescription[pustrDescription->Length/sizeof(WCHAR)] = 0;
611 
612     /* Lock loader */
613     EngAcquireSemaphore(ghsemGraphicsDeviceList);
614 
615     /* Insert the device into the global list */
616     EngpLinkGraphicsDevice(pGraphicsDevice);
617 
618     /* Increment the device number */
619     giDevNum++;
620 
621     /* Unlock loader */
622     EngReleaseSemaphore(ghsemGraphicsDeviceList);
623     TRACE("Prepared %lu modes for %ls\n", pGraphicsDevice->cDevModes, pGraphicsDevice->pwszDescription);
624 
625     /* HACK: already in graphic mode; display wallpaper on this new display */
626     if (ScreenDeviceContext)
627     {
628         UNICODE_STRING DriverName = RTL_CONSTANT_STRING(L"DISPLAY");
629         UNICODE_STRING DisplayName;
630         HDC hdc;
631         RtlInitUnicodeString(&DisplayName, pGraphicsDevice->szWinDeviceName);
632         hdc = IntGdiCreateDC(&DriverName, &DisplayName, NULL, NULL, FALSE);
633         IntPaintDesktop(hdc);
634     }
635 
636     return pGraphicsDevice;
637 }
638 
639 PGRAPHICS_DEVICE
640 NTAPI
641 EngpFindGraphicsDevice(
642     _In_opt_ PUNICODE_STRING pustrDevice,
643     _In_ ULONG iDevNum)
644 {
645     UNICODE_STRING ustrCurrent;
646     PGRAPHICS_DEVICE pGraphicsDevice;
647     ULONG i;
648     TRACE("EngpFindGraphicsDevice('%wZ', %lu)\n",
649            pustrDevice, iDevNum);
650 
651     /* Lock list */
652     EngAcquireSemaphore(ghsemGraphicsDeviceList);
653 
654     if (pustrDevice && pustrDevice->Buffer)
655     {
656         /* Find specified video adapter by name */
657         for (pGraphicsDevice = gpGraphicsDeviceFirst;
658              pGraphicsDevice;
659              pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice)
660         {
661             /* Compare the device name */
662             RtlInitUnicodeString(&ustrCurrent, pGraphicsDevice->szWinDeviceName);
663             if (RtlEqualUnicodeString(&ustrCurrent, pustrDevice, FALSE))
664             {
665                 break;
666             }
667         }
668 
669         if (pGraphicsDevice)
670         {
671             /* Validate selected monitor number */
672 #if 0
673             if (iDevNum >= pGraphicsDevice->dwMonCnt)
674                 pGraphicsDevice = NULL;
675 #else
676             /* FIXME: dwMonCnt not initialized, see EngpRegisterGraphicsDevice */
677 #endif
678         }
679     }
680     else
681     {
682         /* Select video adapter by device number */
683         for (pGraphicsDevice = gpGraphicsDeviceFirst, i = 0;
684              pGraphicsDevice && i < iDevNum;
685              pGraphicsDevice = pGraphicsDevice->pNextGraphicsDevice, i++);
686     }
687 
688     /* Unlock list */
689     EngReleaseSemaphore(ghsemGraphicsDeviceList);
690 
691     return pGraphicsDevice;
692 }
693 
694 static
695 NTSTATUS
696 EngpFileIoRequest(
697     _In_ PFILE_OBJECT pFileObject,
698     _In_ ULONG ulMajorFunction,
699     _In_reads_(nBufferSize) PVOID lpBuffer,
700     _In_ SIZE_T nBufferSize,
701     _In_ ULONGLONG ullStartOffset,
702     _Out_ PULONG_PTR lpInformation)
703 {
704     PDEVICE_OBJECT pDeviceObject;
705     KEVENT Event;
706     PIRP pIrp;
707     IO_STATUS_BLOCK Iosb;
708     NTSTATUS Status;
709     LARGE_INTEGER liStartOffset;
710 
711     /* Get corresponding device object */
712     pDeviceObject = IoGetRelatedDeviceObject(pFileObject);
713     if (!pDeviceObject)
714     {
715         return STATUS_INVALID_PARAMETER;
716     }
717 
718     /* Initialize an event */
719     KeInitializeEvent(&Event, SynchronizationEvent, FALSE);
720 
721     /* Build IRP */
722     liStartOffset.QuadPart = ullStartOffset;
723     pIrp = IoBuildSynchronousFsdRequest(ulMajorFunction,
724                                         pDeviceObject,
725                                         lpBuffer,
726                                         (ULONG)nBufferSize,
727                                         &liStartOffset,
728                                         &Event,
729                                         &Iosb);
730     if (!pIrp)
731     {
732         return STATUS_INSUFFICIENT_RESOURCES;
733     }
734 
735     /* Call the driver */
736     Status = IoCallDriver(pDeviceObject, pIrp);
737 
738     /* Wait if neccessary */
739     if (STATUS_PENDING == Status)
740     {
741         KeWaitForSingleObject(&Event, Executive, KernelMode, TRUE, 0);
742         Status = Iosb.Status;
743     }
744 
745     /* Return information to the caller about the operation. */
746     *lpInformation = Iosb.Information;
747 
748     /* Return NTSTATUS */
749     return Status;
750 }
751 
752 VOID
753 APIENTRY
754 EngFileWrite(
755     _In_ PFILE_OBJECT pFileObject,
756     _In_reads_(nLength) PVOID lpBuffer,
757     _In_ SIZE_T nLength,
758     _Out_ PSIZE_T lpBytesWritten)
759 {
760     NTSTATUS status;
761 
762     status = EngpFileIoRequest(pFileObject,
763                                IRP_MJ_WRITE,
764                                lpBuffer,
765                                nLength,
766                                0,
767                                lpBytesWritten);
768     if (!NT_SUCCESS(status))
769     {
770         *lpBytesWritten = 0;
771     }
772 }
773 
774 _Success_(return>=0)
775 NTSTATUS
776 APIENTRY
777 EngFileIoControl(
778     _In_ PFILE_OBJECT pFileObject,
779     _In_ DWORD dwIoControlCode,
780     _In_reads_(nInBufferSize) PVOID lpInBuffer,
781     _In_ SIZE_T nInBufferSize,
782     _Out_writes_(nOutBufferSize) PVOID lpOutBuffer,
783     _In_ SIZE_T nOutBufferSize,
784     _Out_ PULONG_PTR lpInformation)
785 {
786     PDEVICE_OBJECT pDeviceObject;
787     KEVENT Event;
788     PIRP pIrp;
789     IO_STATUS_BLOCK Iosb;
790     NTSTATUS Status;
791 
792     /* Get corresponding device object */
793     pDeviceObject = IoGetRelatedDeviceObject(pFileObject);
794     if (!pDeviceObject)
795     {
796         return STATUS_INVALID_PARAMETER;
797     }
798 
799     /* Initialize an event */
800     KeInitializeEvent(&Event, SynchronizationEvent, FALSE);
801 
802     /* Build IO control IRP */
803     pIrp = IoBuildDeviceIoControlRequest(dwIoControlCode,
804                                          pDeviceObject,
805                                          lpInBuffer,
806                                          (ULONG)nInBufferSize,
807                                          lpOutBuffer,
808                                          (ULONG)nOutBufferSize,
809                                          FALSE,
810                                          &Event,
811                                          &Iosb);
812     if (!pIrp)
813     {
814         return STATUS_INSUFFICIENT_RESOURCES;
815     }
816 
817     /* Call the driver */
818     Status = IoCallDriver(pDeviceObject, pIrp);
819 
820     /* Wait if neccessary */
821     if (Status == STATUS_PENDING)
822     {
823         KeWaitForSingleObject(&Event, Executive, KernelMode, TRUE, 0);
824         Status = Iosb.Status;
825     }
826 
827     /* Return information to the caller about the operation. */
828     *lpInformation = Iosb.Information;
829 
830     /* This function returns NTSTATUS */
831     return Status;
832 }
833 
834 /*
835  * @implemented
836  */
837 _Success_(return==0)
838 DWORD
839 APIENTRY
840 EngDeviceIoControl(
841     _In_ HANDLE hDevice,
842     _In_ DWORD dwIoControlCode,
843     _In_reads_bytes_opt_(cjInBufferSize) LPVOID lpInBuffer,
844     _In_ DWORD cjInBufferSize,
845     _Out_writes_bytes_opt_(cjOutBufferSize) LPVOID lpOutBuffer,
846     _In_ DWORD cjOutBufferSize,
847     _Out_ LPDWORD lpBytesReturned)
848 {
849     PIRP Irp;
850     NTSTATUS Status;
851     KEVENT Event;
852     IO_STATUS_BLOCK Iosb;
853     PDEVICE_OBJECT DeviceObject;
854 
855     TRACE("EngDeviceIoControl() called\n");
856 
857     if (!hDevice)
858     {
859         return ERROR_INVALID_HANDLE;
860     }
861 
862     KeInitializeEvent(&Event, SynchronizationEvent, FALSE);
863 
864     DeviceObject = (PDEVICE_OBJECT) hDevice;
865 
866     Irp = IoBuildDeviceIoControlRequest(dwIoControlCode,
867                                         DeviceObject,
868                                         lpInBuffer,
869                                         cjInBufferSize,
870                                         lpOutBuffer,
871                                         cjOutBufferSize,
872                                         FALSE,
873                                         &Event,
874                                         &Iosb);
875     if (!Irp) return ERROR_NOT_ENOUGH_MEMORY;
876 
877     Status = IoCallDriver(DeviceObject, Irp);
878 
879     if (Status == STATUS_PENDING)
880     {
881         (VOID)KeWaitForSingleObject(&Event, Executive, KernelMode, TRUE, 0);
882         Status = Iosb.Status;
883     }
884 
885     TRACE("EngDeviceIoControl(): Returning %X/%X\n", Iosb.Status,
886            Iosb.Information);
887 
888     /* Return information to the caller about the operation. */
889     *lpBytesReturned = (DWORD)Iosb.Information;
890 
891     /* Convert NT status values to win32 error codes. */
892     switch (Status)
893     {
894         case STATUS_INSUFFICIENT_RESOURCES:
895             return ERROR_NOT_ENOUGH_MEMORY;
896 
897         case STATUS_BUFFER_OVERFLOW:
898             return ERROR_MORE_DATA;
899 
900         case STATUS_NOT_IMPLEMENTED:
901             return ERROR_INVALID_FUNCTION;
902 
903         case STATUS_INVALID_PARAMETER:
904             return ERROR_INVALID_PARAMETER;
905 
906         case STATUS_BUFFER_TOO_SMALL:
907             return ERROR_INSUFFICIENT_BUFFER;
908 
909         case STATUS_DEVICE_DOES_NOT_EXIST:
910             return ERROR_DEV_NOT_EXIST;
911 
912         case STATUS_PENDING:
913             return ERROR_IO_PENDING;
914     }
915 
916     return Status;
917 }
918 
919 /* EOF */
920