dumpmethods.pp 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. program DumpMethods;
  2. {$mode objfpc}{$H+}
  3. uses
  4. Classes, SysUtils;
  5. const
  6. VMT_COUNT = 100;
  7. type
  8. TMethodNameTableEntry = packed record
  9. Name: PShortstring;
  10. Addr: Pointer;
  11. end;
  12. TMethodNameTable = packed record
  13. Count: DWord;
  14. Entries: packed array[0..9999999] of TMethodNameTableEntry;
  15. end;
  16. PMethodNameTable = ^TMethodNameTable;
  17. TPointerArray = packed array[0..9999999] of Pointer;
  18. PPointerArray = ^TPointerArray;
  19. {$M+}
  20. TMyTest = class(TObject)
  21. // published
  22. procedure P1; virtual;
  23. procedure P2; virtual;
  24. end;
  25. {$M-}
  26. TMyTest2 = class(TMyTest)
  27. // published
  28. procedure P2; override;
  29. procedure P3; virtual;
  30. end;
  31. TMyPersistent = class(TPersistent)
  32. // published
  33. procedure P1; virtual;
  34. procedure P2; virtual;
  35. end;
  36. procedure TMyTest.P1;
  37. begin
  38. end;
  39. procedure TMyTest.P2;
  40. begin
  41. end;
  42. procedure TMyTest2.P2;
  43. begin
  44. end;
  45. procedure TMyTest2.P3;
  46. begin
  47. end;
  48. procedure TMyPersistent.P1;
  49. begin
  50. end;
  51. procedure TMyPersistent.P2;
  52. begin
  53. end;
  54. procedure DumpClass(AClass: TClass);
  55. var
  56. Cvmt: PPointerArray;
  57. Cmnt: PMethodNameTable;
  58. Indent: String;
  59. n, idx: Integer;
  60. SearchAddr: Pointer;
  61. begin
  62. WriteLn('---------------------------------------------');
  63. WriteLn('Dump of ', AClass.ClassName);
  64. WriteLn('---------------------------------------------');
  65. Indent := '';
  66. while AClass <> nil do
  67. begin
  68. WriteLn(Indent, 'Processing ', AClass.Classname);
  69. Indent := Indent + ' ';
  70. Cmnt := PPointer(Pointer(AClass) + vmtMethodTable)^;
  71. if Cmnt <> nil
  72. then begin
  73. WriteLn(Indent, 'Method count: ', IntToStr(Cmnt^.Count));
  74. Cvmt := Pointer(AClass) + vmtMethodStart;
  75. for n := 0 to Cmnt^.Count - 1 do
  76. begin
  77. WriteLn(Indent, 'Search: ', Cmnt^.Entries[n].Name^);
  78. SearchAddr := Cmnt^.Entries[n].Addr;
  79. for idx := 0 to VMT_COUNT - 1 do
  80. begin
  81. if Cvmt^[idx] = SearchAddr
  82. then begin
  83. WriteLn(Indent, 'Found at index: ', IntToStr(idx));
  84. Break;
  85. end;
  86. if idx = VMT_COUNT - 1
  87. then begin
  88. WriteLn('[WARNING] VMT entry "', Cmnt^.Entries[n].Name^, '" not found in "', AClass.ClassName, '"');
  89. Break;
  90. end;
  91. end;
  92. end;
  93. end;
  94. AClass := AClass.ClassParent;
  95. end;
  96. end;
  97. begin
  98. DumpClass(TMyTest);
  99. DumpClass(TMyTest2);
  100. DumpClass(TPersistent);
  101. DumpClass(TMyPersistent);
  102. end.