mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
368 lines
9.7 KiB
C
368 lines
9.7 KiB
C
/*
|
|
* PROJECT: ReactOS win32 kernel mode subsystem
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* FILE: win32ss/gdi/ntgdi/gdikdbgext.c
|
|
* PURPOSE: KDBG extension for GDI
|
|
* PROGRAMMERS: Timo Kreuzer
|
|
*/
|
|
|
|
/* INCLUDES ******************************************************************/
|
|
|
|
#include <win32k.h>
|
|
//#define NDEBUG
|
|
//#include <debug.h>
|
|
|
|
extern PENTRY gpentHmgr;
|
|
extern PULONG gpaulRefCount;
|
|
extern ULONG gulFirstUnused;
|
|
|
|
|
|
static const char * gpszObjectTypes[] =
|
|
{
|
|
"FREE", "DC", "UNUSED1", "UNUSED2", "RGN", "SURF", "CLIENTOBJ", "PATH",
|
|
"PAL", "ICMLCS", "LFONT", "RFONT", "PFE", "PFT", "ICMCXF", "SPRITE",
|
|
"BRUSH", "UMPD", "UNUSED4", "SPACE", "UNUSED5", "META", "EFSTATE",
|
|
"BMFD", "VTFD", "TTFD", "RC", "TEMP", "DRVOBJ", "DCIOBJ", "SPOOL",
|
|
"RESERVED", "ALL"
|
|
};
|
|
|
|
|
|
BOOLEAN
|
|
KdbIsMemoryValid(PVOID pvBase, ULONG cjSize)
|
|
{
|
|
PUCHAR pjAddress;
|
|
|
|
pjAddress = ALIGN_DOWN_POINTER_BY(pvBase, PAGE_SIZE);
|
|
|
|
while (pjAddress < (PUCHAR)pvBase + cjSize)
|
|
{
|
|
if (!MmIsAddressValid(pjAddress)) return FALSE;
|
|
pjAddress += PAGE_SIZE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static
|
|
BOOL
|
|
KdbGetHexNumber(char *pszNum, ULONG_PTR *pulValue)
|
|
{
|
|
char *endptr;
|
|
|
|
/* Skip optional '0x' prefix */
|
|
if ((pszNum[0] == '0') && ((pszNum[1] == 'x') || (pszNum[1] == 'X')))
|
|
pszNum += 2;
|
|
|
|
/* Make a number from the string (hex) */
|
|
*pulValue = strtoul(pszNum, &endptr, 16);
|
|
|
|
return (*endptr == '\0');
|
|
}
|
|
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_help(VOID)
|
|
{
|
|
DbgPrint("GDI KDBG extension.\nAvailable commands:\n"
|
|
"- help - Displays this screen.\n"
|
|
"- dumpht [<type>] - Dumps all handles of <type> or lists all types\n"
|
|
"- handle <handle> - Displays information about a handle\n"
|
|
"- entry <entry> - Displays an ENTRY, <entry> can be a pointer or index\n"
|
|
"- baseobject <object> - Displays a BASEOBJECT\n"
|
|
#if DBG_ENABLE_EVENT_LOGGING
|
|
"- eventlist <object> - Displays the eventlist for an object\n"
|
|
#endif
|
|
);
|
|
}
|
|
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_dumpht(ULONG argc, char *argv[])
|
|
{
|
|
ULONG i;
|
|
UCHAR Objt, jReqestedType;
|
|
PENTRY pentry;
|
|
POBJ pobj;
|
|
KAPC_STATE ApcState;
|
|
ULONG_PTR ulArg;
|
|
|
|
/* No CSRSS, no handle table */
|
|
if (!gpepCSRSS) return;
|
|
KeStackAttachProcess(&gpepCSRSS->Pcb, &ApcState);
|
|
|
|
if (argc == 0)
|
|
{
|
|
USHORT Counts[GDIObjType_MAX_TYPE + 2] = {0};
|
|
|
|
/* Loop all possibly used entries in the handle table */
|
|
for (i = RESERVE_ENTRIES_COUNT; i < gulFirstUnused; i++)
|
|
{
|
|
if (KdbIsMemoryValid(&gpentHmgr[i], sizeof(ENTRY)))
|
|
{
|
|
Objt = gpentHmgr[i].Objt & 0x1F;
|
|
Counts[Objt]++;
|
|
}
|
|
}
|
|
|
|
DbgPrint("Type Count\n");
|
|
DbgPrint("-------------------\n");
|
|
for (i = 0; i <= GDIObjType_MAX_TYPE; i++)
|
|
{
|
|
DbgPrint("%02x %-9s %d\n",
|
|
i, gpszObjectTypes[i], Counts[i]);
|
|
}
|
|
DbgPrint("\n");
|
|
}
|
|
else
|
|
{
|
|
/* Loop all object types */
|
|
for (i = 0; i <= GDIObjType_MAX_TYPE + 1; i++)
|
|
{
|
|
/* Check if this object type was requested */
|
|
if (stricmp(argv[0], gpszObjectTypes[i]) == 0) break;
|
|
}
|
|
|
|
/* Check if we didn't find it yet */
|
|
if (i > GDIObjType_MAX_TYPE + 1)
|
|
{
|
|
/* Try if it's a number */
|
|
if (!KdbGetHexNumber(argv[0], &ulArg))
|
|
{
|
|
DbgPrint("Invalid parameter: %s\n", argv[0]);
|
|
return;
|
|
}
|
|
|
|
/* Check if it's inside the allowed range */
|
|
if (i > GDIObjType_MAX_TYPE)
|
|
{
|
|
DbgPrint("Unknown object type: %s\n", argv[0]);
|
|
goto leave;
|
|
}
|
|
}
|
|
|
|
jReqestedType = i;
|
|
|
|
/* Print header */
|
|
DbgPrint("Index Handle Type pObject ThreadId cLocks ulRefCount\n");
|
|
DbgPrint("---------------------------------------------------------------\n");
|
|
|
|
/* Loop all possibly used entries in the handle table */
|
|
for (i = RESERVE_ENTRIES_COUNT; i < gulFirstUnused; i++)
|
|
{
|
|
/* Get the entry and the object */
|
|
pentry = &gpentHmgr[i];
|
|
|
|
if (!MmIsAddressValid(pentry)) continue;
|
|
|
|
pobj = pentry->einfo.pobj;
|
|
Objt = pentry->Objt & 0x1F;
|
|
|
|
/* Check if ALL objects are requested, or the object type matches */
|
|
if ((jReqestedType == GDIObjType_MAX_TYPE + 1) ||
|
|
(Objt == jReqestedType))
|
|
{
|
|
DbgPrint("%04lx %p %-9s 0x%p 0x%06lx %-6ld ",
|
|
i, pobj->hHmgr, gpszObjectTypes[Objt], pobj,
|
|
pobj->dwThreadId, pobj->cExclusiveLock);
|
|
if (MmIsAddressValid(&gpaulRefCount[i]))
|
|
DbgPrint("0x%08lx\n", gpaulRefCount[i]);
|
|
else
|
|
DbgPrint("??????????\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
leave:
|
|
KeUnstackDetachProcess(&ApcState);
|
|
}
|
|
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_handle(char *argv)
|
|
{
|
|
ULONG_PTR ulObject;
|
|
BASEOBJECT *pobj;
|
|
ENTRY *pentry;
|
|
USHORT usIndex;
|
|
KAPC_STATE ApcState;
|
|
|
|
/* Convert the parameter into a number */
|
|
if (!KdbGetHexNumber(argv, &ulObject))
|
|
{
|
|
DbgPrint("Invalid parameter: %s\n", argv);
|
|
return;
|
|
}
|
|
|
|
/* No CSRSS, no handle table */
|
|
if (!gpepCSRSS) return;
|
|
KeStackAttachProcess(&gpepCSRSS->Pcb, &ApcState);
|
|
|
|
usIndex = ulObject & 0xFFFF;
|
|
pentry = &gpentHmgr[usIndex];
|
|
|
|
if (MmIsAddressValid(pentry))
|
|
{
|
|
pobj = pentry->einfo.pobj;
|
|
|
|
DbgPrint("GDI handle=%p, type=%s, index=0x%lx, pentry=%p.\n",
|
|
ulObject, gpszObjectTypes[(ulObject >> 16) & 0x1f],
|
|
usIndex, pentry);
|
|
DbgPrint(" ENTRY = {.pobj = %p, ObjectOwner = 0x%lx, FullUnique = 0x%04x,\n"
|
|
" Objt=0x%02x, Flags = 0x%02x, pUser = 0x%p}\n",
|
|
pentry->einfo.pobj, pentry->ObjectOwner.ulObj, pentry->FullUnique,
|
|
pentry->Objt, pentry->Flags, pentry->pUser);
|
|
DbgPrint(" BASEOBJECT = {hHmgr = %p, dwThreadId = 0x%lx,\n"
|
|
" cExclusiveLock = %ld, BaseFlags = 0x%lx}\n",
|
|
pobj->hHmgr, pobj->dwThreadId,
|
|
pobj->cExclusiveLock, pobj->BaseFlags);
|
|
if (MmIsAddressValid(&gpaulRefCount[usIndex]))
|
|
DbgPrint(" gpaulRefCount[idx] = %ld\n", gpaulRefCount[usIndex]);
|
|
}
|
|
else
|
|
{
|
|
DbgPrint("Coudn't access ENTRY. Probably paged out.\n");
|
|
}
|
|
|
|
KeUnstackDetachProcess(&ApcState);
|
|
}
|
|
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_entry(char *argv)
|
|
{
|
|
ULONG_PTR ulValue;
|
|
PENTRY pentry;
|
|
KAPC_STATE ApcState;
|
|
|
|
/* Convert the parameter into a number */
|
|
if (!KdbGetHexNumber(argv, &ulValue))
|
|
{
|
|
DbgPrint("Invalid parameter: %s\n", argv);
|
|
return;
|
|
}
|
|
|
|
/* No CSRSS, no handle table */
|
|
if (!gpepCSRSS) return;
|
|
KeStackAttachProcess(&gpepCSRSS->Pcb, &ApcState);
|
|
|
|
/* If the parameter is smaller than 0x10000, it's an index */
|
|
pentry = (ulValue <= 0xFFFF) ? &gpentHmgr[ulValue] : (PENTRY)ulValue;
|
|
|
|
/* Check if the address is readable */
|
|
if (!MmIsAddressValid(pentry))
|
|
{
|
|
DbgPrint("Cannot access entry at %p\n", pentry);
|
|
goto cleanup;
|
|
}
|
|
|
|
/* print the entry */
|
|
DbgPrint("Dumping ENTRY #%ld, @%p:\n", (pentry - gpentHmgr), pentry);
|
|
if (pentry->Objt != 0)
|
|
DbgPrint(" pobj = 0x%p\n", pentry->einfo.pobj);
|
|
else
|
|
DbgPrint(" hFree = 0x%p\n", pentry->einfo.hFree);
|
|
DbgPrint(" ObjectOwner = 0x%p\n", pentry->ObjectOwner.ulObj);
|
|
DbgPrint(" FullUnique = 0x%x\n", pentry->FullUnique);
|
|
DbgPrint(" Objt = 0x%x (%s)\n", pentry->Objt,
|
|
pentry->Objt <= 0x1E ? gpszObjectTypes[pentry->Objt] : "invalid");
|
|
DbgPrint(" Flags = 0x%x\n", pentry->Flags);
|
|
DbgPrint(" pUser = 0x%p\n", pentry->pUser);
|
|
|
|
cleanup:
|
|
KeUnstackDetachProcess(&ApcState);
|
|
}
|
|
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_baseobject(char *argv)
|
|
{
|
|
}
|
|
|
|
#if DBG_ENABLE_EVENT_LOGGING
|
|
static
|
|
VOID
|
|
KdbCommand_Gdi_eventlist(char *argv)
|
|
{
|
|
ULONG_PTR ulValue;
|
|
POBJ pobj;
|
|
PSLIST_ENTRY psle, psleFirst;
|
|
PLOGENTRY pLogEntry;
|
|
|
|
/* Convert the parameter into a number */
|
|
if (!KdbGetHexNumber(argv, &ulValue))
|
|
{
|
|
DbgPrint("Invalid parameter: %s\n", argv);
|
|
return;
|
|
}
|
|
|
|
pobj = (POBJ)ulValue;
|
|
|
|
/* Check if the address is readable */
|
|
if (!KdbIsMemoryValid(pobj, sizeof(BASEOBJECT)))
|
|
{
|
|
DbgPrint("Cannot access BASEOBJECT at %p\n", pobj);
|
|
return;
|
|
}
|
|
|
|
/* The kernel doesn't export RtlFirstEntrySList :( */
|
|
psleFirst = InterlockedFlushSList(&pobj->slhLog);
|
|
|
|
/* Loop all events, but don't remove them */
|
|
for (psle = psleFirst; psle != NULL; psle = psle->Next)
|
|
{
|
|
pLogEntry = CONTAINING_RECORD(psle, LOGENTRY, sleLink);
|
|
DbgPrintEvent(pLogEntry);
|
|
}
|
|
|
|
/* Put the log back in place */
|
|
InterlockedPushEntrySList(&pobj->slhLog, psleFirst);
|
|
}
|
|
#endif
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
DbgGdiKdbgCliCallback(
|
|
IN PCHAR pszCommand,
|
|
IN ULONG argc,
|
|
IN PCH argv[])
|
|
{
|
|
|
|
if (stricmp(argv[0], "!gdi.help") == 0)
|
|
{
|
|
KdbCommand_Gdi_help();
|
|
}
|
|
else if (stricmp(argv[0], "!gdi.dumpht") == 0)
|
|
{
|
|
KdbCommand_Gdi_dumpht(argc - 1, argv + 1);
|
|
}
|
|
else if (stricmp(argv[0], "!gdi.handle") == 0)
|
|
{
|
|
KdbCommand_Gdi_handle(argv[1]);
|
|
}
|
|
else if (stricmp(argv[0], "!gdi.entry") == 0)
|
|
{
|
|
KdbCommand_Gdi_entry(argv[1]);
|
|
}
|
|
else if (stricmp(argv[0], "!gdi.baseobject") == 0)
|
|
{
|
|
KdbCommand_Gdi_baseobject(argv[1]);
|
|
}
|
|
#if DBG_ENABLE_EVENT_LOGGING
|
|
else if (stricmp(argv[0], "!gdi.eventlist") == 0)
|
|
{
|
|
KdbCommand_Gdi_eventlist(argv[1]);
|
|
}
|
|
#endif
|
|
else
|
|
{
|
|
/* Not handled */
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
|
|
|