|
|
#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; }
|