1 /* 2 * PROJECT: ReactOS Win32k subsystem 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: win32ss/user/ntuser/windc.c 5 * PURPOSE: Window DC management 6 * COPYRIGHT: Copyright 2007 ReactOS Team 7 */ 8 9 #include <win32k.h> 10 DBG_DEFAULT_CHANNEL(UserDce); 11 12 /* GLOBALS *******************************************************************/ 13 14 /* NOTE: I think we should store this per window station (including GDI objects) */ 15 /* Answer: No, use the DCE pMonitor to compare with! */ 16 17 static LIST_ENTRY LEDce; 18 static INT DCECount = 0; // Count of DCE in system. 19 20 #define DCX_CACHECOMPAREMASK (DCX_CLIPSIBLINGS | DCX_CLIPCHILDREN | \ 21 DCX_NORESETATTRS | DCX_LOCKWINDOWUPDATE | \ 22 DCX_LAYEREDWIN | DCX_CACHE | DCX_WINDOW | \ 23 DCX_PARENTCLIP) 24 25 /* FUNCTIONS *****************************************************************/ 26 27 CODE_SEG("INIT") 28 NTSTATUS 29 NTAPI 30 InitDCEImpl(VOID) 31 { 32 InitializeListHead(&LEDce); 33 return STATUS_SUCCESS; 34 } 35 36 // 37 // This should be moved to dc.c or dcutil.c. 38 // 39 HDC FASTCALL 40 DceCreateDisplayDC(VOID) 41 { 42 UNICODE_STRING DriverName = RTL_CONSTANT_STRING(L"DISPLAY"); 43 44 if (!co_IntGraphicsCheck(TRUE)) 45 KeBugCheckEx(VIDEO_DRIVER_INIT_FAILURE, 0, 0, 0, USER_VERSION); 46 47 return IntGdiCreateDC(&DriverName, NULL, NULL, NULL, FALSE); 48 } 49 50 /* Returns the DCE pointer from the HDC handle */ 51 DCE* 52 FASTCALL 53 DceGetDceFromDC(HDC hdc) 54 { 55 PLIST_ENTRY ListEntry; 56 DCE* dce; 57 58 ListEntry = LEDce.Flink; 59 while (ListEntry != &LEDce) 60 { 61 dce = CONTAINING_RECORD(ListEntry, DCE, List); 62 ListEntry = ListEntry->Flink; 63 if (dce->hDC == hdc) 64 return dce; 65 } 66 67 return NULL; 68 } 69 70 static 71 PREGION FASTCALL 72 DceGetVisRgn(PWND Window, ULONG Flags, HWND hWndChild, ULONG CFlags) 73 { 74 PREGION Rgn; 75 Rgn = VIS_ComputeVisibleRegion( Window, 76 0 == (Flags & DCX_WINDOW), 77 0 != (Flags & DCX_CLIPCHILDREN), 78 0 != (Flags & DCX_CLIPSIBLINGS)); 79 /* Caller expects a non-null region */ 80 if (!Rgn) 81 Rgn = IntSysCreateRectpRgn(0, 0, 0, 0); 82 return Rgn; 83 } 84 85 PDCE FASTCALL 86 DceAllocDCE(PWND Window OPTIONAL, DCE_TYPE Type) 87 { 88 PDCE pDce; 89 90 pDce = ExAllocatePoolWithTag(PagedPool, sizeof(DCE), USERTAG_DCE); 91 if(!pDce) 92 return NULL; 93 94 pDce->hDC = DceCreateDisplayDC(); 95 if (!pDce->hDC) 96 { 97 ExFreePoolWithTag(pDce, USERTAG_DCE); 98 return NULL; 99 } 100 DCECount++; 101 TRACE("Alloc DCE's! %d\n",DCECount); 102 pDce->hwndCurrent = (Window ? Window->head.h : NULL); 103 pDce->pwndOrg = Window; 104 pDce->pwndClip = Window; 105 pDce->hrgnClip = NULL; 106 pDce->hrgnClipPublic = NULL; 107 pDce->hrgnSavedVis = NULL; 108 pDce->ppiOwner = NULL; 109 110 InsertTailList(&LEDce, &pDce->List); 111 112 DCU_SetDcUndeletable(pDce->hDC); 113 114 if (Type == DCE_WINDOW_DC || Type == DCE_CLASS_DC) // Window DCE have ownership. 115 { 116 pDce->ptiOwner = GetW32ThreadInfo(); 117 } 118 else 119 { 120 TRACE("FREE DCATTR!!!! NOT DCE_WINDOW_DC!!!!! hDC-> %p\n", pDce->hDC); 121 GreSetDCOwner(pDce->hDC, GDI_OBJ_HMGR_NONE); 122 pDce->ptiOwner = NULL; 123 } 124 125 if (Type == DCE_CACHE_DC) 126 { 127 pDce->DCXFlags = DCX_CACHE | DCX_DCEEMPTY; 128 } 129 else 130 { 131 pDce->DCXFlags = DCX_DCEBUSY; 132 if (Window) 133 { 134 if (Type == DCE_WINDOW_DC) 135 { 136 if (Window->style & WS_CLIPCHILDREN) pDce->DCXFlags |= DCX_CLIPCHILDREN; 137 if (Window->style & WS_CLIPSIBLINGS) pDce->DCXFlags |= DCX_CLIPSIBLINGS; 138 } 139 } 140 } 141 return(pDce); 142 } 143 144 static VOID APIENTRY 145 DceSetDrawable( PWND Window OPTIONAL, 146 HDC hDC, 147 ULONG Flags, 148 BOOL SetClipOrigin) 149 { 150 RECTL rect = {0,0,0,0}; 151 152 if (Window) 153 { 154 if (Flags & DCX_WINDOW) 155 { 156 rect = Window->rcWindow; 157 } 158 else 159 { 160 rect = Window->rcClient; 161 } 162 } 163 164 /* Set DC Origin and Window Rectangle */ 165 GreSetDCOrg( hDC, rect.left, rect.top, &rect); 166 } 167 168 169 static VOID FASTCALL 170 DceDeleteClipRgn(DCE* Dce) 171 { 172 Dce->DCXFlags &= ~(DCX_EXCLUDERGN | DCX_INTERSECTRGN); 173 174 if (Dce->DCXFlags & DCX_KEEPCLIPRGN ) 175 { 176 Dce->DCXFlags &= ~DCX_KEEPCLIPRGN; 177 } 178 else if (Dce->hrgnClip != NULL) 179 { 180 GreDeleteObject(Dce->hrgnClip); 181 } 182 183 Dce->hrgnClip = NULL; 184 185 /* Make it dirty so that the vis rgn gets recomputed next time */ 186 Dce->DCXFlags |= DCX_DCEDIRTY; 187 IntGdiSetHookFlags(Dce->hDC, DCHF_INVALIDATEVISRGN); 188 } 189 190 VOID 191 FASTCALL 192 DceUpdateVisRgn(DCE *Dce, PWND Window, ULONG Flags) 193 { 194 PREGION RgnVisible = NULL; 195 ULONG DcxFlags; 196 PWND DesktopWindow; 197 198 if (Flags & DCX_PARENTCLIP) 199 { 200 PWND Parent; 201 202 Parent = Window->spwndParent; 203 if (!Parent) 204 { 205 RgnVisible = NULL; 206 goto noparent; 207 } 208 209 if (Parent->style & WS_CLIPSIBLINGS) 210 { 211 DcxFlags = DCX_CLIPSIBLINGS | 212 (Flags & ~(DCX_CLIPCHILDREN | DCX_WINDOW)); 213 } 214 else 215 { 216 DcxFlags = Flags & ~(DCX_CLIPSIBLINGS | DCX_CLIPCHILDREN | DCX_WINDOW); 217 } 218 RgnVisible = DceGetVisRgn(Parent, DcxFlags, Window->head.h, Flags); 219 } 220 else if (Window == NULL) 221 { 222 DesktopWindow = UserGetWindowObject(IntGetDesktopWindow()); 223 if (NULL != DesktopWindow) 224 { 225 RgnVisible = IntSysCreateRectpRgnIndirect(&DesktopWindow->rcWindow); 226 } 227 else 228 { 229 RgnVisible = NULL; 230 } 231 } 232 else 233 { 234 RgnVisible = DceGetVisRgn(Window, Flags, 0, 0); 235 } 236 237 noparent: 238 if (Flags & DCX_INTERSECTRGN) 239 { 240 PREGION RgnClip = NULL; 241 242 if (Dce->hrgnClip != NULL) 243 RgnClip = REGION_LockRgn(Dce->hrgnClip); 244 245 if (RgnClip) 246 { 247 IntGdiCombineRgn(RgnVisible, RgnVisible, RgnClip, RGN_AND); 248 REGION_UnlockRgn(RgnClip); 249 } 250 else 251 { 252 if (RgnVisible != NULL) 253 { 254 REGION_Delete(RgnVisible); 255 } 256 RgnVisible = IntSysCreateRectpRgn(0, 0, 0, 0); 257 } 258 } 259 else if ((Flags & DCX_EXCLUDERGN) && Dce->hrgnClip != NULL) 260 { 261 PREGION RgnClip = REGION_LockRgn(Dce->hrgnClip); 262 IntGdiCombineRgn(RgnVisible, RgnVisible, RgnClip, RGN_DIFF); 263 REGION_UnlockRgn(RgnClip); 264 } 265 266 Dce->DCXFlags &= ~DCX_DCEDIRTY; 267 GdiSelectVisRgn(Dce->hDC, RgnVisible); 268 /* Tell GDI driver */ 269 if (Window) 270 IntEngWindowChanged(Window, WOC_RGN_CLIENT); 271 272 if (RgnVisible != NULL) 273 { 274 REGION_Delete(RgnVisible); 275 } 276 } 277 278 static INT FASTCALL 279 DceReleaseDC(DCE* dce, BOOL EndPaint) 280 { 281 if (DCX_DCEBUSY != (dce->DCXFlags & (DCX_INDESTROY | DCX_DCEEMPTY | DCX_DCEBUSY))) 282 { 283 return 0; 284 } 285 286 /* Restore previous visible region */ 287 if (EndPaint) 288 { 289 DceUpdateVisRgn(dce, dce->pwndOrg, dce->DCXFlags); 290 } 291 292 if ((dce->DCXFlags & (DCX_INTERSECTRGN | DCX_EXCLUDERGN)) && 293 ((dce->DCXFlags & DCX_CACHE) || EndPaint)) 294 { 295 DceDeleteClipRgn(dce); 296 } 297 298 if (dce->DCXFlags & DCX_CACHE) 299 { 300 if (!(dce->DCXFlags & DCX_NORESETATTRS)) 301 { 302 // Clean the DC 303 if (!IntGdiCleanDC(dce->hDC)) return 0; 304 305 if (dce->DCXFlags & DCX_DCEDIRTY) 306 { 307 /* Don't keep around invalidated entries 308 * because SetDCState() disables hVisRgn updates 309 * by removing dirty bit. */ 310 dce->hwndCurrent = 0; 311 dce->pwndOrg = NULL; 312 dce->pwndClip = NULL; 313 dce->DCXFlags &= DCX_CACHE; 314 dce->DCXFlags |= DCX_DCEEMPTY; 315 } 316 } 317 dce->DCXFlags &= ~DCX_DCEBUSY; 318 TRACE("Exit!!!!! DCX_CACHE!!!!!! hDC-> %p \n", dce->hDC); 319 if (!GreSetDCOwner(dce->hDC, GDI_OBJ_HMGR_NONE)) 320 return 0; 321 dce->ptiOwner = NULL; // Reset ownership. 322 dce->ppiOwner = NULL; 323 324 #if 0 // Need to research and fix before this is a "growing" issue. 325 if (++DCECache > 32) 326 { 327 ListEntry = LEDce.Flink; 328 while (ListEntry != &LEDce) 329 { 330 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 331 ListEntry = ListEntry->Flink; 332 if (!(pDCE->DCXFlags & DCX_DCEBUSY)) 333 { /* Free the unused cache DCEs. */ 334 DceFreeDCE(pDCE, TRUE); 335 } 336 } 337 } 338 #endif 339 } 340 return 1; // Released! 341 } 342 343 344 HDC FASTCALL 345 UserGetDCEx(PWND Wnd OPTIONAL, HANDLE ClipRegion, ULONG Flags) 346 { 347 PWND Parent; 348 ULONG DcxFlags; 349 DCE* Dce = NULL; 350 BOOL UpdateClipOrigin = FALSE; 351 BOOL bUpdateVisRgn = TRUE; 352 HDC hDC = NULL; 353 PPROCESSINFO ppi; 354 PLIST_ENTRY ListEntry; 355 356 if (NULL == Wnd) 357 { 358 Flags &= ~DCX_USESTYLE; 359 Flags |= DCX_CACHE; 360 } 361 362 if (Flags & DCX_PARENTCLIP) Flags |= DCX_CACHE; 363 364 // When GetDC is called with hWnd nz, DCX_CACHE & _WINDOW are clear w _USESTYLE set. 365 if (Flags & DCX_USESTYLE) 366 { 367 Flags &= ~(DCX_CLIPCHILDREN | DCX_CLIPSIBLINGS | DCX_PARENTCLIP); 368 if (!(Flags & DCX_WINDOW)) // Not window rectangle 369 { 370 if (Wnd->pcls->style & CS_PARENTDC) 371 { 372 Flags |= DCX_PARENTCLIP; 373 } 374 375 if (!(Flags & DCX_CACHE) && // Not on the cheap wine list. 376 !(Wnd->pcls->style & CS_OWNDC) ) 377 { 378 if (!(Wnd->pcls->style & CS_CLASSDC)) 379 // The window is not POWNED or has any CLASS, so we are looking for cheap wine. 380 Flags |= DCX_CACHE; 381 else 382 { 383 if (Wnd->pcls->pdce) hDC = ((PDCE)Wnd->pcls->pdce)->hDC; 384 TRACE("We have CLASS!!\n"); 385 } 386 } 387 388 if (Wnd->style & WS_CLIPSIBLINGS) 389 { 390 Flags |= DCX_CLIPSIBLINGS; 391 } 392 393 if (Wnd->style & WS_CLIPCHILDREN && 394 !(Wnd->style & WS_MINIMIZE)) 395 { 396 Flags |= DCX_CLIPCHILDREN; 397 } 398 /* If minized with icon in the set, we are forced to be cheap! */ 399 if (Wnd->style & WS_MINIMIZE && Wnd->pcls->spicn) 400 { 401 Flags |= DCX_CACHE; 402 } 403 } 404 else 405 { 406 if (Wnd->style & WS_CLIPSIBLINGS) Flags |= DCX_CLIPSIBLINGS; 407 Flags |= DCX_CACHE; 408 } 409 } 410 411 if (Flags & DCX_WINDOW) Flags &= ~DCX_CLIPCHILDREN; 412 413 if (Flags & DCX_NOCLIPCHILDREN) 414 { 415 Flags |= DCX_CACHE; 416 Flags &= ~(DCX_PARENTCLIP | DCX_CLIPCHILDREN); 417 } 418 419 Parent = (Wnd ? Wnd->spwndParent : NULL); 420 421 if (NULL == Wnd || !(Wnd->style & WS_CHILD) || NULL == Parent) 422 { 423 Flags &= ~DCX_PARENTCLIP; 424 Flags |= DCX_CLIPSIBLINGS; 425 } 426 427 /* It seems parent clip is ignored when clipping siblings or children */ 428 if (Flags & (DCX_CLIPSIBLINGS | DCX_CLIPCHILDREN)) Flags &= ~DCX_PARENTCLIP; 429 430 if (Flags & DCX_PARENTCLIP) 431 { 432 if ((Wnd->style & WS_VISIBLE) && 433 (Parent->style & WS_VISIBLE)) 434 { 435 Flags &= ~DCX_CLIPCHILDREN; 436 if (Parent->style & WS_CLIPSIBLINGS) 437 { 438 Flags |= DCX_CLIPSIBLINGS; 439 } 440 } 441 } 442 443 // Window nz, check to see if we still own this or it is just cheap wine tonight. 444 if (!(Flags & DCX_CACHE)) 445 { 446 if ( Wnd->head.pti != GetW32ThreadInfo()) 447 Flags |= DCX_CACHE; // Ah~ Not Powned! Forced to be cheap~ 448 } 449 450 DcxFlags = Flags & DCX_CACHECOMPAREMASK; 451 452 if (Flags & DCX_CACHE) 453 { // Scan the cheap wine list for our match. 454 DCE* DceEmpty = NULL; 455 DCE* DceUnused = NULL; 456 KeEnterCriticalRegion(); 457 ListEntry = LEDce.Flink; 458 while (ListEntry != &LEDce) 459 { 460 Dce = CONTAINING_RECORD(ListEntry, DCE, List); 461 ListEntry = ListEntry->Flink; 462 // 463 // The way I understand this, you can have more than one DC per window. 464 // Only one Owned if one was requested and saved and one Cached. 465 // 466 if ((Dce->DCXFlags & (DCX_CACHE | DCX_DCEBUSY)) == DCX_CACHE) 467 { 468 DceUnused = Dce; 469 if (Dce->DCXFlags & DCX_DCEEMPTY) 470 { 471 DceEmpty = Dce; 472 } 473 else if (Dce->hwndCurrent == (Wnd ? Wnd->head.h : NULL) && 474 ((Dce->DCXFlags & DCX_CACHECOMPAREMASK) == DcxFlags)) 475 { 476 UpdateClipOrigin = TRUE; 477 break; 478 } 479 } 480 Dce = NULL; // Loop issue? 481 } 482 KeLeaveCriticalRegion(); 483 484 Dce = (DceEmpty == NULL) ? DceUnused : DceEmpty; 485 486 if (Dce == NULL) 487 { 488 Dce = DceAllocDCE(NULL, DCE_CACHE_DC); 489 } 490 if (Dce == NULL) return NULL; 491 492 Dce->hwndCurrent = (Wnd ? Wnd->head.h : NULL); 493 Dce->pwndOrg = Dce->pwndClip = Wnd; 494 } 495 else // If we are here, we are POWNED or having CLASS. 496 { 497 KeEnterCriticalRegion(); 498 ListEntry = LEDce.Flink; 499 while (ListEntry != &LEDce) 500 { 501 Dce = CONTAINING_RECORD(ListEntry, DCE, List); 502 ListEntry = ListEntry->Flink; 503 504 // Skip Cache DCE entries. 505 if (!(Dce->DCXFlags & DCX_CACHE)) 506 { 507 // Check for Window handle than HDC match for CLASS. 508 if (Dce->hwndCurrent == Wnd->head.h) 509 { 510 bUpdateVisRgn = FALSE; 511 break; 512 } 513 else if (Dce->hDC == hDC) break; 514 } 515 Dce = NULL; // Loop issue? 516 } 517 KeLeaveCriticalRegion(); 518 519 if (Dce == NULL) 520 { 521 return(NULL); 522 } 523 524 if ( (Flags & (DCX_INTERSECTRGN|DCX_EXCLUDERGN)) && 525 (Dce->DCXFlags & (DCX_INTERSECTRGN|DCX_EXCLUDERGN)) ) 526 { 527 DceDeleteClipRgn(Dce); 528 } 529 } 530 // First time use hax, need to use DceAllocDCE during window display init. 531 if (NULL == Dce) 532 { 533 return(NULL); 534 } 535 536 if (!GreIsHandleValid(Dce->hDC)) 537 { 538 ERR("FIXME: Got DCE with invalid hDC! %p\n", Dce->hDC); 539 Dce->hDC = DceCreateDisplayDC(); 540 /* FIXME: Handle error */ 541 } 542 543 Dce->DCXFlags = Flags | DCX_DCEBUSY; 544 545 /* 546 * Bump it up! This prevents the random errors in wine dce tests and with 547 * proper bits set in DCX_CACHECOMPAREMASK. 548 * Reference: 549 * https://reactos.org/archives/public/ros-dev/2008-July/010498.html 550 * https://reactos.org/archives/public/ros-dev/2008-July/010499.html 551 */ 552 RemoveEntryList(&Dce->List); 553 InsertHeadList(&LEDce, &Dce->List); 554 555 /* Introduced in rev 6691 and modified later. */ 556 if ( (Flags & DCX_INTERSECTUPDATE) && !ClipRegion ) 557 { 558 Flags |= DCX_INTERSECTRGN | DCX_KEEPCLIPRGN; 559 Dce->DCXFlags |= DCX_INTERSECTRGN | DCX_KEEPCLIPRGN; 560 ClipRegion = Wnd->hrgnUpdate; 561 bUpdateVisRgn = TRUE; 562 } 563 564 if (ClipRegion == HRGN_WINDOW) 565 { 566 if (!(Flags & DCX_WINDOW)) 567 { 568 Dce->hrgnClip = NtGdiCreateRectRgn( 569 Wnd->rcClient.left, 570 Wnd->rcClient.top, 571 Wnd->rcClient.right, 572 Wnd->rcClient.bottom); 573 } 574 else 575 { 576 Dce->hrgnClip = NtGdiCreateRectRgn( 577 Wnd->rcWindow.left, 578 Wnd->rcWindow.top, 579 Wnd->rcWindow.right, 580 Wnd->rcWindow.bottom); 581 } 582 Dce->DCXFlags &= ~DCX_KEEPCLIPRGN; 583 bUpdateVisRgn = TRUE; 584 } 585 else if (ClipRegion != NULL) 586 { 587 if (Dce->hrgnClip != NULL) 588 { 589 ERR("Should not be called!!\n"); 590 GreDeleteObject(Dce->hrgnClip); 591 Dce->hrgnClip = NULL; 592 } 593 Dce->hrgnClip = ClipRegion; 594 bUpdateVisRgn = TRUE; 595 } 596 597 if (IntGdiSetHookFlags(Dce->hDC, DCHF_VALIDATEVISRGN)) bUpdateVisRgn = TRUE; 598 599 DceSetDrawable(Wnd, Dce->hDC, Flags, UpdateClipOrigin); 600 601 if (bUpdateVisRgn) DceUpdateVisRgn(Dce, Wnd, Flags); 602 603 if (Dce->DCXFlags & DCX_CACHE) 604 { 605 TRACE("ENTER!!!!!! DCX_CACHE!!!!!! hDC-> %p\n", Dce->hDC); 606 // Need to set ownership so Sync dcattr will work. 607 GreSetDCOwner(Dce->hDC, GDI_OBJ_HMGR_POWNED); 608 Dce->ptiOwner = GetW32ThreadInfo(); // Set the temp owning 609 } 610 611 if ( Wnd && 612 Wnd->ExStyle & WS_EX_LAYOUTRTL && 613 !(Flags & DCX_KEEPLAYOUT) ) 614 { 615 NtGdiSetLayout(Dce->hDC, -1, LAYOUT_RTL); 616 } 617 618 if (Dce->DCXFlags & DCX_PROCESSOWNED) 619 { 620 ppi = PsGetCurrentProcessWin32Process(); 621 ppi->W32PF_flags |= W32PF_OWNDCCLEANUP; 622 Dce->ptiOwner = NULL; 623 Dce->ppiOwner = ppi; 624 } 625 626 return(Dce->hDC); 627 } 628 629 /*********************************************************************** 630 * DceFreeDCE 631 */ 632 void FASTCALL 633 DceFreeDCE(PDCE pdce, BOOLEAN Force) 634 { 635 BOOL Hit = FALSE; 636 637 ASSERT(pdce != NULL); 638 if (NULL == pdce) return; 639 640 pdce->DCXFlags |= DCX_INDESTROY; 641 642 if (Force && 643 GreGetObjectOwner(pdce->hDC) != GDI_OBJ_HMGR_POWNED) 644 { 645 TRACE("Change ownership for DCE! -> %p\n" , pdce); 646 // NOTE: Windows sets W32PF_OWNDCCLEANUP and moves on. 647 if (GreIsHandleValid(pdce->hDC)) 648 { 649 GreSetDCOwner(pdce->hDC, GDI_OBJ_HMGR_POWNED); 650 } 651 else 652 { 653 ERR("Attempted to change ownership of an DCEhDC %p currently being destroyed!!!\n", 654 pdce->hDC); 655 Hit = TRUE; 656 } 657 } 658 else 659 { 660 if (GreGetObjectOwner(pdce->hDC) == GDI_OBJ_HMGR_PUBLIC) 661 GreSetDCOwner(pdce->hDC, GDI_OBJ_HMGR_POWNED); 662 } 663 664 if (!Hit) IntGdiDeleteDC(pdce->hDC, TRUE); 665 666 if (pdce->hrgnClip && !(pdce->DCXFlags & DCX_KEEPCLIPRGN)) 667 { 668 GreDeleteObject(pdce->hrgnClip); 669 pdce->hrgnClip = NULL; 670 } 671 672 RemoveEntryList(&pdce->List); 673 674 ExFreePoolWithTag(pdce, USERTAG_DCE); 675 676 DCECount--; 677 TRACE("Freed DCE's! %d \n", DCECount); 678 } 679 680 /*********************************************************************** 681 * DceFreeWindowDCE 682 * 683 * Remove owned DCE and reset unreleased cache DCEs. 684 */ 685 void FASTCALL 686 DceFreeWindowDCE(PWND Window) 687 { 688 PDCE pDCE; 689 PLIST_ENTRY ListEntry; 690 691 if (DCECount <= 0) 692 { 693 ERR("FreeWindowDCE No Entry! %d\n",DCECount); 694 return; 695 } 696 697 ListEntry = LEDce.Flink; 698 while (ListEntry != &LEDce) 699 { 700 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 701 ListEntry = ListEntry->Flink; 702 if ( pDCE->hwndCurrent == Window->head.h && 703 !(pDCE->DCXFlags & DCX_DCEEMPTY) ) 704 { 705 if (!(pDCE->DCXFlags & DCX_CACHE)) /* Owned or Class DCE */ 706 { 707 if (Window->pcls->style & CS_CLASSDC) /* Test Class first */ 708 { 709 if (pDCE->DCXFlags & (DCX_INTERSECTRGN | DCX_EXCLUDERGN)) /* Class DCE */ 710 DceDeleteClipRgn(pDCE); 711 // Update and reset Vis Rgn and clear the dirty bit. 712 // Should release VisRgn than reset it to default. 713 DceUpdateVisRgn(pDCE, Window, pDCE->DCXFlags); 714 pDCE->DCXFlags = DCX_DCEEMPTY|DCX_CACHE; 715 pDCE->hwndCurrent = 0; 716 pDCE->pwndOrg = pDCE->pwndClip = NULL; 717 718 TRACE("POWNED DCE going Cheap!! DCX_CACHE!! hDC-> %p \n", 719 pDCE->hDC); 720 if (!GreSetDCOwner( pDCE->hDC, GDI_OBJ_HMGR_NONE)) 721 { 722 ERR("Fail Owner Switch hDC-> %p \n", pDCE->hDC); 723 break; 724 } 725 /* Do not change owner so thread can clean up! */ 726 } 727 else if (Window->pcls->style & CS_OWNDC) /* Owned DCE */ 728 { 729 DceFreeDCE(pDCE, FALSE); 730 continue; 731 } 732 else 733 { 734 ERR("Not POWNED or CLASSDC hwndCurrent -> %p \n", 735 pDCE->hwndCurrent); 736 // ASSERT(FALSE); /* bug 5320 */ 737 } 738 } 739 else 740 { 741 if (pDCE->DCXFlags & DCX_DCEBUSY) /* Shared cache DCE */ 742 { 743 /* FIXME: AFAICS we are doing the right thing here so 744 * this should be a TRACE. But this is best left as an ERR 745 * because the 'application error' is likely to come from 746 * another part of Wine (i.e. it's our fault after all). 747 * We should change this to TRACE when ReactOS is more stable 748 * (for 1.0?). 749 */ 750 ERR("[%p] GetDC() without ReleaseDC()!\n", Window->head.h); 751 DceReleaseDC(pDCE, FALSE); 752 } 753 pDCE->DCXFlags |= DCX_DCEEMPTY; 754 pDCE->hwndCurrent = 0; 755 pDCE->pwndOrg = pDCE->pwndClip = NULL; 756 } 757 } 758 } 759 } 760 761 void FASTCALL 762 DceFreeClassDCE(PDCE pdceClass) 763 { 764 PDCE pDCE; 765 PLIST_ENTRY ListEntry; 766 767 ListEntry = LEDce.Flink; 768 while (ListEntry != &LEDce) 769 { 770 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 771 ListEntry = ListEntry->Flink; 772 if (pDCE == pdceClass) 773 { 774 DceFreeDCE(pDCE, TRUE); // Might have gone cheap! 775 } 776 } 777 } 778 779 void FASTCALL 780 DceFreeThreadDCE(PTHREADINFO pti) 781 { 782 PDCE pDCE; 783 PLIST_ENTRY ListEntry; 784 785 ListEntry = LEDce.Flink; 786 while (ListEntry != &LEDce) 787 { 788 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 789 ListEntry = ListEntry->Flink; 790 if (pDCE->ptiOwner == pti) 791 { 792 if (pDCE->DCXFlags & DCX_CACHE) 793 { 794 DceFreeDCE(pDCE, TRUE); 795 } 796 } 797 } 798 } 799 800 VOID FASTCALL 801 DceEmptyCache(VOID) 802 { 803 PDCE pDCE; 804 PLIST_ENTRY ListEntry; 805 806 ListEntry = LEDce.Flink; 807 while (ListEntry != &LEDce) 808 { 809 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 810 ListEntry = ListEntry->Flink; 811 DceFreeDCE(pDCE, TRUE); 812 } 813 } 814 815 VOID FASTCALL 816 DceResetActiveDCEs(PWND Window) 817 { 818 DCE *pDCE; 819 PDC dc; 820 PWND CurrentWindow; 821 INT DeltaX; 822 INT DeltaY; 823 PLIST_ENTRY ListEntry; 824 825 if (NULL == Window) 826 { 827 return; 828 } 829 830 ListEntry = LEDce.Flink; 831 while (ListEntry != &LEDce) 832 { 833 pDCE = CONTAINING_RECORD(ListEntry, DCE, List); 834 ListEntry = ListEntry->Flink; 835 if (0 == (pDCE->DCXFlags & (DCX_DCEEMPTY|DCX_INDESTROY))) 836 { 837 if (Window->head.h == pDCE->hwndCurrent) 838 { 839 CurrentWindow = Window; 840 } 841 else 842 { 843 if (!pDCE->hwndCurrent) 844 CurrentWindow = NULL; 845 else 846 CurrentWindow = UserGetWindowObject(pDCE->hwndCurrent); 847 if (NULL == CurrentWindow) 848 { 849 continue; 850 } 851 } 852 853 if (!GreIsHandleValid(pDCE->hDC) || 854 (dc = DC_LockDc(pDCE->hDC)) == NULL) 855 { 856 continue; 857 } 858 if (Window == CurrentWindow || IntIsChildWindow(Window, CurrentWindow)) 859 { 860 if (pDCE->DCXFlags & DCX_WINDOW) 861 { 862 DeltaX = CurrentWindow->rcWindow.left - dc->ptlDCOrig.x; 863 DeltaY = CurrentWindow->rcWindow.top - dc->ptlDCOrig.y; 864 dc->ptlDCOrig.x = CurrentWindow->rcWindow.left; 865 dc->ptlDCOrig.y = CurrentWindow->rcWindow.top; 866 } 867 else 868 { 869 DeltaX = CurrentWindow->rcClient.left - dc->ptlDCOrig.x; 870 DeltaY = CurrentWindow->rcClient.top - dc->ptlDCOrig.y; 871 dc->ptlDCOrig.x = CurrentWindow->rcClient.left; 872 dc->ptlDCOrig.y = CurrentWindow->rcClient.top; 873 } 874 875 if (NULL != dc->dclevel.prgnClip) 876 { 877 REGION_bOffsetRgn(dc->dclevel.prgnClip, DeltaX, DeltaY); 878 dc->fs |= DC_DIRTY_RAO; 879 } 880 if (NULL != pDCE->hrgnClip) 881 { 882 NtGdiOffsetRgn(pDCE->hrgnClip, DeltaX, DeltaY); 883 } 884 } 885 DC_UnlockDc(dc); 886 887 DceUpdateVisRgn(pDCE, CurrentWindow, pDCE->DCXFlags); 888 IntGdiSetHookFlags(pDCE->hDC, DCHF_VALIDATEVISRGN); 889 } 890 } 891 } 892 893 HWND FASTCALL 894 IntWindowFromDC(HDC hDc) 895 { 896 DCE *Dce; 897 PLIST_ENTRY ListEntry; 898 HWND Ret = NULL; 899 900 ListEntry = LEDce.Flink; 901 while (ListEntry != &LEDce) 902 { 903 Dce = CONTAINING_RECORD(ListEntry, DCE, List); 904 ListEntry = ListEntry->Flink; 905 if (Dce->hDC == hDc) 906 { 907 if (Dce->DCXFlags & DCX_INDESTROY) 908 Ret = NULL; 909 else 910 Ret = Dce->hwndCurrent; 911 break; 912 } 913 } 914 return Ret; 915 } 916 917 INT FASTCALL 918 UserReleaseDC(PWND Window, HDC hDc, BOOL EndPaint) 919 { 920 PDCE dce; 921 PLIST_ENTRY ListEntry; 922 INT nRet = 0; 923 BOOL Hit = FALSE; 924 925 TRACE("%p %p\n", Window, hDc); 926 ListEntry = LEDce.Flink; 927 while (ListEntry != &LEDce) 928 { 929 dce = CONTAINING_RECORD(ListEntry, DCE, List); 930 ListEntry = ListEntry->Flink; 931 if (dce->hDC == hDc) 932 { 933 Hit = TRUE; 934 break; 935 } 936 } 937 938 if ( Hit && (dce->DCXFlags & DCX_DCEBUSY)) 939 { 940 nRet = DceReleaseDC(dce, EndPaint); 941 } 942 943 return nRet; 944 } 945 946 HDC FASTCALL 947 UserGetWindowDC(PWND Wnd) 948 { 949 return UserGetDCEx(Wnd, 0, DCX_USESTYLE | DCX_WINDOW); 950 } 951 952 HWND FASTCALL 953 UserGethWnd( HDC hdc, PWNDOBJ *pwndo) 954 { 955 EWNDOBJ* Clip; 956 PWND Wnd; 957 HWND hWnd; 958 959 hWnd = IntWindowFromDC(hdc); 960 961 if (hWnd && (Wnd = UserGetWindowObject(hWnd))) 962 { 963 Clip = (EWNDOBJ*)UserGetProp(Wnd, AtomWndObj, TRUE); 964 965 if ( Clip && Clip->Hwnd == hWnd ) 966 { 967 if (pwndo) *pwndo = (PWNDOBJ)Clip; 968 } 969 } 970 return hWnd; 971 } 972 973 HDC APIENTRY 974 NtUserGetDCEx(HWND hWnd OPTIONAL, HANDLE ClipRegion, ULONG Flags) 975 { 976 PWND Wnd=NULL; 977 DECLARE_RETURN(HDC); 978 979 TRACE("Enter NtUserGetDCEx: hWnd %p, ClipRegion %p, Flags %x.\n", 980 hWnd, ClipRegion, Flags); 981 UserEnterExclusive(); 982 983 if (hWnd && !(Wnd = UserGetWindowObject(hWnd))) 984 { 985 RETURN(NULL); 986 } 987 RETURN( UserGetDCEx(Wnd, ClipRegion, Flags)); 988 989 CLEANUP: 990 TRACE("Leave NtUserGetDCEx, ret=%p\n", _ret_); 991 UserLeave(); 992 END_CLEANUP; 993 } 994 995 /* 996 * NtUserGetWindowDC 997 * 998 * The NtUserGetWindowDC function retrieves the device context (DC) for the 999 * entire window, including title bar, menus, and scroll bars. A window device 1000 * context permits painting anywhere in a window, because the origin of the 1001 * device context is the upper-left corner of the window instead of the client 1002 * area. 1003 * 1004 * Status 1005 * @implemented 1006 */ 1007 HDC APIENTRY 1008 NtUserGetWindowDC(HWND hWnd) 1009 { 1010 return NtUserGetDCEx(hWnd, 0, DCX_USESTYLE | DCX_WINDOW); 1011 } 1012 1013 HDC APIENTRY 1014 NtUserGetDC(HWND hWnd) 1015 { 1016 TRACE("NtUGetDC -> %p:%x\n", hWnd, !hWnd ? DCX_CACHE | DCX_WINDOW : DCX_USESTYLE); 1017 1018 return NtUserGetDCEx(hWnd, NULL, NULL == hWnd ? DCX_CACHE | DCX_WINDOW : DCX_USESTYLE); 1019 } 1020 1021 /*! 1022 * Select logical palette into device context. 1023 * \param hDC handle to the device context 1024 * \param hpal handle to the palette 1025 * \param ForceBackground If this value is FALSE the logical palette will be copied to the device palette only when the application 1026 * is in the foreground. If this value is TRUE then map the colors in the logical palette to the device 1027 * palette colors in the best way. 1028 * \return old palette 1029 * 1030 * \todo implement ForceBackground == TRUE 1031 */ 1032 HPALETTE 1033 APIENTRY 1034 NtUserSelectPalette(HDC hDC, 1035 HPALETTE hpal, 1036 BOOL ForceBackground) 1037 { 1038 HPALETTE oldPal; 1039 UserEnterExclusive(); 1040 // Implement window checks 1041 oldPal = GdiSelectPalette( hDC, hpal, ForceBackground); 1042 UserLeave(); 1043 return oldPal; 1044 } 1045 1046 /* EOF */ 1047