2009-07-10 23:44:05 +00:00
|
|
|
/*
|
1999-08-01 11:21:05 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS kernel
|
|
|
|
* PURPOSE: GDI Driver Brush Functions
|
2015-11-10 17:41:55 +00:00
|
|
|
* FILE: win32ss/gdi/eng/engbrush.c
|
1999-08-01 11:21:05 +00:00
|
|
|
* PROGRAMER: Jason Filby
|
2009-03-23 00:38:59 +00:00
|
|
|
* Timo Kreuzer
|
1999-08-01 11:21:05 +00:00
|
|
|
*/
|
2005-06-29 07:09:25 +00:00
|
|
|
|
2010-04-26 13:58:46 +00:00
|
|
|
#include <win32k.h>
|
1999-08-01 11:21:05 +00:00
|
|
|
|
2015-03-25 22:26:52 +00:00
|
|
|
DBG_DEFAULT_CHANNEL(EngBrush);
|
2005-06-29 07:09:25 +00:00
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
static const ULONG gaulHatchBrushes[HS_DDI_MAX][8] =
|
|
|
|
{
|
|
|
|
{0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0xFF}, /* HS_HORIZONTAL */
|
|
|
|
{0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7}, /* HS_VERTICAL */
|
|
|
|
{0xFE, 0xFD, 0xFB, 0xF7, 0xEF, 0xDF, 0xBF, 0x7F}, /* HS_FDIAGONAL */
|
|
|
|
{0x7F, 0xBF, 0xDF, 0xEF, 0xF7, 0xFB, 0xFD, 0xFE}, /* HS_BDIAGONAL */
|
|
|
|
{0xF7, 0xF7, 0xF7, 0xF7, 0x00, 0xF7, 0xF7, 0xF7}, /* HS_CROSS */
|
|
|
|
{0x7E, 0xBD, 0xDB, 0xE7, 0xE7, 0xDB, 0xBD, 0x7E} /* HS_DIAGCROSS */
|
|
|
|
};
|
|
|
|
|
|
|
|
HSURF gahsurfHatch[HS_DDI_MAX];
|
|
|
|
|
2009-03-20 23:40:59 +00:00
|
|
|
/** Internal functions ********************************************************/
|
|
|
|
|
2020-10-06 19:44:01 +00:00
|
|
|
CODE_SEG("INIT")
|
2012-05-06 08:17:48 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
InitBrushImpl(VOID)
|
|
|
|
{
|
|
|
|
ULONG i;
|
|
|
|
SIZEL sizl = {8, 8};
|
|
|
|
|
|
|
|
/* Loop all hatch styles */
|
|
|
|
for (i = 0; i < HS_DDI_MAX; i++)
|
|
|
|
{
|
|
|
|
/* Create a default hatch bitmap */
|
|
|
|
gahsurfHatch[i] = (HSURF)EngCreateBitmap(sizl,
|
|
|
|
0,
|
|
|
|
BMF_1BPP,
|
|
|
|
0,
|
|
|
|
(PVOID)gaulHatchBrushes[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2009-03-23 00:38:59 +00:00
|
|
|
VOID
|
2009-07-10 23:44:05 +00:00
|
|
|
NTAPI
|
2012-10-18 18:23:47 +00:00
|
|
|
EBRUSHOBJ_vInit(EBRUSHOBJ *pebo,
|
|
|
|
PBRUSH pbrush,
|
|
|
|
PSURFACE psurf,
|
|
|
|
COLORREF crBackgroundClr,
|
|
|
|
COLORREF crForegroundClr,
|
|
|
|
PPALETTE ppalDC)
|
2009-03-20 23:40:59 +00:00
|
|
|
{
|
|
|
|
ASSERT(pebo);
|
|
|
|
ASSERT(pbrush);
|
2009-07-10 23:44:05 +00:00
|
|
|
|
|
|
|
pebo->BrushObject.flColorType = 0;
|
2009-08-04 20:37:10 +00:00
|
|
|
pebo->BrushObject.pvRbrush = NULL;
|
2009-07-10 23:44:05 +00:00
|
|
|
pebo->pbrush = pbrush;
|
2009-08-04 20:37:10 +00:00
|
|
|
pebo->pengbrush = NULL;
|
2009-07-10 23:44:05 +00:00
|
|
|
pebo->flattrs = pbrush->flAttrs;
|
2015-02-12 09:18:20 +00:00
|
|
|
pebo->psoMask = NULL;
|
2009-08-04 20:37:10 +00:00
|
|
|
|
|
|
|
/* Initialize 1 bpp fore and back colors */
|
2012-10-18 18:23:47 +00:00
|
|
|
pebo->crCurrentBack = crBackgroundClr;
|
|
|
|
pebo->crCurrentText = crForegroundClr;
|
2009-08-04 20:37:10 +00:00
|
|
|
|
2012-10-18 18:23:47 +00:00
|
|
|
pebo->psurfTrg = psurf;
|
2012-07-23 10:12:53 +00:00
|
|
|
/* We are initializing for a new memory DC */
|
|
|
|
if(!pebo->psurfTrg)
|
|
|
|
pebo->psurfTrg = psurfDefaultBitmap;
|
2010-05-08 22:10:41 +00:00
|
|
|
ASSERT(pebo->psurfTrg);
|
2010-05-22 14:54:48 +00:00
|
|
|
ASSERT(pebo->psurfTrg->ppal);
|
2010-05-08 22:10:41 +00:00
|
|
|
|
2012-10-18 18:23:47 +00:00
|
|
|
/* Initialize palettes */
|
2010-05-22 14:54:48 +00:00
|
|
|
pebo->ppalSurf = pebo->psurfTrg->ppal;
|
[WIN32K]
Rewrite of the GDI handle manager
- The old handle manager used a completely retarded spinlock in combination with KeDelayExecutionThread() for both exclusive
and shared locks. This is probably the most uneffective algorithm possible. It was also duplicating code everywhere and it was a overall mess It
is now replaced with a lock-free reference counter for shared locks and a pushlock for exclusive locks. -> Better performance and scalability.
- Allocate user mode object attributes from the new gdi pool. This way, we don't need any caching, since the pool serves as a cache. Its also
much faster and uses much less memory.
- Allow object allocations of different size, instead of fixed size from a table. This way a single allocation can take care of actual needs.
- Allow allcoating objects without a handle and insert them into the handle table later
- Properly synchronize the process GDIHandleCount. Now gdiview and taskmanager show the correct number of gdi handles.
- Implement a new event tracking system, that is capable of tracking all reverences and locks of objects and pool allocations to help track
possible leaks
- Make sure that all objects of a process are deleted in cleanup
- Make sure all usermode memory allocations are freed, when cleaning up the process pool.
- Make sure that each object type is using the correct type of lock (either shared or exclusive, not a mixture)
- Fix some object / reference leaks
- Lots of inferface improvements
- Use global variables for certain things instead of members in the mapped gdi handle table
- Make IntSysCreateRectpRgn create a region without a handle
- Fix detection od source and mask use in GreStretchBltMask
- Use GDIOBJ_bLockMultipleObjects in NtGdiCombineRegion to avoid possible deadlocks
- Fix NtGdiAbortPath to reset DCPATH_ACTIVE flag in the dc and only bail out on error, instead of always
- Replace DC_AllocateDcAttr and DC_AllocDcAttr with DC_bAllocDcAttr using the new user mode pool
- Remove DCU_SyncDcAttrtoUser and DCU_SynchDcAttrtoUser. Those functions were unused and didn't do anything useful anyway,
- Replace IntGdiSetDCOwnerEx and DC_SetOwnership with GreSetDCOwner, remove unused NoSetBrush parameter
- Replace GDIOBJ_bValidateHandle and IsObjectDead with GreIsHandleValid
- Chage GDIOBJ_bLockMultipleObjects: pass object type, return a BOOL, whether all objects could be locked, cleanup on failure
svn path=/trunk/; revision=51470
2011-04-28 08:26:46 +00:00
|
|
|
GDIOBJ_vReferenceObjectByPointer(&pebo->ppalSurf->BaseObject);
|
2012-10-18 18:23:47 +00:00
|
|
|
pebo->ppalDC = ppalDC;
|
|
|
|
if(!pebo->ppalDC)
|
|
|
|
pebo->ppalDC = gppalDefault;
|
2012-05-06 08:17:48 +00:00
|
|
|
GDIOBJ_vReferenceObjectByPointer(&pebo->ppalDC->BaseObject);
|
2012-10-18 18:23:47 +00:00
|
|
|
pebo->ppalDIB = NULL;
|
2009-03-20 23:40:59 +00:00
|
|
|
|
2012-05-04 17:37:07 +00:00
|
|
|
if (pbrush->flAttrs & BR_IS_NULL)
|
2009-03-20 23:40:59 +00:00
|
|
|
{
|
2009-08-04 20:37:10 +00:00
|
|
|
/* NULL brushes don't need a color */
|
2009-03-20 23:40:59 +00:00
|
|
|
pebo->BrushObject.iSolidColor = 0;
|
|
|
|
}
|
2012-05-04 17:37:07 +00:00
|
|
|
else if (pbrush->flAttrs & BR_IS_SOLID)
|
2009-03-20 23:40:59 +00:00
|
|
|
{
|
2009-03-23 00:38:59 +00:00
|
|
|
/* Set the RGB color */
|
2012-04-26 16:19:22 +00:00
|
|
|
EBRUSHOBJ_vSetSolidRGBColor(pebo, pbrush->BrushAttr.lbColor);
|
2009-03-20 23:40:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-03-23 00:38:59 +00:00
|
|
|
/* This is a pattern brush that needs realization */
|
2009-03-20 23:40:59 +00:00
|
|
|
pebo->BrushObject.iSolidColor = 0xFFFFFFFF;
|
2009-08-04 20:37:10 +00:00
|
|
|
|
|
|
|
/* Use foreground color of hatch brushes */
|
2012-05-04 17:37:07 +00:00
|
|
|
if (pbrush->flAttrs & BR_IS_HATCH)
|
2009-08-04 20:37:10 +00:00
|
|
|
pebo->crCurrentText = pbrush->BrushAttr.lbColor;
|
2009-03-20 23:40:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-18 18:23:47 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
EBRUSHOBJ_vInitFromDC(EBRUSHOBJ *pebo,
|
|
|
|
PBRUSH pbrush, PDC pdc)
|
|
|
|
{
|
|
|
|
EBRUSHOBJ_vInit(pebo, pbrush, pdc->dclevel.pSurface,
|
|
|
|
pdc->pdcattr->crBackgroundClr, pdc->pdcattr->crForegroundClr,
|
|
|
|
pdc->dclevel.ppal);
|
|
|
|
}
|
|
|
|
|
2009-03-23 00:38:59 +00:00
|
|
|
VOID
|
|
|
|
FASTCALL
|
2012-04-26 16:19:22 +00:00
|
|
|
EBRUSHOBJ_vSetSolidRGBColor(EBRUSHOBJ *pebo, COLORREF crColor)
|
2009-03-23 00:38:59 +00:00
|
|
|
{
|
2009-04-01 01:49:18 +00:00
|
|
|
ULONG iSolidColor;
|
2009-08-04 20:37:10 +00:00
|
|
|
EXLATEOBJ exlo;
|
2009-04-01 01:49:18 +00:00
|
|
|
|
2009-03-23 00:38:59 +00:00
|
|
|
/* Never use with non-solid brushes */
|
2012-05-04 17:37:07 +00:00
|
|
|
ASSERT(pebo->flattrs & BR_IS_SOLID);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2009-04-01 01:49:18 +00:00
|
|
|
/* Set the RGB color */
|
2015-02-11 21:34:11 +00:00
|
|
|
crColor &= 0xFFFFFF;
|
2009-04-01 01:49:18 +00:00
|
|
|
pebo->crRealize = crColor;
|
|
|
|
pebo->ulRGBColor = crColor;
|
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
/* Initialize an XLATEOBJ RGB -> surface */
|
2010-05-19 01:54:43 +00:00
|
|
|
EXLATEOBJ_vInitialize(&exlo,
|
|
|
|
&gpalRGB,
|
|
|
|
pebo->ppalSurf,
|
|
|
|
pebo->crCurrentBack,
|
|
|
|
0,
|
|
|
|
0);
|
2009-08-04 20:37:10 +00:00
|
|
|
|
2009-04-01 01:49:18 +00:00
|
|
|
/* Translate the brush color to the target format */
|
2009-08-04 20:37:10 +00:00
|
|
|
iSolidColor = XLATEOBJ_iXlate(&exlo.xlo, crColor);
|
2009-04-01 01:49:18 +00:00
|
|
|
pebo->BrushObject.iSolidColor = iSolidColor;
|
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
/* Clean up the XLATEOBJ */
|
|
|
|
EXLATEOBJ_vCleanup(&exlo);
|
2009-03-23 00:38:59 +00:00
|
|
|
}
|
|
|
|
|
2009-07-11 01:19:39 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
EBRUSHOBJ_vCleanup(EBRUSHOBJ *pebo)
|
|
|
|
{
|
|
|
|
/* Check if there's a GDI realisation */
|
|
|
|
if (pebo->pengbrush)
|
|
|
|
{
|
2012-03-24 17:19:34 +00:00
|
|
|
/* Unlock the bitmap again */
|
|
|
|
SURFACE_ShareUnlockSurface(pebo->pengbrush);
|
2009-07-11 01:19:39 +00:00
|
|
|
pebo->pengbrush = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if there's a driver's realisation */
|
|
|
|
if (pebo->BrushObject.pvRbrush)
|
|
|
|
{
|
|
|
|
/* Free allocated driver memory */
|
|
|
|
EngFreeMem(pebo->BrushObject.pvRbrush);
|
|
|
|
pebo->BrushObject.pvRbrush = NULL;
|
|
|
|
}
|
2009-08-04 20:37:10 +00:00
|
|
|
|
2015-02-11 21:34:11 +00:00
|
|
|
if (pebo->psoMask != NULL)
|
|
|
|
{
|
|
|
|
SURFACE_ShareUnlockSurface(pebo->psoMask);
|
|
|
|
pebo->psoMask = NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
/* Dereference the palettes */
|
2017-04-30 18:41:56 +00:00
|
|
|
if (pebo->ppalSurf)
|
|
|
|
{
|
|
|
|
PALETTE_ShareUnlockPalette(pebo->ppalSurf);
|
|
|
|
}
|
|
|
|
if (pebo->ppalDC)
|
|
|
|
{
|
|
|
|
PALETTE_ShareUnlockPalette(pebo->ppalDC);
|
|
|
|
}
|
|
|
|
if (pebo->ppalDIB)
|
|
|
|
{
|
|
|
|
PALETTE_ShareUnlockPalette(pebo->ppalDIB);
|
|
|
|
}
|
2009-07-11 01:19:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2015-02-11 21:34:11 +00:00
|
|
|
EBRUSHOBJ_vUpdateFromDC(
|
|
|
|
EBRUSHOBJ *pebo,
|
2012-10-18 18:23:47 +00:00
|
|
|
PBRUSH pbrush,
|
|
|
|
PDC pdc)
|
2009-07-11 01:19:39 +00:00
|
|
|
{
|
|
|
|
/* Cleanup the brush */
|
|
|
|
EBRUSHOBJ_vCleanup(pebo);
|
|
|
|
|
|
|
|
/* Reinitialize */
|
2012-10-18 18:23:47 +00:00
|
|
|
EBRUSHOBJ_vInitFromDC(pebo, pbrush, pdc);
|
2009-07-11 01:19:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is not exported, because it makes no sense for
|
|
|
|
* The driver to punt back to this function */
|
|
|
|
BOOL
|
|
|
|
APIENTRY
|
|
|
|
EngRealizeBrush(
|
|
|
|
BRUSHOBJ *pbo,
|
|
|
|
SURFOBJ *psoDst,
|
|
|
|
SURFOBJ *psoPattern,
|
|
|
|
SURFOBJ *psoMask,
|
|
|
|
XLATEOBJ *pxlo,
|
|
|
|
ULONG iHatch)
|
|
|
|
{
|
|
|
|
EBRUSHOBJ *pebo;
|
|
|
|
HBITMAP hbmpRealize;
|
|
|
|
SURFOBJ *psoRealize;
|
2012-03-24 17:19:34 +00:00
|
|
|
PSURFACE psurfRealize;
|
2009-07-11 01:19:39 +00:00
|
|
|
POINTL ptlSrc = {0, 0};
|
|
|
|
RECTL rclDest;
|
|
|
|
ULONG lWidth;
|
|
|
|
|
|
|
|
/* Calculate width in bytes of the realized brush */
|
2010-09-01 16:52:23 +00:00
|
|
|
lWidth = WIDTH_BYTES_ALIGN32(psoPattern->sizlBitmap.cx,
|
2009-07-11 01:19:39 +00:00
|
|
|
BitsPerFormat(psoDst->iBitmapFormat));
|
|
|
|
|
|
|
|
/* Allocate a bitmap */
|
|
|
|
hbmpRealize = EngCreateBitmap(psoPattern->sizlBitmap,
|
|
|
|
lWidth,
|
|
|
|
psoDst->iBitmapFormat,
|
|
|
|
BMF_NOZEROINIT,
|
|
|
|
NULL);
|
|
|
|
if (!hbmpRealize)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Lock the bitmap */
|
2012-03-24 17:19:34 +00:00
|
|
|
psurfRealize = SURFACE_ShareLockSurface(hbmpRealize);
|
|
|
|
|
|
|
|
/* Already delete the pattern bitmap (will be kept until dereferenced) */
|
2012-05-02 19:29:33 +00:00
|
|
|
EngDeleteSurface((HSURF)hbmpRealize);
|
2012-03-24 17:19:34 +00:00
|
|
|
|
|
|
|
if (!psurfRealize)
|
2009-07-11 01:19:39 +00:00
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the bits to the new format bitmap */
|
2009-07-31 18:21:24 +00:00
|
|
|
rclDest.left = rclDest.top = 0;
|
|
|
|
rclDest.right = psoPattern->sizlBitmap.cx;
|
|
|
|
rclDest.bottom = psoPattern->sizlBitmap.cy;
|
2012-03-24 17:19:34 +00:00
|
|
|
psoRealize = &psurfRealize->SurfObj;
|
2009-07-11 01:19:39 +00:00
|
|
|
EngCopyBits(psoRealize, psoPattern, NULL, pxlo, &rclDest, &ptlSrc);
|
|
|
|
|
|
|
|
|
|
|
|
pebo = CONTAINING_RECORD(pbo, EBRUSHOBJ, BrushObject);
|
2012-03-24 17:19:34 +00:00
|
|
|
pebo->pengbrush = (PVOID)psurfRealize;
|
2009-07-11 01:19:39 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
static
|
|
|
|
PPALETTE
|
|
|
|
FixupDIBBrushPalette(
|
|
|
|
_In_ PPALETTE ppalDIB,
|
|
|
|
_In_ PPALETTE ppalDC)
|
|
|
|
{
|
|
|
|
PPALETTE ppalNew;
|
|
|
|
ULONG i, iPalIndex, crColor;
|
|
|
|
|
|
|
|
/* Allocate a new palette */
|
|
|
|
ppalNew = PALETTE_AllocPalette(PAL_INDEXED,
|
|
|
|
ppalDIB->NumColors,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0);
|
2015-03-25 22:26:52 +00:00
|
|
|
if (ppalNew == NULL)
|
|
|
|
{
|
|
|
|
ERR("Failed to allcate palette for brush\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-05-06 08:17:48 +00:00
|
|
|
|
|
|
|
/* Loop all colors */
|
|
|
|
for (i = 0; i < ppalDIB->NumColors; i++)
|
|
|
|
{
|
|
|
|
/* Get the RGB color, which is the index into the DC palette */
|
|
|
|
iPalIndex = PALETTE_ulGetRGBColorFromIndex(ppalDIB, i);
|
|
|
|
|
|
|
|
/* Roll over when index is too big */
|
|
|
|
iPalIndex %= ppalDC->NumColors;
|
|
|
|
|
|
|
|
/* Set the indexed DC color as the new color */
|
|
|
|
crColor = PALETTE_ulGetRGBColorFromIndex(ppalDC, iPalIndex);
|
|
|
|
PALETTE_vSetRGBColorForIndex(ppalNew, i, crColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return the new palette */
|
|
|
|
return ppalNew;
|
|
|
|
}
|
|
|
|
|
2009-03-23 00:38:59 +00:00
|
|
|
BOOL
|
2009-07-10 23:44:05 +00:00
|
|
|
NTAPI
|
2009-07-11 01:19:39 +00:00
|
|
|
EBRUSHOBJ_bRealizeBrush(EBRUSHOBJ *pebo, BOOL bCallDriver)
|
2009-03-23 00:38:59 +00:00
|
|
|
{
|
|
|
|
BOOL bResult;
|
2013-01-01 09:40:48 +00:00
|
|
|
PFN_DrvRealizeBrush pfnRealizeBrush = NULL;
|
2015-02-11 21:34:11 +00:00
|
|
|
PSURFACE psurfPattern;
|
|
|
|
SURFOBJ *psoMask;
|
2012-05-06 08:17:48 +00:00
|
|
|
PPDEVOBJ ppdev;
|
2009-08-04 20:37:10 +00:00
|
|
|
EXLATEOBJ exlo;
|
2012-05-02 18:55:23 +00:00
|
|
|
PPALETTE ppalPattern;
|
2012-05-06 08:17:48 +00:00
|
|
|
PBRUSH pbr = pebo->pbrush;
|
|
|
|
HBITMAP hbmPattern;
|
|
|
|
ULONG iHatch;
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2010-05-22 14:54:48 +00:00
|
|
|
/* All EBRUSHOBJs have a surface, see EBRUSHOBJ_vInit */
|
|
|
|
ASSERT(pebo->psurfTrg);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
ppdev = (PPDEVOBJ)pebo->psurfTrg->SurfObj.hdev;
|
2015-02-11 21:34:11 +00:00
|
|
|
if (!ppdev)
|
|
|
|
ppdev = gppdevPrimary;
|
2009-08-04 20:37:10 +00:00
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
if (bCallDriver)
|
2015-02-11 21:34:11 +00:00
|
|
|
{
|
|
|
|
/* Get the Drv function */
|
2013-01-01 09:40:48 +00:00
|
|
|
pfnRealizeBrush = ppdev->DriverFunctions.RealizeBrush;
|
2015-02-11 21:34:11 +00:00
|
|
|
if (pfnRealizeBrush == NULL)
|
|
|
|
{
|
|
|
|
ERR("No DrvRealizeBrush. Cannot realize brush\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2009-08-04 20:37:10 +00:00
|
|
|
|
2015-02-11 21:34:11 +00:00
|
|
|
/* Get the mask */
|
|
|
|
psoMask = EBRUSHOBJ_psoMask(pebo);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Use the Eng function */
|
2013-01-01 09:40:48 +00:00
|
|
|
pfnRealizeBrush = EngRealizeBrush;
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2015-02-11 21:34:11 +00:00
|
|
|
/* We don't handle the mask bitmap here. We do this only on demand */
|
|
|
|
psoMask = NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
/* Check if this is a hatch brush */
|
|
|
|
if (pbr->flAttrs & BR_IS_HATCH)
|
|
|
|
{
|
|
|
|
/* Get the hatch brush pattern from the PDEV */
|
2015-02-11 21:33:50 +00:00
|
|
|
hbmPattern = (HBITMAP)ppdev->ahsurf[pbr->iHatch];
|
|
|
|
iHatch = pbr->iHatch;
|
2012-05-06 08:17:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Use the brushes pattern */
|
|
|
|
hbmPattern = pbr->hbmPattern;
|
|
|
|
iHatch = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
psurfPattern = SURFACE_ShareLockSurface(hbmPattern);
|
2009-08-04 20:37:10 +00:00
|
|
|
ASSERT(psurfPattern);
|
2010-05-19 16:23:33 +00:00
|
|
|
ASSERT(psurfPattern->ppal);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2012-05-02 18:55:23 +00:00
|
|
|
/* DIB brushes with DIB_PAL_COLORS usage need a new palette */
|
2012-05-06 08:17:48 +00:00
|
|
|
if (pbr->flAttrs & BR_IS_DIBPALCOLORS)
|
2012-05-02 18:55:23 +00:00
|
|
|
{
|
2012-05-07 22:57:41 +00:00
|
|
|
/* Create a palette with the colors from the DC */
|
2012-05-06 08:17:48 +00:00
|
|
|
ppalPattern = FixupDIBBrushPalette(psurfPattern->ppal, pebo->ppalDC);
|
2015-03-25 22:26:52 +00:00
|
|
|
if (ppalPattern == NULL)
|
|
|
|
{
|
|
|
|
ERR("FixupDIBBrushPalette() failed.\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-05-06 08:17:48 +00:00
|
|
|
pebo->ppalDIB = ppalPattern;
|
2012-05-02 18:55:23 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* The palette is already as it should be */
|
|
|
|
ppalPattern = psurfPattern->ppal;
|
|
|
|
}
|
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
/* Initialize XLATEOBJ for the brush */
|
2012-03-24 17:19:34 +00:00
|
|
|
EXLATEOBJ_vInitialize(&exlo,
|
2012-05-02 18:55:23 +00:00
|
|
|
ppalPattern,
|
2010-05-19 16:23:33 +00:00
|
|
|
pebo->psurfTrg->ppal,
|
|
|
|
0,
|
|
|
|
pebo->crCurrentBack,
|
|
|
|
pebo->crCurrentText);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
/* Create the realization */
|
2013-01-01 09:40:48 +00:00
|
|
|
bResult = pfnRealizeBrush(&pebo->BrushObject,
|
|
|
|
&pebo->psurfTrg->SurfObj,
|
|
|
|
&psurfPattern->SurfObj,
|
2015-02-11 21:34:11 +00:00
|
|
|
psoMask,
|
2013-01-01 09:40:48 +00:00
|
|
|
&exlo.xlo,
|
|
|
|
iHatch);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2009-08-04 20:37:10 +00:00
|
|
|
/* Cleanup the XLATEOBJ */
|
|
|
|
EXLATEOBJ_vCleanup(&exlo);
|
2009-07-10 23:44:05 +00:00
|
|
|
|
2015-02-11 21:34:11 +00:00
|
|
|
/* Unlock surface */
|
2014-05-01 08:30:41 +00:00
|
|
|
SURFACE_ShareUnlockSurface(psurfPattern);
|
2009-03-23 00:38:59 +00:00
|
|
|
|
|
|
|
return bResult;
|
|
|
|
}
|
|
|
|
|
2009-07-10 23:44:05 +00:00
|
|
|
PVOID
|
|
|
|
NTAPI
|
|
|
|
EBRUSHOBJ_pvGetEngBrush(EBRUSHOBJ *pebo)
|
|
|
|
{
|
|
|
|
BOOL bResult;
|
|
|
|
|
|
|
|
if (!pebo->pengbrush)
|
|
|
|
{
|
2009-07-11 01:19:39 +00:00
|
|
|
bResult = EBRUSHOBJ_bRealizeBrush(pebo, FALSE);
|
2009-07-10 23:44:05 +00:00
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
if (pebo->pengbrush)
|
|
|
|
EngDeleteSurface(pebo->pengbrush);
|
|
|
|
pebo->pengbrush = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pebo->pengbrush;
|
|
|
|
}
|
2009-03-23 00:38:59 +00:00
|
|
|
|
2012-03-24 17:19:34 +00:00
|
|
|
SURFOBJ*
|
|
|
|
NTAPI
|
|
|
|
EBRUSHOBJ_psoPattern(EBRUSHOBJ *pebo)
|
|
|
|
{
|
|
|
|
PSURFACE psurfPattern;
|
|
|
|
|
|
|
|
psurfPattern = EBRUSHOBJ_pvGetEngBrush(pebo);
|
|
|
|
|
|
|
|
return psurfPattern ? &psurfPattern->SurfObj : NULL;
|
|
|
|
}
|
|
|
|
|
2015-02-11 21:34:11 +00:00
|
|
|
SURFOBJ*
|
|
|
|
NTAPI
|
|
|
|
EBRUSHOBJ_psoMask(EBRUSHOBJ *pebo)
|
|
|
|
{
|
2015-02-11 21:35:48 +00:00
|
|
|
HBITMAP hbmMask;
|
|
|
|
PSURFACE psurfMask;
|
|
|
|
PPDEVOBJ ppdev;
|
|
|
|
|
|
|
|
/* Check if we don't have a mask yet */
|
|
|
|
if (pebo->psoMask == NULL)
|
|
|
|
{
|
|
|
|
/* Check if this is a hatch brush */
|
|
|
|
if (pebo->flattrs & BR_IS_HATCH)
|
|
|
|
{
|
|
|
|
/* Get the PDEV */
|
|
|
|
ppdev = (PPDEVOBJ)pebo->psurfTrg->SurfObj.hdev;
|
|
|
|
if (!ppdev)
|
|
|
|
ppdev = gppdevPrimary;
|
|
|
|
|
|
|
|
/* Use the hatch bitmap as the mask */
|
|
|
|
hbmMask = (HBITMAP)ppdev->ahsurf[pebo->pbrush->iHatch];
|
|
|
|
psurfMask = SURFACE_ShareLockSurface(hbmMask);
|
|
|
|
if (psurfMask == NULL)
|
|
|
|
{
|
|
|
|
ERR("Failed to lock hatch brush for PDEV %p, iHatch %lu\n",
|
|
|
|
ppdev, pebo->pbrush->iHatch);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
NT_ASSERT(psurfMask->SurfObj.iBitmapFormat == BMF_1BPP);
|
|
|
|
pebo->psoMask = &psurfMask->SurfObj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pebo->psoMask;
|
2015-02-11 21:34:11 +00:00
|
|
|
}
|
2009-03-20 23:40:59 +00:00
|
|
|
|
|
|
|
/** Exported DDI functions ****************************************************/
|
|
|
|
|
2003-07-11 15:59:37 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 22:48:58 +00:00
|
|
|
PVOID APIENTRY
|
2009-03-23 00:38:59 +00:00
|
|
|
BRUSHOBJ_pvAllocRbrush(
|
|
|
|
IN BRUSHOBJ *pbo,
|
|
|
|
IN ULONG cj)
|
1999-08-01 11:21:05 +00:00
|
|
|
{
|
2010-06-10 20:45:58 +00:00
|
|
|
pbo->pvRbrush = EngAllocMem(0, cj, GDITAG_RBRUSH);
|
2009-03-23 00:38:59 +00:00
|
|
|
return pbo->pvRbrush;
|
1999-08-01 11:21:05 +00:00
|
|
|
}
|
|
|
|
|
2003-07-11 15:59:37 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 22:48:58 +00:00
|
|
|
PVOID APIENTRY
|
2009-03-23 00:38:59 +00:00
|
|
|
BRUSHOBJ_pvGetRbrush(
|
|
|
|
IN BRUSHOBJ *pbo)
|
1999-08-01 11:21:05 +00:00
|
|
|
{
|
2009-07-10 23:44:05 +00:00
|
|
|
EBRUSHOBJ *pebo = CONTAINING_RECORD(pbo, EBRUSHOBJ, BrushObject);
|
|
|
|
BOOL bResult;
|
|
|
|
|
|
|
|
if (!pbo->pvRbrush)
|
|
|
|
{
|
2009-07-11 01:19:39 +00:00
|
|
|
bResult = EBRUSHOBJ_bRealizeBrush(pebo, TRUE);
|
2009-07-10 23:44:05 +00:00
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
if (pbo->pvRbrush)
|
2009-07-11 20:15:37 +00:00
|
|
|
{
|
2009-07-10 23:44:05 +00:00
|
|
|
EngFreeMem(pbo->pvRbrush);
|
2009-07-11 20:15:37 +00:00
|
|
|
pbo->pvRbrush = NULL;
|
|
|
|
}
|
2009-07-10 23:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-23 00:38:59 +00:00
|
|
|
return pbo->pvRbrush;
|
1999-08-01 11:21:05 +00:00
|
|
|
}
|
2004-01-30 16:36:16 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 22:48:58 +00:00
|
|
|
ULONG APIENTRY
|
2009-03-23 00:38:59 +00:00
|
|
|
BRUSHOBJ_ulGetBrushColor(
|
|
|
|
IN BRUSHOBJ *pbo)
|
2004-01-30 16:36:16 +00:00
|
|
|
{
|
2009-03-23 00:38:59 +00:00
|
|
|
EBRUSHOBJ *pebo = CONTAINING_RECORD(pbo, EBRUSHOBJ, BrushObject);
|
|
|
|
return pebo->ulRGBColor;
|
2004-01-30 16:36:16 +00:00
|
|
|
}
|
|
|
|
|
2003-05-18 17:16:18 +00:00
|
|
|
/* EOF */
|