记得先在form上放个RichEdit。
==================
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls, ComCtrls;
type
TForm1 = class(TForm)
RichEdit1: TRichEdit;
procedure FormCreate(Sender: TObject);
procedure FormDestroy(Sender: TObject);
private
{ Private declarations }
procedure WithDropFiles(var Msg: TMessage); message WM_DROPFILES;
procedure ShowMsgAlign(s: string; Indent: integer = 0; Align: integer= 34);
procedure ShowMsg(s: string; Indent: integer = 0);
procedure DoDump;
procedure DumpImportTable;
procedure DumpExportTable;
procedure DumpRelocationTable;
procedure DumpDelayImportTable;
procedure DumpBoundImportTable;
procedure DumpCOMTable;
procedure DumpCopyrightTable;
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.dfm}
uses StrUtils, ShellAPI, JwaWinNt;
type
PImageDelayImportDirectory = ^TImageDelayImportDirectory;
TImageDelayImportDirectory = record
Attr: DWORD;
DLLName: DWORD;
Hmod: DWORD;
IAT: DWORD;
INT: DWORD;
BoundIAT: DWORD;
UnloadIAT: DWORD;
TimeDateStamp: DWORD;
end;
type
TImageDirectoryArray = record
RVA: DWORD;
Size: DWORD;
end;
type
TDWORDArray= Array of DWORD;
TWORDArray= Array of WORD;
const
TableArray: Array [0..IMAGE_NUMBEROF_DIRECTORY_ENTRIES - 1] of string =
('ExportTable', 'ImportTable', 'Resource', 'Exception', 'Security',
'Relocation', 'Debug' ,'Copyright', 'GlobalPtr', 'TLSTable', 'LoadConfig',
'BoundImport' ,'IAT', 'DelayImport', 'COM', 'Reserved');
var
fn: String;
MStream: TMemoryStream;
PPE: Pointer;
PFileHeader: PImageFileHeader;
POptionalHeader32: PImageOptionalHeader32;
PDataDirectory: JwaWinNt.PImageDataDirectory;
PSectionHeader: PImageSectionHeader;
DirectoryArray: Array [0..IMAGE_NUMBEROF_DIRECTORY_ENTRIES - 1] of TImageDirectoryArray;
PImportDecriptor: PImageImportDecriptor;
PExportDirectory: PImageExportDirectory;
PBaseRelocation: PImageBaseRelocation;
DelayImportDirectory: PImageDelayImportDirectory;
PBoundImportDescriptor: PImageBoundImportDescriptor;
PCor20Header: PImageCor20Header;
procedure TForm1.WithDropFiles(var Msg: TMessage);
Var
Buffer: PChar;
i: integer;
begin
i := DragQueryFile( Msg.WParam, 0, nil, 0) + 1;
GetMem(Buffer, i);
try
DragQueryFile(Msg.WParam, 0, Buffer, i);
MStream.Clear;
RichEdit1.Clear;
fn := Buffer;
MStream.LoadFromFile(fn);
DoDump;
SendMessage(Richedit1.Handle, WM_VSCROLL, SB_BOTTOM, 0);
DragFinish(Msg.WParam);
SetForegroundWindow(Handle);
Richedit1.SetFocus;
finally
FreeMem(Buffer);
end;
end;
procedure TForm1.ShowMsg(s: string; Indent: integer = 0);
begin
RichEdit1.Lines.Add(DupeString(' ', Indent) + s);
end;
procedure TForm1.ShowMsgAlign(s: string; Indent: integer = 0; Align: integer= 34);
var
j: integer;
s1: string;
begin
j := pos(';', s);
s1 := LeftStr(s, j - 1) + DupeString(' ', Align - Indent - j) + RightStr(s, length(s) - j);
RichEdit1.Lines.Add(DupeString(' ', Indent) + s1);
end;
procedure TForm1.FormCreate(Sender: TObject);
begin
fn := 'explorer.exe';
DragAcceptFiles(Handle, true);
MStream := TMemoryStream.Create;
MStream.LoadFromFile(fn);
DoDump;
SendMessage(Richedit1.Handle, WM_VSCROLL, SB_BOTTOM, 0);
end;
function RVAtoOffset(RVA: Integer): DWORD;
var
i: integer;
PTmpSectionHeader: PImageSectionHeader;
begin
i := 0;
PTmpSectionHeader := PImageSectionHeader(DWORD(POptionalHeader32) + PFileHeader^.SizeOfOptionalHeader);
result := RVA;
while i < PFileHeader^.NumberOfSections do
begin
inc(i);
if (RVA >= PTmpSectionHeader.VirtualAddress) and
(RVA - PTmpSectionHeader.VirtualAddress < PTmpSectionHeader^.Misc.VirtualSize) then
begin
result := RVA - PTmpSectionHeader.VirtualAddress + PTmpSectionHeader.PointerToRawData;
break;
end;
inc(DWORD(PTmpSectionHeader), IMAGE_SIZEOF_SECTION_HEADER);
end;
end;
procedure TForm1.DumpImportTable;
var
i, j: integer;
PFuncName: PImageImportByName;
begin
ShowMsg(DupeString('=', 21) + 'Import Section' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_IMPORT].RVA = 0 then
begin
ShowMsg('No import function found!');
exit;
end;
PImportDecriptor := PImageImportDecriptor(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_IMPORT].RVA));
i := 1;
while (PImportDecriptor^.Union.OriginalFirstThunk <> 0) or (PImportDecriptor^.TimeDateStamp <> 0) or
(PImportDecriptor^.ForwarderChain <> 0) or (PImportDecriptor^.Name <> 0) or (PImportDecriptor^.FirstThunk <> 0) do
begin
ShowMsg(inttostr(i) + '. ImageImportDescriptor:');
ShowMsgAlign('OriginalFirstThunk;' + inttohex(PImportDecriptor^.Union.OriginalFirstThunk, 8), 3);
if PImportDecriptor^.TimeDateStamp = 0 then
ShowMsgAlign('TimeDateStamp;' + inttohex(PImportDecriptor^.TimeDateStamp, 8), 3)
else
ShowMsgAlign('TimeDateStamp;' + inttohex(PImportDecriptor^.TimeDateStamp, 8) + ' (Bound)', 3);
if (PImportDecriptor^.ForwarderChain = 0) or (PImportDecriptor^.ForwarderChain = $FFFFFFFF) then
ShowMsgAlign('ForwarderChain;' + inttohex(PImportDecriptor^.ForwarderChain, 8), 3)
else
ShowMsgAlign('ForwarderChain;' + inttohex(PImportDecriptor^.ForwarderChain, 8) + ' (Forwarder)', 3);
ShowMsgAlign('Dll name;' + inttohex(PImportDecriptor^.Name, 8) + ' ("'+
PChar(DWORD(PPE) + RVAtoOffset(PImportDecriptor^.Name)) +
'"
', 3);
ShowMsgAlign('FirstThunk;' + inttohex(PImportDecriptor^.FirstThunk, 8), 3);
//Loader 会依OriginalFirstThunk(或FirstThunk)指向的函数的顺序填充FirstThunk指向的DWORD,以全零结束。
//即,load后,FirstThunk不再是指向函数名的指针,而是指向函数的地址序列。
ShowMsg('');
ShowMsg('Ordinal/Hint API name', 3);
ShowMsg('------------ ' + DupeString('-', 20), 3);
if PImportDecriptor^.Union.OriginalFirstThunk <> 0 then
j := DWORD(PPE) + RVAtoOffset(PImportDecriptor^.Union.OriginalFirstThunk)
else
j := DWORD(PPE) + RVAtoOffset(PImportDecriptor^.FirstThunk);
while PDWORD(j)^ <> 0 do
begin
if PDWORD(j)^ and IMAGE_ORDINAL_FLAG32 = 0 then
begin
PFuncName := PImageImportByName(DWORD(PPE) + RVAtoOffset(PDWORD(j)^));
ShowMsgAlign('0x' + inttohex(PFuncName^.Hint, 4) + ';"' + PChar(@PFuncName^.Name) + '"', 3, 13);
end
else
begin
ShowMsgAlign('0x' + inttohex(PDWORD(j)^ and $7FFFFFFF, 4) + ';(Ordinal only)', 3, 13);
end;
inc(j, 4);
end;
ShowMsg('');
inc(i);
inc(PImportDecriptor);
end;
end;
procedure TForm1.DumpExportTable;
var
i, j: Cardinal;
ExpFuncAddr: TDWORDArray;
ExpFuncName: TDWORDArray;
ExpFuncOrdinal: TWORDArray;
TmpPchar: string;
ExpNameFound: Boolean;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'Export Section' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_EXPORT].RVA = 0 then
begin
ShowMsg('No export function found!');
exit;
end;
PExportDirectory := PImageExportDirectory(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_EXPORT].RVA));
ShowMsgAlign('Characteristics;0x' + inttohex(PExportDirectory^.Characteristics, 8), 3);
ShowMsgAlign('TimeDateStamp;0x' + inttohex(PExportDirectory^.TimeDateStamp, 8), 3);
ShowMsgAlign('MajorVersion;0x' + inttohex(PExportDirectory^.MajorVersion, 8), 3);
ShowMsgAlign('MinorVersion;0x' + inttohex(PExportDirectory^.MinorVersion, 8), 3);
ShowMsgAlign('Name;0x' + inttohex(PExportDirectory^.Name, 8) + '"(' +
PChar(DWORD(PPE) + RVAtoOffset(PExportDirectory^.Name)) + '"
', 3);
ShowMsgAlign('Base;0x' + inttohex(PExportDirectory^.Base, 8), 3);
ShowMsgAlign('NumberOfFunctions;0x' + inttohex(PExportDirectory^.NumberOfFunctions, 8), 3);
ShowMsgAlign('NumberOfNames;0x' + inttohex(PExportDirectory^.NumberOfNames, 8), 3);
ShowMsgAlign('AddressOfFunctions;0x' + inttohex(PExportDirectory^.AddressOfFunctions, 8), 3);
ShowMsgAlign('AddressOfNames;0x' + inttohex(PExportDirectory^.AddressOfNames, 8), 3);
ShowMsgAlign('AddressOfNameOrdinals;0x' + inttohex(PExportDirectory^.AddressOfNameOrdinals, 8), 3);
ShowMsg('');
ShowMsg('Ordinal' + DupeString(' ', 2) + 'RVA' + DupeString(' ', 8) + 'Symbol Name', 3);
ShowMsg('--------' + DupeString(' ', 1) + '----------' + DupeString(' ', 1) + DupeString('-', 15), 3);
DWORD(ExpFuncAddr) := DWORD(PPE) + RVAtoOffset(PExportDirectory^.AddressOfFunctions);
DWORD(ExpFuncName) := DWORD(PPE) + RVAtoOffset(PExportDirectory^.AddressOfNames);
DWORD(ExpFuncOrdinal) := DWORD(PPE) + RVAtoOffset(PExportDirectory^.AddressOfNameOrdinals);
for i := 0 to PExportDirectory^.NumberOfFunctions - 1 do
begin
if (ExpFuncAddr
<> 0) then
begin
TmpPchar := 'n/a';
ExpNameFound := false;
for j := 0 to PExportDirectory^.NumberOfNames - 1 do
if ExpFuncOrdinal[j] = i then
begin
TmpPchar := PChar(DWORD(PPE) + RVAtoOffset(ExpFuncName[j]));
ExpNameFound := true;
break;
end;
if (ExpFuncAddr >= DirectoryArray[0].RVA) and
(ExpFuncAddr < DirectoryArray[0].RVA + DirectoryArray[0].Size) then
begin
TmpPchar := TmpPchar + ' (Forward)';
end; // Forward ?
if ExpNameFound then
ShowMsg('0x' + inttohex(ExpFuncOrdinal[j] + PExportDirectory^.Base, 4) + ' 0x' +
inttohex(ExpFuncAddr, 8) + ' "' + TmpPchar + '"', 3)
else
ShowMsg('0x' + inttohex(i + PExportDirectory^.Base, 4) + ' 0x' +
inttohex(ExpFuncAddr, 8) + ' "' + TmpPchar + '"', 3);
end;
end;
end;
procedure TForm1.DumpRelocationTable;
var
i, j, k: integer;
RelArray: TWORDArray;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'Reloction Section' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_BASERELOC ].RVA = 0 then
begin
ShowMsg('Reloction table not found!');
exit;
end;
ShowMsg('to display relocation table will take too much time! SKIPPED!');
exit;
Richedit1.Lines.BeginUpdate;
PBaseRelocation := PImageBaseRelocation(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_BASERELOC].RVA));
j := 1;
k := DirectoryArray[IMAGE_DIRECTORY_ENTRY_BASERELOC].Size;
while k > 0 do
begin
DWORD(RelArray) := DWORD(PBaseRelocation) + 2 * SizeOf(DWORD);
ShowMsgAlign(inttostr(j) + '. Relocation block:;');
ShowMsgAlign('RVA;' + inttohex(PBaseRelocation^.VirtualAddress, 8), 3);
ShowMsgAlign('Size;' + inttohex(PBaseRelocation^.SizeOfBlock, 8), 3);
ShowMsg('Offset TYPE', 3);
ShowMsg('------------ ' + DupeString('-', 20), 3);
for i := 0 to (PBaseRelocation^.SizeOfBlock - 2 * SizeOf(DWORD)) div 2 - 1 do
begin
if (RelArray shr 12) = IMAGE_REL_BASED_HIGHLOW then
begin
ShowMsgAlign('0x' + inttohex(RelArray and $0FFF, 4) + ';HIGHLOW', 3, 17);
end;
end;
dec(k, PBaseRelocation^.SizeOfBlock);
inc(j);
inc(DWORD(PBaseRelocation), PBaseRelocation^.SizeOfBlock);
end;
Richedit1.Lines.EndUpdate;
end;
procedure TForm1.DumpDelayImportTable;
var
i, j: integer;
PFuncName: PImageImportByName;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'DelayImport Table' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_DELAY_IMPORT ].RVA = 0 then
begin
ShowMsg('DelayImport table not found!');
exit;
end;
DelayImportDirectory := PImageDelayImportDirectory(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_DELAY_IMPORT].RVA));
i := 1;
while DelayImportDirectory^.DLLName <> 0 do
begin
ShowMsgAlign(inttostr(i) + '. DelayImportDescriptor:;');
ShowMsgAlign('Attributes;' + inttohex(DelayImportDirectory^.Attr, 8), 3);
ShowMsgAlign('Dll name;' + inttohex(DelayImportDirectory^.DLLName, 8) + ' "' +
PChar(DWORD(PPE) + RVAtoOffset(DelayImportDirectory^.DLLName - POptionalHeader32^.ImageBase)) + '"', 3);
ShowMsgAlign('hMod;' + inttohex(DelayImportDirectory^.Hmod, 8), 3);
ShowMsgAlign('IAT;' + inttohex(DelayImportDirectory^.IAT, 8), 3);
ShowMsgAlign('INT;' + inttohex(DelayImportDirectory^.INT, 8), 3);
ShowMsgAlign('BoundIAT;' + inttohex(DelayImportDirectory^.BoundIAT, 8), 3);
ShowMsgAlign('UnloadIAT;' + inttohex(DelayImportDirectory^.UnloadIAT, 8), 3);
ShowMsgAlign('TimeDateStamp;' + inttohex(DelayImportDirectory^.TimeDateStamp, 8), 3);
ShowMsg('');
ShowMsg('Ordinal/Hint API name', 3);
ShowMsg('------------ ' + DupeString('-', 20), 3);
j := DWORD(PPE) + RVAtoOffset(DelayImportDirectory^.INT - POptionalHeader32^.ImageBase);
while PDWORD(j)^ <> 0 do
begin
PFuncName := PImageImportByName(DWORD(PPE) + RVAtoOffset(PDWORD(j)^ - POptionalHeader32^.ImageBase));
if PDWORD(j)^ and IMAGE_ORDINAL_FLAG32 = 0 then
begin
ShowMsgAlign('0x' + inttohex(PFuncName^.Hint, 4) + ';"' + PChar(@PFuncName^.Name) + '"', 3, 13);
end
else
begin
ShowMsgAlign('0x' + inttohex(PDWORD(j)^ and $7FFFFFFF, 4) + ';(Ordinal only)', 3, 13);
end;
inc(j, 4);
end;
ShowMsg('');
inc(i);
inc(DelayImportDirectory);
end;
end;
procedure TForm1.DumpBoundImportTable;
var
i, j: integer;
bid: DWORD;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'BoundImport Table' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_BOUND_IMPORT ].RVA = 0 then
begin
ShowMsg('BoundImport table not found!');
exit;
end;
PBoundImportDescriptor := PImageBoundImportDescriptor(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_BOUND_IMPORT].RVA));
bid := DWORD(PBoundImportDescriptor);
j := 1;
while PBoundImportDescriptor^.TimeDateStamp <> 0 do
begin
ShowMsgAlign(inttostr(j) + '. Bound Import Descriptor:;');
ShowMsgAlign('TimeDateStamp;' + inttohex(PBoundImportDescriptor^.TimeDateStamp, 8), 3);
ShowMsgAlign('Dll name;' + inttohex(PBoundImportDescriptor^.OffsetModuleName, 8) + ' "' +
PChar(bid + PBoundImportDescriptor^.OffsetModuleName) + '"', 3);
ShowMsgAlign('NumberOfModuleForwarderRefs;' + inttohex(PBoundImportDescriptor^.NumberOfModuleForwarderRefs, 8), 3);
for i := 0 to PBoundImportDescriptor^.NumberOfModuleForwarderRefs - 1 do
begin
inc(PBoundImportDescriptor);
ShowMsgAlign(inttostr(i) + '. TimeDateStamp;' + inttohex(PBoundImportDescriptor^.TimeDateStamp, 8),3);
ShowMsgAlign('Dll name;' + inttohex(PBoundImportDescriptor^.OffsetModuleName, 8) + ' "' +
PChar(DWORD(bid) + PBoundImportDescriptor^.OffsetModuleName) + '"', 6);
ShowMsgAlign('Resversed;' + inttohex(PBoundImportDescriptor^.NumberOfModuleForwarderRefs, 8), 6);
end;
ShowMsg('');
inc(j);
inc(PBoundImportDescriptor);
end;
end;
procedure TForm1.DumpCOMTable;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'COM Table' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR ].RVA = 0 then
begin
ShowMsg('COM table not found!');
exit;
end;
PCor20Header := PImageCor20Header(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR].RVA));
ShowMsgAlign('COM table Descriptor:;');
ShowMsgAlign('cb;' + inttohex(PCor20Header^.cb, 8), 3);
ShowMsgAlign('MajorRuntimeVersion;' + inttohex(PCor20Header^.MajorRuntimeVersion, 8), 3);
ShowMsgAlign('MinorRuntimeVersion;' + inttohex(PCor20Header^.MinorRuntimeVersion, 8), 3);
ShowMsgAlign('MetaData.VirtualAddress;' + inttohex(PCor20Header^.MetaData.VirtualAddress, 8), 3);
ShowMsgAlign('MetaData.Size;' + inttohex(PCor20Header^.MetaData.Size, 8), 3);
ShowMsgAlign('Flags;' + inttohex(PCor20Header^.Flags, 8), 3);
ShowMsgAlign('EntryPointToken;' + inttohex(PCor20Header^.EntryPointToken, 8), 3);
ShowMsg('');
ShowMsgAlign('Resources.VirtualAddress;' + inttohex(PCor20Header^.Resources.VirtualAddress, 8), 3);
ShowMsgAlign('Resources.Size;' + inttohex(PCor20Header^.Resources.Size, 8), 3);
ShowMsgAlign('StrongNameSignature.VirtualAddress;' + inttohex(PCor20Header^.StrongNameSignature.VirtualAddress, 8), 3);
ShowMsgAlign('StrongNameSignature.Size;' + inttohex(PCor20Header^.StrongNameSignature.Size, 8), 3);
ShowMsgAlign('CodeManagerTable.VirtualAddress;' + inttohex(PCor20Header^.CodeManagerTable.VirtualAddress, 8), 3);
ShowMsgAlign('CodeManagerTable.Size;' + inttohex(PCor20Header^.CodeManagerTable.Size, 8), 3);
ShowMsgAlign('VTableFixups.VirtualAddress;' + inttohex(PCor20Header^.VTableFixups.VirtualAddress, 8), 3);
ShowMsgAlign('VTableFixups.Size;' + inttohex(PCor20Header^.VTableFixups.Size, 8), 3);
ShowMsgAlign('ExportAddressTableJumps.VirtualAddress;' + inttohex(PCor20Header^.ExportAddressTableJumps.VirtualAddress, 8), 3);
ShowMsgAlign('ExportAddressTableJumps.Size;' + inttohex(PCor20Header^.ExportAddressTableJumps.Size, 8), 3);
ShowMsgAlign('ManagedNativeHeader.VirtualAddress;' + inttohex(PCor20Header^.ManagedNativeHeader.VirtualAddress, 8), 3);
ShowMsgAlign('ManagedNativeHeader.Size;' + inttohex(PCor20Header^.ManagedNativeHeader.Size, 8), 3);
end;
procedure TForm1.DumpCopyrightTable;
var
ptr: Pointer;
s: string;
begin
ShowMsg('');
ShowMsg(DupeString('=', 21) + 'Copyright Table' + DupeString('=', 21));
if DirectoryArray[IMAGE_DIRECTORY_ENTRY_ARCHITECTURE ].RVA = 0 then
begin
ShowMsg('Copyright table not found!');
exit;
end;
ptr := Pointer(DWORD(PPE) + RVAtoOffset(DirectoryArray[IMAGE_DIRECTORY_ENTRY_ARCHITECTURE].RVA));
s := copy(PChar(ptr), 0, DirectoryArray[IMAGE_DIRECTORY_ENTRY_ARCHITECTURE].Size);
ShowMsg('Copyright info:');
ShowMsg(s, 3);
end;
procedure TForm1.DoDump;
var
i: integer;
ScetionName: String[8];
begin
Caption := fn;
PPE := MStream.Memory;
if MStream.Size = 0 then
begin
Showmessage('0 file!');
exit;
end;
if (PImageDosHeader(PPE)^.e_magic <> $5A4D) then
begin
Showmessage('Invalid PE file!');
exit;
end;
if (PDWORD(Longint(PPE) + PImageDosHeader(PPE)^.e_lfanew)^ <> $4550) then
begin
Showmessage('Invalid PE file!');
exit;
end;
ShowMsgAlign('DosHeader e-magic;0x' + inttohex(PImageDosHeader(PPE)^.e_magic, 4));
ShowMsg(DupeString('-', 56));
PFileHeader := PImageFileHeader(Longint(PPE) + PImageDosHeader(PPE)^.e_lfanew + 4);
ShowMsgAlign('PE Signature;0x' + inttohex(PDWORD(Longint(PPE) + PImageDosHeader(PPE)^.e_lfanew)^, 4));
ShowMsgAlign('NumberOfSections;0x' + inttohex(PFileHeader^.NumberOfSections, 4));
ShowMsg(DupeString('-', 56));
POptionalHeader32 := PImageOptionalHeader32(DWORD(PFileHeader) + IMAGE_SIZEOF_FILE_HEADER);
ShowMsgAlign('OptionalHeader magic;0x' + inttohex(POptionalHeader32^.Magic, 4));
ShowMsgAlign('SectionAlignment;0x' + inttohex(POptionalHeader32^.SectionAlignment, 4));
ShowMsgAlign('FileAlignment;0x' + inttohex(POptionalHeader32^.FileAlignment, 4));
ShowMsgAlign('NUMBEROF_DIRECTORY_ENTRIES;0x' + inttohex(POptionalHeader32^.NumberOfRvaAndSizes, 4));
ShowMsg('');
PDataDirectory := PImageDataDirectory(@POptionalHeader32^.DataDirectory[0]);
ShowMsg('DataDirectory (0x10)' + DupeString(' ', 16) + 'RVA' + DupeString(' ', 9) + 'Size');
ShowMsg('------------------' + DupeString(' ', 16) + '----------' + DupeString(' ', 2) + '----------');
for i := 0 to IMAGE_NUMBEROF_DIRECTORY_ENTRIES - 1 do
begin
ShowMsgAlign(TableArray + ';0x' + inttohex(PDataDirectory^.VirtualAddress, 8) +
' 0x' + inttohex(PDataDirectory^.Size, 8));
DirectoryArray.RVA := PDataDirectory^.VirtualAddress;
DirectoryArray.Size := PDataDirectory^.Size;
inc(DWORD(PDataDirectory), SizeOf(TImageDataDirectory));
end;
ShowMsg(DupeString('-', 56));
PSectionHeader := PImageSectionHeader(DWORD(POptionalHeader32) + PFileHeader^.SizeOfOptionalHeader);
for i := 0 to PFileHeader^.NumberOfSections - 1 do
begin
ScetionName := Copy(PChar(@PSectionHeader^.Name), 1, 8);
ShowMsgAlign('Section Name;' + ScetionName);
ShowMsgAlign('VOffset;' + inttohex(PSectionHeader^.VirtualAddress, 8), 2);
ShowMsgAlign('VSize;' + inttohex(PSectionHeader^.Misc.VirtualSize, 8), 2);
ShowMsgAlign('ROffset(in file);' + inttohex(PSectionHeader^.PointerToRawData, 8), 2);
ShowMsgAlign('RSize (in file);' + inttohex(PSectionHeader^.SizeOfRawData, 8), 2);
ShowMsgAlign('Characteristics;' + inttohex(PSectionHeader^.Characteristics, 8), 2);
ShowMsg('');
inc(DWORD(PSectionHeader), IMAGE_SIZEOF_SECTION_HEADER);
end;
//--------------------------------------------------------------------------------
DumpImportTable;
//--------------------------------------------------------------------------------
DumpExportTable;
//--------------------------------------------------------------------------------
DumpRelocationTable;
//--------------------------------------------------------------------------------
DumpDelayImportTable;
//--------------------------------------------------------------------------------
DumpBoundImportTable;
//--------------------------------------------------------------------------------
DumpCOMTable;
//--------------------------------------------------------------------------------
DumpCopyrightTable;
end;
procedure TForm1.FormDestroy(Sender: TObject);
begin
MStream.Free;
end;
end.