亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? vmtutils.pas

?? SrcDecompiler is about creating a Delphi program decompiler. The program is written for Delphi 4 or
?? PAS
字號:
unit VMTUtils;

interface

uses
  SysUtils, Classes, TypInfo;

type
  EVMTError = class(Exception);

{ virtual methods methods }
function GetVirtualMethodCount(AClass: TClass): Integer;
function GetVirtualMethod(AClass: TClass; const Index: Integer): Pointer;
procedure SetVirtualMethod(AClass: TClass; const Index: Integer; const Method: Pointer);

{ dynamic methods methods }
type
  TDynamicIndexList = array[0..High(Word)] of Word;
  PDynamicIndexList = ^TDynamicIndexList;
  TDynamicAddressList = array[0..High(Word)] of pointer;
  PDynamicAddressList = ^TDynamicAddressList;

function GetDynamicMethodCount(AClass: TClass): Integer;
function GetDynamicIndexList(AClass: TClass): PDynamicIndexList;
function GetDynamicAddressList(AClass: TClass): PDynamicAddressList;
function HasDynamicMethod(AClass: TClass; Index: Integer): Boolean;

{ init table methods }
function GetInitTable(AClass: TClass): PTypeInfo;

{ field table methods }

type
  PFieldEntry = ^TFieldEntry;
  TFieldEntry = packed record
    OffSet: Integer;
    IDX: Word;
    Name: ShortString;
  end;

  PFieldClassTable = ^TFieldClassTable;
  TFieldClassTable = packed record
    Count: Smallint;
    Classes: array[0..8191] of ^TPersistentClass;
  end;

  PFieldTable = ^TFieldTable;
  TFieldTable = packed record
    EntryCount: Word;
    FieldClassTable: PFieldClassTable;
    FirstEntry: TFieldEntry;
   {Entries: array[1..65534] of TFieldEntry;}
  end;

function GetFieldTable(AClass: TClass): PFieldTable;

{ method table }

type
  PMethodEntry = ^TMethodEntry;
  TMethodEntry = packed record
    EntrySize: Word;
    Address: Pointer;
    Name: ShortString;
  end;

  PMethodTable = ^TMethodTable;
  TMethodTable = packed record
    Count: Word;
    FirstEntry: TMethodEntry;
   {Entries: array[1..65534] of TMethodEntry;}
  end;

function GetMethodTable(AClass: TClass): PMethodTable;
function GetMethodEntry(MethodTable: PMethodTable; Index: Integer): PMethodEntry;

{ class parent methods }
procedure SetClassParent(AClass: TClass; NewClassParent: TClass);
function GetClassParent(AClass: TClass): TClass;

function IsClass(Address: Pointer): Boolean;

implementation

uses
  Windows;

resourcestring
  SMemoryWriteError = 'Error writing VMT memory (%s).';
  STypeInfoError = 'Error reading type info.';

type
  PLongint= ^Longint;
  PPointer = ^Pointer;

function GetVirtualMethodCount(AClass: TClass): Integer;
var
  BeginVMT: Longint;
  EndVMT: Longint;
  TablePointer: Longint;
  I: Integer;
begin
  BeginVMT := Longint(AClass);

  // Scan the offset entries in the class table for the various fields,
  // namely vmtIntfTable, vmtAutoTable, ..., vmtDynamicTable
  // The last entry is always the vmtClassName, so stop once we got there
  //
  // This assumes that all these tables come after each other, basically:
  //P: Wrong: They don't come after each other, you have to scan them all.
  //
  //    First VMT table entry         (pointed to by vmtSelfPtr = -76)
  //        ...
  //    Last VMT table entry
  //    First IntfTable table entry   (pointed to by vmtIntfTable = -72)
  //        ...
  //    Last IntfTable table entry
  //    First AutoTable table entry   (pointed to by vmtAutoTable = - 68)
  //        ...
  //    Last AutoTable table entry
  // ...
  //    ClassName ShortString         (pointed to by vmtClassName = -44)
  //

  EndVMT := PLongint(LongInt(AClass) + vmtClassName)^;
  // Set iterator to first item behind VMT table pointer
  I := vmtSelfPtr + SizeOf(Pointer);
  repeat
    TablePointer := PLongint(Longint(AClass) + I)^;
    if (TablePointer <> 0) and (TablePointer >= BeginVMT) and
       (TablePointer < EndVMT) then
    begin
      EndVMT := Longint(TablePointer);
