-
Notifications
You must be signed in to change notification settings - Fork 4
/
CodeRedirect.pas
228 lines (198 loc) · 6.11 KB
/
CodeRedirect.pas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
{$WEAKPACKAGEUNIT ON}
unit CodeRedirect;
interface
uses
System.SysUtils;
type
TInstruction = class abstract
const
Default_OffSetLimit = 8192;
public
class function GetAddress(aIP: Pointer; aOffSet: NativeInt = 0): Pointer; overload;
class function GetAddress(aIP: pointer; aSignature: array of byte; OffsetLimit:
Integer = Default_OffSetLimit): Pointer; overload;
class function GetAddressOfCall(aIP: pointer): Pointer; overload;
class function GetAddressOfCall(aIP: pointer; aSignature: array of byte;
OffsetLimit: Integer = Default_OffSetLimit): Pointer; overload;
class procedure Patch(aAddr: Pointer; aPatchValues: array of Byte);
end;
TCodeRedirect = class(TObject)
const
Default_OffSetLimit = 8192;
private
FOriginalCode: TBytes;
FSrcAddr: Pointer;
FPatchCode: TBytes;
public
constructor CreateWithMethod(const aOldMethod, aNewMethod: Pointer); deprecated
'Use https://github.com/MahdiSafsafi/DDetours';
constructor Create(const aSrcAddr: Pointer; const aPatchCode: array of byte);
procedure BeforeDestruction; override;
procedure StopPatch;
procedure StartPatch;
end;
function IsRuntimePackaged: boolean;
function RuntimeArchitecture: TOSVersion.TArchitecture;
implementation
uses
Winapi.Windows;
function IsRuntimePackaged: boolean;
begin
Result := FindClassHInstance(TObject) <> MainInstance;
end;
function RuntimeArchitecture: TOSVersion.TArchitecture;
begin
{$ifdef Win32}Exit(TOSVersion.TArchitecture.arIntelX86);{$endif}
{$ifdef Win64}Exit(TOSVersion.TArchitecture.arIntelX64);{$endif}
raise ENotSupportedException.Create('Platform not supported');
end;
class function TInstruction.GetAddress(aIP: Pointer; aOffSet: NativeInt = 0):
Pointer;
type
PAbsoluteIndirectJmp = ^TAbsoluteIndirectJmp;
TAbsoluteIndirectJmp = packed record
OpCode: Word; //$FF25(Jmp, FF /4)
Addr: Cardinal;
end;
var J: PAbsoluteIndirectJmp;
{$ifdef Win32}
type
PWin9xDebugThunk = ^TWin9xDebugThunk;
TWin9xDebugThunk = packed record
PUSH: Byte;
Addr: Pointer;
JMP: Byte;
Offset: Integer;
end;
function IsWin9xDebugThunk(AAddr: Pointer): Boolean;
begin
Result := (AAddr <> nil) and
(PWin9xDebugThunk(AAddr).PUSH = $68) and
(PWin9xDebugThunk(AAddr).JMP = $E9);
end;
begin
if aIP = nil then Exit(aIP);
if (Win32Platform <> VER_PLATFORM_WIN32_NT) and IsWin9xDebugThunk(aIP) then
aIP := PWin9xDebugThunk(aIP).Addr;
J := PAbsoluteIndirectJmp(aIP);
if (J.OpCode = $25FF) then
Result := PPointer(J.Addr)^
else
Result := aIP;
Result := PByte(Result) + aOffset;
end;
{$endif}
{$ifdef Win64}
begin
if aIP = nil then Exit(aIP);
J := PAbsoluteIndirectJmp(aIP);
if (J.OpCode = $25FF) then
Result := PPointer(NativeUInt(aIP) + J.Addr + 6{Instruction Size})^
else
Result := aIP;
Result := PByte(Result) + aOffset;
end;
{$endif}
class function TInstruction.GetAddress(aIP: pointer; aSignature: array of byte;
OffsetLimit: Integer = Default_OffSetLimit): Pointer;
var P: ^NativeInt;
i: integer;
bFound: boolean;
begin
i := 0;
P := TInstruction.GetAddress(aIP);
bFound := True;
while not CompareMem(P, @aSignature, Length(aSignature)) do begin
Inc(NativeUInt(P));
Inc(i);
if (i > OffsetLimit) then begin
bFound := False;
Break;
end;
end;
if bFound then
Result := P
else
Result := nil;
end;
class function TInstruction.GetAddressOfCall(aIP: pointer): Pointer;
var P: PByteArray;
begin
P := aIP;
if Assigned(P) then
Result := Pointer(NativeInt(P) + 5{Instruction Size} + PInteger(@P[1])^)
else
Result := nil;
end;
class function TInstruction.GetAddressOfCall(aIP: pointer; aSignature: array of
byte; OffsetLimit: Integer = Default_OffSetLimit): Pointer;
var P: PByteArray;
begin
P := TInstruction.GetAddress(aIP, aSignature, OffsetLimit);
if Assigned(P) then
Result := Pointer(NativeInt(P) + 5{Instruction Size} + PInteger(@P[1])^)
else
Result := nil;
end;
class procedure TInstruction.Patch(aAddr: Pointer; aPatchValues: array of Byte);
var P: Cardinal;
n: Integer;
begin
n := Length(aPatchValues);
if (n > 0) and VirtualProtect(aAddr, n, PAGE_EXECUTE_READWRITE, P) then begin
Move(aPatchValues[0], aAddr^, n);
VirtualProtect(aAddr, n, P, @P);
FlushInstructionCache(GetCurrentProcess, aAddr, n);
end;
end;
procedure TCodeRedirect.BeforeDestruction;
begin
inherited;
StopPatch;
end;
constructor TCodeRedirect.Create(const aSrcAddr: Pointer; const aPatchCode:
array of byte);
begin
inherited Create;
FSrcAddr := aSrcAddr;
SetLength(FPatchCode, Length(aPatchCode));
Move(aPatchCode[0], FPatchCode[0], Length(aPatchCode));
StartPatch;
end;
procedure TCodeRedirect.StopPatch;
var n: NativeUInt;
begin
if not WriteProcessMemory(GetCurrentProcess, FSrcAddr, @FOriginalCode[0], Length(FOriginalCode), n) or (NativeInt(n) <> Length(FOriginalCode)) then
RaiseLastOSError;
end;
constructor TCodeRedirect.CreateWithMethod(const aOldMethod, aNewMethod:
Pointer);
var P: pointer;
B: TBytes;
O: PInteger;
begin
if (aOldMethod = nil) or (aNewMethod = nil) then
raise Exception.Create('Unknown method address');
SetLength(B, 1{Jmp Instruction Size} + 4{Address size});
P := TInstruction.GetAddress(aOldMethod);
B[0] := $E9; // Jmp instruction
O := @B[1];
O^ := Integer(NativeInt(aNewMethod) - (NativeInt(P) + NativeInt(Length(B))));
Create(P, B);
end;
procedure TCodeRedirect.StartPatch;
var OldProtect: Cardinal;
n: Integer;
begin
n := Length(FPatchCode);
if VirtualProtect(FSrcAddr, n, PAGE_EXECUTE_READWRITE, OldProtect) then begin
// Store original code
SetLength(FOriginalCode, n);
Move(FSrcAddr^, FOriginalCode[0], n);
// Patch original code
Move(FPatchCode[0], FSrcAddr^, n);
VirtualProtect(FSrcAddr, n, OldProtect, @OldProtect);
FlushInstructionCache(GetCurrentProcess, FSrcAddr, n);
end;
end;
end.