{----------------------------------------------------------------------------- The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/MPL-1.1.html Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is: JvHook.PAS, released on 2002-07-04. The Initial Developers of the Original Code are: Fedor Koshevnikov, Igor Pavluk and Serge Korolev Copyright (c) 1997, 1998 Fedor Koshevnikov, Igor Pavluk and Serge Korolev Copyright (c) 2001,2002 SGB Software All Rights Reserved. Last Modified: 2002-07-04 You may retrieve the latest version of this file at the Project JEDI's JVCL home page, located at http://jvcl.sourceforge.net Known Issues: (rom) this file should definitely be merged with JclSysUtils.pas -----------------------------------------------------------------------------} {$I JVCL.INC} unit JvHook; interface uses Windows, SysUtils; {, JvComponent} type PClass = ^TClass; function GetVirtualMethodCount(AClass: TClass): Integer; function GetVirtualMethodAddress(AClass: TClass; AIndex: Integer): Pointer; function SetVirtualMethodAddress(AClass: TClass; AIndex: Integer; NewAddress: Pointer): Pointer; function FindVirtualMethodIndex(AClass: TClass; MethodAddr: Pointer): Integer; implementation { SetVirtualMethodAddress procedure. Destroy destructor has index 0, first user defined virtual method has index 1. } type PPointer = ^Pointer; // (rom) copied from JCL 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 // After the last virtual method there is one of these entries. 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 EndVMT := Longint(TablePointer); Inc(I, SizeOf(Pointer)); until I >= vmtClassName; Result := (EndVMT - BeginVMT) div SizeOf(Pointer); end; function GetVirtualMethodAddress(AClass: TClass; AIndex: Integer): Pointer; var Table: PPointer; begin Table := PPointer(AClass); Inc(Table, AIndex - 1); Result := Table^; end; function SetVirtualMethodAddress(AClass: TClass; AIndex: Integer; NewAddress: Pointer): Pointer; const PageSize = SizeOf(Pointer); var Table: PPointer; SaveFlag: DWORD; begin Table := PPointer(AClass); Inc(Table, AIndex - 1); Result := Table^; if VirtualProtect(Table, PageSize, PAGE_EXECUTE_READWRITE, @SaveFlag) then try Table^ := NewAddress; finally VirtualProtect(Table, PageSize, SaveFlag, @SaveFlag); end; end; // (rom) reimplemented using GetVirtualMethodCount function FindVirtualMethodIndex(AClass: TClass; MethodAddr: Pointer): Integer; var I: Integer; begin Result := -1; // (rom) API change! for I := 0 to GetVirtualMethodCount(AClass)-1 do if GetVirtualMethodAddress(AClass, Result) = MethodAddr then begin Result := I; Break; end; end; end.