//      Break;
    end;
    Inc(I, SizeOf(Pointer));
  until I >= vmtClassName;

  //P: In case we don't have a class name?
{  if EndVMT = 0 then
    raise EVMTError.Create(SErrorRetrievingVmtCount);}

  Result := (EndVMT - BeginVMT) div SizeOf(Pointer);
end;

function GetVirtualMethod(AClass: TClass; const Index: Integer): Pointer;
begin
  Result := PPointer(Integer(AClass) + Index * SizeOf(Pointer))^;
end;

procedure SetVirtualMethod(AClass: TClass; const Index: Integer; const Method: Pointer);
var
  WrittenBytes: DWORD;
  PatchAddress: Pointer;
begin
  PatchAddress := PPointer(Integer(AClass) + Index * SizeOf(Pointer))^;

  //! StH: WriteProcessMemory IMO is not exactly the politically correct approach;
  // better VirtualProtect, direct patch, VirtualProtect
  if not WriteProcessMemory( GetCurrentProcess,
                             PatchAddress, @Method,
                             SizeOf(Pointer), WrittenBytes) then
  begin
    raise EVMTError.CreateFmt(SMemoryWriteError, [SysErrorMessage(GetLastError)]);
  end;

  if WrittenBytes <> SizeOf(Pointer) then
    raise EVMTError.CreateFmt(SMemoryWriteError, [IntToStr(WrittenBytes)]);

  // make sure that everything keeps working in a dual processor setting
  FlushInstructionCache(GetCurrentProcess, PatchAddress, SizeOf(Pointer));
end;

{ Dynamic method routines }

type
  TvmtDynamicTable = packed record
    Count: Word;
   {IndexList: array[1..Count] of Word;
    AddressList: array[1..Count] of Pointer;}
  end;

function GetDynamicMethodCount(AClass: TClass): Integer;
asm
       MOV     EAX, [EAX].vmtDynamicTable
       TEST    EAX, EAX
       JE      @@exit
       MOVZX   EAX, Word ptr [EAX]
@@exit:
end;

function GetDynamicIndexList(AClass: TClass): PDynamicIndexList;
asm
       MOV      EAX, [EAX].vmtDynamicTable
       ADD      EAX, 2
end;

function GetDynamicAddressList(AClass: TClass): PDynamicAddressList;
asm
       MOV      EAX, [EAX].vmtDynamicTable
       MOVZX    EDX, Word ptr [EAX]
       ADD      EAX, EDX
       ADD      EAX, EDX
       ADD      EAX, 2
end;

function HasDynamicMethod(AClass: TClass; Index: Integer): Boolean;
asm
        { ->    EAX     vmt of class            }
        {       DX      dynamic method index    }

        PUSH    EDI
        XCHG    EAX,EDX
        JMP     @@haveVMT
@@outerLoop:
        MOV     EDX,[EDX]
@@haveVMT:
        MOV     EDI,[EDX].vmtDynamicTable
        TEST    EDI,EDI
        JE      @@parent
        MOVZX   ECX,word ptr [EDI]
        PUSH    ECX
        ADD     EDI,2
        REPNE   SCASW
        JE      @@found
        POP     ECX
@@parent:
        MOV     EDX,[EDX].vmtParent
        TEST    EDX,EDX
        JNE     @@outerLoop
        MOV     EAX, 0
        JMP     @@exit

@@found:
        POP     EAX
        MOV     EAX, 1
@@exit:
        POP     EDI
end;

{ Interface table methods }

{ Init table methods }

function GetInitTable(AClass: TClass): PTypeInfo;
asm
       MOV    EAX, [EAX].vmtInitTable
end;

{ Field Table methods }

function GetFieldTable(AClass: TClass): PFieldTable;
asm
      MOV     EAX, [EAX].vmtFieldTable
end;

{ Method Table }

function GetMethodTable(AClass: TClass): PMethodTable;
asm
        MOV     EAX, [EAX].vmtMethodTable
end;

