2005-03-07 12:03:53 +01:00
|
|
|
/*
|
|
|
|
* MiniDump dumping utility
|
|
|
|
*
|
|
|
|
* Copyright 2005 Eric Pouech
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#define NONAMELESSUNION
|
|
|
|
#define NONAMELESSSTRUCT
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "winver.h"
|
|
|
|
#include "dbghelp.h"
|
|
|
|
#include "winedump.h"
|
|
|
|
|
|
|
|
static void dump_mdmp_data(const MINIDUMP_LOCATION_DESCRIPTOR* md, const char* pfx)
|
|
|
|
{
|
|
|
|
if (md->DataSize)
|
|
|
|
dump_data(PRD(md->Rva, md->DataSize), md->DataSize, pfx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dump_mdmp_string(DWORD rva)
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_STRING* ms = PRD(rva, sizeof(MINIDUMP_STRING));
|
2005-03-07 12:03:53 +01:00
|
|
|
if (ms)
|
|
|
|
dump_unicode_str( ms->Buffer, ms->Length / sizeof(WCHAR) );
|
|
|
|
else
|
|
|
|
printf("<<?>>");
|
|
|
|
}
|
|
|
|
|
2006-04-10 20:53:50 +02:00
|
|
|
static const MINIDUMP_DIRECTORY* get_mdmp_dir(const MINIDUMP_HEADER* hdr, int str_idx)
|
2005-03-07 12:03:53 +01:00
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_DIRECTORY* dir;
|
|
|
|
unsigned int i;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
for (i = 0; i < hdr->NumberOfStreams; i++)
|
|
|
|
{
|
|
|
|
dir = PRD(hdr->StreamDirectoryRva + i * sizeof(MINIDUMP_DIRECTORY),
|
|
|
|
sizeof(MINIDUMP_DIRECTORY));
|
|
|
|
if (!dir) continue;
|
|
|
|
if (dir->StreamType == str_idx) return dir;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mdmp_dump(void)
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_HEADER* hdr = (const MINIDUMP_HEADER*)PRD(0, sizeof(MINIDUMP_HEADER));
|
|
|
|
ULONG idx, ndir = 0;
|
|
|
|
const MINIDUMP_DIRECTORY* dir;
|
|
|
|
const void* stream;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
if (!hdr)
|
|
|
|
{
|
|
|
|
printf("Cannot get Minidump header\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-04-10 20:53:50 +02:00
|
|
|
printf("Signature: %lu (%.4s)\n", hdr->Signature, (const char*)&hdr->Signature);
|
2005-03-07 12:03:53 +01:00
|
|
|
printf("Version: %lx\n", hdr->Version);
|
|
|
|
printf("NumberOfStreams: %lu\n", hdr->NumberOfStreams);
|
|
|
|
printf("StreamDirectoryRva: %lu\n", hdr->StreamDirectoryRva);
|
|
|
|
printf("CheckSum: %lu\n", hdr->CheckSum);
|
|
|
|
printf("TimeDateStamp: %s\n", get_time_str(hdr->u.TimeDateStamp));
|
|
|
|
printf("Flags: %llx\n", hdr->Flags);
|
|
|
|
|
|
|
|
for (idx = 0; idx <= LastReservedStream; idx++)
|
|
|
|
{
|
|
|
|
if (!(dir = get_mdmp_dir(hdr, idx))) continue;
|
|
|
|
|
|
|
|
stream = PRD(dir->Location.Rva, dir->Location.DataSize);
|
|
|
|
printf("Directory [%lu]: ", ndir++);
|
|
|
|
switch (dir->StreamType)
|
|
|
|
{
|
|
|
|
case ThreadListStream:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_THREAD_LIST* mtl = (const MINIDUMP_THREAD_LIST*)stream;
|
|
|
|
const MINIDUMP_THREAD* mt = &mtl->Threads[0];
|
|
|
|
unsigned int i;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
printf("Threads: %lu\n", mtl->NumberOfThreads);
|
|
|
|
for (i = 0; i < mtl->NumberOfThreads; i++, mt++)
|
|
|
|
{
|
|
|
|
printf(" Thread: #%d\n", i);
|
|
|
|
printf(" ThreadId: %lu\n", mt->ThreadId);
|
|
|
|
printf(" SuspendCount: %lu\n", mt->SuspendCount);
|
|
|
|
printf(" PriorityClass: %lu\n", mt->PriorityClass);
|
|
|
|
printf(" Priority: %lu\n", mt->Priority);
|
|
|
|
printf(" Teb: 0x%llx\n", mt->Teb);
|
|
|
|
printf(" Stack: 0x%llx-0x%llx\n",
|
|
|
|
mt->Stack.StartOfMemoryRange,
|
|
|
|
mt->Stack.StartOfMemoryRange + mt->Stack.Memory.DataSize);
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&mt->Stack.Memory, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
printf(" ThreadContext:\n");
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&mt->ThreadContext, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ModuleListStream:
|
|
|
|
case 0xFFF0:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_MODULE_LIST* mml = (const MINIDUMP_MODULE_LIST*)stream;
|
|
|
|
const MINIDUMP_MODULE* mm = &mml->Modules[0];
|
|
|
|
unsigned int i;
|
|
|
|
const char* p1;
|
|
|
|
const char* p2;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
printf("Modules (%s): %lu\n",
|
|
|
|
dir->StreamType == ModuleListStream ? "PE" : "ELF",
|
|
|
|
mml->NumberOfModules);
|
|
|
|
for (i = 0; i < mml->NumberOfModules; i++, mm++)
|
|
|
|
{
|
|
|
|
printf(" Module #%d:\n", i);
|
|
|
|
printf(" BaseOfImage: 0x%llx\n", mm->BaseOfImage);
|
|
|
|
printf(" SizeOfImage: %lu\n", mm->SizeOfImage);
|
|
|
|
printf(" CheckSum: %lu\n", mm->CheckSum);
|
|
|
|
printf(" TimeDateStamp: %s\n", get_time_str(mm->TimeDateStamp));
|
|
|
|
printf(" ModuleName: ");
|
|
|
|
dump_mdmp_string(mm->ModuleNameRva);
|
|
|
|
printf("\n");
|
|
|
|
printf(" VersionInfo:\n");
|
|
|
|
printf(" dwSignature: %lx\n", mm->VersionInfo.dwSignature);
|
|
|
|
printf(" dwStrucVersion: %lx\n",
|
|
|
|
mm->VersionInfo.dwStrucVersion);
|
|
|
|
printf(" dwFileVersion: %d,%d,%d,%d\n",
|
|
|
|
HIWORD(mm->VersionInfo.dwFileVersionMS),
|
|
|
|
LOWORD(mm->VersionInfo.dwFileVersionMS),
|
|
|
|
HIWORD(mm->VersionInfo.dwFileVersionLS),
|
|
|
|
LOWORD(mm->VersionInfo.dwFileVersionLS));
|
|
|
|
printf(" dwProductVersion %d,%d,%d,%d\n",
|
|
|
|
HIWORD(mm->VersionInfo.dwProductVersionMS),
|
|
|
|
LOWORD(mm->VersionInfo.dwProductVersionMS),
|
|
|
|
HIWORD(mm->VersionInfo.dwProductVersionLS),
|
|
|
|
LOWORD(mm->VersionInfo.dwProductVersionLS));
|
|
|
|
printf(" dwFileFlagsMask: %lu\n",
|
|
|
|
mm->VersionInfo.dwFileFlagsMask);
|
|
|
|
printf(" dwFileFlags: %s%s%s%s%s%s\n",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_DEBUG ? "Debug " : "",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_INFOINFERRED ? "Inferred " : "",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_PATCHED ? "Patched " : "",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_PRERELEASE ? "PreRelease " : "",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_PRIVATEBUILD ? "PrivateBuild " : "",
|
|
|
|
mm->VersionInfo.dwFileFlags & VS_FF_SPECIALBUILD ? "SpecialBuild " : "");
|
|
|
|
if (mm->VersionInfo.dwFileOS)
|
|
|
|
{
|
|
|
|
switch (mm->VersionInfo.dwFileOS & 0x000F)
|
|
|
|
{
|
|
|
|
case VOS__BASE: p1 = "_base"; break;
|
|
|
|
case VOS__WINDOWS16:p1 = "16 bit Windows"; break;
|
|
|
|
case VOS__PM16: p1 = "16 bit Presentation Manager"; break;
|
|
|
|
case VOS__PM32: p1 = "32 bit Presentation Manager"; break;
|
|
|
|
case VOS__WINDOWS32:p1 = "32 bit Windows"; break;
|
|
|
|
default: p1 = "---"; break;
|
|
|
|
}
|
|
|
|
switch (mm->VersionInfo.dwFileOS & 0xF0000)
|
|
|
|
{
|
|
|
|
case VOS_UNKNOWN: p2 = "unknown"; break;
|
|
|
|
case VOS_DOS: p2 = "DOS"; break;
|
|
|
|
case VOS_OS216: p2 = "16 bit OS/2"; break;
|
|
|
|
case VOS_OS232: p2 = "32 bit OS/2"; break;
|
|
|
|
case VOS_NT: p2 = "Windows NT"; break;
|
|
|
|
default: p2 = "---"; break;
|
|
|
|
}
|
|
|
|
printf(" dwFileOS: %s running on %s\n", p1, p2);
|
|
|
|
}
|
|
|
|
else printf(" dwFileOS: 0\n");
|
|
|
|
switch (mm->VersionInfo.dwFileType)
|
|
|
|
{
|
|
|
|
case VFT_UNKNOWN: p1 = "Unknown"; break;
|
|
|
|
case VFT_APP: p1 = "Application"; break;
|
|
|
|
case VFT_DLL: p1 = "DLL"; break;
|
|
|
|
case VFT_DRV: p1 = "Driver"; break;
|
|
|
|
case VFT_FONT: p1 = "Font"; break;
|
|
|
|
case VFT_VXD: p1 = "VxD"; break;
|
|
|
|
case VFT_STATIC_LIB: p1 = "Static Library"; break;
|
|
|
|
default: p1 = "---"; break;
|
|
|
|
}
|
|
|
|
printf(" dwFileType: %s\n", p1);
|
|
|
|
printf(" dwFileSubtype: %lu\n",
|
|
|
|
mm->VersionInfo.dwFileSubtype);
|
|
|
|
printf(" dwFileDate: %lx%08lx\n",
|
|
|
|
mm->VersionInfo.dwFileDateMS, mm->VersionInfo.dwFileDateLS);
|
|
|
|
printf(" CvRecord: <%lu>\n", mm->CvRecord.DataSize);
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&mm->CvRecord, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
printf(" MiscRecord: <%lu>\n", mm->MiscRecord.DataSize);
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&mm->MiscRecord, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
printf(" Reserved0: %llu\n", mm->Reserved0);
|
|
|
|
printf(" Reserved1: %llu\n", mm->Reserved1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MemoryListStream:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_MEMORY_LIST* mml = (const MINIDUMP_MEMORY_LIST*)stream;
|
|
|
|
const MINIDUMP_MEMORY_DESCRIPTOR* mmd = &mml->MemoryRanges[0];
|
|
|
|
unsigned int i;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
printf("Memory Ranges: %lu\n", mml->NumberOfMemoryRanges);
|
|
|
|
for (i = 0; i < mml->NumberOfMemoryRanges; i++, mmd++)
|
|
|
|
{
|
|
|
|
printf(" Memory Range #%d:\n", i);
|
|
|
|
printf(" Range: 0x%llx-0x%llx\n",
|
|
|
|
mmd->StartOfMemoryRange,
|
|
|
|
mmd->StartOfMemoryRange + mmd->Memory.DataSize);
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&mmd->Memory, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SystemInfoStream:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_SYSTEM_INFO* msi = (const MINIDUMP_SYSTEM_INFO*)stream;
|
2005-05-23 12:28:17 +02:00
|
|
|
const char* str;
|
2005-03-07 12:03:53 +01:00
|
|
|
char tmp[128];
|
|
|
|
|
|
|
|
printf("System Information:\n");
|
|
|
|
switch (msi->ProcessorArchitecture)
|
|
|
|
{
|
|
|
|
case PROCESSOR_ARCHITECTURE_UNKNOWN:
|
|
|
|
str = "Unknown";
|
|
|
|
break;
|
|
|
|
case PROCESSOR_ARCHITECTURE_INTEL:
|
|
|
|
strcpy(tmp, "Intel ");
|
|
|
|
switch (msi->ProcessorLevel)
|
|
|
|
{
|
|
|
|
case 3: str = "80386"; break;
|
|
|
|
case 4: str = "80486"; break;
|
|
|
|
case 5: str = "Pentium"; break;
|
|
|
|
case 6: str = "Pentium Pro/II"; break;
|
|
|
|
default: str = "???"; break;
|
|
|
|
}
|
|
|
|
strcat(tmp, str);
|
|
|
|
if (msi->ProcessorLevel == 3 || msi->ProcessorLevel == 4)
|
|
|
|
{
|
|
|
|
if (HIWORD(msi->ProcessorRevision) == 0xFF)
|
|
|
|
sprintf(tmp + strlen(tmp), "-%c%d", 'A' + HIBYTE(LOWORD(msi->ProcessorRevision)), LOBYTE(LOWORD(msi->ProcessorRevision)));
|
|
|
|
else
|
|
|
|
sprintf(tmp + strlen(tmp), "-%c%d", 'A' + HIWORD(msi->ProcessorRevision), LOWORD(msi->ProcessorRevision));
|
|
|
|
}
|
|
|
|
else sprintf(tmp + strlen(tmp), "-%d.%d", HIWORD(msi->ProcessorRevision), LOWORD(msi->ProcessorRevision));
|
|
|
|
str = tmp;
|
|
|
|
break;
|
|
|
|
case PROCESSOR_ARCHITECTURE_MIPS:
|
|
|
|
str = "Mips";
|
|
|
|
break;
|
|
|
|
case PROCESSOR_ARCHITECTURE_ALPHA:
|
|
|
|
str = "Alpha";
|
|
|
|
break;
|
|
|
|
case PROCESSOR_ARCHITECTURE_PPC:
|
|
|
|
str = "PowerPC";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
str = "???";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
printf(" Processor: %s (#%d CPUs)\n", str, msi->u.s.NumberOfProcessors);
|
|
|
|
switch (msi->MajorVersion)
|
|
|
|
{
|
|
|
|
case 3:
|
|
|
|
switch (msi->MinorVersion)
|
|
|
|
{
|
|
|
|
case 51: str = "NT 3.51"; break;
|
|
|
|
default: str = "3-????"; break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
switch (msi->MinorVersion)
|
|
|
|
{
|
|
|
|
case 0: str = (msi->PlatformId == VER_PLATFORM_WIN32_NT) ? "NT 4.0" : "95"; break;
|
|
|
|
case 10: str = "98"; break;
|
|
|
|
case 90: str = "ME"; break;
|
|
|
|
default: str = "5-????"; break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
switch (msi->MinorVersion)
|
|
|
|
{
|
|
|
|
case 0: str = "2000"; break;
|
|
|
|
case 1: str = "XP"; break;
|
|
|
|
case 2: str = "Server 2003"; break;
|
|
|
|
default: str = "5-????"; break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: str = "???"; break;
|
|
|
|
}
|
|
|
|
printf(" Version: Windows %s (%lu)\n", str, msi->BuildNumber);
|
|
|
|
printf(" PlatformId: %lu\n", msi->PlatformId);
|
|
|
|
printf(" CSD: ");
|
|
|
|
dump_mdmp_string(msi->CSDVersionRva);
|
|
|
|
printf("\n");
|
|
|
|
printf(" Reserved1: %lu\n", msi->u1.Reserved1);
|
|
|
|
if (msi->ProcessorArchitecture == PROCESSOR_ARCHITECTURE_INTEL)
|
|
|
|
{
|
|
|
|
printf(" x86.VendorId: %.12s\n",
|
2006-04-10 20:53:50 +02:00
|
|
|
(const char*)&msi->Cpu.X86CpuInfo.VendorId[0]);
|
2005-03-07 12:03:53 +01:00
|
|
|
printf(" x86.VersionInformation: %lx\n",
|
|
|
|
msi->Cpu.X86CpuInfo.VersionInformation);
|
|
|
|
printf(" x86.FeatureInformation: %lx\n",
|
|
|
|
msi->Cpu.X86CpuInfo.FeatureInformation);
|
|
|
|
printf(" x86.AMDExtendedCpuFeatures: %lu\n",
|
|
|
|
msi->Cpu.X86CpuInfo.AMDExtendedCpuFeatures);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MiscInfoStream:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_MISC_INFO* mmi = (const MINIDUMP_MISC_INFO*)stream;
|
|
|
|
|
2005-03-07 12:03:53 +01:00
|
|
|
printf("Misc Information\n");
|
|
|
|
printf(" Size: %lu\n", mmi->SizeOfInfo);
|
|
|
|
printf(" Flags: %s%s\n",
|
|
|
|
mmi->Flags1 & MINIDUMP_MISC1_PROCESS_ID ? "ProcessId " : "",
|
|
|
|
mmi->Flags1 & MINIDUMP_MISC1_PROCESS_TIMES ? "ProcessTimes " : "");
|
|
|
|
if (mmi->Flags1 & MINIDUMP_MISC1_PROCESS_ID)
|
|
|
|
printf(" ProcessId: %lu\n", mmi->ProcessId);
|
|
|
|
if (mmi->Flags1 & MINIDUMP_MISC1_PROCESS_TIMES)
|
|
|
|
{
|
|
|
|
printf(" ProcessCreateTime: %lu\n", mmi->ProcessCreateTime);
|
|
|
|
printf(" ProcessUserTime: %lu\n", mmi->ProcessUserTime);
|
|
|
|
printf(" ProcessKernelTime: %lu\n", mmi->ProcessKernelTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ExceptionStream:
|
|
|
|
{
|
2006-04-10 20:53:50 +02:00
|
|
|
const MINIDUMP_EXCEPTION_STREAM* mes = (const MINIDUMP_EXCEPTION_STREAM*)stream;
|
|
|
|
unsigned int i;
|
2005-03-07 12:03:53 +01:00
|
|
|
|
|
|
|
printf("Exception:\n");
|
|
|
|
printf(" ThreadId: %08lx\n", mes->ThreadId);
|
|
|
|
printf(" ExceptionRecord:\n");
|
|
|
|
printf(" ExceptionCode: %lu\n", mes->ExceptionRecord.ExceptionCode);
|
|
|
|
printf(" ExceptionFlags: %lu\n", mes->ExceptionRecord.ExceptionFlags);
|
|
|
|
printf(" ExceptionRecord: 0x%llx\n",
|
|
|
|
mes->ExceptionRecord.ExceptionRecord);
|
|
|
|
printf(" ExceptionAddress: 0x%llx\n",
|
|
|
|
mes->ExceptionRecord.ExceptionAddress);
|
|
|
|
printf(" ExceptionNumberParameters: %lu\n",
|
|
|
|
mes->ExceptionRecord.NumberParameters);
|
|
|
|
for (i = 0; i < mes->ExceptionRecord.NumberParameters; i++)
|
|
|
|
{
|
|
|
|
printf(" [%d]: 0x%llx\n", i,
|
|
|
|
mes->ExceptionRecord.ExceptionInformation[i]);
|
|
|
|
}
|
|
|
|
printf(" ThreadContext:\n");
|
|
|
|
dump_mdmp_data(&mes->ThreadContext, " ");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("NIY %ld\n", dir->StreamType);
|
|
|
|
printf(" RVA: %lu\n", dir->Location.Rva);
|
|
|
|
printf(" Size: %lu\n", dir->Location.DataSize);
|
2005-05-24 13:45:14 +02:00
|
|
|
dump_mdmp_data(&dir->Location, " ");
|
2005-03-07 12:03:53 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|