2020-09-30 16:53:55 +02:00

261 lines
7.9 KiB
C++

/******************************Module*Header*******************************\
* Module Name: DdHmgr.h
*
* This file contains all the prototypes for the DirectDraw handle mangager.
*
* Added header: 30-Apr-1999 16:31:46
* Author: Lindsay Steventon (linstev)
*
* Copyright (c) 1999 Microsoft Corporation
\**************************************************************************/
// <--full unique-->
// +--------+------+----------------+
// | unique | type | index |
// +--------+------+----------------+
//
// TYPE - types used by DDRAW & D3D
//
// UNIQUE - bits that are incremented for each instance of the handle
// FULLUNIQUE - bits used for comparing identical handles. This includes the TYPE
//
// INDEX - index into server side handle table
//
#define DD_TABLESIZE_DELTA ((PAGE_SIZE*4) / sizeof(DD_ENTRY)) // 4 pages of entries each time
#define DD_DEF_TYPE 0
#define DD_DIRECTDRAW_TYPE 1
#define DD_SURFACE_TYPE 2
#define D3D_HANDLE_TYPE 3
#define DD_VIDEOPORT_TYPE 4
#define DD_MOTIONCOMP_TYPE 5
#define DD_MAX_TYPE 5
#define DD_INDEX_BITS 21 // 2^21 ~ 2 million handles
#define DD_TYPE_BITS 3 // 2^3 = 8, we only need 6
#define DD_UNIQUE_BITS 8 // identifies each new handle
#define DD_NONINDEX_BITS (32 - DD_INDEX_BITS)
#define DD_INDEX_SHIFT 0
#define DD_TYPE_SHIFT (DD_INDEX_BITS)
#define DD_UNIQUE_SHIFT (DD_TYPE_SHIFT + DD_TYPE_BITS)
// MASKS contain the bits of the handle used for the paricular field
#define DD_NONINDEX_MASK(shift,cbits) ( ((1 << (cbits)) - 1) << (shift) )
#define DD_INDEX_MASK ((1 << DD_INDEX_BITS) - 1)
#define DD_TYPE_MASK (DD_NONINDEX_MASK(DD_TYPE_SHIFT, DD_TYPE_BITS))
#define DD_UNIQUE_MASK (DD_NONINDEX_MASK(DD_UNIQUE_SHIFT, DD_UNIQUE_BITS))
#define DD_FULLUNIQUE_MASK (DD_UNIQUE_MASK | DD_TYPE_MASK)
#define DD_MAKE_HMGR_HANDLE(Index,Unique) LongToHandle(((((LONG) Unique) << DD_INDEX_BITS) | ((LONG) Index)))
// NOTE that DD_UNIQUE_INCREMENT is based on the uniqueness being a short, not a full handle
#define DD_UNIQUE_INCREMENT (1 << (DD_UNIQUE_SHIFT - DD_INDEX_BITS))
#define DdHmgIfromH(h) (ULONG)((ULONG_PTR)(h) & DD_INDEX_MASK)
#define DdHmgUfromH(h) ((USHORT) (((ULONG_PTR)(h) & DD_FULLUNIQUE_MASK) >> DD_TYPE_SHIFT))
#define DdHmgObjtype(h) ((DD_OBJTYPE)(((ULONG_PTR)(h) & DD_TYPE_MASK) >> DD_TYPE_SHIFT))
// given a usUnique and a type, modify it to contain a new type
#define DD_USUNIQUE(u,t) (USHORT)((u & ((ULONG)DD_UNIQUE_MASK >> (ULONG)DD_INDEX_BITS)) | \
(t << (DD_TYPE_SHIFT - DD_INDEX_BITS)))
#define DD_MAX_HANDLE_COUNT (1 << (32 - DD_NONINDEX_BITS))
#define DD_HMGR_HANDLE_BASE 1
ULONG FASTCALL DdHmgQueryLock(HDD_OBJ hobj);
BOOL DdHmgCreate();
BOOL DdHmgDestroy();
BOOL DdHmgCloseProcess(W32PID W32Pid);
HDD_OBJ DdHmgAlloc(ULONGSIZE_T,DD_OBJTYPE,USHORT);
VOID DdHmgFree(HDD_OBJ);
VOID DdFreeObject(PVOID pvFree, ULONG ulType);
PDD_OBJ FASTCALL DdHmgLock(HDD_OBJ,DD_OBJTYPE,BOOL);
PDD_OBJ FASTCALL DdHmgNextObjt(HDD_OBJ hobj, DD_OBJTYPE objt);
PVOID DdHmgRemoveObject(HDD_OBJ,LONG,LONG,BOOL,DD_OBJTYPE);
VOID DdHmgAcquireHmgrSemaphore();
VOID DdHmgReleaseHmgrSemaphore();
// DirectDraw Handle Manager data.
extern ULONG gcSizeDdHmgr;
extern DD_ENTRY *gpentDdHmgr;
extern HDD_OBJ ghFreeDdHmgr;
extern ULONG gcMaxDdHmgr;
extern PLARGE_INTEGER gpLockShortDelay;
/*********************************MACRO************************************\
* INC_EXCLUSIVE_REF_CNT - increment object's exclusive reference count
* DEC_EXCLUSIVE_REF_CNT - decrement object's exclusive reference count
*
* Note that the InterlockedIncrement/Decrement treats the cExclusiveLock
* as a ULONG. cExclusiveLock is declared as a USHORT and the increment
* overlaps with the BASEOBJECT::BaseFlags. If the BaseFlags were ever changed,
* this code may have to be changed to use an InterlockedCompareExchange loop.
* See BASEOBJECT declaration.
*
*
* Arguments:
*
* pObj - pointer to object
*
* Return Value:
*
* None
*
\**************************************************************************/
#define INC_EXCLUSIVE_REF_CNT(pObj) \
InterlockedIncrement((LONG *)& (((PDD_OBJ) pObj)->cExclusiveLock))
#define DEC_EXCLUSIVE_REF_CNT(pObj) \
InterlockedDecrement((LONG *)& (((PDD_OBJ) pObj)->cExclusiveLock))
/*******************************Function***********************************\
* VerifyObjectOwner
*
* Verifies ownership of the object passed in via the PDD_ENTRY.
*
* History:
*
* 21-Feb-1996 -by- Mark Enstrom [marke]
* 12-Mar-2001 -by- Scott Mackowski [ScottMa] (taken from DDHANDLELOCK)
*
\**************************************************************************/
inline
BOOL VerifyObjectOwner(PDD_ENTRY pentry)
{
DD_OBJECTOWNER Obj;
Obj = pentry->ObjectOwner;
if ((DD_OBJECTOWNER_PID(Obj) != DD_W32GetCurrentPID()) &&
(DD_OBJECTOWNER_PID(Obj) != OBJECT_OWNER_PUBLIC) )
{
return FALSE;
}
return TRUE;
}
// Notes on entry structure
//
// The internal entry in the handle manager appears as follows
//
// +-------------------+
// | einfo.pobj, hfree | 4 bytes
// +-------------------+
// | ObjectOwner | 4 bytes
// +-------------------+
// | FullUnique | 2 bytes
// +-------------------+
// | Objt | 1 byte
// +-------------------+
// | Flags | 1 byte
// +-------------------+
// | dwReserved | 4 bytes
// +-------------------+
// 16 bytes total space
#define HMGR_ALLOC_LOCK 0x0001
#define HMGR_ALLOC_ALT_LOCK 0x0002
#define HMGR_NO_ZERO_INIT 0x0004
#define HMGR_MAKE_PUBLIC 0x0008
class DD_ENTRYOBJ : public _DD_ENTRY
{
public:
DD_ENTRYOBJ() { }
~DD_ENTRYOBJ() { }
VOID vSetup(PDD_OBJ pObj, DD_OBJTYPE objt_, FSHORT fs)
{
DD_OBJECTOWNER ObjNew;
ObjNew = ObjectOwner;
einfo.pobj = (PDD_OBJ) pObj;
Objt = objt_;
Flags = 0;
dwReserved = NULL;
if (fs & HMGR_MAKE_PUBLIC)
{
DD_SET_OBJECTOWNER_PID(ObjNew,OBJECT_OWNER_PUBLIC);
}
else
{
DD_SET_OBJECTOWNER_PID(ObjNew,DD_W32GetCurrentPID());
}
if (fs & HMGR_ALLOC_LOCK)
{
pObj->Tid = (ULONG_PTR)PsGetCurrentThread();
}
pObj->cExclusiveLock = (USHORT)(fs & HMGR_ALLOC_LOCK);
pObj->ulShareCount = (USHORT)0;
//
// Update the ObjectOwner.
//
ObjectOwner = ObjNew;
}
VOID vFree(UINT uiIndex)
{
//
// handle must already be locked
//
DD_ENTRY *pentry = &gpentDdHmgr[uiIndex];
DD_OBJECTOWNER ObjNew = pentry->ObjectOwner;
//
// Insert the specified handle in the free list.
//
pentry->einfo.hFree = ghFreeDdHmgr;
ghFreeDdHmgr = (HDD_OBJ) (ULONG_PTR)uiIndex;
//
// Set the object type to the default type so all handle translations
// will fail and increment the uniqueness value.
//
Objt = (DD_OBJTYPE) DD_DEF_TYPE;
FullUnique += DD_UNIQUE_INCREMENT;
//
// clear user date pointer
//
dwReserved = NULL;
//
// Clear shared count, set initial pid. Caller
// must unlock handle.
//
DD_SET_OBJECTOWNER_PID(ObjNew,0);
pentry->ObjectOwner = ObjNew;
}
BOOL bOwnedBy(W32PID pid_)
{
return((Objt != DD_DEF_TYPE) && (DD_OBJECTOWNER_PID(ObjectOwner) == (pid_ & DD_PID_BITS)));
}
};
typedef DD_ENTRYOBJ *PDD_ENTRYOBJ;