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

2149 lines
62 KiB
C

#include "ext.h"
#include "globals.h"
#include <cmnutil.hpp>
// globals
EXT_API_VERSION ExtApiVersion = { (VER_PRODUCTVERSION_W >> 8), (VER_PRODUCTVERSION_W & 0xff), EXT_API_VERSION_NUMBER64, 0 };
WINDBG_EXTENSION_APIS ExtensionApis;
USHORT SavedMajorVersion;
USHORT SavedMinorVersion;
typedef struct {
DWORD64 base;
DWORD64 end;
char name[64];
} LMINFO, *PLMINFO;
typedef struct {
CHAR name[4098];
DWORD64 addr;
CHAR image[4098];
DWORD machine;
USHORT HdrType;
ULONG DebugType;
ULONG64 DebugDataVA;
ULONG nDebugDirs;
ULONG SymType;
time_t TimeDateStamp;
ULONG CheckSum;
ULONG SizeOfImage;
ULONG Characteristics;
ULONG SymLoadError;
BOOL omap;
CHAR PdbFileName[MAX_PATH + 1];
ULONG PdbSrc;
CHAR ImageFileName[MAX_PATH + 1];
ULONG ImageType;
ULONG ImageSrc;
ULONG numsyms;
CVDD cvrec;
} MODULE_INFO, *PMODULE_INFO, *PMODULE_INFOx;
typedef struct _MACHINE_TYPE {
ULONG MachineId;
PCHAR MachineName;
} MACHINE_TYPE;
typedef struct _ERROR_TYPE {
ULONG ErrorVal;
PCHAR Desc;
} ERROR_TYPE;
const ERROR_TYPE SymLoadErrorDesc[] = {
{SYMLOAD_OK, "Symbols loaded successfully"},
{SYMLOAD_PDBUNMATCHED, "Unmatched PDB"},
{SYMLOAD_PDBNOTFOUND, "PDB not found"},
{SYMLOAD_DBGNOTFOUND, "DBG not found"},
{SYMLOAD_OTHERERROR, "Error in load symbols"},
{SYMLOAD_OUTOFMEMORY, "DBGHELP Out of memory"},
{SYMLOAD_HEADERPAGEDOUT, "Image header paged out"},
{(EC_FORMAT << 8), "Unrecognized pdb format"},
{(EC_CORRUPT << 8), "Cvinfo is corrupt"},
{(EC_ACCESS_DENIED << 8), "Pdb read access denied"},
{SYMLOAD_DEFERRED, "No error - symbol load deferred"},
};
MACHINE_TYPE Machines[] = {
{IMAGE_FILE_MACHINE_UNKNOWN, "UNKNOWN"},
{IMAGE_FILE_MACHINE_I386, "I386"},
{IMAGE_FILE_MACHINE_R3000, "R3000"},
{IMAGE_FILE_MACHINE_R4000, "R4000"},
{IMAGE_FILE_MACHINE_R10000, "R10000"},
{IMAGE_FILE_MACHINE_WCEMIPSV2, "WCEMIPSV2"},
{IMAGE_FILE_MACHINE_ALPHA, "ALPHA"},
{IMAGE_FILE_MACHINE_POWERPC, "POWERPC"},
{IMAGE_FILE_MACHINE_POWERPCFP, "POWERPCFP"},
{IMAGE_FILE_MACHINE_SH3, "SH3"},
{IMAGE_FILE_MACHINE_SH3DSP, "SH3DSP"},
{IMAGE_FILE_MACHINE_SH3E, "SH3E"},
{IMAGE_FILE_MACHINE_SH4, "SH4"},
{IMAGE_FILE_MACHINE_SH5, "SH5"},
{IMAGE_FILE_MACHINE_ARM, "ARM"},
{IMAGE_FILE_MACHINE_AM33, "AM33"},
{IMAGE_FILE_MACHINE_THUMB, "THUMB"},
{IMAGE_FILE_MACHINE_IA64, "IA64"},
{IMAGE_FILE_MACHINE_MIPS16, "MIPS16"},
{IMAGE_FILE_MACHINE_MIPSFPU, "MIPSFPU"},
{IMAGE_FILE_MACHINE_MIPSFPU16, "MIPSFPU16"},
{IMAGE_FILE_MACHINE_ALPHA64, "ALPHA64"},
{IMAGE_FILE_MACHINE_TRICORE, "TRICORE"},
{IMAGE_FILE_MACHINE_CEF, "CEF"},
{IMAGE_FILE_MACHINE_CEE, "CEE"},
{IMAGE_FILE_MACHINE_AMD64, "AMD X86-64"},
};
char *ImageDebugType[] = {
"UNKNOWN",
"COFF",
"CODEVIEW",
"FPO",
"MISC",
"EXCEPTION",
"FIXUP",
"OMAP TO SRC",
"OMAP FROM SRC"
"BORLAND",
"RESERVED10",
"CLSID",
};
char *gSymTypeLabel[NumSymTypes] = {
"NONE", "COFF", "CV", "PDB", "EXPORT", "DEFERRED", "SYM16", "DIA PDB", "VIRTUAL"
};
char *gSrcLabel[] = {
"", // srcNone
"symbol search path", // srcSearchPath
"image path", // srcImagePath
"dbg file path", // srcDbgPath
"symbol server", // srcSymSrv
"image header", // srcCVRec
"debugger", // srcHandle
"loaded memory" // srcMemory
};
char *gImageTypeLabel[] = {
"DEFERRED", // dsNone,
"MEMORY", // dsInProc,
"FILE", // dsImage,
"DBG", // dsDbg,
"PDB" // dsPdb
};
void TruncateArgs(LPSTR args);
void lmiDumpModuleInfo(HANDLE hp,PMODULE_INFO mdi);
BOOL lmiGetModuleDumpInfo(HANDLE hp, PMODULE_ENTRY me, PMODULE_INFO mdi);
#ifdef __cplusplus
extern "C" {
#endif
typedef enum DFT
{
dftUnknown,
dftObject,
dftPE,
dftROM,
dftDBG,
dftPEF,
} DFT;
IMAGE_NT_HEADERS64 ImageNtHeaders;
PIMAGE_FILE_HEADER ImageFileHdr;
PIMAGE_OPTIONAL_HEADER64 ImageOptionalHdr;
PIMAGE_SECTION_HEADER SectionHdrs;
ULONG NumSections;
ULONG64 Base;
ULONG64 ImageNtHeadersAddr, SectionHdrsAddr;// , ImageFileHdrAddr, ImageOptionalHdrAddr,
DFT dft;
BOOL
TranslateFilePointerToVirtualAddress(
IN ULONG FilePointer,
OUT PULONG VirtualAddress
);
LPEXT_API_VERSION
ExtensionApiVersion(
VOID
)
{
return &ExtApiVersion;
}
VOID
WinDbgExtensionDllInit(
PWINDBG_EXTENSION_APIS64 lpExtensionApis,
USHORT MajorVersion,
USHORT MinorVersion
)
{
ExtensionApis = *lpExtensionApis;
SavedMajorVersion = MajorVersion;
SavedMinorVersion = MinorVersion;
}
#ifdef __cplusplus
}
#endif
BOOL
ReadFilePtr(
LPCSTR path,
LPSTR contents
)
{
BOOL rc;
HANDLE hptr;
DWORD fsize;
DWORD cb;
LPSTR p;
char ptrfile[MAX_PATH + 1];
char file[MAX_PATH + 1];
rc = false;
if (!path || !*path)
return rc;
// check for existance of file pointer
if (!CopyString(ptrfile, path, _MAX_PATH))
return false;
if (!fileexists(ptrfile))
return false;
hptr = CreateFile(ptrfile,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (hptr == INVALID_HANDLE_VALUE)
return false;
// test validity of file pointer
fsize = GetFileSize(hptr, NULL);
if (!fsize || fsize > MAX_PATH)
goto cleanup;
// read it
ZeroMemory(file, _MAX_PATH * sizeof(path[0]));
if (!ReadFile(hptr, file, fsize, &cb, 0))
goto cleanup;
if (cb != fsize)
goto cleanup;
rc = true;
// trim string down to the CR
for (p = file; *p; p++) {
if (*p == 10 || *p == 13)
{
*p = 0;
break;
}
}
CopyString(contents, file, MAX_PATH + 1);
cleanup:
// done
if (hptr)
CloseHandle(hptr);
return rc;
}
DECLARE_API(fptr)
{
char contents[MAX_PATH + 1] = "";
ReadFilePtr(args, contents);
if (*contents)
dprintf("%s\n", contents);
}
DECLARE_API(vc7fpo)
{
g_vc7fpo = !g_vc7fpo;
dprintf((g_vc7fpo) ? "VC7FPO - Enabled\n" : "VC7FPO - Disabled\n");
}
DECLARE_API(stackdbg)
{
if (*args && *args != ';') {
for (;;) {
while (*args == ' ' || *args == '\t') {
args++;
}
if (*args == '-' || *args == '/') {
switch(*(args + 1)) {
case 'c':
g_StackDebugIo = SDB_CALLBACK_OUT;
break;
case 'd':
g_StackDebugIo = SDB_DEBUG_OUT;
break;
default:
// Assume it's the beginning of an expression.
goto Expr;
}
args += 2;
} else {
break;
}
}
Expr:
g_StackDebugMask = (ULONG)GetExpression(args);
}
if (g_StackDebugMask == 0) {
dprintf("Stack debugging is off\n");
} else {
dprintf("Stack debugging mask is 0x%08x, output via %s\n",
g_StackDebugMask, g_StackDebugIo == SDB_DEBUG_OUT ?
"debug output" : "callback");
}
}
DECLARE_API(sym)
{
if (strstr(args, "noisy")) {
SymSetOptions(g.SymOptions | SYMOPT_DEBUG);
symsrvSetCallback(true);
} else if (strstr(args, "quiet")) {
SymSetOptions(g.SymOptions & ~SYMOPT_DEBUG);
if (!g.hLog)
symsrvSetCallback(false);
} else if (strstr(args, "prompts off")) {
if (!option(SYMOPT_NO_PROMPTS)) {
SymSetOptions(g.SymOptions | SYMOPT_NO_PROMPTS);
symsrvSetPrompts();
}
} else if (strstr(args, "prompts")) {
if (option(SYMOPT_NO_PROMPTS)) {
SymSetOptions(g.SymOptions & ~SYMOPT_NO_PROMPTS);
symsrvClose();
}
} else {
dprintf("!sym <noisy/quiet - prompts/prompts off> - ");
}
dprintf(option(SYMOPT_DEBUG) ? "noisy mode" : "quiet mode");
dprintf(option(SYMOPT_NO_PROMPTS) ? " - symbol prompts off\n" : " - symbol prompts on\n");
}
DECLARE_API(symsrv)
{
if (strstr(args, "close")) {
symsrvClose();
dprintf("symbol server client has been closed\n");
} else
dprintf("!symsrv close - closes the symbol server client so it can be updated\n");
}
int __cdecl
CompareBase(
const void *e1,
const void *e2
)
{
PLMINFO mod1 = (PLMINFO)e1;
PLMINFO mod2 = (PLMINFO)e2;
LONGLONG diff = mod1->base - mod2->base;
if (diff < 0)
return -1;
else if (diff > 0)
return 1;
else
return 0;
}
#define MAX_FORMAT_STRINGS 8
LPSTR
FormatAddr64(
ULONG64 addr,
BOOL format64
)
{
static CHAR strings[MAX_FORMAT_STRINGS][18];
static int next = 0;
LPSTR string;
string = strings[next];
++next;
if (next >= MAX_FORMAT_STRINGS)
next = 0;
if (format64)
PrintString(string, 18, "%08x`%08x", (ULONG)(addr>>32), (ULONG)addr);
else
PrintString(string, 18, "%08x", (ULONG)addr);
return string;
}
int __cdecl
CompareNames(
const void *e1,
const void *e2
)
{
PLMINFO mod1 = (PLMINFO)e1;
PLMINFO mod2 = (PLMINFO)e2;
return strcmp( mod1->name, mod2->name );
}
DECLARE_API(lm)
{
PPROCESS_ENTRY pe;
HANDLE hp;
PLIST_ENTRY next;
PMODULE_ENTRY mi;
PLMINFO mods;
PLMINFO mod;
DWORD count;
BOOL format64;
GetCurrentProcessHandle (&hp);
if (!hp) {
dprintf("Couldn't get process handle\n");
return;
}
pe = FindProcessEntry(hp);
if (!pe) {
dprintf("Couldn't find process 0x%x\n", hp);
SetLastError(ERROR_INVALID_HANDLE);
return;
}
next = pe->ModuleList.Flink;
if (!next)
return;
for (count = 0; (PVOID)next != (PVOID)&pe->ModuleList; count++) {
mi = CONTAINING_RECORD( next, MODULE_ENTRY, ListEntry );
next = mi->ListEntry.Flink;
}
mods = (PLMINFO)MemAlloc(count * sizeof(LMINFO));
if (!mods)
return;
ZeroMemory(mods, count * sizeof(LMINFO));
format64 = false;
next = pe->ModuleList.Flink;
for (mod = mods; (PVOID)next != (PVOID)&pe->ModuleList; mod++) {
mi = CONTAINING_RECORD( next, MODULE_ENTRY, ListEntry );
mod->base = mi->BaseOfDll;
mod->end = mod->base + mi->DllSize;
CopyStrArray(mod->name, mi->ModuleName);
format64 = IsImageMachineType64(mi->MachineType);
next = mi->ListEntry.Flink;
}
qsort(mods, count, sizeof(LMINFO), CompareBase);
dprintf("%d loaded modules...\n", count);
if (format64)
dprintf(" base - end name\n", mod->base, mod->end, mod->name);
else
dprintf(" base - end name\n", mod->base, mod->end, mod->name);
for (mod = mods; count > 0; mod++, count--) {
dprintf("0x%s - ", FormatAddr64(mod->base, format64));
dprintf("0x%s ", FormatAddr64(mod->end, format64));
dprintf("%s\n", mod->name);
}
MemFree(mods);
}
DECLARE_API(lmi)
{
PPROCESS_ENTRY pe;
PMODULE_ENTRY mi = NULL;
MODULE_INFO mdi;
DWORD64 addr;
HANDLE hp = 0;
char argstr[1024];
char *pc;
if (!args[0] || !CopyStrArray(argstr, args)) {
dprintf("You must specify a module\n");
return;
}
_strlwr(argstr);
TruncateArgs(argstr);
dprintf("Loaded Module Info: [%s] ", argstr);
GetCurrentProcessHandle(&hp);
if (!hp) {
dprintf("couldn't get process handle\n");
return;
}
pe = FindProcessEntry(hp);
if (!pe) {
dprintf("Couldn't find process 0x%x while looking for %s\n", hp, argstr);
SetLastError(ERROR_INVALID_HANDLE);
return;
}
dprintf("\n");
if (mi = FindModule(hp, pe, argstr, false)) {
if (lmiGetModuleDumpInfo(hp, mi, &mdi)) {
lmiDumpModuleInfo(hp, &mdi);
} else {
// dprintf("Cannot get module info for %s\n", argstr);
}
if (SectionHdrs) {
free(SectionHdrs);
SectionHdrs = NULL;
}
dprintf(" Load Report: %s\n", SymbolStatus(mi, 17));
return;
}
GetExpressionEx(args, &addr, NULL);
if (!addr) {
dprintf("%s not found\n", argstr);
SetLastError(ERROR_MOD_NOT_FOUND);
return;
}
mi = GetModuleForPC( pe, addr, false );
if (!mi) {
dprintf("%I64lx is not a valid address\n", addr);
return;
}
if (lmiGetModuleDumpInfo(hp, mi, &mdi)) {
lmiDumpModuleInfo(hp, &mdi);
}
if (SectionHdrs) {
free(SectionHdrs);
SectionHdrs = NULL;
}
dprintf(" Load Report: %s\n", SymbolStatus(mi, 17));
}
DECLARE_API(omap)
{
PPROCESS_ENTRY pe;
PMODULE_ENTRY mi = NULL;
HANDLE hp = 0;
char argstr[1024];
POMAP pomap;
DWORD i;
if (!CopyStrArray(argstr, args))
return;
_strlwr(argstr);
TruncateArgs(argstr);
dprintf("Dump OMAP: [%s] ", argstr);
GetCurrentProcessHandle (&hp);
if (!hp) {
dprintf("couldn't get process handle\n");
return;
}
pe = FindProcessEntry(hp);
if (!pe) {
dprintf("Couldn't find process 0x%x while looking for %s\n", hp, argstr);
SetLastError(ERROR_INVALID_HANDLE);
return;
}
dprintf("\n");
mi = FindModule(hp, pe, argstr, false);
if (!mi)
return;
i = sizeof(DWORD);
if (!mi->pOmapFrom)
return;
dprintf("\nOMAP FROM:\n");
for(i = 0, pomap = mi->pOmapFrom;
i < 100; // mi->cOmapFrom;
i++, pomap++)
{
dprintf("%8x <-%8x\n", pomap->rva, pomap->rvaTo);
}
if (!mi->pOmapTo)
return;
dprintf("\nOMAP TO:\n");
for(i = 0, pomap = mi->pOmapTo;
i < 100; // mi->cOmapTo;
i++, pomap++)
{
dprintf("%8x ->%8x\n", pomap->rva, pomap->rvaTo);
}
}
BOOL
cbSrcFiles(
PSOURCEFILE pSourceFile,
PVOID UserContext
)
{
PMODULE_ENTRY mi;
PCHAR mname;
if (!pSourceFile)
return false;
mi = GetModFromAddr((PPROCESS_ENTRY)UserContext, pSourceFile->ModBase);
if (!mi)
return true;
dprintf(" %s!%s\n", (*mi->AliasName) ? mi->AliasName : mi->ModuleName, pSourceFile->FileName);
return true;
}
DECLARE_API(srcfiles)
{
HANDLE hp = 0;
char argstr[1024];
BOOL rc;
PPROCESS_ENTRY pe;
if (!CopyStrArray(argstr, args))
return;
_strlwr(argstr);
TruncateArgs(argstr);
dprintf("Source Files: [%s]\n", argstr);
GetCurrentProcessHandle (&hp);
if (!hp) {
dprintf("couldn't get process handle\n");
return;
}
pe = FindProcessEntry(hp);
if (!pe) {
dprintf("Couldn't find process 0x%x while looking for %s\n", hp, argstr);
SetLastError(ERROR_INVALID_HANDLE);
return;
}
rc = SymEnumSourceFiles(hp, 0, argstr, cbSrcFiles, pe);
}
VOID
DumpSecHdr (
IN DWORD i,
IN PIMAGE_SECTION_HEADER Sh
)
{
dprintf(" %2d %s\n", i, Sh->Name);
}
VOID
DumpSecs(
BOOL fFullDump
)
{
IMAGE_SECTION_HEADER sh;
const char *p;
DWORD li;
DWORD cb;
BOOL Ok;
int i, j;
CHAR szName[IMAGE_SIZEOF_SHORT_NAME + 1];
if (ImageFileHdr->NumberOfSections < 1)
return;
dprintf(" Sections: # Name\n");
for (i = 1; i <= ImageFileHdr->NumberOfSections; i++) {
sh = SectionHdrs[i-1];
CopyStrArray(szName, (char *) sh.Name);
DumpSecHdr(i, &sh);
}
}
BOOL
lmiGetModuleDumpInfo(
HANDLE hp,
PMODULE_ENTRY me,
PMODULE_INFOx mdi)
{
BOOL rc;
DWORD cb;
ULONG nDebugDirs;
ULONG64 ddva;
IMAGE_SEPARATE_DEBUG_HEADER sdh;
IMAGE_DOS_HEADER DosHeader;
IMAGE_NT_HEADERS32 NtHeader32;
IMAGE_NT_HEADERS64 NtHeader64;
PIMAGE_FILE_HEADER FileHeader;
PIMAGE_ROM_OPTIONAL_HEADER rom;
PIMAGE_DATA_DIRECTORY datadir;
DWORD64 offset;
BOOL b64;
ZeroMemory(mdi, sizeof(MODULE_INFO));
CopyStrArray(mdi->name, me->ModuleName);
mdi->addr = me->BaseOfDll;
CopyString(mdi->image, me->ImageName, DIMA(mdi->image));
if (!mdi->addr) {
dprintf("Module does not have base address\n");
return false;
}
mdi->SymType = me->SymType;
mdi->SymLoadError = me->SymLoadError;
if (me->SymType == SymVirtual)
return true;
if (me->SymType == SymDeferred) {
mdi->SymLoadError = SYMLOAD_DEFERRED;
}
rc = ReadMemory(mdi->addr, &DosHeader, sizeof(DosHeader), &cb);
if (!rc || cb != sizeof(DosHeader)) {
dprintf("Cannot read Image header @ %p\n", mdi->addr);
return false;
}
mdi->HdrType = DosHeader.e_magic;
mdi->omap = me->cOmapFrom ? true : false;
mdi->PdbSrc = me->PdbSrc;
if (me->LoadedPdbName)
CopyStrArray(mdi->PdbFileName, me->LoadedPdbName);
mdi->ImageSrc = me->ImageSrc;
if (me->LoadedImageName)
CopyStrArray(mdi->ImageFileName, me->LoadedImageName);
mdi->ImageType = me->ImageType;
if (DosHeader.e_magic == IMAGE_DOS_SIGNATURE) {
rc = ReadMemory(mdi->addr + DosHeader.e_lfanew, &NtHeader32, sizeof(NtHeader32), &cb);
if (!rc || cb != sizeof(NtHeader32)) {
dprintf("Cannot read Image NT header @ %p\n", mdi->addr + DosHeader.e_lfanew);
return false;
}
mdi->machine = NtHeader32.FileHeader.Machine;
mdi->TimeDateStamp = NtHeader32.FileHeader.TimeDateStamp;
if (NtHeader32.Signature != IMAGE_NT_SIGNATURE) {
// if header is not NT sig, this is a ROM image
rom = (PIMAGE_ROM_OPTIONAL_HEADER)&NtHeader32.OptionalHeader;
if (rom->Magic == IMAGE_ROM_OPTIONAL_HDR_MAGIC) {
ImageFileHdr = &NtHeader32.FileHeader;
mdi->SizeOfImage = rom->SizeOfCode;
mdi->CheckSum = 0;
nDebugDirs = 0;
if (!(ImageFileHdr->Characteristics & IMAGE_FILE_DEBUG_STRIPPED)) {
// Get the debug dir VA
}
} else {
dprintf("Unknown NT Image signature\n");
return false;
}
} else {
// otherwise, get info from appropriate header type for 32 or 64 bit
if (IsImageMachineType64(NtHeader32.FileHeader.Machine)) {
// Reread the header as a 64bit header.
rc = ReadMemory(mdi->addr + DosHeader.e_lfanew, &NtHeader64, sizeof(NtHeader64), &cb);
if (!rc || cb != sizeof(NtHeader64)) {
dprintf("Cannot read Image NT header @ %p\n", mdi->addr + DosHeader.e_lfanew);
return false;
}
ImageFileHdr = &NtHeader64.FileHeader;
mdi->CheckSum = NtHeader64.OptionalHeader.CheckSum;
mdi->SizeOfImage = NtHeader64.OptionalHeader.SizeOfImage;
datadir = NtHeader64.OptionalHeader.DataDirectory;
} else {
ImageFileHdr = &NtHeader32.FileHeader;
datadir = NtHeader32.OptionalHeader.DataDirectory;
mdi->SizeOfImage = NtHeader32.OptionalHeader.SizeOfImage;
mdi->CheckSum = NtHeader32.OptionalHeader.CheckSum;
}
mdi->DebugDataVA = mdi->addr + datadir[IMAGE_DIRECTORY_ENTRY_DEBUG].VirtualAddress;
mdi->nDebugDirs = datadir[IMAGE_DIRECTORY_ENTRY_DEBUG].Size / sizeof(IMAGE_DEBUG_DIRECTORY);
}
// read the section headers
mdi->Characteristics = ImageFileHdr->Characteristics;
SectionHdrs = (PIMAGE_SECTION_HEADER) malloc((NumSections = ImageFileHdr->NumberOfSections)*
sizeof(IMAGE_SECTION_HEADER));
if (!SectionHdrs) {
dprintf("Cannot allocate memory for reading sections\n");
return false;
}
offset = mdi->addr + DosHeader.e_lfanew;
b64 = IsImageMachineType64(ImageFileHdr->Machine);
SectionHdrsAddr = offset + (b64 ? sizeof(IMAGE_NT_HEADERS64) : sizeof(IMAGE_NT_HEADERS32)) +
ImageFileHdr->SizeOfOptionalHeader - (b64 ? sizeof(IMAGE_OPTIONAL_HEADER64) : sizeof(IMAGE_OPTIONAL_HEADER32));
rc = ReadMemory(SectionHdrsAddr,
SectionHdrs,
(NumSections) * sizeof(IMAGE_SECTION_HEADER),
&cb);
if (!rc) {
dprintf("Can't read section headers\n");
} else {
if (cb != NumSections * sizeof(IMAGE_SECTION_HEADER)) {
dprintf("\n***\n*** Some section headers may be missing ***\n***\n\n");
NumSections = (USHORT)(cb / sizeof(IMAGE_SECTION_HEADER));
}
#if 0
DumpSecs(false);
#endif
}
} else if (DosHeader.e_magic == IMAGE_SEPARATE_DEBUG_SIGNATURE) {
rc = ReadMemory(mdi->addr, &sdh, sizeof(sdh), &cb);
if (!rc || cb != sizeof(sdh)) {
dprintf("Cannot read Image Debug header @ %p\n", mdi->addr);
return false;
}
mdi->machine = sdh.Machine;
mdi->TimeDateStamp = sdh.TimeDateStamp;
mdi->CheckSum = sdh.CheckSum;
mdi->SizeOfImage = sdh.SizeOfImage;
mdi->Characteristics = sdh.Characteristics;
if (sdh.DebugDirectorySize) {
mdi->nDebugDirs = (int)(sdh.DebugDirectorySize / sizeof(IMAGE_DEBUG_DIRECTORY));
mdi->DebugDataVA = sizeof(IMAGE_SEPARATE_DEBUG_HEADER)
+ (sdh.NumberOfSections * sizeof(IMAGE_SECTION_HEADER))
+ sdh.ExportedNamesSize;
}
} else {
dprintf("Unknown image\n");
return false;
}
mdi->numsyms = me->numsyms;
memcpy(&mdi->cvrec, &me->CVRec, min(sizeof(mdi->cvrec), sizeof(me->CVRec)));
return true;
}
BOOL
DumpDbgDirectories(
HANDLE hp,
PMODULE_INFOx mdi
)
{
ULONG rc, cb;
IMAGE_DEBUG_DIRECTORY dd;
IMAGE_DEBUG_MISC md;
ULONG64 ddva;
ULONG nDebugDirs;
PCVDD pcv;
ULONG64 cvAddr;
ULONG cvSize;
PCHAR pCV;
CHAR ImgData[MAX_PATH];
IMAGE_COFF_SYMBOLS_HEADER CoffHdr;
ULONG va;
nDebugDirs = mdi->nDebugDirs;
ddva = mdi->DebugDataVA;
dprintf("Debug Data Dirs: Type Size VA Pointer\n");
for (;nDebugDirs; dprintf("\n"), nDebugDirs--) {
rc = ReadMemory(ddva, &dd, sizeof(dd), &cb);
if (!rc || cb != sizeof(dd))
return false;
if (dd.Type) {
dprintf("%21s ", // dd.Type,
(dd.Type < sizeof (ImageDebugType) / sizeof(char *)) ? ImageDebugType[dd.Type] : "??");
dprintf(
"%5lx, %5lx, %7lx ",
dd.SizeOfData,
dd.AddressOfRawData,
dd.PointerToRawData);
if (!TranslateFilePointerToVirtualAddress(dd.PointerToRawData, &va)) {
dprintf(" [Debug data not mapped]");
if (dd.Type == IMAGE_DEBUG_TYPE_CODEVIEW)
dprintf(" - Can't validate symbols, if present.");
goto nextDebugDir;
}
switch(dd.Type)
{
case IMAGE_DEBUG_TYPE_MISC:
if (!dd.PointerToRawData) {
dprintf("[Data not mapped]");
break;
}
rc = ReadMemory(mdi->addr + dd.PointerToRawData, &md, sizeof(md), &cb);
if (!rc || cb != sizeof(md) || md.DataType != IMAGE_DEBUG_MISC_EXENAME) {
dprintf("[Data not mapped]");
goto nextDebugDir;
}
rc = ReadMemory(mdi->addr + dd.PointerToRawData + FIELD_OFFSET(IMAGE_DEBUG_MISC, Data),
ImgData, MAX_PATH, &cb);
if (rc && cb)
dprintf(" %s", ImgData);
break;
case IMAGE_DEBUG_TYPE_CODEVIEW:
if (dd.AddressOfRawData) {
cvAddr = mdi->addr + dd.AddressOfRawData;
} else if (dd.PointerToRawData) {
cvAddr = mdi->addr + dd.PointerToRawData;
} else {
break;
}
cvSize = dd.SizeOfData;
if (!(pCV = (PCHAR)MemAlloc(dd.SizeOfData + 1)))
break;
pcv = (PCVDD)pCV;
rc = ReadMemory(cvAddr,pCV, cvSize, &cb);
if (rc && cb == cvSize) {
char *c = (char *)&pcv->dwSig;
dprintf("%c%c%c%c - ", *c, *(c + 1), *(c + 2), *(c + 3));
} else {
pcv->dwSig = 0;
}
switch (pcv->dwSig)
{
case 0:
dprintf("[Debug data not mapped] - can't validate symbols, if present.");
break;
case '01BN':
pCV[cvSize] = 0;
dprintf("Sig: %lx, Age: %lx,%sPdb: %s",
pcv->nb10i.sig,
pcv->nb10i.age,
(strlen(pCV) > 14 ? "\n " : " "),
pcv->nb10i.szPdb);
break;
case 'SDSR':
pCV[cvSize] = 0;
dprintf("GUID: (0x%8x, 0x%4x, 0x%4x, 0x%2x, 0x%2x, 0x%2x, 0x%2x, 0x%2x, 0x%2x, 0x%2x, 0x%2x)\n",
pcv->rsdsi.guidSig.Data1,
pcv->rsdsi.guidSig.Data2,
pcv->rsdsi.guidSig.Data3,
pcv->rsdsi.guidSig.Data4[0],
pcv->rsdsi.guidSig.Data4[1],
pcv->rsdsi.guidSig.Data4[2],
pcv->rsdsi.guidSig.Data4[3],
pcv->rsdsi.guidSig.Data4[4],
pcv->rsdsi.guidSig.Data4[5],
pcv->rsdsi.guidSig.Data4[6],
pcv->rsdsi.guidSig.Data4[7]);
dprintf(" Age: %lx, Pdb: %s",
pcv->rsdsi.age,
pcv->rsdsi.szPdb);
break;
case '80BN':
case '90BN':
case '11BN':
break;
default:
dprintf("unrecognized symbol format ID");
break;
}
MemFree(pCV);
break;
case IMAGE_DEBUG_TYPE_COFF:
if (!dd.PointerToRawData) {
dprintf("[Data paged out] - unable to load COFF info.");
break;
}
rc = ReadMemory(mdi->addr + dd.PointerToRawData, &CoffHdr, sizeof(CoffHdr), &cb);
if (!rc || cb != sizeof(CoffHdr)) {
dprintf("[Data paged out] - unable to load COFF info.");
break;
}
dprintf("NumSyms %#lx, Numlines %#lx",
mdi->numsyms, // CoffHdr.NumberOfSymbols,
CoffHdr.NumberOfLinenumbers);
break;
case IMAGE_DEBUG_TYPE_OMAP_TO_SRC:
dprintf("BBT Optimized");
break;
default:
dprintf("[Data not mapped]");
break;
}
}
nextDebugDir:
ddva += sizeof (dd);
}
return true;
}
void lmiDumpModuleInfo(
HANDLE hp,
PMODULE_INFO mdi
)
{
ULONG i;
const char *time;
dprintf(" Module: %s\n", mdi->name);
dprintf(" Base Address: %p%s", mdi->addr, mdi->addr ? "\n" : " is INVALID\n");
dprintf(" Image Name: %s\n", mdi->image);
if (mdi->SymType == SymVirtual)
goto symboltype;
dprintf(" Machine Type: %d", mdi->machine);
for (i=0;i<sizeof(Machines)/sizeof(MACHINE_TYPE);i++) {
if (mdi->machine == Machines[i].MachineId) {
dprintf(" (%s)", Machines[i].MachineName);
break;
}
}
dprintf("\n Time Stamp: %lx", mdi->TimeDateStamp);
if ((time = ctime((time_t *) &mdi->TimeDateStamp)) != NULL) {
dprintf( " %s", time);
} else
dprintf("\n");
dprintf(" Size: %x\n", mdi->SizeOfImage);
dprintf(" CheckSum: %lx\n", mdi->CheckSum);
dprintf("Characteristics: %lx %s %s\n",
mdi->Characteristics,
((mdi->Characteristics & IMAGE_FILE_DEBUG_STRIPPED) ? "stripped":""),
(mdi->omap ? "perf" : "")
);
if (mdi->nDebugDirs) {
DumpDbgDirectories(hp, mdi);
} else {
dprintf("Debug Directories not present\n");
}
switch (mdi->ImageType)
{
case dsInProc:
case dsImage:
if (mdi->Characteristics & IMAGE_FILE_DEBUG_STRIPPED)
break;
case dsDbg:
case dsPdb:
dprintf(" Image Type: %-9s", gImageTypeLabel[mdi->ImageType]);
dprintf("- Image read successfully from %s.", gSrcLabel[mdi->ImageSrc]);
if (mdi->ImageSrc != srcNone && mdi->ImageSrc != srcMemory)
dprintf("\n %s", mdi->ImageFileName);
dprintf("\n");
break;
case dsNone:
default:
break;
}
symboltype:
dprintf(" Symbol Type: %-9s", gSymTypeLabel[mdi->SymType]);
for (i=0;i< sizeof(SymLoadErrorDesc) / sizeof (ERROR_TYPE); i++) {
if (mdi->SymLoadError == SymLoadErrorDesc[i].ErrorVal) {
dprintf("- %s", SymLoadErrorDesc[i].Desc);
break;
}
}
if (mdi->PdbSrc != srcNone) {
dprintf(" from %s.", gSrcLabel[mdi->PdbSrc]);
if (*mdi->PdbFileName)
dprintf("\n %s", mdi->PdbFileName);
dprintf("\n");
} else {
dprintf("\n");
}
}
void TruncateArgs(
LPSTR sz
)
{
PSTR p;
for (p = sz; !isspace(*p); p++) {
if (!*p)
break;
}
*p = 0;
}
// STYP_ flags values for MIPS ROM images
#define STYP_REG 0x00000000
#define STYP_TEXT 0x00000020
#define STYP_INIT 0x80000000
#define STYP_RDATA 0x00000100
#define STYP_DATA 0x00000040
#define STYP_LIT8 0x08000000
#define STYP_LIT4 0x10000000
#define STYP_SDATA 0x00000200
#define STYP_SBSS 0x00000080
#define STYP_BSS 0x00000400
#define STYP_LIB 0x40000000
#define STYP_UCODE 0x00000800
#define S_NRELOC_OVFL 0x20000000
#define IMAGE_SCN_MEM_SYSHEAP 0x00010000 // Obsolete
#define IMAGE_SCN_MEM_PROTECTED 0x00004000 // Obsolete
const static char * const MachineName[] = {
"Unknown",
"i386",
"Alpha AXP",
"Alpha AXP64",
"Intel IA64",
"AMD X86-64",
"ARM 32-bit",
};
const static char * const SubsystemName[] = {
"Unknown",
"Native",
"Windows GUI",
"Windows CUI",
"Posix CUI",
};
const static char * const DirectoryEntryName[] = {
"Export",
"Import",
"Resource",
"Exception",
"Security",
"Base Relocation",
"Debug",
"Description",
"Special",
"Thread Storage",
"Load Configuration",
"Bound Import",
"Import Address Table",
"Reserved",
"Reserved",
"Reserved",
0
};
VOID
dhDumpHeaders (
VOID
);
VOID
dhDumpSections(
VOID
);
VOID
dhDumpImage(
ULONG64 xBase,
BOOL DoHeaders,
BOOL DoSections
);
VOID
dhdh(
IN PSTR lpArgs
)
{
BOOL DoAll;
BOOL DoSections;
BOOL DoHeaders;
CHAR c;
PCHAR p;
ULONG64 xBase;
//
// Evaluate the argument string to get the address of the
// image to dump.
//
DoAll = true;
DoHeaders = false;
DoSections = false;
xBase = 0;
while (*lpArgs) {
while (isspace(*lpArgs)) {
lpArgs++;
}
if (*lpArgs == '/' || *lpArgs == '-') {
// process switch
switch (*++lpArgs) {
case 'a': // dump everything we can
case 'A':
++lpArgs;
DoAll = true;
break;
default: // invalid switch
case 'h': // help
case 'H':
case '?':
dprintf("Usage: dh [options] address\n");
dprintf("\n");
dprintf("Dumps headers from an image based at address\n");
dprintf("\n");
dprintf("Options:\n");
dprintf("\n");
dprintf(" -a Dump everything\n");
dprintf(" -f Dump file headers\n");
dprintf(" -s Dump section headers\n");
dprintf("\n");
return;
case 'f':
case 'F':
++lpArgs;
DoAll = false;
DoHeaders = true;
break;
case 's':
case 'S':
++lpArgs;
DoAll = false;
DoSections = true;
break;
}
} else if (*lpArgs) {
if (xBase != 0) {
dprintf("Invalid extra argument\n");
return;
}
p = lpArgs;
while (*p && !isspace(*p)) {
p++;
}
c = *p;
*p = 0;
xBase = GetExpression(lpArgs);
*p = c;
lpArgs=p;
}
}
if ( !xBase ) {
return;
}
dhDumpImage(xBase, DoAll || DoHeaders, DoAll || DoSections);
}
DECLARE_API(dh)
{
dhdh( (PSTR)args );
}
BOOL
dhReadNtHeader(
ULONG64 Address,
PIMAGE_NT_HEADERS64 pNtHdrs
)
{
ULONG cb;
BOOL Ok;
Ok = ReadMemory(Address, pNtHdrs, sizeof(*pNtHdrs), &cb);
if (IsImageMachineType64(pNtHdrs->FileHeader.Machine))
{
Ok = Ok && (cb == sizeof(*pNtHdrs));
}
else
{
IMAGE_NT_HEADERS32 nthdr32;
Ok = ReadMemory(Address, &nthdr32, sizeof(nthdr32), &cb);
Ok = Ok && (cb == sizeof(nthdr32));
ImageNtHdr32To64(&nthdr32, pNtHdrs);
}
return Ok;
}
VOID
dhDumpImage(
ULONG64 xBase,
BOOL DoHeaders,
BOOL DoSections
)
{
IMAGE_DOS_HEADER DosHeader;
ULONG cb;
ULONG64 Offset;
BOOL Ok;
Base = xBase;
Ok = ReadMemory(Base, &DosHeader, sizeof(DosHeader), &cb);
if (!Ok) {
dprintf("Can't read file header: error == %d\n", GetLastError());
return;
}
if (cb != sizeof(DosHeader) || DosHeader.e_magic != IMAGE_DOS_SIGNATURE) {
dprintf("No file header\n");
return;
}
Offset = Base + DosHeader.e_lfanew;
if (!dhReadNtHeader(ImageNtHeadersAddr=Offset, &ImageNtHeaders)) {
dprintf("Bad file header\n");
return;
}
ImageFileHdr = &ImageNtHeaders.FileHeader;
ImageOptionalHdr = &ImageNtHeaders.OptionalHeader;
if (ImageFileHdr->SizeOfOptionalHeader == sizeof(IMAGE_ROM_OPTIONAL_HEADER)) {
dft = dftROM;
} else if (ImageFileHdr->Characteristics & IMAGE_FILE_DLL) {
dft = dftPE;
} else if (ImageFileHdr->Characteristics & IMAGE_FILE_EXECUTABLE_IMAGE) {
dft = dftPE;
} else if (ImageFileHdr->SizeOfOptionalHeader == 0) {
dft = dftObject;
} else {
dft = dftUnknown;
}
if (DoHeaders) {
dhDumpHeaders();
}
if (DoSections) {
SectionHdrs = (PIMAGE_SECTION_HEADER) malloc((NumSections = ImageFileHdr->NumberOfSections)*
sizeof(IMAGE_SECTION_HEADER));
if (!SectionHdrs) {
dprintf("Cannot allocate memory for dumping sections\n");
return;
}
__try {
BOOL b64 = IsImageMachineType64(ImageFileHdr->Machine);
SectionHdrsAddr = Offset + (b64 ? sizeof(IMAGE_NT_HEADERS64) : sizeof(IMAGE_NT_HEADERS32)) +
ImageFileHdr->SizeOfOptionalHeader - (b64 ? sizeof(IMAGE_OPTIONAL_HEADER64) : sizeof(IMAGE_OPTIONAL_HEADER32));
Ok = ReadMemory(
SectionHdrsAddr,
SectionHdrs,
(NumSections) * sizeof(IMAGE_SECTION_HEADER),
&cb);
if (!Ok) {
dprintf("Can't read section headers\n");
} else {
if (cb != NumSections * sizeof(IMAGE_SECTION_HEADER)) {
dprintf("\n***\n*** Some section headers may be missing ***\n***\n\n");
NumSections = (USHORT)(cb / sizeof(IMAGE_SECTION_HEADER));
}
dhDumpSections();
}
}
__finally {
if (SectionHdrs) {
free(SectionHdrs);
SectionHdrs = 0;
}
}
}
}
VOID
dhDumpHeaders (
VOID
)
/*++
Routine Description:
Formats the file header and optional header.
Arguments:
None.
Return Value:
None.
--*/
{
int i, j;
const char *time;
const char *name;
DWORD dw;
// Print out file type
switch (dft) {
case dftObject :
dprintf("\nFile Type: COFF OBJECT\n");
break;
case dftPE :
if (ImageFileHdr->Characteristics & IMAGE_FILE_DLL) {
dprintf("\nFile Type: DLL\n");
} else {
dprintf("\nFile Type: EXECUTABLE IMAGE\n");
}
break;
case dftROM :
dprintf("\nFile Type: ROM IMAGE\n");
break;
default :
dprintf("\nFile Type: UNKNOWN\n");
break;
}
switch (ImageFileHdr->Machine) {
case IMAGE_FILE_MACHINE_I386 : i = 1; break;
case IMAGE_FILE_MACHINE_ALPHA : i = 2; break;
case IMAGE_FILE_MACHINE_ALPHA64 : i = 3; break;
case IMAGE_FILE_MACHINE_IA64 : i = 4; break;
case IMAGE_FILE_MACHINE_AMD64 : i = 5; break;
case IMAGE_FILE_MACHINE_ARM : i = 6; break;
default : i = 0;
}
dprintf(
"FILE HEADER VALUES\n"
"%8hX machine (%s)\n"
"%8hX number of sections\n"
"%8lX time date stamp",
ImageFileHdr->Machine,
MachineName[i],
ImageFileHdr->NumberOfSections,
ImageFileHdr->TimeDateStamp);
if ((time = ctime((time_t *) &ImageFileHdr->TimeDateStamp)) != NULL) {
dprintf( " %s", time);
}
dprintf("\n");
dprintf(
"%8lX file pointer to symbol table\n"
"%8lX number of symbols\n"
"%8hX size of optional header\n"
"%8hX characteristics\n",
ImageFileHdr->PointerToSymbolTable,
ImageFileHdr->NumberOfSymbols,
ImageFileHdr->SizeOfOptionalHeader,
ImageFileHdr->Characteristics);
for (dw = ImageFileHdr->Characteristics, j = 0; dw; dw >>= 1, j++) {
if (dw & 1) {
switch (1 << j) {
case IMAGE_FILE_RELOCS_STRIPPED : name = "Relocations stripped"; break;
case IMAGE_FILE_EXECUTABLE_IMAGE : name = "Executable"; break;
case IMAGE_FILE_LINE_NUMS_STRIPPED : name = "Line numbers stripped"; break;
case IMAGE_FILE_LOCAL_SYMS_STRIPPED : name = "Symbols stripped"; break;
case IMAGE_FILE_LARGE_ADDRESS_AWARE : name = "App can handle >2gb addresses"; break;
case IMAGE_FILE_BYTES_REVERSED_LO : name = "Bytes reversed"; break;
case IMAGE_FILE_32BIT_MACHINE : name = "32 bit word machine"; break;
case IMAGE_FILE_DEBUG_STRIPPED : name = "Debug information stripped"; break;
case IMAGE_FILE_SYSTEM : name = "System"; break;
case IMAGE_FILE_DLL : name = "DLL"; break;
case IMAGE_FILE_BYTES_REVERSED_HI : name = ""; break;
default : name = "RESERVED - UNKNOWN";
}
if (*name) {
dprintf( " %s\n", name);
}
}
}
if (ImageFileHdr->SizeOfOptionalHeader != 0) {
char szLinkerVersion[30];
PrintString(szLinkerVersion,
DIMA(szLinkerVersion),
"%u.%02u",
ImageOptionalHdr->MajorLinkerVersion,
ImageOptionalHdr->MinorLinkerVersion);
dprintf(
"\n"
"OPTIONAL HEADER VALUES\n"
"%8hX magic #\n"
"%8s linker version\n"
"%8lX size of code\n"
"%8lX size of initialized data\n"
"%8lX size of uninitialized data\n"
"%8lX address of entry point\n"
"%8lX base of code\n"
,
ImageOptionalHdr->Magic,
szLinkerVersion,
ImageOptionalHdr->SizeOfCode,
ImageOptionalHdr->SizeOfInitializedData,
ImageOptionalHdr->SizeOfUninitializedData,
ImageOptionalHdr->AddressOfEntryPoint,
ImageOptionalHdr->BaseOfCode
);
// dprintf("%p base of image\n",
// ImageOptionalHdr->ImageBase
// );
}
if (dft == dftROM) {
PIMAGE_ROM_OPTIONAL_HEADER romOptionalHdr;
romOptionalHdr = (PIMAGE_ROM_OPTIONAL_HEADER) &ImageOptionalHdr;
dprintf(
" ----- rom -----\n"
"%8lX base of bss\n"
"%8lX gpr mask\n"
" cpr mask\n"
" %08lX %08lX %08lX %08lX\n"
"%8hX gp value\n",
romOptionalHdr->BaseOfBss,
romOptionalHdr->GprMask,
romOptionalHdr->CprMask[0],
romOptionalHdr->CprMask[1],
romOptionalHdr->CprMask[2],
romOptionalHdr->CprMask[3],
romOptionalHdr->GpValue);
}
if ((ImageFileHdr->SizeOfOptionalHeader == sizeof(IMAGE_OPTIONAL_HEADER32)) ||
(ImageFileHdr->SizeOfOptionalHeader == sizeof(IMAGE_OPTIONAL_HEADER64)))
{
char szOSVersion[30];
char szImageVersion[30];
char szSubsystemVersion[30];
switch (ImageOptionalHdr->Subsystem) {
case IMAGE_SUBSYSTEM_POSIX_CUI : i = 4; break;
case IMAGE_SUBSYSTEM_WINDOWS_CUI : i = 3; break;
case IMAGE_SUBSYSTEM_WINDOWS_GUI : i = 2; break;
case IMAGE_SUBSYSTEM_NATIVE : i = 1; break;
default : i = 0;
}
PrintString(szOSVersion,
DIMA(szOSVersion),
"%hu.%02hu",
ImageOptionalHdr->MajorOperatingSystemVersion,
ImageOptionalHdr->MinorOperatingSystemVersion);
PrintString(szImageVersion,
DIMA(szImageVersion),
"%hu.%02hu",
ImageOptionalHdr->MajorImageVersion,
ImageOptionalHdr->MinorImageVersion);
PrintString(szSubsystemVersion,
DIMA(szSubsystemVersion),
"%hu.%02hu",
ImageOptionalHdr->MajorSubsystemVersion,
ImageOptionalHdr->MinorSubsystemVersion);
dprintf(
" ----- new -----\n"
"%p image base\n"
"%8lX section alignment\n"
"%8lX file alignment\n"
"%8hX subsystem (%s)\n"
"%8s operating system version\n"
"%8s image version\n"
"%8s subsystem version\n"
"%8lX size of image\n"
"%8lX size of headers\n"
"%8lX checksum\n",
ImageOptionalHdr->ImageBase,
ImageOptionalHdr->SectionAlignment,
ImageOptionalHdr->FileAlignment,
ImageOptionalHdr->Subsystem,
SubsystemName[i],
szOSVersion,
szImageVersion,
szSubsystemVersion,
ImageOptionalHdr->SizeOfImage,
ImageOptionalHdr->SizeOfHeaders,
ImageOptionalHdr->CheckSum);
dprintf(
"%p size of stack reserve\n"
"%p size of stack commit\n"
"%p size of heap reserve\n"
"%p size of heap commit\n",
ImageOptionalHdr->SizeOfStackReserve,
ImageOptionalHdr->SizeOfStackCommit,
ImageOptionalHdr->SizeOfHeapReserve,
ImageOptionalHdr->SizeOfHeapCommit);
for (i = 0; i < IMAGE_NUMBEROF_DIRECTORY_ENTRIES; i++) {
if (!DirectoryEntryName[i]) {
break;
}
dprintf( "%8lX [%8lX] address [size] of %s Directory\n",
ImageOptionalHdr->DataDirectory[i].VirtualAddress,
ImageOptionalHdr->DataDirectory[i].Size,
DirectoryEntryName[i]
);
}
dprintf( "\n" );
}
}
VOID
DumpSectionHeader (
IN DWORD i,
IN PIMAGE_SECTION_HEADER Sh
)
{
const char *name;
char *szUnDName;
DWORD li, lj;
WORD memFlags;
dprintf("\nSECTION HEADER #%hX\n%8.8s name", i, Sh->Name);
#if 0
if (Sh->Name[0] == '/') {
name = SzObjSectionName((char *) Sh->Name, (char *) DumpStringTable);
dprintf(" (%s)", name);
}
#endif
dprintf( "\n");
dprintf( "%8lX %s\n"
"%8lX virtual address\n"
"%8lX size of raw data\n"
"%8lX file pointer to raw data\n"
"%8lX file pointer to relocation table\n",
Sh->Misc.PhysicalAddress,
(dft == dftObject) ? "physical address" : "virtual size",
Sh->VirtualAddress,
Sh->SizeOfRawData,
Sh->PointerToRawData,
Sh->PointerToRelocations);
dprintf( "%8lX file pointer to line numbers\n"
"%8hX number of relocations\n"
"%8hX number of line numbers\n"
"%8lX flags\n",
Sh->PointerToLinenumbers,
Sh->NumberOfRelocations,
Sh->NumberOfLinenumbers,
Sh->Characteristics);
memFlags = 0;
li = Sh->Characteristics;
if (dft == dftROM) {
for (lj = 0L; li; li = li >> 1, lj++) {
if (li & 1) {
switch ((li & 1) << lj) {
case STYP_REG : name = "Regular"; break;
case STYP_TEXT : name = "Text"; memFlags = 1; break;
case STYP_INIT : name = "Init Code"; memFlags = 1; break;
case STYP_RDATA : name = "Data"; memFlags = 2; break;
case STYP_DATA : name = "Data"; memFlags = 6; break;
case STYP_LIT8 : name = "Literal 8"; break;
case STYP_LIT4 : name = "Literal 4"; break;
case STYP_SDATA : name = "GP Init Data"; memFlags = 6; break;
case STYP_SBSS : name = "GP Uninit Data"; memFlags = 6; break;
case STYP_BSS : name = "Uninit Data"; memFlags = 6; break;
case STYP_LIB : name = "Library"; break;
case STYP_UCODE : name = "UCode"; break;
case S_NRELOC_OVFL : name = "Non-Relocatable overlay"; memFlags = 1; break;
default : name = "RESERVED - UNKNOWN";
}
dprintf( " %s\n", name);
}
}
} else {
// Clear the padding bits
li &= ~0x00700000;
for (lj = 0L; li; li = li >> 1, lj++) {
if (li & 1) {
switch ((li & 1) << lj) {
case IMAGE_SCN_TYPE_NO_PAD : name = "No Pad"; break;
case IMAGE_SCN_CNT_CODE : name = "Code"; break;
case IMAGE_SCN_CNT_INITIALIZED_DATA : name = "Initialized Data"; break;
case IMAGE_SCN_CNT_UNINITIALIZED_DATA : name = "Uninitialized Data"; break;
case IMAGE_SCN_LNK_OTHER : name = "Other"; break;
case IMAGE_SCN_LNK_INFO : name = "Info"; break;
case IMAGE_SCN_LNK_REMOVE : name = "Remove"; break;
case IMAGE_SCN_LNK_COMDAT : name = "Communal"; break;
case IMAGE_SCN_MEM_DISCARDABLE: name = "Discardable"; break;
case IMAGE_SCN_MEM_NOT_CACHED: name = "Not Cached"; break;
case IMAGE_SCN_MEM_NOT_PAGED: name = "Not Paged"; break;
case IMAGE_SCN_MEM_SHARED : name = "Shared"; break;
case IMAGE_SCN_MEM_EXECUTE : name = ""; memFlags |= 1; break;
case IMAGE_SCN_MEM_READ : name = ""; memFlags |= 2; break;
case IMAGE_SCN_MEM_WRITE : name = ""; memFlags |= 4; break;
case IMAGE_SCN_MEM_FARDATA : name = "Far Data"; break;
case IMAGE_SCN_MEM_SYSHEAP : name = "Sys Heap"; break;
case IMAGE_SCN_MEM_PURGEABLE: name = "Purgeable or 16-Bit"; break;
case IMAGE_SCN_MEM_LOCKED : name = "Locked"; break;
case IMAGE_SCN_MEM_PRELOAD : name = "Preload"; break;
case IMAGE_SCN_MEM_PROTECTED: name = "Protected"; break;
default : name = "RESERVED - UNKNOWN";
}
if (*name) {
dprintf( " %s\n", name);
}
}
}
// print alignment
switch (Sh->Characteristics & 0x00700000) {
default: name = "(no align specified)"; break;
case IMAGE_SCN_ALIGN_1BYTES: name = "1 byte align"; break;
case IMAGE_SCN_ALIGN_2BYTES: name = "2 byte align"; break;
case IMAGE_SCN_ALIGN_4BYTES: name = "4 byte align"; break;
case IMAGE_SCN_ALIGN_8BYTES: name = "8 byte align"; break;
case IMAGE_SCN_ALIGN_16BYTES: name = "16 byte align"; break;
case IMAGE_SCN_ALIGN_32BYTES: name = "32 byte align"; break;
case IMAGE_SCN_ALIGN_64BYTES: name = "64 byte align"; break;
}
dprintf( " %s\n", name);
}
if (memFlags) {
switch(memFlags) {
case 1 : name = "Execute Only"; break;
case 2 : name = "Read Only"; break;
case 3 : name = "Execute Read"; break;
case 4 : name = "Write Only"; break;
case 5 : name = "Execute Write"; break;
case 6 : name = "Read Write"; break;
case 7 : name = "Execute Read Write"; break;
default : name = "Unknown Memory Flags"; break;
}
dprintf( " %s\n", name);
}
}
VOID
DumpDebugDirectory (
IN PIMAGE_DEBUG_DIRECTORY DebugDir
)
{
BOOL Ok;
DWORD cb;
CVDD cv;
PIMAGE_DEBUG_MISC miscData;
PIMAGE_DEBUG_MISC miscDataCur;
ULONG VirtualAddress;
DWORD len;
switch (DebugDir->Type){
case IMAGE_DEBUG_TYPE_COFF:
dprintf( "\tcoff ");
break;
case IMAGE_DEBUG_TYPE_CODEVIEW:
dprintf( "\tcv ");
break;
case IMAGE_DEBUG_TYPE_FPO:
dprintf( "\tfpo ");
break;
case IMAGE_DEBUG_TYPE_MISC:
dprintf( "\tmisc ");
break;
case IMAGE_DEBUG_TYPE_FIXUP:
dprintf( "\tfixup ");
break;
case IMAGE_DEBUG_TYPE_OMAP_TO_SRC:
dprintf( "\t-> src ");
break;
case IMAGE_DEBUG_TYPE_OMAP_FROM_SRC:
dprintf( "\tsrc -> ");
break;
case IMAGE_DEBUG_TYPE_EXCEPTION:
dprintf( "\tpdata ");
break;
default:
dprintf( "\t(%6lu)", DebugDir->Type);
break;
}
dprintf( "%8x %8x %8x",
DebugDir->SizeOfData,
DebugDir->AddressOfRawData,
DebugDir->PointerToRawData);
if (DebugDir->PointerToRawData &&
DebugDir->Type == IMAGE_DEBUG_TYPE_MISC)
{
if (!TranslateFilePointerToVirtualAddress(DebugDir->PointerToRawData, &VirtualAddress)) {
dprintf(" [Debug data not mapped]\n");
} else {
len = DebugDir->SizeOfData;
miscData = (PIMAGE_DEBUG_MISC) malloc(len);
if (!miscData) {
goto DebugTypeCodeView;
}
__try {
Ok = ReadMemory(Base + VirtualAddress, miscData, len, &cb);
if (!Ok || cb != len) {
dprintf("Can't read debug data\n");
} else {
miscDataCur = miscData;
do {
if (miscDataCur->DataType == IMAGE_DEBUG_MISC_EXENAME) {
if (ImageOptionalHdr->MajorLinkerVersion == 2 &&
ImageOptionalHdr->MinorLinkerVersion < 37) {
dprintf( "\tImage Name: %s", miscDataCur->Reserved);
} else {
dprintf( "\tImage Name: %s", miscDataCur->Data);
}
break;
}
len -= miscDataCur->Length;
miscDataCur = (PIMAGE_DEBUG_MISC) ((PCHAR) miscDataCur + miscData->Length);
} while (len > 0);
}
}
__finally {
if (miscData) {
free(miscData);
}
}
}
}
DebugTypeCodeView:
if (DebugDir->PointerToRawData &&
DebugDir->Type == IMAGE_DEBUG_TYPE_CODEVIEW) {
if (DebugDir->AddressOfRawData) {
VirtualAddress = DebugDir->AddressOfRawData;
}
if (!DebugDir->AddressOfRawData &&
!TranslateFilePointerToVirtualAddress(DebugDir->PointerToRawData, &VirtualAddress)) {
dprintf(" [Debug data not mapped]\n");
} else {
len = DebugDir->SizeOfData;
Ok = ReadMemory(Base + VirtualAddress, &cv, len, &cb);
if (!Ok || cb != len) {
dprintf("\tCan't read debug data cb=%lx\n", cb);
} else {
if (cv.dwSig == '01BN') {
dprintf( "\tFormat: NB10, %x, %x, %s", cv.nb10i.sig, cv.nb10i.age, cv.nb10i.szPdb);
} else if (cv.dwSig == 'SDSR') {
dprintf( "\tFormat: RSDS, guid, %x, %s", cv.rsdsi.age, cv.rsdsi.szPdb);
} else {
dprintf( "\tFormat: UNKNOWN");
}
}
}
}
dprintf( "\n");
}
VOID
DumpDebugDirectories (
PIMAGE_SECTION_HEADER sh
)
/*++
Routine Description:
Print out the contents of all debug directories
Arguments:
sh - Section header for section that contains debug dirs
Return Value:
None.
--*/
{
int numDebugDirs;
IMAGE_DEBUG_DIRECTORY debugDir;
ULONG64 DebugDirAddr;
ULONG64 pc;
DWORD cb;
BOOL Ok;
if (dft == dftROM) {
DebugDirAddr = (Base + sh->VirtualAddress);
pc = DebugDirAddr;
Ok = ReadMemory(pc, &debugDir, sizeof(IMAGE_DEBUG_DIRECTORY), &cb);
if (!Ok || cb != sizeof(IMAGE_DEBUG_DIRECTORY)) {
dprintf("Can't read debug dir\n");
return;
}
numDebugDirs = 0;
while (debugDir.Type != 0) {
numDebugDirs++;
pc += sizeof(IMAGE_DEBUG_DIRECTORY);
Ok = ReadMemory(pc, &debugDir, sizeof(IMAGE_DEBUG_DIRECTORY), &cb);
if (!Ok || cb != sizeof(IMAGE_DEBUG_DIRECTORY)) {
break;
}
}
} else {
DebugDirAddr = (Base + ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_DEBUG].VirtualAddress);
numDebugDirs = ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_DEBUG].Size / sizeof(IMAGE_DEBUG_DIRECTORY);
}
dprintf("\n\nDebug Directories(%d)\n",numDebugDirs);
dprintf("\tType Size Address Pointer\n\n");
pc = DebugDirAddr;
while (numDebugDirs) {
Ok = ReadMemory(pc, &debugDir, sizeof(IMAGE_DEBUG_DIRECTORY), &cb);
if (!Ok || cb != sizeof(IMAGE_DEBUG_DIRECTORY)) {
dprintf("Can't read debug dir\n");
break;
}
pc += sizeof(IMAGE_DEBUG_DIRECTORY);
DumpDebugDirectory(&debugDir);
numDebugDirs--;
}
}
VOID
dhDumpSections(
)
{
IMAGE_SECTION_HEADER sh;
const char *p;
DWORD li;
DWORD cb;
BOOL Ok;
int i, j;
CHAR szName[IMAGE_SIZEOF_SHORT_NAME + 1];
for (i = 1; i <= ImageFileHdr->NumberOfSections; i++) {
sh = SectionHdrs[i-1];
//szName = SzObjSectionName((char *) sh.Name, (char *) DumpStringTable);
CopyStrArray(szName, (char *) sh.Name);
DumpSectionHeader(i, &sh);
if (dft == dftROM) {
if (!(ImageFileHdr->Characteristics & IMAGE_FILE_DEBUG_STRIPPED)) {
// If we're looking at the .rdata section and the symbols
// aren't stripped, the debug directory must be here.
if (!strcmp(szName, ".rdata")) {
DumpDebugDirectories(&sh);
//DumpDebugData(&sh);
}
}
} else if (dft == dftPE) {
if ((li = ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_DEBUG].VirtualAddress) != 0) {
if (li >= sh.VirtualAddress && li < sh.VirtualAddress+sh.SizeOfRawData) {
DumpDebugDirectories(&sh);
//DumpDebugData(&sh);
}
}
#if 0
if (Switch.Dump.PData) {
li = ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_EXCEPTION].VirtualAddress;
if ((li != 0) && (li >= sh.VirtualAddress) && (li < sh.VirtualAddress+sh.SizeOfRawData)) {
DumpFunctionTable(pimage, rgsym, (char *) DumpStringTable, &sh);
}
}
if (Switch.Dump.Imports) {
li = ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress;
if ((li != 0) && (li >= sh.VirtualAddress) && (li < sh.VirtualAddress+sh.SizeOfRawData)) {
DumpImports(&sh);
}
}
if (Switch.Dump.Exports) {
li = ImageOptionalHdr->DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress;
if ((li != 0) && (li >= sh.VirtualAddress) && (li < sh.VirtualAddress+sh.SizeOfRawData)) {
// UNDONE: Is this check really necessary?
if (ImageFileHdr->Machine != IMAGE_FILE_MACHINE_MPPC_601) {
DumpExports(&sh);
}
}
}
#endif
}
}
}
BOOL
TranslateFilePointerToVirtualAddress(
IN ULONG FilePointer,
OUT PULONG VirtualAddress
)
{
ULONG i;
PIMAGE_SECTION_HEADER sh;
*VirtualAddress = 0;
for (i = 1; i <= NumSections; i++) {
sh = &SectionHdrs[i-1];
if (sh->PointerToRawData <= FilePointer &&
FilePointer < sh->PointerToRawData + sh->SizeOfRawData) {
*VirtualAddress = FilePointer - sh->PointerToRawData + sh->VirtualAddress;
return true;
}
}
return false;
}