1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS Win32k subsystem 4 * PURPOSE: Support for logical devices 5 * FILE: win32ss/gdi/eng/ldevobj.c 6 * PROGRAMER: Timo Kreuzer (timo.kreuzer@reactos.org) 7 */ 8 9 #include <win32k.h> 10 #define NDEBUG 11 #include <debug.h> 12 DBG_DEFAULT_CHANNEL(EngLDev); 13 14 #ifndef RVA_TO_ADDR 15 #define RVA_TO_ADDR(Base,Rva) ((PVOID)(((ULONG_PTR)(Base)) + (Rva))) 16 #endif 17 18 /** Globals *******************************************************************/ 19 20 static HSEMAPHORE ghsemLDEVList; 21 static LIST_ENTRY gleLdevListHead; 22 static LDEVOBJ *gpldevWin32k = NULL; 23 24 25 /** Private functions *********************************************************/ 26 27 CODE_SEG("INIT") 28 NTSTATUS 29 NTAPI 30 InitLDEVImpl(VOID) 31 { 32 ULONG cbSize; 33 34 /* Initialize the LDEV list head */ 35 InitializeListHead(&gleLdevListHead); 36 37 /* Initialize the loader lock */ 38 ghsemLDEVList = EngCreateSemaphore(); 39 if (!ghsemLDEVList) 40 { 41 ERR("Failed to create ghsemLDEVList\n"); 42 return STATUS_INSUFFICIENT_RESOURCES; 43 } 44 45 /* Allocate a LDEVOBJ for win32k */ 46 gpldevWin32k = ExAllocatePoolWithTag(PagedPool, 47 sizeof(LDEVOBJ) + 48 sizeof(SYSTEM_GDI_DRIVER_INFORMATION), 49 GDITAG_LDEV); 50 if (!gpldevWin32k) 51 { 52 return STATUS_NO_MEMORY; 53 } 54 55 /* Initialize the LDEVOBJ for win32k */ 56 gpldevWin32k->leLink.Flink = NULL; 57 gpldevWin32k->leLink.Blink = NULL; 58 gpldevWin32k->ldevtype = LDEV_DEVICE_DISPLAY; 59 gpldevWin32k->cRefs = 1; 60 gpldevWin32k->ulDriverVersion = GDI_ENGINE_VERSION; 61 gpldevWin32k->pGdiDriverInfo = (PVOID)(gpldevWin32k + 1); 62 RtlInitUnicodeString(&gpldevWin32k->pGdiDriverInfo->DriverName, 63 L"\\SystemRoot\\System32\\win32k.sys"); 64 gpldevWin32k->pGdiDriverInfo->ImageAddress = &__ImageBase; 65 gpldevWin32k->pGdiDriverInfo->SectionPointer = NULL; 66 gpldevWin32k->pGdiDriverInfo->EntryPoint = (PVOID)DriverEntry; 67 gpldevWin32k->pGdiDriverInfo->ExportSectionPointer = 68 RtlImageDirectoryEntryToData(&__ImageBase, 69 TRUE, 70 IMAGE_DIRECTORY_ENTRY_EXPORT, 71 &cbSize); 72 gpldevWin32k->pGdiDriverInfo->ImageLength = 0; // FIXME 73 74 return STATUS_SUCCESS; 75 } 76 77 static 78 PLDEVOBJ 79 LDEVOBJ_AllocLDEV( 80 _In_ LDEVTYPE ldevtype) 81 { 82 PLDEVOBJ pldev; 83 84 /* Allocate the structure from paged pool */ 85 pldev = ExAllocatePoolWithTag(PagedPool, sizeof(LDEVOBJ), GDITAG_LDEV); 86 if (!pldev) 87 { 88 ERR("Failed to allocate LDEVOBJ.\n"); 89 return NULL; 90 } 91 92 /* Zero out the structure */ 93 RtlZeroMemory(pldev, sizeof(LDEVOBJ)); 94 95 /* Set the ldevtype */ 96 pldev->ldevtype = ldevtype; 97 98 return pldev; 99 } 100 101 static 102 VOID 103 LDEVOBJ_vFreeLDEV( 104 _In_ _Post_ptr_invalid_ PLDEVOBJ pldev) 105 { 106 /* Make sure we don't have a driver loaded */ 107 ASSERT(pldev && pldev->pGdiDriverInfo == NULL); 108 ASSERT(pldev->cRefs == 0); 109 110 /* Free the memory */ 111 ExFreePoolWithTag(pldev, GDITAG_LDEV); 112 } 113 114 static 115 BOOL 116 LDEVOBJ_bLoadImage( 117 _Inout_ PLDEVOBJ pldev, 118 _In_ PUNICODE_STRING pustrPathName) 119 { 120 PSYSTEM_GDI_DRIVER_INFORMATION pDriverInfo; 121 NTSTATUS Status; 122 ULONG cbSize; 123 124 /* Make sure no image is loaded yet */ 125 ASSERT(pldev && pldev->pGdiDriverInfo == NULL); 126 127 /* Allocate a SYSTEM_GDI_DRIVER_INFORMATION structure */ 128 cbSize = sizeof(SYSTEM_GDI_DRIVER_INFORMATION) + pustrPathName->Length; 129 pDriverInfo = ExAllocatePoolWithTag(PagedPool, cbSize, GDITAG_LDEV); 130 if (!pDriverInfo) 131 { 132 ERR("Failed to allocate SYSTEM_GDI_DRIVER_INFORMATION\n"); 133 return FALSE; 134 } 135 136 /* Initialize the UNICODE_STRING and copy the driver name */ 137 RtlInitEmptyUnicodeString(&pDriverInfo->DriverName, 138 (PWSTR)(pDriverInfo + 1), 139 pustrPathName->Length); 140 RtlCopyUnicodeString(&pDriverInfo->DriverName, pustrPathName); 141 142 /* Try to load the driver */ 143 Status = ZwSetSystemInformation(SystemLoadGdiDriverInformation, 144 pDriverInfo, 145 sizeof(SYSTEM_GDI_DRIVER_INFORMATION)); 146 if (!NT_SUCCESS(Status)) 147 { 148 ERR("Failed to load a GDI driver: '%wZ', Status = 0x%lx\n", 149 pustrPathName, Status); 150 151 /* Free the allocated memory */ 152 ExFreePoolWithTag(pDriverInfo, GDITAG_LDEV); 153 return FALSE; 154 } 155 156 /* Set the driver info */ 157 pldev->pGdiDriverInfo = pDriverInfo; 158 159 /* Return success. */ 160 return TRUE; 161 } 162 163 static 164 BOOL 165 LDEVOBJ_bEnableDriver( 166 _Inout_ PLDEVOBJ pldev, 167 _In_ PFN_DrvEnableDriver pfnEnableDriver) 168 { 169 DRVENABLEDATA ded; 170 ULONG i; 171 172 TRACE("LDEVOBJ_bEnableDriver('%wZ')\n", &pldev->pGdiDriverInfo->DriverName); 173 174 ASSERT(pldev); 175 ASSERT(pldev->cRefs == 0); 176 177 if (pldev->ldevtype == LDEV_IMAGE) 178 return TRUE; 179 180 /* Call the drivers DrvEnableDriver function */ 181 RtlZeroMemory(&ded, sizeof(ded)); 182 if (!pfnEnableDriver(GDI_ENGINE_VERSION, sizeof(ded), &ded)) 183 { 184 ERR("DrvEnableDriver failed\n"); 185 return FALSE; 186 } 187 188 /* Copy the returned driver version */ 189 pldev->ulDriverVersion = ded.iDriverVersion; 190 191 /* Fill the driver function array */ 192 for (i = 0; i < ded.c; i++) 193 { 194 pldev->apfn[ded.pdrvfn[i].iFunc] = ded.pdrvfn[i].pfn; 195 } 196 197 /* Return success. */ 198 return TRUE; 199 } 200 201 static 202 VOID 203 LDEVOBJ_vDisableDriver( 204 _Inout_ PLDEVOBJ pldev) 205 { 206 ASSERT(pldev); 207 ASSERT(pldev->cRefs == 0); 208 209 TRACE("LDEVOBJ_vDisableDriver('%wZ')\n", &pldev->pGdiDriverInfo->DriverName); 210 211 if (pldev->ldevtype == LDEV_IMAGE) 212 return; 213 214 if (pldev->pfn.DisableDriver) 215 { 216 /* Call the unload function */ 217 pldev->pfn.DisableDriver(); 218 } 219 } 220 221 static 222 PVOID 223 LDEVOBJ_pvFindImageProcAddress( 224 _In_ PLDEVOBJ pldev, 225 _In_z_ LPSTR pszProcName) 226 { 227 PVOID pvImageBase; 228 PIMAGE_EXPORT_DIRECTORY pExportDir; 229 PVOID pvProcAdress = NULL; 230 PUSHORT pOrdinals; 231 PULONG pNames, pAddresses; 232 ULONG i; 233 234 /* Make sure we have a driver info */ 235 ASSERT(pldev && pldev->pGdiDriverInfo != NULL); 236 237 /* Get the pointer to the export directory */ 238 pvImageBase = pldev->pGdiDriverInfo->ImageAddress; 239 pExportDir = pldev->pGdiDriverInfo->ExportSectionPointer; 240 if (!pExportDir) 241 { 242 ERR("LDEVOBJ_pvFindImageProcAddress: no export section found\n"); 243 return NULL; 244 } 245 246 /* Get pointers to some tables */ 247 pNames = RVA_TO_ADDR(pvImageBase, pExportDir->AddressOfNames); 248 pOrdinals = RVA_TO_ADDR(pvImageBase, pExportDir->AddressOfNameOrdinals); 249 pAddresses = RVA_TO_ADDR(pvImageBase, pExportDir->AddressOfFunctions); 250 251 /* Loop the export table */ 252 for (i = 0; i < pExportDir->NumberOfNames; i++) 253 { 254 /* Compare the name */ 255 if (_stricmp(pszProcName, RVA_TO_ADDR(pvImageBase, pNames[i])) == 0) 256 { 257 /* Found! Calculate the procedure address */ 258 pvProcAdress = RVA_TO_ADDR(pvImageBase, pAddresses[pOrdinals[i]]); 259 break; 260 } 261 } 262 263 /* Return the address */ 264 return pvProcAdress; 265 } 266 267 static 268 BOOL 269 LDEVOBJ_bUnloadImage( 270 _Inout_ PLDEVOBJ pldev) 271 { 272 NTSTATUS Status; 273 274 /* Make sure we have a driver info */ 275 ASSERT(pldev && pldev->pGdiDriverInfo != NULL); 276 ASSERT(pldev->cRefs == 0); 277 278 TRACE("LDEVOBJ_bUnloadImage('%wZ')\n", &pldev->pGdiDriverInfo->DriverName); 279 280 /* Unload the driver */ 281 #if 0 282 Status = ZwSetSystemInformation(SystemUnloadGdiDriverInformation, 283 &pldev->pGdiDriverInfo->SectionPointer, 284 sizeof(HANDLE)); 285 #else 286 /* Unfortunately, ntoskrnl allows unloading a driver, but fails loading 287 * it again with STATUS_IMAGE_ALREADY_LOADED. Prevent this problem by 288 * never unloading any driver. 289 */ 290 UNIMPLEMENTED; 291 Status = STATUS_NOT_IMPLEMENTED; 292 #endif 293 if (!NT_SUCCESS(Status)) 294 return FALSE; 295 296 ExFreePoolWithTag(pldev->pGdiDriverInfo, GDITAG_LDEV); 297 pldev->pGdiDriverInfo = NULL; 298 299 return TRUE; 300 } 301 302 PLDEVOBJ 303 LDEVOBJ_pLoadInternal( 304 _In_ PFN_DrvEnableDriver pfnEnableDriver, 305 _In_ ULONG ldevtype) 306 { 307 PLDEVOBJ pldev; 308 309 TRACE("LDEVOBJ_pLoadInternal(%lu)\n", ldevtype); 310 311 /* Lock loader */ 312 EngAcquireSemaphore(ghsemLDEVList); 313 314 /* Allocate a new LDEVOBJ */ 315 pldev = LDEVOBJ_AllocLDEV(ldevtype); 316 if (!pldev) 317 { 318 ERR("Could not allocate LDEV\n"); 319 goto leave; 320 } 321 322 /* Load the driver */ 323 if (!LDEVOBJ_bEnableDriver(pldev, pfnEnableDriver)) 324 { 325 ERR("LDEVOBJ_bEnableDriver failed\n"); 326 LDEVOBJ_vFreeLDEV(pldev); 327 pldev = NULL; 328 goto leave; 329 } 330 331 /* Insert the LDEV into the global list */ 332 InsertHeadList(&gleLdevListHead, &pldev->leLink); 333 334 /* Increase ref count */ 335 pldev->cRefs++; 336 337 leave: 338 /* Unlock loader */ 339 EngReleaseSemaphore(ghsemLDEVList); 340 341 TRACE("LDEVOBJ_pLoadInternal returning %p\n", pldev); 342 return pldev; 343 } 344 345 PLDEVOBJ 346 NTAPI 347 LDEVOBJ_pLoadDriver( 348 _In_z_ LPWSTR pwszDriverName, 349 _In_ ULONG ldevtype) 350 { 351 WCHAR acwBuffer[MAX_PATH]; 352 PLIST_ENTRY pleLink; 353 PLDEVOBJ pldev; 354 UNICODE_STRING strDriverName; 355 SIZE_T cwcLength; 356 LPWSTR pwsz; 357 358 TRACE("LDEVOBJ_pLoadDriver(%ls, %lu)\n", pwszDriverName, ldevtype); 359 ASSERT(pwszDriverName); 360 361 /* Initialize buffer for the the driver name */ 362 RtlInitEmptyUnicodeString(&strDriverName, acwBuffer, sizeof(acwBuffer)); 363 364 /* Start path with systemroot */ 365 RtlAppendUnicodeToString(&strDriverName, L"\\SystemRoot\\System32\\"); 366 367 /* Get Length of given string */ 368 cwcLength = wcslen(pwszDriverName); 369 370 /* Check if we have a system32 path given */ 371 pwsz = pwszDriverName + cwcLength; 372 while (pwsz > pwszDriverName) 373 { 374 if ((*pwsz == L'\\') && (_wcsnicmp(pwsz, L"\\system32\\", 10) == 0)) 375 { 376 /* Driver name starts after system32 */ 377 pwsz += 10; 378 break; 379 } 380 pwsz--; 381 } 382 383 /* Append the driver name */ 384 RtlAppendUnicodeToString(&strDriverName, pwsz); 385 386 /* MSDN says "The driver must include this suffix in the pwszDriver string." 387 But in fact it's optional. The function can also load .sys files without 388 appending the .dll extension. */ 389 if ((cwcLength < 4) || 390 ((_wcsnicmp(pwszDriverName + cwcLength - 4, L".dll", 4) != 0) && 391 (_wcsnicmp(pwszDriverName + cwcLength - 4, L".sys", 4) != 0)) ) 392 { 393 /* Append the .dll suffix */ 394 RtlAppendUnicodeToString(&strDriverName, L".dll"); 395 } 396 397 /* Lock loader */ 398 EngAcquireSemaphore(ghsemLDEVList); 399 400 /* Search the List of LDEVS for the driver name */ 401 for (pleLink = gleLdevListHead.Flink; 402 pleLink != &gleLdevListHead; 403 pleLink = pleLink->Flink) 404 { 405 pldev = CONTAINING_RECORD(pleLink, LDEVOBJ, leLink); 406 407 /* Check if the ldev is associated with a file */ 408 if (pldev->pGdiDriverInfo) 409 { 410 /* Check for match (case insensative) */ 411 if (RtlEqualUnicodeString(&pldev->pGdiDriverInfo->DriverName, &strDriverName, TRUE)) 412 { 413 /* Image found in LDEV list */ 414 break; 415 } 416 } 417 } 418 419 /* Did we find one? */ 420 if (pleLink == &gleLdevListHead) 421 { 422 /* No, allocate a new LDEVOBJ */ 423 pldev = LDEVOBJ_AllocLDEV(ldevtype); 424 if (!pldev) 425 { 426 ERR("Could not allocate LDEV\n"); 427 goto leave; 428 } 429 430 /* Load the image */ 431 if (!LDEVOBJ_bLoadImage(pldev, &strDriverName)) 432 { 433 LDEVOBJ_vFreeLDEV(pldev); 434 pldev = NULL; 435 ERR("LDEVOBJ_bLoadImage failed\n"); 436 goto leave; 437 } 438 439 /* Load the driver */ 440 if (!LDEVOBJ_bEnableDriver(pldev, pldev->pGdiDriverInfo->EntryPoint)) 441 { 442 ERR("LDEVOBJ_bEnableDriver failed\n"); 443 444 /* Unload the image. */ 445 if (LDEVOBJ_bUnloadImage(pldev)) 446 LDEVOBJ_vFreeLDEV(pldev); 447 else 448 ERR("Could not unload driver. Leaking memory\n"); 449 pldev = NULL; 450 goto leave; 451 } 452 453 /* Insert the LDEV into the global list */ 454 InsertHeadList(&gleLdevListHead, &pldev->leLink); 455 } 456 457 /* Increase ref count */ 458 pldev->cRefs++; 459 460 leave: 461 /* Unlock loader */ 462 EngReleaseSemaphore(ghsemLDEVList); 463 464 TRACE("LDEVOBJ_pLoadDriver returning %p\n", pldev); 465 return pldev; 466 } 467 468 static 469 VOID 470 LDEVOBJ_vDereference( 471 _Inout_ PLDEVOBJ pldev) 472 { 473 /* Lock loader */ 474 EngAcquireSemaphore(ghsemLDEVList); 475 476 /* Decrement reference count */ 477 ASSERT(pldev->cRefs > 0); 478 pldev->cRefs--; 479 480 /* More references left? */ 481 if (pldev->cRefs > 0) 482 { 483 EngReleaseSemaphore(ghsemLDEVList); 484 return; 485 } 486 487 LDEVOBJ_vDisableDriver(pldev); 488 489 if (LDEVOBJ_bUnloadImage(pldev)) 490 { 491 /* Remove ldev from the list */ 492 RemoveEntryList(&pldev->leLink); 493 494 /* Free the driver info structure */ 495 LDEVOBJ_vFreeLDEV(pldev); 496 } 497 else 498 { 499 WARN("Failed to unload driver '%wZ', trying to re-enable it.\n", &pldev->pGdiDriverInfo->DriverName); 500 LDEVOBJ_bEnableDriver(pldev, pldev->pGdiDriverInfo->EntryPoint); 501 502 /* Increment again reference count */ 503 pldev->cRefs++; 504 } 505 506 /* Unlock loader */ 507 EngReleaseSemaphore(ghsemLDEVList); 508 } 509 510 ULONG 511 LDEVOBJ_ulGetDriverModes( 512 _In_ LPWSTR pwszDriverName, 513 _In_ HANDLE hDriver, 514 _Out_ PDEVMODEW *ppdm) 515 { 516 PLDEVOBJ pldev = NULL; 517 ULONG cbSize = 0; 518 PDEVMODEW pdm = NULL; 519 520 TRACE("LDEVOBJ_ulGetDriverModes('%ls', %p)\n", pwszDriverName, hDriver); 521 522 pldev = LDEVOBJ_pLoadDriver(pwszDriverName, LDEV_DEVICE_DISPLAY); 523 if (!pldev) 524 goto cleanup; 525 526 /* Mirror drivers may omit this function */ 527 if (!pldev->pfn.GetModes) 528 goto cleanup; 529 530 /* Call the driver to get the required size */ 531 cbSize = pldev->pfn.GetModes(hDriver, 0, NULL); 532 if (!cbSize) 533 { 534 ERR("DrvGetModes returned 0\n"); 535 goto cleanup; 536 } 537 538 /* Allocate a buffer for the DEVMODE array */ 539 pdm = ExAllocatePoolWithTag(PagedPool, cbSize, GDITAG_DEVMODE); 540 if (!pdm) 541 { 542 ERR("Could not allocate devmodeinfo\n"); 543 goto cleanup; 544 } 545 546 /* Call the driver again to fill the buffer */ 547 cbSize = pldev->pfn.GetModes(hDriver, cbSize, pdm); 548 if (!cbSize) 549 { 550 /* Could not get modes */ 551 ERR("DrvrGetModes returned 0 on second call\n"); 552 ExFreePoolWithTag(pdm, GDITAG_DEVMODE); 553 pdm = NULL; 554 } 555 556 cleanup: 557 if (pldev) 558 LDEVOBJ_vDereference(pldev); 559 560 *ppdm = pdm; 561 return cbSize; 562 } 563 564 BOOL 565 LDEVOBJ_bBuildDevmodeList( 566 _Inout_ PGRAPHICS_DEVICE pGraphicsDevice) 567 { 568 PWSTR pwsz; 569 PDEVMODEINFO pdminfo; 570 PDEVMODEW pdm, pdmEnd; 571 ULONG i, cModes = 0; 572 ULONG cbSize, cbFull; 573 574 if (pGraphicsDevice->pdevmodeInfo) 575 return TRUE; 576 ASSERT(pGraphicsDevice->pDevModeList == NULL); 577 578 pwsz = pGraphicsDevice->pDiplayDrivers; 579 580 /* Loop through the driver names 581 * This is a REG_MULTI_SZ string */ 582 for (; *pwsz; pwsz += wcslen(pwsz) + 1) 583 { 584 /* Get the mode list from the driver */ 585 TRACE("Trying driver: %ls\n", pwsz); 586 cbSize = LDEVOBJ_ulGetDriverModes(pwsz, pGraphicsDevice->DeviceObject, &pdm); 587 if (!cbSize) 588 { 589 WARN("Driver %ls returned no valid mode\n", pwsz); 590 continue; 591 } 592 593 /* Add space for the header */ 594 cbFull = cbSize + FIELD_OFFSET(DEVMODEINFO, adevmode); 595 596 /* Allocate a buffer for the DEVMODE array */ 597 pdminfo = ExAllocatePoolWithTag(PagedPool, cbFull, GDITAG_DEVMODE); 598 if (!pdminfo) 599 { 600 ERR("Could not allocate devmodeinfo\n"); 601 ExFreePoolWithTag(pdm, GDITAG_DEVMODE); 602 continue; 603 } 604 605 pdminfo->cbdevmode = cbSize; 606 RtlCopyMemory(pdminfo->adevmode, pdm, cbSize); 607 ExFreePoolWithTag(pdm, GDITAG_DEVMODE); 608 609 /* Attach the mode info to the device */ 610 pdminfo->pdmiNext = pGraphicsDevice->pdevmodeInfo; 611 pGraphicsDevice->pdevmodeInfo = pdminfo; 612 613 /* Loop all DEVMODEs */ 614 pdmEnd = (DEVMODEW*)((PCHAR)pdminfo->adevmode + pdminfo->cbdevmode); 615 for (pdm = pdminfo->adevmode; 616 (pdm + 1 <= pdmEnd) && (pdm->dmSize != 0); 617 pdm = (DEVMODEW*)((PCHAR)pdm + pdm->dmSize + pdm->dmDriverExtra)) 618 { 619 /* Count this DEVMODE */ 620 cModes++; 621 622 /* Some drivers like the VBox driver don't fill the dmDeviceName 623 with the name of the display driver. So fix that here. */ 624 RtlStringCbCopyW(pdm->dmDeviceName, sizeof(pdm->dmDeviceName), pwsz); 625 } 626 } 627 628 if (!pGraphicsDevice->pdevmodeInfo || cModes == 0) 629 { 630 ERR("No devmodes\n"); 631 return FALSE; 632 } 633 634 /* Allocate an index buffer */ 635 pGraphicsDevice->cDevModes = cModes; 636 pGraphicsDevice->pDevModeList = ExAllocatePoolWithTag(PagedPool, 637 cModes * sizeof(DEVMODEENTRY), 638 GDITAG_GDEVICE); 639 if (!pGraphicsDevice->pDevModeList) 640 { 641 ERR("No devmode list\n"); 642 return FALSE; 643 } 644 645 /* Loop through all DEVMODEINFOs */ 646 for (pdminfo = pGraphicsDevice->pdevmodeInfo, i = 0; 647 pdminfo; 648 pdminfo = pdminfo->pdmiNext) 649 { 650 /* Calculate End of the DEVMODEs */ 651 pdmEnd = (DEVMODEW*)((PCHAR)pdminfo->adevmode + pdminfo->cbdevmode); 652 653 /* Loop through the DEVMODEs */ 654 for (pdm = pdminfo->adevmode; 655 (pdm + 1 <= pdmEnd) && (pdm->dmSize != 0); 656 pdm = (PDEVMODEW)((PCHAR)pdm + pdm->dmSize + pdm->dmDriverExtra)) 657 { 658 TRACE(" %S has mode %lux%lux%lu(%lu Hz)\n", 659 pdm->dmDeviceName, 660 pdm->dmPelsWidth, 661 pdm->dmPelsHeight, 662 pdm->dmBitsPerPel, 663 pdm->dmDisplayFrequency); 664 665 /* Initialize the entry */ 666 pGraphicsDevice->pDevModeList[i].dwFlags = 0; 667 pGraphicsDevice->pDevModeList[i].pdm = pdm; 668 i++; 669 } 670 } 671 return TRUE; 672 } 673 674 /* Search the closest display mode according to some settings. 675 * Note that we don't care about the DM_* flags in dmFields, but check if value != 0 instead */ 676 static 677 BOOL 678 LDEVOBJ_bGetClosestMode( 679 _Inout_ PGRAPHICS_DEVICE pGraphicsDevice, 680 _In_ PDEVMODEW RequestedMode, 681 _Out_ PDEVMODEW *pSelectedMode) 682 { 683 DEVMODEW dmDiff; 684 PDEVMODEW pdmCurrent, pdmBest = NULL; 685 ULONG i; 686 687 /* Use a DEVMODE to keep the differences between best mode found and expected mode. 688 * Initialize fields to max value so we can find better modes. */ 689 dmDiff.dmPelsWidth = 0xffffffff; 690 dmDiff.dmPelsHeight = 0xffffffff; 691 dmDiff.dmBitsPerPel = 0xffffffff; 692 dmDiff.dmDisplayFrequency = 0xffffffff; 693 694 /* Search the closest mode */ 695 #define DM_DIFF(field) (RequestedMode->field > pdmCurrent->field ? (RequestedMode->field - pdmCurrent->field) : (pdmCurrent->field - RequestedMode->field)) 696 for (i = 0; i < pGraphicsDevice->cDevModes; i++) 697 { 698 pdmCurrent = pGraphicsDevice->pDevModeList[i].pdm; 699 700 /* Skip current mode if it is worse than best mode found */ 701 if (RequestedMode->dmPelsWidth != 0 && DM_DIFF(dmPelsWidth) > dmDiff.dmPelsWidth) 702 continue; 703 if (RequestedMode->dmPelsHeight != 0 && DM_DIFF(dmPelsHeight) > dmDiff.dmPelsHeight) 704 continue; 705 if (RequestedMode->dmBitsPerPel != 0 && DM_DIFF(dmBitsPerPel) > dmDiff.dmBitsPerPel) 706 continue; 707 if (RequestedMode->dmDisplayFrequency != 0 && DM_DIFF(dmDisplayFrequency) > dmDiff.dmDisplayFrequency) 708 continue; 709 710 /* Better (or equivalent) mode found. Update differences */ 711 dmDiff.dmPelsWidth = DM_DIFF(dmPelsWidth); 712 dmDiff.dmPelsHeight = DM_DIFF(dmPelsHeight); 713 dmDiff.dmBitsPerPel = DM_DIFF(dmBitsPerPel); 714 dmDiff.dmDisplayFrequency = DM_DIFF(dmDisplayFrequency); 715 pdmBest = pdmCurrent; 716 } 717 #undef DM_DIFF 718 719 if (pdmBest) 720 { 721 TRACE("Closest display mode to '%dx%dx%d %d Hz' is '%dx%dx%d %d Hz'\n", 722 RequestedMode->dmPelsWidth, 723 RequestedMode->dmPelsHeight, 724 RequestedMode->dmBitsPerPel, 725 RequestedMode->dmDisplayFrequency, 726 pdmBest->dmPelsWidth, 727 pdmBest->dmPelsHeight, 728 pdmBest->dmBitsPerPel, 729 pdmBest->dmDisplayFrequency); 730 } 731 732 *pSelectedMode = pdmBest; 733 return pdmBest != NULL; 734 } 735 736 BOOL 737 LDEVOBJ_bProbeAndCaptureDevmode( 738 _Inout_ PGRAPHICS_DEVICE pGraphicsDevice, 739 _In_ PDEVMODEW RequestedMode, 740 _Out_ PDEVMODEW *pSelectedMode, 741 _In_ BOOL bSearchClosestMode) 742 { 743 DEVMODEW dmSearch; 744 PDEVMODEW pdmCurrent, pdm, pdmSelected = NULL; 745 ULONG i; 746 ULONG ulVirtualWidth = 0, ulVirtualHeight = 0; 747 BOOL bResult = TRUE; 748 NTSTATUS Status; 749 750 if (!LDEVOBJ_bBuildDevmodeList(pGraphicsDevice)) 751 return FALSE; 752 753 /* At first, load information from registry */ 754 RtlZeroMemory(&dmSearch, sizeof(dmSearch)); 755 Status = EngpGetDisplayDriverParameters(pGraphicsDevice, &dmSearch); 756 if (!NT_SUCCESS(Status)) 757 { 758 ERR("EngpGetDisplayDriverParameters() failed with status 0x%08x\n", Status); 759 return FALSE; 760 } 761 762 /* Override values with the new ones provided */ 763 764 _SEH2_TRY 765 { 766 bSearchClosestMode |= RequestedMode->dmFields == 0; 767 768 /* Copy standard fields (if provided) */ 769 if (RequestedMode->dmFields & DM_BITSPERPEL && RequestedMode->dmBitsPerPel != 0) 770 dmSearch.dmBitsPerPel = RequestedMode->dmBitsPerPel; 771 if (RequestedMode->dmFields & DM_PELSWIDTH && RequestedMode->dmPelsWidth != 0) 772 dmSearch.dmPelsWidth = RequestedMode->dmPelsWidth; 773 if (RequestedMode->dmFields & DM_PELSHEIGHT && RequestedMode->dmPelsHeight != 0) 774 dmSearch.dmPelsHeight = RequestedMode->dmPelsHeight; 775 if (RequestedMode->dmFields & DM_DISPLAYFREQUENCY && RequestedMode->dmDisplayFrequency != 0) 776 dmSearch.dmDisplayFrequency = RequestedMode->dmDisplayFrequency; 777 778 if ((RequestedMode->dmFields & (DM_PANNINGWIDTH | DM_PANNINGHEIGHT)) == (DM_PANNINGWIDTH | DM_PANNINGHEIGHT) && 779 RequestedMode->dmPanningWidth != 0 && RequestedMode->dmPanningHeight != 0 && 780 RequestedMode->dmPanningWidth < dmSearch.dmPelsWidth && 781 RequestedMode->dmPanningHeight < dmSearch.dmPelsHeight) 782 { 783 /* Get new panning values */ 784 ulVirtualWidth = RequestedMode->dmPelsWidth; 785 ulVirtualHeight = RequestedMode->dmPelsHeight; 786 dmSearch.dmPelsWidth = RequestedMode->dmPanningWidth; 787 dmSearch.dmPelsHeight = RequestedMode->dmPanningHeight; 788 } 789 else if (dmSearch.dmPanningWidth != 0 && dmSearch.dmPanningHeight != 0 && 790 dmSearch.dmPanningWidth < dmSearch.dmPelsWidth && 791 dmSearch.dmPanningHeight < dmSearch.dmPelsHeight) 792 { 793 /* Keep existing panning values */ 794 ulVirtualWidth = dmSearch.dmPelsWidth; 795 ulVirtualHeight = dmSearch.dmPelsHeight; 796 dmSearch.dmPelsWidth = dmSearch.dmPanningWidth; 797 dmSearch.dmPelsHeight = dmSearch.dmPanningHeight; 798 } 799 } 800 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 801 { 802 bResult = FALSE; 803 } 804 _SEH2_END; 805 806 if (!bResult) 807 return FALSE; 808 809 if (LDEVOBJ_bGetClosestMode(pGraphicsDevice, &dmSearch, &pdmSelected)) 810 { 811 if (bSearchClosestMode) 812 { 813 /* Ok, found a closest mode. Update search */ 814 dmSearch.dmBitsPerPel = pdmSelected->dmBitsPerPel; 815 dmSearch.dmPelsWidth = pdmSelected->dmPelsWidth; 816 dmSearch.dmPelsHeight = pdmSelected->dmPelsHeight; 817 dmSearch.dmDisplayFrequency = pdmSelected->dmDisplayFrequency; 818 } 819 else 820 { 821 /* Only update not provided fields */ 822 _SEH2_TRY 823 { 824 if (!(RequestedMode->dmFields & DM_BITSPERPEL) || RequestedMode->dmBitsPerPel == 0) 825 dmSearch.dmBitsPerPel = pdmSelected->dmBitsPerPel; 826 if (!(RequestedMode->dmFields & DM_PELSWIDTH) || RequestedMode->dmPelsWidth == 0) 827 dmSearch.dmPelsWidth = pdmSelected->dmPelsWidth; 828 if (!(RequestedMode->dmFields & DM_PELSHEIGHT) || RequestedMode->dmPelsHeight == 0) 829 dmSearch.dmPelsHeight = pdmSelected->dmPelsHeight; 830 if (!(RequestedMode->dmFields & DM_DISPLAYFREQUENCY) || RequestedMode->dmDisplayFrequency == 0) 831 dmSearch.dmDisplayFrequency = pdmSelected->dmDisplayFrequency; 832 } 833 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 834 { 835 bResult = FALSE; 836 } 837 _SEH2_END; 838 839 if (!bResult) 840 return FALSE; 841 } 842 } 843 844 /* Now, search the exact mode to return to caller */ 845 for (i = 0; i < pGraphicsDevice->cDevModes; i++) 846 { 847 pdmCurrent = pGraphicsDevice->pDevModeList[i].pdm; 848 849 /* For now, we only need those */ 850 if (pdmCurrent->dmBitsPerPel != dmSearch.dmBitsPerPel) 851 continue; 852 if (pdmCurrent->dmPelsWidth != dmSearch.dmPelsWidth) 853 continue; 854 if (pdmCurrent->dmPelsHeight != dmSearch.dmPelsHeight) 855 continue; 856 if (pdmCurrent->dmDisplayFrequency != dmSearch.dmDisplayFrequency) 857 continue; 858 859 pdmSelected = pdmCurrent; 860 break; 861 } 862 863 if (!pdmSelected) 864 { 865 ERR("Requested mode not found (%dx%dx%d %d Hz)\n", 866 dmSearch.dmPelsWidth, 867 dmSearch.dmPelsHeight, 868 dmSearch.dmBitsPerPel, 869 dmSearch.dmDisplayFrequency); 870 return FALSE; 871 } 872 873 /* Allocate memory for output */ 874 pdm = ExAllocatePoolZero(PagedPool, pdmSelected->dmSize + pdmSelected->dmDriverExtra, GDITAG_DEVMODE); 875 if (!pdm) 876 return FALSE; 877 878 /* Copy selected mode */ 879 RtlCopyMemory(pdm, pdmSelected, pdmSelected->dmSize); 880 RtlCopyMemory((PVOID)((ULONG_PTR)pdm + pdm->dmSize), 881 (PVOID)((ULONG_PTR)pdmSelected + pdmSelected->dmSize), 882 pdmSelected->dmDriverExtra); 883 884 /* Add back panning */ 885 if (ulVirtualWidth != 0 && ulVirtualHeight != 0 && 886 pdm->dmPelsWidth < ulVirtualWidth && 887 pdm->dmPelsHeight < ulVirtualHeight) 888 { 889 pdm->dmFields |= DM_PANNINGWIDTH | DM_PANNINGHEIGHT; 890 pdm->dmPanningWidth = pdm->dmPelsWidth; 891 pdm->dmPanningHeight = pdm->dmPelsHeight; 892 pdm->dmPelsWidth = ulVirtualWidth; 893 pdm->dmPelsHeight = ulVirtualHeight; 894 } 895 896 *pSelectedMode = pdm; 897 return TRUE; 898 } 899 900 /** Exported functions ********************************************************/ 901 902 HANDLE 903 APIENTRY 904 EngLoadImage( 905 _In_ LPWSTR pwszDriverName) 906 { 907 return (HANDLE)LDEVOBJ_pLoadDriver(pwszDriverName, LDEV_IMAGE); 908 } 909 910 911 VOID 912 APIENTRY 913 EngUnloadImage( 914 _In_ HANDLE hModule) 915 { 916 PLDEVOBJ pldev = (PLDEVOBJ)hModule; 917 918 /* Make sure the LDEV is in the list */ 919 ASSERT((pldev->leLink.Flink != NULL) && (pldev->leLink.Blink != NULL)); 920 921 LDEVOBJ_vDereference(pldev); 922 } 923 924 925 PVOID 926 APIENTRY 927 EngFindImageProcAddress( 928 _In_ HANDLE hModule, 929 _In_ LPSTR lpProcName) 930 { 931 PLDEVOBJ pldev = (PLDEVOBJ)hModule; 932 933 ASSERT(gpldevWin32k != NULL); 934 935 /* Check if win32k is requested */ 936 if (!pldev) 937 { 938 pldev = gpldevWin32k; 939 } 940 941 /* Check if the drivers entry point is requested */ 942 if (_strnicmp(lpProcName, "DrvEnableDriver", 15) == 0) 943 { 944 return pldev->pGdiDriverInfo->EntryPoint; 945 } 946 947 /* Try to find the address */ 948 return LDEVOBJ_pvFindImageProcAddress(pldev, lpProcName); 949 } 950 951 /* EOF */ 952