/* * COPYRIGHT: See COPYING in the top level directory * PROJECT: ReactOS System Libraries * FILE: win32ss/gdi/gdi32/include/gdi32p.h * PURPOSE: User-Mode Win32 GDI Library Private Header * PROGRAMMER: Alex Ionescu (alex@relsoft.net) */ #pragma once /* DATA **********************************************************************/ extern PGDI_TABLE_ENTRY GdiHandleTable; extern PGDI_SHARED_HANDLE_TABLE GdiSharedHandleTable; extern HANDLE hProcessHeap; extern HANDLE CurrentProcessId; extern DWORD GDI_BatchLimit; extern PDEVCAPS GdiDevCaps; extern BOOL gbLpk; // Global bool LanguagePack extern HANDLE ghSpooler; extern RTL_CRITICAL_SECTION semLocal; typedef INT (CALLBACK* EMFPLAYPROC)( HDC hdc, INT iFunction, HANDLE hPageQuery ); typedef BOOL (WINAPI* LPKETO)( HDC hdc, int x, int y, UINT fuOptions, const RECT *lprc, LPCWSTR lpString, UINT uCount, const INT *lpDx, INT unknown ); typedef DWORD (WINAPI* LPKGCP)( HDC hdc, LPCWSTR lpString, INT uCount, INT nMaxExtent, LPGCP_RESULTSW lpResults, DWORD dwFlags, DWORD dwUnused ); typedef BOOL (WINAPI* LPKGTEP)( HDC hdc, LPCWSTR lpString, INT cString, INT nMaxExtent, LPINT lpnFit, LPINT lpnDx, LPSIZE lpSize, DWORD dwUnused, int unknown ); extern HINSTANCE hLpk; extern LPKETO LpkExtTextOut; extern LPKGCP LpkGetCharacterPlacement; extern LPKGTEP LpkGetTextExtentExPoint; /* DEFINES *******************************************************************/ #define HANDLE_LIST_INC 20 #define METAFILE_MEMORY 1 #define METAFILE_DISK 2 #define SAPCALLBACKDELAY 244 #define LPK_INIT 1 #define LPK_ETO 2 #define LPK_GCP 3 #define LPK_GTEP 4 /* MACRO ********************************************************************/ #define ROP_USES_SOURCE(Rop) (((Rop) << 2 ^ Rop) & 0xCC0000) #define RCAST(_Type, _Value) (*((_Type*)&_Value)) /* TYPES *********************************************************************/ // Based on wmfapi.h and Wine. typedef struct tagMETAFILEDC { PVOID pvMetaBuffer; HANDLE hFile; DWORD Size; DWORD dwWritten; METAHEADER mh; WORD reserved; HLOCAL MFObjList; HPEN hPen; HBRUSH hBrush; HDC hDc; HGDIOBJ hMetaDc; HPALETTE hPalette; HFONT hFont; HBITMAP hBitmap; HRGN hRegion; HGDIOBJ hMetafile; HGDIOBJ hMemDc; HPEN hExtPen; HGDIOBJ hEnhMetaDc; HGDIOBJ hEnhMetaFile; HCOLORSPACE hColorSpace; WCHAR Filename[MAX_PATH+2]; } METAFILEDC,*PMETAFILEDC; // Metafile Entry handle typedef struct tagMF_ENTRY { LIST_ENTRY List; HGDIOBJ hmDC; // Handle return from NtGdiCreateClientObj. PMETAFILEDC pmfDC; } MF_ENTRY, *PMF_ENTRY; typedef struct tagENHMETAFILE { PVOID pvMetaBuffer; HANDLE hFile; /* Handle for disk based MetaFile */ DWORD Size; INT iType; PENHMETAHEADER emf; UINT handles_size, cur_handles; HGDIOBJ *handles; INT horzres, vertres; INT horzsize, vertsize; INT logpixelsx, logpixelsy; INT bitspixel; INT textcaps; INT rastercaps; INT technology; INT planes; } ENHMETAFILE,*PENHMETAFILE; #define PDEV_UMPD_ID 0xFEDCBA98 // UMPDEV flags #define UMPDEV_NO_ESCAPE 0x0002 #define UMPDEV_SUPPORT_ESCAPE 0x0004 typedef struct _UMPDEV { DWORD_PTR Sig; // Init with PDEV_UMPD_ID struct _UMPDEV *pumpdNext; PDRIVER_INFO_5W pdi5Info; HMODULE hModule; DWORD dwFlags; DWORD dwDriverAttributes; DWORD dwConfigVersion; // Number of times the configuration // file for this driver has been upgraded // or downgraded since the last spooler restart. DWORD dwDriverCount; // After init should be 2 DWORD WOW64_UMPDev; DWORD WOW64_hMod; DWORD Unknown; PVOID apfn[INDEX_LAST]; // Print Driver pfn } UMPDEV, *PUMPDEV; #define LOCALFONT_COUNT 10 typedef struct _LOCALFONT { FONT_ATTR lfa[LOCALFONT_COUNT]; } LOCALFONT, *PLOCALFONT; // sdk/winspool.h typedef BOOL (WINAPI *ABORTPRINTER) (HANDLE); typedef BOOL (WINAPI *CLOSEPRINTER) (HANDLE); typedef BOOL (WINAPI *CLOSESPOOLFILEHANDLE) (HANDLE, HANDLE); // W2k8 typedef HANDLE (WINAPI *COMMITSPOOLDATA) (HANDLE,HANDLE,DWORD); // W2k8 typedef LONG (WINAPI *DOCUMENTPROPERTIESW) (HWND,HANDLE,LPWSTR,PDEVMODEW,PDEVMODEW,DWORD); typedef BOOL (WINAPI *ENDDOCPRINTER) (HANDLE); typedef BOOL (WINAPI *ENDPAGEPRINTER) (HANDLE); typedef BOOL (WINAPI *GETPRINTERW) (HANDLE,DWORD,LPBYTE,DWORD,LPDWORD); typedef BOOL (WINAPI *GETPRINTERDRIVERW) (HANDLE,LPWSTR,DWORD,LPBYTE,DWORD,LPDWORD); typedef HANDLE (WINAPI *GETSPOOLFILEHANDLE) (HANDLE); // W2k8 typedef BOOL (WINAPI *ISVALIDDEVMODEW) (PDEVMODEW,size_t); typedef BOOL (WINAPI *OPENPRINTERW) (LPWSTR,PHANDLE,LPPRINTER_DEFAULTSW); typedef BOOL (WINAPI *READPRINTER) (HANDLE,PVOID,DWORD,PDWORD); typedef BOOL (WINAPI *RESETPRINTERW) (HANDLE,LPPRINTER_DEFAULTSW); typedef LPWSTR (WINAPI *STARTDOCDLGW) (HANDLE,DOCINFOW *); typedef DWORD (WINAPI *STARTDOCPRINTERW) (HANDLE,DWORD,PBYTE); typedef BOOL (WINAPI *STARTPAGEPRINTER) (HANDLE); // ddk/winsplp.h typedef BOOL (WINAPI *SEEKPRINTER) (HANDLE,LARGE_INTEGER,PLARGE_INTEGER,DWORD,BOOL); typedef BOOL (WINAPI *SPLREADPRINTER) (HANDLE,LPBYTE *,DWORD); // Same as ddk/winsplp.h DriverUnloadComplete? typedef BOOL (WINAPI *SPLDRIVERUNLOADCOMPLETE) (LPWSTR); // Driver support: // DrvDocumentEvent api/winddiui.h not W2k8 DocumentEventAW typedef INT (WINAPI *DOCUMENTEVENT) (HANDLE,HDC,INT,ULONG,PVOID,ULONG,PVOID); // DrvQueryColorProfile typedef BOOL (WINAPI *QUERYCOLORPROFILE) (HANDLE,PDEVMODEW,ULONG,VOID*,ULONG,FLONG); // Unknown: typedef DWORD (WINAPI *QUERYSPOOLMODE) (HANDLE,DWORD,DWORD); typedef DWORD (WINAPI *QUERYREMOTEFONTS) (DWORD,DWORD,DWORD); extern CLOSEPRINTER fpClosePrinter; extern OPENPRINTERW fpOpenPrinterW; /* FUNCTIONS *****************************************************************/ PVOID HEAP_alloc(DWORD len); NTSTATUS HEAP_strdupA2W( LPWSTR* ppszW, LPCSTR lpszA ); VOID HEAP_free(LPVOID memory); VOID FASTCALL FONT_TextMetricWToA( const TEXTMETRICW *ptmW, LPTEXTMETRICA ptmA ); VOID FASTCALL NewTextMetricW2A( NEWTEXTMETRICA *tma, NEWTEXTMETRICW *tmw ); VOID FASTCALL NewTextMetricExW2A( NEWTEXTMETRICEXA *tma, NEWTEXTMETRICEXW *tmw ); BOOL FASTCALL DeleteRegion( HRGN ); BOOL WINAPI GdiValidateHandle(HGDIOBJ); BOOL GdiGetHandleUserData( HGDIOBJ hGdiObj, DWORD ObjectType, PVOID *UserData ); PLDC FASTCALL GdiGetLDC(HDC hDC); BOOL FASTCALL GdiSetLDC(HDC hdc, PVOID pvLDC); HGDIOBJ WINAPI GdiFixUpHandle(HGDIOBJ hGO); BOOL WINAPI CalculateColorTableSize( CONST BITMAPINFOHEADER *BitmapInfoHeader, UINT *ColorSpec, UINT *ColorTableSize ); LPBITMAPINFO WINAPI ConvertBitmapInfo( CONST BITMAPINFO *BitmapInfo, UINT ColorSpec, UINT *BitmapInfoSize, BOOL FollowedByData ); DWORD WINAPI GetAndSetDCDWord( _In_ HDC hdc, _In_ UINT u, _In_ DWORD dwIn, _In_ ULONG ulMFId, _In_ USHORT usMF16Id, _In_ DWORD dwError); DWORD WINAPI GetDCDWord( _In_ HDC hdc, _In_ UINT u, _In_ DWORD dwError); HGDIOBJ WINAPI GetDCObject( HDC, INT); VOID NTAPI LogFontA2W( LPLOGFONTW pW, CONST LOGFONTA *pA ); VOID NTAPI LogFontW2A( LPLOGFONTA pA, CONST LOGFONTW *pW ); VOID WINAPI EnumLogFontExW2A( LPENUMLOGFONTEXA fontA, CONST ENUMLOGFONTEXW *fontW ); BOOL WINAPI LoadLPK( INT LpkFunctionID ); VOID WINAPI GdiInitializeLanguagePack( _In_ DWORD InitParam); VOID WINAPI InitializeLpkHooks( _In_ FARPROC *hookfuncs); BOOL WINAPI GetETM(HDC hdc, EXTTEXTMETRIC *petm); /* FIXME: Put in some public header */ UINT WINAPI UserRealizePalette(HDC hDC); int WINAPI GdiAddFontResourceW(LPCWSTR lpszFilename,FLONG fl,DESIGNVECTOR *pdv); VOID WINAPI GdiSetLastError( DWORD dwErrCode ); DWORD WINAPI GdiGetCodePage(HDC); int WINAPI GdiGetBitmapBitsSize(BITMAPINFO *lpbmi); VOID GdiSAPCallback(PLDC pldc); HGDIOBJ FASTCALL hGetPEBHandle(HANDLECACHETYPE,COLORREF); int FASTCALL DocumentEventEx(PVOID,HANDLE,HDC,int,ULONG,PVOID,ULONG,PVOID); BOOL FASTCALL EndPagePrinterEx(PVOID,HANDLE); BOOL FASTCALL LoadTheSpoolerDrv(VOID); FORCEINLINE PVOID GdiAllocBatchCommand( HDC hdc, USHORT Cmd) { PTEB pTeb; USHORT cjSize; PGDIBATCHHDR pHdr; /* Get a pointer to the TEB */ pTeb = NtCurrentTeb(); /* Check if we have a valid environment */ if (!pTeb || !pTeb->Win32ThreadInfo) return NULL; /* Do we use a DC? */ if (hdc) { /* If the batch DC is NULL, we set this one as the new one */ if (!pTeb->GdiTebBatch.HDC) pTeb->GdiTebBatch.HDC = hdc; /* If not, check if the batch DC equal to our DC */ else if (pTeb->GdiTebBatch.HDC != hdc) return NULL; } /* Get the size of the entry */ if (Cmd == GdiBCPatBlt) cjSize = sizeof(GDIBSPATBLT); else if (Cmd == GdiBCPolyPatBlt) cjSize = sizeof(GDIBSPPATBLT); else if (Cmd == GdiBCTextOut) cjSize = sizeof(GDIBSTEXTOUT); else if (Cmd == GdiBCExtTextOut) cjSize = sizeof(GDIBSEXTTEXTOUT); else if (Cmd == GdiBCSetBrushOrg) cjSize = sizeof(GDIBSSETBRHORG); else if (Cmd == GdiBCExtSelClipRgn) cjSize = 0; else if (Cmd == GdiBCSelObj) cjSize = sizeof(GDIBSOBJECT); else if (Cmd == GdiBCDelRgn) cjSize = sizeof(GDIBSOBJECT); else if (Cmd == GdiBCDelObj) cjSize = sizeof(GDIBSOBJECT); else cjSize = 0; /* Unsupported operation */ if (cjSize == 0) return NULL; /* Check if the buffer is full */ if ((pTeb->GdiBatchCount >= GDI_BatchLimit) || ((pTeb->GdiTebBatch.Offset + cjSize) > GDIBATCHBUFSIZE)) { /* Call win32k, the kernel will call NtGdiFlushUserBatch to flush the current batch */ NtGdiFlush(); // If Flushed, lose the hDC for this batch job! See CORE-15839. if (hdc) { if (!pTeb->GdiTebBatch.HDC) pTeb->GdiTebBatch.HDC = hdc; } } /* Get the head of the entry */ pHdr = (PVOID)((PUCHAR)pTeb->GdiTebBatch.Buffer + pTeb->GdiTebBatch.Offset); /* Update Offset and batch count */ pTeb->GdiTebBatch.Offset += cjSize; pTeb->GdiBatchCount++; /* Fill in the core fields */ pHdr->Cmd = Cmd; pHdr->Size = cjSize; return pHdr; } FORCEINLINE PDC_ATTR GdiGetDcAttr(HDC hdc) { GDILOOBJTYPE eDcObjType; PDC_ATTR pdcattr; /* Check DC object type */ eDcObjType = GDI_HANDLE_GET_TYPE(hdc); if ((eDcObjType != GDILoObjType_LO_DC_TYPE) && (eDcObjType != GDILoObjType_LO_ALTDC_TYPE)) { return NULL; } /* Get the DC attribute */ if (!GdiGetHandleUserData((HGDIOBJ)hdc, eDcObjType, (PVOID*)&pdcattr)) { return NULL; } return pdcattr; } FORCEINLINE PRGN_ATTR GdiGetRgnAttr(HRGN hrgn) { PRGN_ATTR prgnattr; /* Get the region attribute */ if (!GdiGetHandleUserData(hrgn, GDILoObjType_LO_REGION_TYPE, (PVOID*)&prgnattr)) { return NULL; } return prgnattr; } #ifdef _M_IX86 FLOATL FASTCALL EFtoF(EFLOAT_S * efp); #define FOtoF(pfo) EFtoF((EFLOAT_S*)pfo) #else #define FOtoF(pfo) (*(pfo)) #endif /* This is an inlined version of lrintf. */ FORCEINLINE int _lrintf(float f) { #if defined(_M_IX86) && defined(__GNUC__) int result; __asm__ __volatile__ ("fistpl %0" : "=m" (result) : "t" (f) : "st"); return result; #elif defined(_M_IX86) && defined(_MSC_VER) int result; __asm { fld f; fistp result; } #else /* slow, but portable */ return (int)(f >= 0 ? f+0.5 : f-0.5); #endif } HBRUSH WINAPI GdiSelectBrush( _In_ HDC hdc, _In_ HBRUSH hbr); HPEN WINAPI GdiSelectPen( _In_ HDC hdc, _In_ HPEN hpen); HFONT WINAPI GdiSelectFont( _In_ HDC hdc, _In_ HFONT hfont); HGDIOBJ WINAPI GdiCreateClientObj( _In_ PVOID pvObject, _In_ GDILOOBJTYPE eObjType); PVOID WINAPI GdiDeleteClientObj( _In_ HGDIOBJ hobj); BOOL WINAPI GdiCreateClientObjLink( _In_ HGDIOBJ hobj, _In_ PVOID pvObject); PVOID WINAPI GdiGetClientObjLink( _In_ HGDIOBJ hobj); PVOID WINAPI GdiRemoveClientObjLink( _In_ HGDIOBJ hobj); extern ULONG gcClientObj; VOID WINAPI METADC_DeleteObject(HGDIOBJ hobj); BOOL WINAPI METADC_DeleteDC( _In_ HDC hdc); INT WINAPI METADC16_Escape( _In_ HDC hdc, _In_ INT nEscape, _In_ INT cbInput, _In_ LPCSTR lpvInData, _Out_ LPVOID lpvOutData); BOOL WINAPI METADC_ExtTextOutW( HDC hdc, INT x, INT y, UINT fuOptions, const RECT *lprc, LPCWSTR lpString, UINT cchString, const INT *lpDx); BOOL WINAPI METADC_PatBlt( _In_ HDC hdc, _In_ INT xLeft, _In_ INT yTop, _In_ INT nWidth, _In_ INT nHeight, _In_ DWORD dwRop); /* The following METADC_* functions follow this pattern: */ #define HANDLE_METADC0P(_RetType, _Func, dwError, hdc, ...) \ if (GDI_HANDLE_GET_TYPE(hdc) != GDILoObjType_LO_DC_TYPE) \ { \ DWORD_PTR dwResult; \ if (METADC_Dispatch(DCFUNC_##_Func, &dwResult, (DWORD_PTR)dwError, hdc)) \ { \ return (_RetType)dwResult; \ } \ } #define HANDLE_METADC(_RetType, _Func, dwError, hdc, ...) \ if (GDI_HANDLE_GET_TYPE(hdc) != GDILoObjType_LO_DC_TYPE) \ { \ DWORD_PTR dwResult = 1; \ if (METADC_Dispatch(DCFUNC_##_Func, &dwResult, (DWORD_PTR)dwError, hdc, __VA_ARGS__)) \ { \ return (_RetType)dwResult; \ } \ } typedef enum _DCFUNC { //DCFUNC_AbortDoc, DCFUNC_AbortPath, DCFUNC_AlphaBlend, // UNIMPLEMENTED DCFUNC_AngleArc, // UNIMPLEMENTED DCFUNC_Arc, DCFUNC_ArcTo, // UNIMPLEMENTED DCFUNC_BeginPath, //DCFUNC_BitBlt, DCFUNC_Chord, DCFUNC_CloseFigure, DCFUNC_Ellipse, DCFUNC_EndPath, DCFUNC_ExcludeClipRect, DCFUNC_ExtEscape, DCFUNC_ExtFloodFill, DCFUNC_ExtSelectClipRgn, DCFUNC_ExtTextOut, DCFUNC_FillPath, DCFUNC_FillRgn, DCFUNC_FlattenPath, DCFUNC_FrameRgn, DCFUNC_GetDeviceCaps, DCFUNC_GdiComment, DCFUNC_GradientFill, // UNIMPLEMENTED DCFUNC_IntersectClipRect, DCFUNC_InvertRgn, DCFUNC_LineTo, DCFUNC_MaskBlt, // UNIMPLEMENTED DCFUNC_ModifyWorldTransform, DCFUNC_MoveTo, DCFUNC_OffsetClipRgn, DCFUNC_OffsetViewportOrgEx, DCFUNC_OffsetWindowOrgEx, DCFUNC_PathToRegion, // UNIMPLEMENTED DCFUNC_PatBlt, DCFUNC_Pie, DCFUNC_PlgBlt, // UNIMPLEMENTED DCFUNC_PolyBezier, DCFUNC_PolyBezierTo, DCFUNC_PolyDraw, DCFUNC_Polygon, DCFUNC_Polyline, DCFUNC_PolylineTo, DCFUNC_PolyPolygon, DCFUNC_PolyPolyline, DCFUNC_RealizePalette, DCFUNC_Rectangle, DCFUNC_RestoreDC, DCFUNC_RoundRect, DCFUNC_SaveDC, DCFUNC_ScaleViewportExtEx, DCFUNC_ScaleWindowExtEx, DCFUNC_SelectBrush, DCFUNC_SelectClipPath, DCFUNC_SelectFont, DCFUNC_SelectPalette, DCFUNC_SelectPen, DCFUNC_SetDCBrushColor, DCFUNC_SetDCPenColor, DCFUNC_SetDIBitsToDevice, DCFUNC_SetBkColor, DCFUNC_SetBkMode, DCFUNC_SetLayout, //DCFUNC_SetMapMode, DCFUNC_SetPixel, DCFUNC_SetPolyFillMode, DCFUNC_SetROP2, DCFUNC_SetStretchBltMode, DCFUNC_SetTextAlign, DCFUNC_SetTextCharacterExtra, DCFUNC_SetTextColor, DCFUNC_SetTextJustification, DCFUNC_SetViewportExtEx, DCFUNC_SetViewportOrgEx, DCFUNC_SetWindowExtEx, DCFUNC_SetWindowOrgEx, DCFUNC_SetWorldTransform, DCFUNC_StretchBlt, DCFUNC_StrokeAndFillPath, DCFUNC_StrokePath, DCFUNC_TransparentBlt, // UNIMPLEMENTED DCFUNC_WidenPath, } DCFUNC; BOOL METADC_Dispatch( _In_ DCFUNC eFunction, _Out_ PDWORD_PTR pdwResult, _In_ DWORD_PTR dwError, _In_ HDC hdc, ...); #define HANDLE_METADC2(_RetType, _Func, hdc, ...) \ if (GDI_HANDLE_GET_TYPE(hdc) != GDILoObjType_LO_DC_TYPE) \ { \ _RetType result; \ if (METADC_##_Func(&result, hdc, __VA_ARGS__)) \ { \ return result; \ } \ } BOOL WINAPI METADC_GetAndSetDCDWord( _Out_ PDWORD pdwResult, _In_ HDC hdc, _In_ UINT u, _In_ DWORD dwIn, _In_ ULONG ulMFId, _In_ USHORT usMF16Id, _In_ DWORD dwError); /* EOF */