function GetMethodEntry(MethodTable: PMethodTable; Index: Integer): PMethodEntry;
begin
  Result := Pointer(Cardinal(MethodTable) +2);
  for Index := Index downto 1 do
    Inc(Cardinal(Result), Result^.EntrySize);
end;

{ Class Parent methods }

procedure SetClassParent(AClass: TClass; NewClassParent: TClass);
var
  WrittenBytes: DWORD;
  PatchAddress: Pointer;
begin
  PatchAddress := PPointer(Integer(AClass) + vmtParent)^;

  //! StH: WriteProcessMemory IMO is not exactly the politically correct approach;
  // better VirtualProtect, direct patch, VirtualProtect
  if not WriteProcessMemory( GetCurrentProcess,
                             PatchAddress, @NewClassParent,
                             SizeOf(Pointer), WrittenBytes) then
  begin
    raise EVMTError.CreateFmt(SMemoryWriteError, [SysErrorMessage(GetLastError)]);
  end;

  if WrittenBytes <> SizeOf(Pointer) then
    raise EVMTError.CreateFmt(SMemoryWriteError, [IntToStr(WrittenBytes)]);

  // make sure that everything keeps working in a dual processor setting
  FlushInstructionCache(GetCurrentProcess, PatchAddress, SizeOf(Pointer));
end;

function GetClassParent(AClass: TClass): TClass;
begin
  Result := TClass(PPointer(Integer(AClass) + vmtParent)^^);
end;

function IsClass(Address: Pointer): Boolean;
asm
        CMP     Address, Address.vmtSelfPtr
        JNZ     @False
        MOV     Result, True
        JMP     @Exit
@False:
        MOV     Result, False
@Exit:
end;

