You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
304 lines
5.9 KiB
304 lines
5.9 KiB
/*++
|
|
|
|
Copyright (c) 1992 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
Lists.c
|
|
|
|
Abstract:
|
|
|
|
WinDbg Extension Api
|
|
|
|
Author:
|
|
|
|
Gary Kimura [GaryKi] 25-Mar-96
|
|
|
|
Environment:
|
|
|
|
User Mode.
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
#define ReadAtAddress(A,V,S,R) \
|
|
(ReadMemory( (A), &(V), (S), &R ) && (R >= (S)))
|
|
|
|
#define BIG_READ (sizeof(ULONG) * 4)
|
|
#define SMALL_READ (sizeof(ULONG) * 2)
|
|
|
|
VOID
|
|
DumpListByLinks (
|
|
IN ULONG64 StartAddress,
|
|
IN ULONG MaxCount,
|
|
IN ULONG Bias,
|
|
IN LOGICAL UseFlink
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Dump a list by its blinks.
|
|
|
|
Arguments:
|
|
|
|
arg - [Address] [count] [bias]
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG64 Address;
|
|
ULONG Buffer[4];
|
|
ULONG ReadSize;
|
|
ULONG BytesRead;
|
|
ULONG Count;
|
|
|
|
//
|
|
// set our starting address and then while the count is greater than zero
|
|
// and the starting address is not equal to the current dumping address
|
|
// we'll read in 4 ulongs, dump them, and then go through the flink&blink
|
|
// using the specified bias.
|
|
//
|
|
|
|
if (!IsPtr64()) {
|
|
StartAddress = (ULONG64) (LONG64) (LONG) StartAddress;
|
|
}
|
|
Address = StartAddress;
|
|
|
|
ReadSize = BIG_READ;
|
|
|
|
for (Count = 0; Count < MaxCount; ) {
|
|
|
|
if (ReadAtAddress( Address, Buffer, ReadSize, BytesRead ) == 0) {
|
|
ReadSize = SMALL_READ;
|
|
if (ReadAtAddress( Address, Buffer, ReadSize, BytesRead ) == 0) {
|
|
dprintf("Can't Read Memory at %08lx\n", Address);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ReadSize == BIG_READ) {
|
|
dprintf("%08p %08lx %08lx %08lx %08lx\n", Address, Buffer[0], Buffer[1], Buffer[2], Buffer[3]);
|
|
}
|
|
else {
|
|
dprintf("%08p %08lx %08lx\n", Address, Buffer[0], Buffer[1]);
|
|
}
|
|
|
|
Count += 1;
|
|
|
|
//
|
|
// the bias tells us which bits to knock out of the pointer
|
|
//
|
|
|
|
if (UseFlink == TRUE) {
|
|
GetFieldValue(Address, "LIST_ENTRY", "Flink", Address);
|
|
Address &= ~((ULONG64)Bias);
|
|
}
|
|
else {
|
|
GetFieldValue(Address, "LIST_ENTRY", "Blink", Address);
|
|
Address &= ~((ULONG64)Bias);
|
|
}
|
|
|
|
if (Address == StartAddress) {
|
|
break;
|
|
}
|
|
|
|
if (((Count & 0xf) == 0) && CheckControlC() ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (Count != 0) {
|
|
dprintf("0x%x entries dumped\n", Count);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
DECLARE_API( dflink )
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Dump a list by its flinks.
|
|
|
|
Arguments:
|
|
|
|
arg - [Address] [count] [bias]
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG64 StartAddress;
|
|
ULONG Count;
|
|
ULONG Bias;
|
|
|
|
StartAddress = 0;
|
|
Count = 0x20;
|
|
Bias = 0;
|
|
|
|
//
|
|
// read in the parameters
|
|
//
|
|
|
|
if (GetExpressionEx(args,&StartAddress, &args)) {
|
|
if (!sscanf(args, "%lx %lx", &Count, &Bias)) {
|
|
Bias = 0;
|
|
}
|
|
}
|
|
|
|
DumpListByLinks (StartAddress, Count, Bias, TRUE);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
DECLARE_API( dblink )
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Dump a list by its blinks.
|
|
|
|
Arguments:
|
|
|
|
arg - [Address] [count] [bias]
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG64 StartAddress;
|
|
ULONG Count;
|
|
ULONG Bias;
|
|
|
|
StartAddress = 0;
|
|
Count = 0x20;
|
|
Bias = 0;
|
|
|
|
//
|
|
// read in the parameters
|
|
//
|
|
|
|
if (GetExpressionEx(args,&StartAddress, &args)) {
|
|
if (!sscanf(args, "%lx %lx", &Count, &Bias)) {
|
|
Bias = 0;
|
|
}
|
|
}
|
|
|
|
DumpListByLinks (StartAddress, Count, Bias, FALSE);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
DECLARE_API( validatelist )
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Validate a doubly linked list
|
|
|
|
Arguments:
|
|
|
|
arg - [Address]
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG64 StartAddress, Address, Flink, Blink, LastAddress, Links[2];
|
|
ULONG Count;
|
|
ULONG ListEntrySize;
|
|
ULONG PointerSize;
|
|
ULONG Status;
|
|
ULONG cb;
|
|
PULONG Links32;
|
|
BOOLEAN First;
|
|
|
|
StartAddress = 0;
|
|
Count = 0;
|
|
|
|
//
|
|
// read in the parameters
|
|
//
|
|
|
|
if (!GetExpressionEx (args, &StartAddress, &args)) {
|
|
dprintf("Failed to process argument\n");
|
|
return S_OK;
|
|
}
|
|
|
|
ListEntrySize = GetTypeSize ("nt!_LIST_ENTRY");
|
|
if (ListEntrySize == 0) {
|
|
dprintf("Failed to get size of nt!_LIST_ENTRY\n");
|
|
return S_OK;
|
|
}
|
|
|
|
PointerSize = GetTypeSize ("nt!PVOID");
|
|
if (PointerSize != 4 && PointerSize != 8) {
|
|
dprintf("Failed to get size of nt!PVOID\n");
|
|
return S_OK;
|
|
}
|
|
|
|
Address = StartAddress;
|
|
LastAddress = 0;
|
|
First = TRUE;
|
|
Links32 = (PULONG)&Links;
|
|
|
|
while (1) {
|
|
Status = ReadMemory (Address, Links, PointerSize * 2, &cb);
|
|
if (!Status) {
|
|
dprintf("Failed to read entry at address %p got from %p\n", Address, LastAddress);
|
|
return S_OK;
|
|
}
|
|
if (PointerSize == 4) {
|
|
Flink = (ULONG64)(LONG)Links32[0];
|
|
Blink = (ULONG64)(LONG)Links32[1];
|
|
} else {
|
|
Flink = Links[0];
|
|
Blink = Links[1];
|
|
}
|
|
if (!First && Blink != LastAddress) {
|
|
dprintf("Blink at address %p does not point back to previous at %p\n", Address, LastAddress);
|
|
return S_OK;
|
|
}
|
|
First = FALSE;
|
|
LastAddress = Address;
|
|
Address = Flink;
|
|
Count++;
|
|
if ((Count%256) == 0) {
|
|
dprintf("Processed %d entries %p %p\n", Count, Address, LastAddress);
|
|
if (CheckControlC()) {
|
|
break;
|
|
}
|
|
}
|
|
if (Address == StartAddress) {
|
|
dprintf("Found list end after %d entries\n", Count - 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|