Windows2003-3790/windows/core/dxkernel/dxg/dxg.cxx
2020-09-30 16:53:55 +02:00

314 lines
13 KiB
C++

/******************************Module*Header*******************************\
* Module Name: dxg.cxx
*
* Contains the kernel-mode code for DirectX graphics.
*
* Created: 20-Apr-2000
* Author: Hideyuki Nagase [hideyukn]
*
* Copyright (c) 2000 Microsoft Corporation
*
\**************************************************************************/
#include <precomp.hxx>
extern "C" {
NTSTATUS
DriverEntry(
PDRIVER_OBJECT DriverObject,
PUNICODE_STRING RegistryPath
);
}
#if defined(ALLOC_PRAGMA)
#pragma alloc_text(INIT,DriverEntry)
#endif
#if defined(_X86_)
ULONG_PTR DxgUserProbeAddress;
#endif
DRVFN gaDxgFuncs[] =
{
{ INDEX_DxDxgGenericThunk, (PFN) DxDxgGenericThunk },
{ INDEX_DxD3dContextCreate, (PFN) DxD3dContextCreate },
{ INDEX_DxD3dContextDestroy, (PFN) DxD3dContextDestroy },
{ INDEX_DxD3dContextDestroyAll, (PFN) DxD3dContextDestroyAll },
{ INDEX_DxD3dValidateTextureStageState, (PFN) DxD3dValidateTextureStageState },
{ INDEX_DxD3dDrawPrimitives2, (PFN) DxD3dDrawPrimitives2 },
{ INDEX_DxDdGetDriverState, (PFN) DxDdGetDriverState },
{ INDEX_DxDdAddAttachedSurface, (PFN) DxDdAddAttachedSurface },
{ INDEX_DxDdAlphaBlt, (PFN) DxDdAlphaBlt },
{ INDEX_DxDdAttachSurface, (PFN) DxDdAttachSurface },
{ INDEX_DxDdBeginMoCompFrame, (PFN) DxDdBeginMoCompFrame },
{ INDEX_DxDdBlt, (PFN) DxDdBlt },
{ INDEX_DxDdCanCreateSurface, (PFN) DxDdCanCreateSurface },
{ INDEX_DxDdCanCreateD3DBuffer, (PFN) DxDdCanCreateD3DBuffer },
{ INDEX_DxDdColorControl, (PFN) DxDdColorControl },
{ INDEX_DxDdCreateDirectDrawObject, (PFN) DxDdCreateDirectDrawObject },
{ INDEX_DxDdCreateSurface, (PFN) DxDdCreateSurface },
{ INDEX_DxDdCreateD3DBuffer, (PFN) DxDdCreateD3DBuffer },
{ INDEX_DxDdCreateMoComp, (PFN) DxDdCreateMoComp },
{ INDEX_DxDdCreateSurfaceObject, (PFN) DxDdCreateSurfaceObject },
{ INDEX_DxDdDeleteDirectDrawObject, (PFN) DxDdDeleteDirectDrawObject },
{ INDEX_DxDdDeleteSurfaceObject, (PFN) DxDdDeleteSurfaceObject },
{ INDEX_DxDdDestroyMoComp, (PFN) DxDdDestroyMoComp },
{ INDEX_DxDdDestroySurface, (PFN) DxDdDestroySurface },
{ INDEX_DxDdDestroyD3DBuffer, (PFN) DxDdDestroyD3DBuffer },
{ INDEX_DxDdEndMoCompFrame, (PFN) DxDdEndMoCompFrame },
{ INDEX_DxDdFlip, (PFN) DxDdFlip },
{ INDEX_DxDdFlipToGDISurface, (PFN) DxDdFlipToGDISurface },
{ INDEX_DxDdGetAvailDriverMemory, (PFN) DxDdGetAvailDriverMemory },
{ INDEX_DxDdGetBltStatus, (PFN) DxDdGetBltStatus },
{ INDEX_DxDdGetDC, (PFN) DxDdGetDC },
{ INDEX_DxDdGetDriverInfo, (PFN) DxDdGetDriverInfo },
{ INDEX_DxDdGetDxHandle, (PFN) DxDdGetDxHandle },
{ INDEX_DxDdGetFlipStatus, (PFN) DxDdGetFlipStatus },
{ INDEX_DxDdGetInternalMoCompInfo, (PFN) DxDdGetInternalMoCompInfo },
{ INDEX_DxDdGetMoCompBuffInfo, (PFN) DxDdGetMoCompBuffInfo },
{ INDEX_DxDdGetMoCompGuids, (PFN) DxDdGetMoCompGuids },
{ INDEX_DxDdGetMoCompFormats, (PFN) DxDdGetMoCompFormats },
{ INDEX_DxDdGetScanLine, (PFN) DxDdGetScanLine },
{ INDEX_DxDdLock, (PFN) DxDdLock },
{ INDEX_DxDdLockD3D, (PFN) DxDdLockD3D },
{ INDEX_DxDdQueryDirectDrawObject, (PFN) DxDdQueryDirectDrawObject },
{ INDEX_DxDdQueryMoCompStatus, (PFN) DxDdQueryMoCompStatus },
{ INDEX_DxDdReenableDirectDrawObject, (PFN) DxDdReenableDirectDrawObject },
{ INDEX_DxDdReleaseDC, (PFN) DxDdReleaseDC },
{ INDEX_DxDdRenderMoComp, (PFN) DxDdRenderMoComp },
{ INDEX_DxDdResetVisrgn, (PFN) DxDdResetVisrgn },
{ INDEX_DxDdSetColorKey, (PFN) DxDdSetColorKey },
{ INDEX_DxDdSetExclusiveMode, (PFN) DxDdSetExclusiveMode },
{ INDEX_DxDdSetGammaRamp, (PFN) DxDdSetGammaRamp },
{ INDEX_DxDdCreateSurfaceEx, (PFN) DxDdCreateSurfaceEx },
{ INDEX_DxDdSetOverlayPosition, (PFN) DxDdSetOverlayPosition },
{ INDEX_DxDdUnattachSurface, (PFN) DxDdUnattachSurface },
{ INDEX_DxDdUnlock, (PFN) DxDdUnlock },
{ INDEX_DxDdUnlockD3D, (PFN) DxDdUnlockD3D },
{ INDEX_DxDdUpdateOverlay, (PFN) DxDdUpdateOverlay },
{ INDEX_DxDdWaitForVerticalBlank, (PFN) DxDdWaitForVerticalBlank },
{ INDEX_DxDvpCanCreateVideoPort, (PFN) DxDvpCanCreateVideoPort },
{ INDEX_DxDvpColorControl, (PFN) DxDvpColorControl },
{ INDEX_DxDvpCreateVideoPort, (PFN) DxDvpCreateVideoPort },
{ INDEX_DxDvpDestroyVideoPort, (PFN) DxDvpDestroyVideoPort },
{ INDEX_DxDvpFlipVideoPort, (PFN) DxDvpFlipVideoPort },
{ INDEX_DxDvpGetVideoPortBandwidth, (PFN) DxDvpGetVideoPortBandwidth },
{ INDEX_DxDvpGetVideoPortField, (PFN) DxDvpGetVideoPortField },
{ INDEX_DxDvpGetVideoPortFlipStatus, (PFN) DxDvpGetVideoPortFlipStatus },
{ INDEX_DxDvpGetVideoPortInputFormats, (PFN) DxDvpGetVideoPortInputFormats },
{ INDEX_DxDvpGetVideoPortLine, (PFN) DxDvpGetVideoPortLine },
{ INDEX_DxDvpGetVideoPortOutputFormats, (PFN) DxDvpGetVideoPortOutputFormats },
{ INDEX_DxDvpGetVideoPortConnectInfo, (PFN) DxDvpGetVideoPortConnectInfo },
{ INDEX_DxDvpGetVideoSignalStatus, (PFN) DxDvpGetVideoSignalStatus },
{ INDEX_DxDvpUpdateVideoPort, (PFN) DxDvpUpdateVideoPort },
{ INDEX_DxDvpWaitForVideoPortSync, (PFN) DxDvpWaitForVideoPortSync },
{ INDEX_DxDvpAcquireNotification, (PFN) DxDvpAcquireNotification },
{ INDEX_DxDvpReleaseNotification, (PFN) DxDvpReleaseNotification },
{ INDEX_DxDdHeapVidMemAllocAligned, (PFN) DxDdHeapVidMemAllocAligned },
{ INDEX_DxDdHeapVidMemFree, (PFN) DxDdHeapVidMemFree },
{ INDEX_DxDdEnableDirectDraw, (PFN) DxDdEnableDirectDraw },
{ INDEX_DxDdDisableDirectDraw, (PFN) DxDdDisableDirectDraw },
{ INDEX_DxDdSuspendDirectDraw, (PFN) DxDdSuspendDirectDraw },
{ INDEX_DxDdResumeDirectDraw, (PFN) DxDdResumeDirectDraw },
{ INDEX_DxDdDynamicModeChange, (PFN) DxDdDynamicModeChange },
{ INDEX_DxDdCloseProcess, (PFN) DxDdCloseProcess },
{ INDEX_DxDdGetDirectDrawBounds, (PFN) DxDdGetDirectDrawBounds },
{ INDEX_DxDdEnableDirectDrawRedirection, (PFN) DxDdEnableDirectDrawRedirection },
{ INDEX_DxDdAllocPrivateUserMem, (PFN) DxDdAllocPrivateUserMem },
{ INDEX_DxDdFreePrivateUserMem, (PFN) DxDdFreePrivateUserMem },
{ INDEX_DxDdLockDirectDrawSurface, (PFN) DxDdLockDirectDrawSurface },
{ INDEX_DxDdUnlockDirectDrawSurface, (PFN) DxDdUnlockDirectDrawSurface },
{ INDEX_DxDdSetAccelLevel, (PFN) DxDdSetAccelLevel },
{ INDEX_DxDdGetSurfaceLock, (PFN) DxDdGetSurfaceLock },
{ INDEX_DxDdEnumLockedSurfaceRect, (PFN) DxDdEnumLockedSurfaceRect },
{ INDEX_DxDdIoctl, (PFN) DxDdIoctl }
};
ULONG gcDxgFuncs = sizeof(gaDxgFuncs) / sizeof(DRVFN);
//
// Pointer to the pointer table to Win32k.sys
//
DRVFN *gpEngFuncs = NULL;
//
// This is the global pointer to the dummy page to which all of the video
// memory is mapped when we need to forcibly unmap it. Instead of causing
// the app to fault accessing unmapped memory, we remap it to this play
// area where it can doodle around till it discovers that it had "lost"
// surfaces.
//
PVOID gpDummyPage;
LONG gcDummyPageRefCnt;
HSEMAPHORE ghsemDummyPage;
PEPROCESS gpepSession = NULL;
/***************************************************************************\
* NTSTATUS DriverEntry
*
* This routine is never actually called, but we need it to link.
*
\***************************************************************************/
extern "C"
NTSTATUS
DriverEntry(
PDRIVER_OBJECT DriverObject,
PUNICODE_STRING RegistryPath
)
{
return(STATUS_SUCCESS);
}
/***************************************************************************\
* NTSTATUS StartupDxGraphics
*
* This routine is called by win32k.sys to initialize dxg.sys.
*
\***************************************************************************/
extern "C"
NTSTATUS
DxDdStartupDxGraphics(
ULONG cjEng,
DRVENABLEDATA *pdedEng,
ULONG cjDxg,
DRVENABLEDATA *pdedDxg,
DWORD *pdwDirectDrawContext,
PEPROCESS pepSession
)
{
if ((cjEng >= sizeof(DRVENABLEDATA)) &&
(cjDxg >= sizeof(DRVENABLEDATA)))
{
//
// Initialize global variables
//
gpDummyPage = NULL;
gcDummyPageRefCnt = 0;
ghsemDummyPage = NULL;
//
// Give back function pointers to GDI, which they will call.
//
pdedDxg->iDriverVersion = 0x00080000; // Supporting until DX8.
pdedDxg->c = gcDxgFuncs;
pdedDxg->pdrvfn = gaDxgFuncs;
//
// pdedEng->iDriverVersion contains win32k version (= OS platform version).
//
// - 0x00050001 for Whistler.
//
//
// Check the function printers from GDI, which we will call.
//
if (pdedEng->c != INDEX_WIN32K_TABLE_SIZE)
{
WARNING("pdedEng->c != INDEX_WIN32K_TABLE_MAX\n");
return STATUS_INTERNAL_ERROR;
}
//
// Make sure all pointers are sorted and nothing missing.
//
for (ULONG i = 1; i < INDEX_WIN32K_TABLE_SIZE; i++)
{
if ((pdedEng->pdrvfn[i].iFunc != i) ||
(pdedEng->pdrvfn[i].pfn == NULL))
{
WARNING("pdedEng->pdrvfn is not well orded or pointer is missing\n");
return STATUS_INTERNAL_ERROR;
}
}
//
// If everything is good, keep the pointer.
//
gpEngFuncs = pdedEng->pdrvfn;
//
// Return size of DirectDraw context, so that GDI can allocate it inside HDEV.
//
*pdwDirectDrawContext = sizeof(EDD_DIRECTDRAW_GLOBAL);
//
// Initialize handle manager
//
if (!DdHmgCreate())
{
goto Error_Exit;
}
//
// Create semaphore to sync dummy page global variable.
//
if ((ghsemDummyPage = EngCreateSemaphore()) == NULL)
{
goto Error_Exit;
}
#if defined(_X86_)
//
// Keep our own copy of this to avoid double indirections on probing
//
DxgUserProbeAddress = *MmUserProbeAddress;
#endif
//
// Keep pointer to CsrSS process for this session.
//
gpepSession = pepSession;
return(STATUS_SUCCESS);
}
return(STATUS_BUFFER_TOO_SMALL);
Error_Exit:
DdHmgDestroy();
if (ghsemDummyPage)
{
EngDeleteSemaphore(ghsemDummyPage);
ghsemDummyPage = NULL;
}
return(STATUS_NO_MEMORY);
}
/***************************************************************************\
* NTSTATUS CleanupDxGraphics
*
* This routine is called by win32k.sys to uninitialize dxg.sys
* just before unload.
*
\***************************************************************************/
extern "C"
NTSTATUS
DxDdCleanupDxGraphics(VOID)
{
DdHmgDestroy();
if (ghsemDummyPage)
{
if (gpDummyPage)
{
ExFreePool(gpDummyPage);
gpDummyPage = NULL;
gcDummyPageRefCnt = 0;
}
EngDeleteSemaphore(ghsemDummyPage);
ghsemDummyPage = NULL;
}
return (STATUS_SUCCESS);
}