mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-12 02:08:09 +02:00
124 lines
2.4 KiB
ObjectPascal
124 lines
2.4 KiB
ObjectPascal
program DumpMethods;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
uses
|
|
Classes, SysUtils;
|
|
|
|
const
|
|
VMT_COUNT = 100;
|
|
ITEM_COUNT = 1000;
|
|
|
|
type
|
|
TMethodNameTableEntry = packed record
|
|
Name: PShortstring;
|
|
Addr: Pointer;
|
|
end;
|
|
|
|
TMethodNameTable = packed record
|
|
Count: DWord;
|
|
Entries: packed array[0..ITEM_COUNT-1] of TMethodNameTableEntry;
|
|
end;
|
|
PMethodNameTable = ^TMethodNameTable;
|
|
|
|
TPointerArray = packed array[0..ITEM_COUNT-1] of Pointer;
|
|
PPointerArray = ^TPointerArray;
|
|
|
|
{$M+}
|
|
TMyTest = class(TObject)
|
|
// published
|
|
procedure P1; virtual;
|
|
procedure P2; virtual;
|
|
end;
|
|
{$M-}
|
|
|
|
TMyTest2 = class(TMyTest)
|
|
// published
|
|
procedure P2; override;
|
|
procedure P3; virtual;
|
|
end;
|
|
|
|
TMyPersistent = class(TPersistent)
|
|
// published
|
|
procedure P1; virtual;
|
|
procedure P2; virtual;
|
|
end;
|
|
|
|
procedure TMyTest.P1;
|
|
begin
|
|
end;
|
|
|
|
procedure TMyTest.P2;
|
|
begin
|
|
end;
|
|
|
|
procedure TMyTest2.P2;
|
|
begin
|
|
end;
|
|
|
|
procedure TMyTest2.P3;
|
|
begin
|
|
end;
|
|
|
|
procedure TMyPersistent.P1;
|
|
begin
|
|
end;
|
|
|
|
procedure TMyPersistent.P2;
|
|
begin
|
|
end;
|
|
|
|
procedure DumpClass(AClass: TClass);
|
|
var
|
|
Cvmt: PPointerArray;
|
|
Cmnt: PMethodNameTable;
|
|
Indent: String;
|
|
n, idx: Integer;
|
|
SearchAddr: Pointer;
|
|
begin
|
|
WriteLn('---------------------------------------------');
|
|
WriteLn('Dump of ', AClass.ClassName);
|
|
WriteLn('---------------------------------------------');
|
|
Indent := '';
|
|
while AClass <> nil do
|
|
begin
|
|
WriteLn(Indent, 'Processing ', AClass.Classname);
|
|
Indent := Indent + ' ';
|
|
Cmnt := PPointer(Pointer(AClass) + vmtMethodTable)^;
|
|
if Cmnt <> nil
|
|
then begin
|
|
WriteLn(Indent, 'Method count: ', IntToStr(Cmnt^.Count));
|
|
|
|
Cvmt := Pointer(AClass) + vmtMethodStart;
|
|
|
|
for n := 0 to Cmnt^.Count - 1 do
|
|
begin
|
|
WriteLn(Indent, 'Search: ', Cmnt^.Entries[n].Name^);
|
|
|
|
SearchAddr := Cmnt^.Entries[n].Addr;
|
|
for idx := 0 to VMT_COUNT - 1 do
|
|
begin
|
|
if Cvmt^[idx] = SearchAddr
|
|
then begin
|
|
WriteLn(Indent, 'Found at index: ', IntToStr(idx));
|
|
Break;
|
|
end;
|
|
if idx = VMT_COUNT - 1
|
|
then begin
|
|
WriteLn('[WARNING] VMT entry "', Cmnt^.Entries[n].Name^, '" not found in "', AClass.ClassName, '"');
|
|
Break;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
AClass := AClass.ClassParent;
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
DumpClass(TMyTest);
|
|
DumpClass(TMyTest2);
|
|
DumpClass(TPersistent);
|
|
DumpClass(TMyPersistent);
|
|
end.
|