end.

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
欧美裸体一区二区三区| 成人激情文学综合网| 亚洲视频小说图片| 亚洲黄色性网站| 中文字幕人成不卡一区| 中文字幕的久久| 国产精品黄色在线观看| 亚洲视频在线观看一区| 亚洲日本va在线观看| 亚洲韩国一区二区三区| 日韩和的一区二区| 捆绑变态av一区二区三区| 精品一区二区三区免费观看 | 亚洲国产另类精品专区| 亚洲国产成人av| 蜜臀av亚洲一区中文字幕| 国内成+人亚洲+欧美+综合在线 | 99re这里只有精品视频首页| 成人精品一区二区三区四区| 色欧美片视频在线观看| 制服丝袜成人动漫| 久久综合九色综合久久久精品综合| 久久亚洲精品小早川怜子| 中文字幕一区二区在线播放| 亚洲福利视频导航| 精品午夜久久福利影院| 91亚洲精品一区二区乱码| 欧美高清性hdvideosex| 精品国产亚洲一区二区三区在线观看| 中文字幕免费在线观看视频一区| 亚洲影视在线观看| 久久se精品一区精品二区| 99国产精品久久久久久久久久 | 风间由美一区二区av101| 色婷婷综合久久久久中文| 欧美一区二区视频观看视频| 中文字幕av一区二区三区高| 日韩综合一区二区| 9色porny自拍视频一区二区| 91精品国产乱| 亚洲乱码国产乱码精品精的特点| 秋霞av亚洲一区二区三| 91浏览器入口在线观看| 久久美女艺术照精彩视频福利播放| 亚洲精品一二三四区| 国产很黄免费观看久久| 欧美精品九九99久久| 亚洲视频你懂的| 国产精品一区二区三区网站| 欧美午夜精品久久久久久超碰 | 亚洲一区二区四区蜜桃| 国产91露脸合集magnet| 日韩一区二区视频在线观看| 一区二区三区不卡视频| 不卡av免费在线观看| 久久日韩精品一区二区五区| 亚洲高清视频的网址| www.av精品| 国产精品嫩草99a| 老汉av免费一区二区三区| 在线看不卡av| 亚洲色图欧美偷拍| 国产91精品久久久久久久网曝门| 日韩欧美在线网站| 国产99久久久精品| 日韩精品一区二区三区视频播放| 一区av在线播放| 欧洲一区在线观看| 国产精品国产三级国产普通话99| 国产一本一道久久香蕉| 精品国产91久久久久久久妲己 | 日本在线播放一区二区三区| 欧美亚洲综合色| 一区二区三区在线观看视频| bt7086福利一区国产| 国产精品久久久久久久久免费樱桃 | 日韩欧美www| 久久66热re国产| 精品国产乱码久久| 国产一区二区三区精品视频 | 成人h动漫精品| 国产精品嫩草99a| 91在线视频官网| 一区二区三区四区蜜桃| 欧美午夜精品免费| 免费看黄色91| 久久日一线二线三线suv| 国产成人自拍网| 欧美高清在线视频| 91亚洲精品久久久蜜桃| 亚洲第一会所有码转帖| 91精品国产欧美一区二区成人| 免费欧美日韩国产三级电影| 精品国产乱码久久久久久老虎 | 色婷婷综合五月| 国产麻豆日韩欧美久久| 中文字幕中文在线不卡住| 日本高清成人免费播放| 麻豆精品一区二区三区| 国产精品网曝门| 欧美日韩一区在线观看| 免费一级片91| 中文字幕中文字幕一区二区| 欧美性生活久久| 国产一区二区免费看| 亚洲欧洲综合另类| 日韩欧美成人激情| 99精品欧美一区二区蜜桃免费| 午夜视频一区二区| 国产女人aaa级久久久级| 精品视频一区三区九区| 国产精品一区一区三区| 亚洲va欧美va人人爽午夜| 久久久久久免费网| 欧美日韩成人一区二区| 成人激情综合网站| 青草av.久久免费一区| 亚洲欧美怡红院| 精品久久人人做人人爽| 欧美特级限制片免费在线观看| 国产精品一线二线三线精华| 亚洲成人激情综合网| 亚洲黄色性网站| 久久久精品国产免费观看同学| 欧美日韩国产在线播放网站| av在线不卡电影| 国产乱一区二区| 日韩电影免费在线| 亚洲激情自拍视频| 日本一区二区动态图| 久久这里只有精品视频网| 欧美另类久久久品| 在线观看一区日韩| 成人免费电影视频| 国产精品亚洲第一区在线暖暖韩国 | 亚洲另类在线一区| 中文字幕乱码一区二区免费| 日韩久久免费av| 欧美精品高清视频| 欧美亚洲综合一区| 在线看国产日韩| 欧美影院一区二区| 色婷婷久久一区二区三区麻豆| 成人免费视频视频在线观看免费| 国产麻豆精品一区二区| 麻豆视频观看网址久久| 琪琪久久久久日韩精品| 午夜电影网亚洲视频| 亚洲成人av一区二区| 亚洲精品国产高清久久伦理二区| 中文字幕制服丝袜一区二区三区 | 日韩欧美国产一区在线观看| 欧美精品日日鲁夜夜添| 91精品久久久久久久91蜜桃| 欧美午夜精品一区二区蜜桃| 欧美日韩国产乱码电影| 69久久夜色精品国产69蝌蚪网| 欧美电影在线免费观看| 日韩欧美区一区二| 久久看人人爽人人| 日本一区二区视频在线| 一区二区中文视频| 一级日本不卡的影视| 日本欧美韩国一区三区| 久久se精品一区二区| 国产一区二区在线影院| 成+人+亚洲+综合天堂| 91亚洲国产成人精品一区二区三| 91视频www| 欧美日韩国产高清一区二区| 欧美一级免费观看| 国产情人综合久久777777| 亚洲图片激情小说| 亚洲r级在线视频| 国产精品影视在线观看| 99国产精品久久久久久久久久| 在线观看日韩精品| 日韩免费成人网| 国产精品护士白丝一区av| 午夜在线电影亚洲一区| 国内成人精品2018免费看| 色综合久久99| 精品日产卡一卡二卡麻豆| 国产精品国产三级国产aⅴ无密码 国产精品国产三级国产aⅴ原创 | 欧美人体做爰大胆视频| 久久久精品免费免费| 一区二区三区在线免费观看| 另类欧美日韩国产在线| 91麻豆福利精品推荐| 日韩欧美一级二级| 中文字幕一区二区在线观看| 日韩成人一区二区| 欧美日韩国产一二三| 国产精品乱人伦一区二区| 亚洲国产成人av网| 99久久免费视频.com| 久久久久久电影| 天堂久久久久va久久久久| 99久久亚洲一区二区三区青草| 日韩欧美中文字幕公布|