2007-11-02 04:23:31 +00:00
|
|
|
|
|
|
|
#include <w32k.h>
|
|
|
|
|
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Gdi Batch Flush support functions.
|
|
|
|
//
|
|
|
|
|
2007-12-04 07:51:03 +00:00
|
|
|
//
|
|
|
|
// DoDeviceSync
|
|
|
|
//
|
|
|
|
// based on IntEngEnter from eng/engmisc.c
|
|
|
|
//
|
|
|
|
VOID
|
|
|
|
FASTCALL
|
|
|
|
DoDeviceSync( SURFOBJ *Surface, PRECTL Rect, FLONG fl)
|
|
|
|
{
|
|
|
|
PGDIDEVICE Device = (GDIDEVICE*)Surface->hdev;
|
|
|
|
// No punting and "Handle to a surface, provided that the surface is device-managed.
|
|
|
|
// Otherwise, dhsurf is zero".
|
|
|
|
if (!(Device->flFlags & PDEV_DRIVER_PUNTED_CALL) && (Surface->dhsurf))
|
|
|
|
{
|
|
|
|
if (Device->DriverFunctions.SynchronizeSurface)
|
|
|
|
return Device->DriverFunctions.SynchronizeSurface(Surface, Rect, fl);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (Device->DriverFunctions.Synchronize)
|
|
|
|
return Device->DriverFunctions.Synchronize(Surface->dhpdev, Rect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
FASTCALL
|
|
|
|
SynchonizeDriver(FLONG Flags)
|
|
|
|
{
|
|
|
|
SURFOBJ *SurfObj;
|
|
|
|
PGDIDEVICE Device;
|
|
|
|
|
|
|
|
if (Flags & GCAPS2_SYNCFLUSH)
|
|
|
|
Flags = DSS_FLUSH_EVENT;
|
|
|
|
if (Flags & GCAPS2_SYNCTIMER)
|
|
|
|
Flags = DSS_TIMER_EVENT;
|
|
|
|
|
|
|
|
Device = IntEnumHDev();
|
|
|
|
|
|
|
|
SurfObj = EngLockSurface((HSURF)Device->Handle);
|
|
|
|
if(!SurfObj) return;
|
|
|
|
DoDeviceSync( SurfObj, NULL, Flags);
|
|
|
|
EngUnlockSurface(SurfObj);
|
|
|
|
return;
|
|
|
|
}
|
2007-11-02 04:23:31 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Process the batch.
|
|
|
|
//
|
|
|
|
ULONG
|
|
|
|
FASTCALL
|
|
|
|
GdiFlushUserBatch(HDC hDC, PGDIBATCHHDR pHdr)
|
|
|
|
{
|
2007-11-05 02:25:43 +00:00
|
|
|
PDC dc = NULL;
|
2007-11-18 13:51:34 +00:00
|
|
|
PDC_ATTR Dc_Attr = NULL;
|
|
|
|
if (hDC)
|
|
|
|
{
|
|
|
|
dc = DC_LockDc(hDC);
|
2007-12-01 13:26:45 +00:00
|
|
|
if (dc)
|
|
|
|
{
|
|
|
|
Dc_Attr = dc->pDc_Attr;
|
|
|
|
if (!Dc_Attr) Dc_Attr = &dc->Dc_Attr;
|
|
|
|
}
|
2007-11-18 13:51:34 +00:00
|
|
|
}
|
2007-11-05 02:25:43 +00:00
|
|
|
// The thread is approaching the end of sunset.
|
2007-11-02 04:23:31 +00:00
|
|
|
switch(pHdr->Cmd)
|
|
|
|
{
|
|
|
|
case GdiBCPatBlt: // Highest pri first!
|
|
|
|
break;
|
|
|
|
case GdiBCPolyPatBlt:
|
|
|
|
break;
|
|
|
|
case GdiBCTextOut:
|
|
|
|
break;
|
|
|
|
case GdiBCExtTextOut:
|
|
|
|
break;
|
|
|
|
case GdiBCSetBrushOrg:
|
|
|
|
{
|
2007-11-27 01:14:38 +00:00
|
|
|
PGDIBSSETBRHORG pgSBO;
|
2007-12-04 15:31:11 +00:00
|
|
|
if(!dc) break;
|
2007-11-26 11:24:48 +00:00
|
|
|
pgSBO = (PGDIBSSETBRHORG) pHdr;
|
2007-11-18 13:51:34 +00:00
|
|
|
Dc_Attr->ptlBrushOrigin = pgSBO->ptlBrushOrigin;
|
2007-11-02 04:23:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case GdiBCExtSelClipRgn:
|
|
|
|
break;
|
|
|
|
case GdiBCSelObj:
|
2007-12-01 02:09:39 +00:00
|
|
|
{
|
|
|
|
PGDIBSOBJECT pgO;
|
|
|
|
if(!dc) break;
|
|
|
|
pgO = (PGDIBSOBJECT) pHdr;
|
|
|
|
if(NT_SUCCESS(TextIntRealizeFont((HFONT) pgO->hgdiobj)))
|
|
|
|
Dc_Attr->hlfntNew = (HFONT) pgO->hgdiobj;
|
|
|
|
}
|
2007-11-02 04:23:31 +00:00
|
|
|
case GdiBCDelObj:
|
|
|
|
case GdiBCDelRgn:
|
2007-11-05 01:50:59 +00:00
|
|
|
{
|
|
|
|
PGDIBSOBJECT pgO = (PGDIBSOBJECT) pHdr;
|
|
|
|
NtGdiDeleteObject( pgO->hgdiobj );
|
2007-11-02 04:23:31 +00:00
|
|
|
break;
|
2007-11-05 01:50:59 +00:00
|
|
|
}
|
2007-11-02 04:23:31 +00:00
|
|
|
default:
|
2007-11-05 02:25:43 +00:00
|
|
|
break;
|
2007-11-02 04:23:31 +00:00
|
|
|
}
|
2007-11-05 02:25:43 +00:00
|
|
|
if (dc) DC_UnlockDc(dc);
|
2007-11-02 04:23:31 +00:00
|
|
|
return pHdr->Size; // Return the full size of the structure.
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* NtGdiFlush
|
|
|
|
*
|
|
|
|
* Flushes the calling thread's current batch.
|
|
|
|
*/
|
|
|
|
VOID
|
|
|
|
APIENTRY
|
|
|
|
NtGdiFlush(VOID)
|
|
|
|
{
|
2007-12-04 07:51:03 +00:00
|
|
|
SynchonizeDriver(GCAPS2_SYNCFLUSH);
|
2007-11-02 04:23:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* NtGdiFlushUserBatch
|
|
|
|
*
|
|
|
|
* Callback for thread batch flush routine.
|
|
|
|
*
|
|
|
|
* Think small & fast!
|
|
|
|
*/
|
|
|
|
NTSTATUS
|
|
|
|
APIENTRY
|
|
|
|
NtGdiFlushUserBatch(VOID)
|
|
|
|
{
|
|
|
|
PTEB pTeb = NtCurrentTeb();
|
|
|
|
ULONG GdiBatchCount = pTeb->GdiBatchCount;
|
2007-11-06 02:33:38 +00:00
|
|
|
|
|
|
|
if( (GdiBatchCount > 0) && (GdiBatchCount <= (GDIBATCHBUFSIZE/4)))
|
2007-11-02 04:23:31 +00:00
|
|
|
{
|
|
|
|
HDC hDC = (HDC) pTeb->GdiTebBatch.HDC;
|
2007-11-05 01:50:59 +00:00
|
|
|
//
|
|
|
|
// If hDC is zero and the buffer fills up with delete objects we need to run
|
|
|
|
// anyway. So, hard code to the system batch limit.
|
|
|
|
//
|
2007-11-06 02:33:38 +00:00
|
|
|
if ((hDC) || ((!hDC) && (GdiBatchCount >= GDI_BATCH_LIMIT)))
|
2007-11-02 04:23:31 +00:00
|
|
|
{
|
|
|
|
PULONG pHdr = &pTeb->GdiTebBatch.Buffer[0];
|
|
|
|
// No need to init anything, just go!
|
|
|
|
for (; GdiBatchCount > 0; GdiBatchCount--)
|
|
|
|
{
|
|
|
|
// Process Gdi Batch!
|
|
|
|
pHdr += GdiFlushUserBatch( hDC, (PGDIBATCHHDR) pHdr );
|
|
|
|
}
|
|
|
|
// Exit and clear out for the next round.
|
|
|
|
pTeb->GdiTebBatch.Offset = 0;
|
|
|
|
pTeb->GdiBatchCount = 0;
|
|
|
|
pTeb->GdiTebBatch.HDC = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-12-01 02:09:39 +00:00
|
|
|
|