VirtualBox

source: vbox/trunk/src/VBox/VMM/PATM/PATM.cpp@ 31604

Last change on this file since 31604 was 31438, checked in by vboxsync, 14 years ago

patmR3HandleDirtyInstr: wrong variable.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 247.5 KB
Line 
1/* $Id: PATM.cpp 31438 2010-08-06 12:19:13Z vboxsync $ */
2/** @file
3 * PATM - Dynamic Guest OS Patching Manager
4 *
5 * NOTE: Never ever reuse patch memory!!
6 */
7
8/*
9 * Copyright (C) 2006-2007 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20/*******************************************************************************
21* Header Files *
22*******************************************************************************/
23#define LOG_GROUP LOG_GROUP_PATM
24#include <VBox/patm.h>
25#include <VBox/stam.h>
26#include <VBox/pgm.h>
27#include <VBox/cpum.h>
28#include <VBox/cpumdis.h>
29#include <VBox/iom.h>
30#include <VBox/mm.h>
31#include <VBox/ssm.h>
32#include <VBox/trpm.h>
33#include <VBox/cfgm.h>
34#include <VBox/param.h>
35#include <VBox/selm.h>
36#include <iprt/avl.h>
37#include "PATMInternal.h"
38#include "PATMPatch.h"
39#include <VBox/vm.h>
40#include <VBox/csam.h>
41#include <VBox/dbg.h>
42#include <VBox/err.h>
43#include <VBox/log.h>
44#include <iprt/assert.h>
45#include <iprt/asm.h>
46#include <VBox/dis.h>
47#include <VBox/disopcode.h>
48#include <include/internal/pgm.h>
49
50#include <iprt/string.h>
51#include "PATMA.h"
52
53//#define PATM_REMOVE_PATCH_ON_TOO_MANY_TRAPS
54//#define PATM_DISABLE_ALL
55
56/*******************************************************************************
57* Internal Functions *
58*******************************************************************************/
59
60static int patmDisableUnusablePatch(PVM pVM, RTRCPTR pInstrGC, RTRCPTR pConflictAddr, PPATCHINFO pPatch);
61static int patmActivateInt3Patch(PVM pVM, PPATCHINFO pPatch);
62static int patmDeactivateInt3Patch(PVM pVM, PPATCHINFO pPatch);
63
64#ifdef LOG_ENABLED // keep gcc quiet
65static bool patmIsCommonIDTHandlerPatch(PVM pVM, RTRCPTR pInstrGC);
66#endif
67#ifdef VBOX_WITH_STATISTICS
68static const char *PATMPatchType(PVM pVM, PPATCHINFO pPatch);
69static void patmResetStat(PVM pVM, void *pvSample);
70static void patmPrintStat(PVM pVM, void *pvSample, char *pszBuf, size_t cchBuf);
71#endif
72
73#define patmPatchHCPtr2PatchGCPtr(pVM, pHC) (pVM->patm.s.pPatchMemGC + (pHC - pVM->patm.s.pPatchMemHC))
74#define patmPatchGCPtr2PatchHCPtr(pVM, pGC) (pVM->patm.s.pPatchMemHC + (pGC - pVM->patm.s.pPatchMemGC))
75
76static int patmReinit(PVM pVM);
77static DECLCALLBACK(int) RelocatePatches(PAVLOU32NODECORE pNode, void *pParam);
78
79#ifdef VBOX_WITH_DEBUGGER
80static DECLCALLBACK(int) DisableAllPatches(PAVLOU32NODECORE pNode, void *pVM);
81static DECLCALLBACK(int) patmr3CmdOn(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
82static DECLCALLBACK(int) patmr3CmdOff(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
83
84/** Command descriptors. */
85static const DBGCCMD g_aCmds[] =
86{
87 /* pszCmd, cArgsMin, cArgsMax, paArgDesc, cArgDescs, pResultDesc, fFlags, pfnHandler pszSyntax, ....pszDescription */
88 { "patmon", 0, 0, NULL, 0, NULL, 0, patmr3CmdOn, "", "Enable patching." },
89 { "patmoff", 0, 0, NULL, 0, NULL, 0, patmr3CmdOff, "", "Disable patching." },
90};
91#endif
92
93/* Don't want to break saved states, so put it here as a global variable. */
94static unsigned int cIDTHandlersDisabled = 0;
95
96/**
97 * Initializes the PATM.
98 *
99 * @returns VBox status code.
100 * @param pVM The VM to operate on.
101 */
102VMMR3DECL(int) PATMR3Init(PVM pVM)
103{
104 int rc;
105
106 Log(("PATMR3Init: Patch record size %d\n", sizeof(PATCHINFO)));
107
108 /* These values can't change as they are hardcoded in patch code (old saved states!) */
109 AssertCompile(VMCPU_FF_TIMER == RT_BIT_32(2));
110 AssertCompile(VM_FF_REQUEST == VMCPU_FF_REQUEST);
111 AssertCompile(VMCPU_FF_INTERRUPT_APIC == RT_BIT_32(0));
112 AssertCompile(VMCPU_FF_INTERRUPT_PIC == RT_BIT_32(1));
113
114 AssertReleaseMsg(PATMInterruptFlag == (VMCPU_FF_INTERRUPT_APIC | VMCPU_FF_INTERRUPT_PIC | VMCPU_FF_TIMER | VMCPU_FF_REQUEST),
115 ("Interrupt flags out of sync!! PATMInterruptFlag=%#x expected %#x. broken assembler?\n", PATMInterruptFlag, VMCPU_FF_INTERRUPT_APIC | VMCPU_FF_INTERRUPT_PIC | VMCPU_FF_TIMER | VMCPU_FF_REQUEST));
116
117 /* Allocate patch memory and GC patch state memory. */
118 pVM->patm.s.cbPatchMem = PATCH_MEMORY_SIZE;
119 /* Add another page in case the generated code is much larger than expected. */
120 /** @todo bad safety precaution */
121 rc = MMR3HyperAllocOnceNoRel(pVM, PATCH_MEMORY_SIZE + PAGE_SIZE + PATM_STACK_TOTAL_SIZE + PAGE_SIZE + PATM_STAT_MEMSIZE, PAGE_SIZE, MM_TAG_PATM, (void **)&pVM->patm.s.pPatchMemHC);
122 if (RT_FAILURE(rc))
123 {
124 Log(("MMHyperAlloc failed with %Rrc\n", rc));
125 return rc;
126 }
127 pVM->patm.s.pPatchMemGC = MMHyperR3ToRC(pVM, pVM->patm.s.pPatchMemHC);
128
129 /* PATM stack page for call instruction execution. (2 parts: one for our private stack and one to store the original return address */
130 pVM->patm.s.pGCStackHC = (RTRCPTR *)(pVM->patm.s.pPatchMemHC + PATCH_MEMORY_SIZE + PAGE_SIZE);
131 pVM->patm.s.pGCStackGC = MMHyperR3ToRC(pVM, pVM->patm.s.pGCStackHC);
132
133 /*
134 * Hypervisor memory for GC status data (read/write)
135 *
136 * Note1: This is non-critical data; if trashed by the guest, then it will only cause problems for itself
137 * Note2: This doesn't really belong here, but we need access to it for relocation purposes
138 *
139 */
140 Assert(sizeof(PATMGCSTATE) < PAGE_SIZE); /* Note: hardcoded dependencies on this exist. */
141 pVM->patm.s.pGCStateHC = (PPATMGCSTATE)((uint8_t *)pVM->patm.s.pGCStackHC + PATM_STACK_TOTAL_SIZE);
142 pVM->patm.s.pGCStateGC = MMHyperR3ToRC(pVM, pVM->patm.s.pGCStateHC);
143
144 /* Hypervisor memory for patch statistics */
145 pVM->patm.s.pStatsHC = (PSTAMRATIOU32)((uint8_t *)pVM->patm.s.pGCStateHC + PAGE_SIZE);
146 pVM->patm.s.pStatsGC = MMHyperR3ToRC(pVM, pVM->patm.s.pStatsHC);
147
148 /* Memory for patch lookup trees. */
149 rc = MMHyperAlloc(pVM, sizeof(*pVM->patm.s.PatchLookupTreeHC), 0, MM_TAG_PATM, (void **)&pVM->patm.s.PatchLookupTreeHC);
150 AssertRCReturn(rc, rc);
151 pVM->patm.s.PatchLookupTreeGC = MMHyperR3ToRC(pVM, pVM->patm.s.PatchLookupTreeHC);
152
153#ifdef RT_ARCH_AMD64 /* see patmReinit(). */
154 /* Check CFGM option. */
155 rc = CFGMR3QueryBool(CFGMR3GetRoot(pVM), "PATMEnabled", &pVM->fPATMEnabled);
156 if (RT_FAILURE(rc))
157# ifdef PATM_DISABLE_ALL
158 pVM->fPATMEnabled = false;
159# else
160 pVM->fPATMEnabled = true;
161# endif
162#endif
163
164 rc = patmReinit(pVM);
165 AssertRC(rc);
166 if (RT_FAILURE(rc))
167 return rc;
168
169 /*
170 * Register save and load state notificators.
171 */
172 rc = SSMR3RegisterInternal(pVM, "PATM", 0, PATM_SSM_VERSION, sizeof(pVM->patm.s) + PATCH_MEMORY_SIZE + PAGE_SIZE + PATM_STACK_TOTAL_SIZE + PAGE_SIZE,
173 NULL, NULL, NULL,
174 NULL, patmR3Save, NULL,
175 NULL, patmR3Load, NULL);
176 AssertRCReturn(rc, rc);
177
178#ifdef VBOX_WITH_DEBUGGER
179 /*
180 * Debugger commands.
181 */
182 static bool s_fRegisteredCmds = false;
183 if (!s_fRegisteredCmds)
184 {
185 int rc2 = DBGCRegisterCommands(&g_aCmds[0], RT_ELEMENTS(g_aCmds));
186 if (RT_SUCCESS(rc2))
187 s_fRegisteredCmds = true;
188 }
189#endif
190
191#ifdef VBOX_WITH_STATISTICS
192 STAM_REG(pVM, &pVM->patm.s.StatNrOpcodeRead, STAMTYPE_COUNTER, "/PATM/OpcodeBytesRead", STAMUNIT_OCCURENCES, "The number of opcode bytes read by the recompiler.");
193 STAM_REG(pVM, &pVM->patm.s.StatPATMMemoryUsed,STAMTYPE_COUNTER, "/PATM/MemoryUsed", STAMUNIT_OCCURENCES, "The amount of hypervisor heap used for patches.");
194 STAM_REG(pVM, &pVM->patm.s.StatDisabled, STAMTYPE_COUNTER, "/PATM/Patch/Disabled", STAMUNIT_OCCURENCES, "Number of times patches were disabled.");
195 STAM_REG(pVM, &pVM->patm.s.StatEnabled, STAMTYPE_COUNTER, "/PATM/Patch/Enabled", STAMUNIT_OCCURENCES, "Number of times patches were enabled.");
196 STAM_REG(pVM, &pVM->patm.s.StatDirty, STAMTYPE_COUNTER, "/PATM/Patch/Dirty", STAMUNIT_OCCURENCES, "Number of times patches were marked dirty.");
197 STAM_REG(pVM, &pVM->patm.s.StatUnusable, STAMTYPE_COUNTER, "/PATM/Patch/Unusable", STAMUNIT_OCCURENCES, "Number of unusable patches (conflicts).");
198 STAM_REG(pVM, &pVM->patm.s.StatInstalled, STAMTYPE_COUNTER, "/PATM/Patch/Installed", STAMUNIT_OCCURENCES, "Number of installed patches.");
199 STAM_REG(pVM, &pVM->patm.s.StatInt3Callable, STAMTYPE_COUNTER, "/PATM/Patch/Int3Callable", STAMUNIT_OCCURENCES, "Number of cli patches turned into int3 patches.");
200
201 STAM_REG(pVM, &pVM->patm.s.StatInt3BlockRun, STAMTYPE_COUNTER, "/PATM/Patch/Run/Int3", STAMUNIT_OCCURENCES, "Number of times an int3 block patch was executed.");
202 STAMR3RegisterF(pVM, &pVM->patm.s.pGCStateHC->uPatchCalls, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Patch/Run/Normal");
203
204 STAM_REG(pVM, &pVM->patm.s.StatInstalledFunctionPatches, STAMTYPE_COUNTER, "/PATM/Patch/Installed/Function", STAMUNIT_OCCURENCES, "Number of installed function duplication patches.");
205 STAM_REG(pVM, &pVM->patm.s.StatInstalledTrampoline, STAMTYPE_COUNTER, "/PATM/Patch/Installed/Trampoline", STAMUNIT_OCCURENCES, "Number of installed trampoline patches.");
206 STAM_REG(pVM, &pVM->patm.s.StatInstalledJump, STAMTYPE_COUNTER, "/PATM/Patch/Installed/Jump", STAMUNIT_OCCURENCES, "Number of installed jump patches.");
207
208 STAM_REG(pVM, &pVM->patm.s.StatOverwritten, STAMTYPE_COUNTER, "/PATM/Patch/Overwritten", STAMUNIT_OCCURENCES, "Number of overwritten patches.");
209 STAM_REG(pVM, &pVM->patm.s.StatFixedConflicts,STAMTYPE_COUNTER, "/PATM/Patch/ConflictFixed", STAMUNIT_OCCURENCES, "Number of fixed conflicts.");
210 STAM_REG(pVM, &pVM->patm.s.StatFlushed, STAMTYPE_COUNTER, "/PATM/Patch/Flushed", STAMUNIT_OCCURENCES, "Number of flushes of pages with patch jumps.");
211 STAM_REG(pVM, &pVM->patm.s.StatMonitored, STAMTYPE_COUNTER, "/PATM/Patch/Monitored", STAMUNIT_OCCURENCES, "Number of patches in monitored patch pages.");
212 STAM_REG(pVM, &pVM->patm.s.StatPageBoundaryCrossed, STAMTYPE_COUNTER, "/PATM/Patch/BoundaryCross", STAMUNIT_OCCURENCES, "Number of refused patches due to patch jump crossing page boundary.");
213
214 STAM_REG(pVM, &pVM->patm.s.StatHandleTrap, STAMTYPE_PROFILE, "/PATM/HandleTrap", STAMUNIT_TICKS_PER_CALL, "Profiling of PATMR3HandleTrap");
215 STAM_REG(pVM, &pVM->patm.s.StatPushTrap, STAMTYPE_COUNTER, "/PATM/HandleTrap/PushWP", STAMUNIT_OCCURENCES, "Number of traps due to monitored stack pages.");
216
217 STAM_REG(pVM, &pVM->patm.s.StatSwitchBack, STAMTYPE_COUNTER, "/PATM/SwitchBack", STAMUNIT_OCCURENCES, "Switch back to original guest code when IF=1 & executing PATM instructions");
218 STAM_REG(pVM, &pVM->patm.s.StatSwitchBackFail,STAMTYPE_COUNTER, "/PATM/SwitchBackFail", STAMUNIT_OCCURENCES, "Failed switch back to original guest code when IF=1 & executing PATM instructions");
219
220 STAM_REG(pVM, &pVM->patm.s.StatDuplicateREQFailed, STAMTYPE_COUNTER, "/PATM/Function/DupREQ/Failed", STAMUNIT_OCCURENCES, "Nr of failed PATMR3DuplicateFunctionRequest calls");
221 STAM_REG(pVM, &pVM->patm.s.StatDuplicateREQSuccess, STAMTYPE_COUNTER, "/PATM/Function/DupREQ/Success", STAMUNIT_OCCURENCES, "Nr of successful PATMR3DuplicateFunctionRequest calls");
222 STAM_REG(pVM, &pVM->patm.s.StatDuplicateUseExisting,STAMTYPE_COUNTER, "/PATM/Function/DupREQ/UseExist", STAMUNIT_OCCURENCES, "Nr of successful PATMR3DuplicateFunctionRequest calls when using an existing patch");
223
224 STAM_REG(pVM, &pVM->patm.s.StatFunctionLookupInsert, STAMTYPE_COUNTER, "/PATM/Function/Lookup/Insert", STAMUNIT_OCCURENCES, "Nr of successful function address insertions");
225 STAM_REG(pVM, &pVM->patm.s.StatFunctionLookupReplace, STAMTYPE_COUNTER, "/PATM/Function/Lookup/Replace", STAMUNIT_OCCURENCES, "Nr of successful function address replacements");
226 STAM_REG(pVM, &pVM->patm.s.StatU32FunctionMaxSlotsUsed, STAMTYPE_U32_RESET,"/PATM/Function/Lookup/MaxSlots", STAMUNIT_OCCURENCES, "Maximum nr of lookup slots used in all call patches");
227
228 STAM_REG(pVM, &pVM->patm.s.StatFunctionFound, STAMTYPE_COUNTER, "/PATM/Function/Found", STAMUNIT_OCCURENCES, "Nr of successful function patch lookups in GC");
229 STAM_REG(pVM, &pVM->patm.s.StatFunctionNotFound, STAMTYPE_COUNTER, "/PATM/Function/NotFound", STAMUNIT_OCCURENCES, "Nr of failed function patch lookups in GC");
230
231 STAM_REG(pVM, &pVM->patm.s.StatPatchWrite, STAMTYPE_PROFILE, "/PATM/Write/Handle", STAMUNIT_TICKS_PER_CALL, "Profiling of PATMR3PatchWrite");
232 STAM_REG(pVM, &pVM->patm.s.StatPatchWriteDetect, STAMTYPE_PROFILE, "/PATM/Write/Detect", STAMUNIT_TICKS_PER_CALL, "Profiling of PATMIsWriteToPatchPage");
233 STAM_REG(pVM, &pVM->patm.s.StatPatchWriteInterpreted, STAMTYPE_COUNTER, "/PATM/Write/Interpreted/Success", STAMUNIT_OCCURENCES, "Nr of interpreted patch writes.");
234 STAM_REG(pVM, &pVM->patm.s.StatPatchWriteInterpretedFailed, STAMTYPE_COUNTER, "/PATM/Write/Interpreted/Failed", STAMUNIT_OCCURENCES, "Nr of failed interpreted patch writes.");
235
236 STAM_REG(pVM, &pVM->patm.s.StatPatchRefreshSuccess, STAMTYPE_COUNTER, "/PATM/Refresh/Success", STAMUNIT_OCCURENCES, "Successful patch refreshes");
237 STAM_REG(pVM, &pVM->patm.s.StatPatchRefreshFailed, STAMTYPE_COUNTER, "/PATM/Refresh/Failure", STAMUNIT_OCCURENCES, "Failed patch refreshes");
238
239 STAM_REG(pVM, &pVM->patm.s.StatPatchPageInserted, STAMTYPE_COUNTER, "/PATM/Page/Inserted", STAMUNIT_OCCURENCES, "Nr of inserted guest pages that were patched");
240 STAM_REG(pVM, &pVM->patm.s.StatPatchPageRemoved, STAMTYPE_COUNTER, "/PATM/Page/Removed", STAMUNIT_OCCURENCES, "Nr of removed guest pages that were patched");
241
242 STAM_REG(pVM, &pVM->patm.s.StatInstrDirty, STAMTYPE_COUNTER, "/PATM/Instr/Dirty/Detected", STAMUNIT_OCCURENCES, "Number of times instructions were marked dirty.");
243 STAM_REG(pVM, &pVM->patm.s.StatInstrDirtyGood, STAMTYPE_COUNTER, "/PATM/Instr/Dirty/Corrected", STAMUNIT_OCCURENCES, "Number of times instructions were marked dirty and corrected later on.");
244 STAM_REG(pVM, &pVM->patm.s.StatInstrDirtyBad, STAMTYPE_COUNTER, "/PATM/Instr/Dirty/Failed", STAMUNIT_OCCURENCES, "Number of times instructions were marked dirty and we were not able to correct them.");
245
246 STAM_REG(pVM, &pVM->patm.s.StatSysEnter, STAMTYPE_COUNTER, "/PATM/Emul/SysEnter", STAMUNIT_OCCURENCES, "Number of times sysenter was emulated.");
247 STAM_REG(pVM, &pVM->patm.s.StatSysExit, STAMTYPE_COUNTER, "/PATM/Emul/SysExit" , STAMUNIT_OCCURENCES, "Number of times sysexit was emulated.");
248 STAM_REG(pVM, &pVM->patm.s.StatEmulIret, STAMTYPE_COUNTER, "/PATM/Emul/Iret/Success", STAMUNIT_OCCURENCES, "Number of times iret was emulated.");
249 STAM_REG(pVM, &pVM->patm.s.StatEmulIretFailed, STAMTYPE_COUNTER, "/PATM/Emul/Iret/Failed", STAMUNIT_OCCURENCES, "Number of times iret was emulated.");
250
251 STAM_REG(pVM, &pVM->patm.s.StatGenRet, STAMTYPE_COUNTER, "/PATM/Gen/Ret" , STAMUNIT_OCCURENCES, "Number of generated ret instructions.");
252 STAM_REG(pVM, &pVM->patm.s.StatGenRetReused, STAMTYPE_COUNTER, "/PATM/Gen/RetReused" , STAMUNIT_OCCURENCES, "Number of reused ret instructions.");
253 STAM_REG(pVM, &pVM->patm.s.StatGenCall, STAMTYPE_COUNTER, "/PATM/Gen/Call", STAMUNIT_OCCURENCES, "Number of generated call instructions.");
254 STAM_REG(pVM, &pVM->patm.s.StatGenJump, STAMTYPE_COUNTER, "/PATM/Gen/Jmp" , STAMUNIT_OCCURENCES, "Number of generated indirect jump instructions.");
255 STAM_REG(pVM, &pVM->patm.s.StatGenPopf, STAMTYPE_COUNTER, "/PATM/Gen/Popf" , STAMUNIT_OCCURENCES, "Number of generated popf instructions.");
256
257 STAM_REG(pVM, &pVM->patm.s.StatCheckPendingIRQ, STAMTYPE_COUNTER, "/PATM/GC/CheckIRQ" , STAMUNIT_OCCURENCES, "Number of traps that ask to check for pending irqs.");
258#endif /* VBOX_WITH_STATISTICS */
259
260 Log(("PATMCallRecord.size %d\n", PATMCallRecord.size));
261 Log(("PATMCallIndirectRecord.size %d\n", PATMCallIndirectRecord.size));
262 Log(("PATMRetRecord.size %d\n", PATMRetRecord.size));
263 Log(("PATMJumpIndirectRecord.size %d\n", PATMJumpIndirectRecord.size));
264 Log(("PATMPopf32Record.size %d\n", PATMPopf32Record.size));
265 Log(("PATMIretRecord.size %d\n", PATMIretRecord.size));
266 Log(("PATMStiRecord.size %d\n", PATMStiRecord.size));
267 Log(("PATMCheckIFRecord.size %d\n", PATMCheckIFRecord.size));
268
269 return rc;
270}
271
272/**
273 * Finalizes HMA page attributes.
274 *
275 * @returns VBox status code.
276 * @param pVM The VM handle.
277 */
278VMMR3DECL(int) PATMR3InitFinalize(PVM pVM)
279{
280 /* The GC state, stack and statistics must be read/write for the guest (supervisor only of course). */
281 int rc = PGMMapSetPage(pVM, pVM->patm.s.pGCStateGC, PAGE_SIZE, X86_PTE_P | X86_PTE_A | X86_PTE_D | X86_PTE_RW);
282 if (RT_FAILURE(rc))
283 Log(("PATMR3InitFinalize: PGMMapSetPage failed with %Rrc!!\n", rc));
284
285 rc = PGMMapSetPage(pVM, pVM->patm.s.pGCStackGC, PATM_STACK_TOTAL_SIZE, X86_PTE_P | X86_PTE_A | X86_PTE_D | X86_PTE_RW);
286 if (RT_FAILURE(rc))
287 Log(("PATMR3InitFinalize: PGMMapSetPage failed with %Rrc!!\n", rc));
288
289 rc = PGMMapSetPage(pVM, pVM->patm.s.pStatsGC, PATM_STAT_MEMSIZE, X86_PTE_P | X86_PTE_A | X86_PTE_D | X86_PTE_RW);
290 if (RT_FAILURE(rc))
291 Log(("PATMR3InitFinalize: PGMMapSetPage failed with %Rrc!!\n", rc));
292
293 return rc;
294}
295
296/**
297 * (Re)initializes PATM
298 *
299 * @param pVM The VM.
300 */
301static int patmReinit(PVM pVM)
302{
303 int rc;
304
305 /*
306 * Assert alignment and sizes.
307 */
308 AssertRelease(!(RT_OFFSETOF(VM, patm.s) & 31));
309 AssertRelease(sizeof(pVM->patm.s) <= sizeof(pVM->patm.padding));
310
311 /*
312 * Setup any fixed pointers and offsets.
313 */
314 pVM->patm.s.offVM = RT_OFFSETOF(VM, patm);
315
316#ifndef RT_ARCH_AMD64 /* would be nice if this was changed everywhere. was driving me crazy on AMD64. */
317#ifndef PATM_DISABLE_ALL
318 pVM->fPATMEnabled = true;
319#endif
320#endif
321
322 Assert(pVM->patm.s.pGCStateHC);
323 memset(pVM->patm.s.pGCStateHC, 0, PAGE_SIZE);
324 AssertReleaseMsg(pVM->patm.s.pGCStateGC, ("Impossible! MMHyperHC2GC(%p) failed!\n", pVM->patm.s.pGCStateGC));
325
326 Log(("Patch memory allocated at %p - %RRv\n", pVM->patm.s.pPatchMemHC, pVM->patm.s.pPatchMemGC));
327 pVM->patm.s.pGCStateHC->uVMFlags = X86_EFL_IF;
328
329 Assert(pVM->patm.s.pGCStackHC);
330 memset(pVM->patm.s.pGCStackHC, 0, PAGE_SIZE);
331 AssertReleaseMsg(pVM->patm.s.pGCStackGC, ("Impossible! MMHyperHC2GC(%p) failed!\n", pVM->patm.s.pGCStackGC));
332 pVM->patm.s.pGCStateHC->Psp = PATM_STACK_SIZE;
333 pVM->patm.s.pGCStateHC->fPIF = 1; /* PATM Interrupt Flag */
334
335 Assert(pVM->patm.s.pStatsHC);
336 memset(pVM->patm.s.pStatsHC, 0, PATM_STAT_MEMSIZE);
337 AssertReleaseMsg(pVM->patm.s.pStatsGC, ("Impossible! MMHyperHC2GC(%p) failed!\n", pVM->patm.s.pStatsGC));
338
339 Assert(pVM->patm.s.pPatchMemHC);
340 Assert(pVM->patm.s.pPatchMemGC == MMHyperR3ToRC(pVM, pVM->patm.s.pPatchMemHC));
341 memset(pVM->patm.s.pPatchMemHC, 0, PATCH_MEMORY_SIZE);
342 AssertReleaseMsg(pVM->patm.s.pPatchMemGC, ("Impossible! MMHyperHC2GC(%p) failed!\n", pVM->patm.s.pPatchMemHC));
343
344 /* Needed for future patching of sldt/sgdt/sidt/str etc. */
345 pVM->patm.s.pCPUMCtxGC = VM_RC_ADDR(pVM, CPUMQueryGuestCtxPtr(VMMGetCpu(pVM)));
346
347 Assert(pVM->patm.s.PatchLookupTreeHC);
348 Assert(pVM->patm.s.PatchLookupTreeGC == MMHyperR3ToRC(pVM, pVM->patm.s.PatchLookupTreeHC));
349
350 /*
351 * (Re)Initialize PATM structure
352 */
353 Assert(!pVM->patm.s.PatchLookupTreeHC->PatchTree);
354 Assert(!pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr);
355 Assert(!pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage);
356 pVM->patm.s.offPatchMem = 16; /* don't start with zero here */
357 pVM->patm.s.uCurrentPatchIdx = 1; /* Index zero is a dummy */
358 pVM->patm.s.pvFaultMonitor = 0;
359 pVM->patm.s.deltaReloc = 0;
360
361 /* Lowest and highest patched instruction */
362 pVM->patm.s.pPatchedInstrGCLowest = ~0;
363 pVM->patm.s.pPatchedInstrGCHighest = 0;
364
365 pVM->patm.s.PatchLookupTreeHC->PatchTree = 0;
366 pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr = 0;
367 pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage = 0;
368
369 pVM->patm.s.pfnSysEnterPatchGC = 0;
370 pVM->patm.s.pfnSysEnterGC = 0;
371
372 pVM->patm.s.fOutOfMemory = false;
373
374 pVM->patm.s.pfnHelperCallGC = 0;
375
376 /* Generate all global functions to be used by future patches. */
377 /* We generate a fake patch in order to use the existing code for relocation. */
378 rc = MMHyperAlloc(pVM, sizeof(PATMPATCHREC), 0, MM_TAG_PATM_PATCH, (void **)&pVM->patm.s.pGlobalPatchRec);
379 if (RT_FAILURE(rc))
380 {
381 Log(("Out of memory!!!!\n"));
382 return VERR_NO_MEMORY;
383 }
384 pVM->patm.s.pGlobalPatchRec->patch.flags = PATMFL_GLOBAL_FUNCTIONS;
385 pVM->patm.s.pGlobalPatchRec->patch.uState = PATCH_ENABLED;
386 pVM->patm.s.pGlobalPatchRec->patch.pPatchBlockOffset = pVM->patm.s.offPatchMem;
387
388 rc = patmPatchGenGlobalFunctions(pVM, &pVM->patm.s.pGlobalPatchRec->patch);
389 AssertRC(rc);
390
391 /* Update free pointer in patch memory. */
392 pVM->patm.s.offPatchMem += pVM->patm.s.pGlobalPatchRec->patch.uCurPatchOffset;
393 /* Round to next 8 byte boundary. */
394 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
395 return rc;
396}
397
398
399/**
400 * Applies relocations to data and code managed by this
401 * component. This function will be called at init and
402 * whenever the VMM need to relocate it self inside the GC.
403 *
404 * The PATM will update the addresses used by the switcher.
405 *
406 * @param pVM The VM.
407 */
408VMMR3DECL(void) PATMR3Relocate(PVM pVM)
409{
410 RTRCPTR GCPtrNew = MMHyperR3ToRC(pVM, pVM->patm.s.pGCStateHC);
411 RTRCINTPTR delta = GCPtrNew - pVM->patm.s.pGCStateGC;
412
413 Log(("PATMR3Relocate from %RRv to %RRv - delta %08X\n", pVM->patm.s.pGCStateGC, GCPtrNew, delta));
414 if (delta)
415 {
416 PCPUMCTX pCtx;
417
418 /* Update CPUMCTX guest context pointer. */
419 pVM->patm.s.pCPUMCtxGC += delta;
420
421 pVM->patm.s.deltaReloc = delta;
422
423 RTAvloU32DoWithAll(&pVM->patm.s.PatchLookupTreeHC->PatchTree, true, RelocatePatches, (void *)pVM);
424
425 pCtx = CPUMQueryGuestCtxPtr(VMMGetCpu(pVM));
426
427 /* If we are running patch code right now, then also adjust EIP. */
428 if (PATMIsPatchGCAddr(pVM, pCtx->eip))
429 pCtx->eip += delta;
430
431 pVM->patm.s.pGCStateGC = GCPtrNew;
432 pVM->patm.s.pPatchMemGC = MMHyperR3ToRC(pVM, pVM->patm.s.pPatchMemHC);
433
434 pVM->patm.s.pGCStackGC = MMHyperR3ToRC(pVM, pVM->patm.s.pGCStackHC);
435
436 pVM->patm.s.pStatsGC = MMHyperR3ToRC(pVM, pVM->patm.s.pStatsHC);
437
438 pVM->patm.s.PatchLookupTreeGC = MMHyperR3ToRC(pVM, pVM->patm.s.PatchLookupTreeHC);
439
440 if (pVM->patm.s.pfnSysEnterPatchGC)
441 pVM->patm.s.pfnSysEnterPatchGC += delta;
442
443 /* Deal with the global patch functions. */
444 pVM->patm.s.pfnHelperCallGC += delta;
445 pVM->patm.s.pfnHelperRetGC += delta;
446 pVM->patm.s.pfnHelperIretGC += delta;
447 pVM->patm.s.pfnHelperJumpGC += delta;
448
449 RelocatePatches(&pVM->patm.s.pGlobalPatchRec->Core, (void *)pVM);
450 }
451}
452
453
454/**
455 * Terminates the PATM.
456 *
457 * Termination means cleaning up and freeing all resources,
458 * the VM it self is at this point powered off or suspended.
459 *
460 * @returns VBox status code.
461 * @param pVM The VM to operate on.
462 */
463VMMR3DECL(int) PATMR3Term(PVM pVM)
464{
465 /* Memory was all allocated from the two MM heaps and requires no freeing. */
466 return VINF_SUCCESS;
467}
468
469
470/**
471 * PATM reset callback.
472 *
473 * @returns VBox status code.
474 * @param pVM The VM which is reset.
475 */
476VMMR3DECL(int) PATMR3Reset(PVM pVM)
477{
478 Log(("PATMR3Reset\n"));
479
480 /* Free all patches. */
481 while (true)
482 {
483 PPATMPATCHREC pPatchRec = (PPATMPATCHREC)RTAvloU32RemoveBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, 0, true);
484 if (pPatchRec)
485 {
486 PATMRemovePatch(pVM, pPatchRec, true);
487 }
488 else
489 break;
490 }
491 Assert(!pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage);
492 Assert(!pVM->patm.s.PatchLookupTreeHC->PatchTree);
493 pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr = 0;
494 pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage = 0;
495
496 int rc = patmReinit(pVM);
497 if (RT_SUCCESS(rc))
498 rc = PATMR3InitFinalize(pVM); /* paranoia */
499
500 return rc;
501}
502
503/**
504 * Read callback for disassembly function; supports reading bytes that cross a page boundary
505 *
506 * @returns VBox status code.
507 * @param pSrc GC source pointer
508 * @param pDest HC destination pointer
509 * @param size Number of bytes to read
510 * @param pvUserdata Callback specific user data (pCpu)
511 *
512 */
513int patmReadBytes(RTUINTPTR pSrc, uint8_t *pDest, unsigned size, void *pvUserdata)
514{
515 DISCPUSTATE *pCpu = (DISCPUSTATE *)pvUserdata;
516 PATMDISASM *pDisInfo = (PATMDISASM *)pCpu->apvUserData[0];
517 int orgsize = size;
518
519 Assert(size);
520 if (size == 0)
521 return VERR_INVALID_PARAMETER;
522
523 /*
524 * Trap/interrupt handler typically call common code on entry. Which might already have patches inserted.
525 * As we currently don't support calling patch code from patch code, we'll let it read the original opcode bytes instead.
526 */
527 /** @todo could change in the future! */
528 if (pDisInfo->fReadFlags & PATMREAD_ORGCODE)
529 {
530 for (int i=0;i<orgsize;i++)
531 {
532 int rc = PATMR3QueryOpcode(pDisInfo->pVM, (RTRCPTR)pSrc, pDest);
533 if (RT_SUCCESS(rc))
534 {
535 pSrc++;
536 pDest++;
537 size--;
538 }
539 else break;
540 }
541 if (size == 0)
542 return VINF_SUCCESS;
543#ifdef VBOX_STRICT
544 if ( !(pDisInfo->pPatchInfo->flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER))
545 && !(pDisInfo->fReadFlags & PATMREAD_NOCHECK))
546 {
547 Assert(PATMR3IsInsidePatchJump(pDisInfo->pVM, pSrc, NULL) == false);
548 Assert(PATMR3IsInsidePatchJump(pDisInfo->pVM, pSrc+size-1, NULL) == false);
549 }
550#endif
551 }
552
553 if ( !pDisInfo->pInstrHC
554 || ( PAGE_ADDRESS(pDisInfo->pInstrGC) != PAGE_ADDRESS(pSrc + size - 1)
555 && !PATMIsPatchGCAddr(pDisInfo->pVM, pSrc)))
556 {
557 Assert(!PATMIsPatchGCAddr(pDisInfo->pVM, pSrc));
558 return PGMPhysSimpleReadGCPtr(&pDisInfo->pVM->aCpus[0], pDest, pSrc, size);
559 }
560 else
561 {
562 Assert(pDisInfo->pInstrHC);
563
564 uint8_t *pInstrHC = pDisInfo->pInstrHC;
565
566 Assert(pInstrHC);
567
568 /* pInstrHC is the base address; adjust according to the GC pointer. */
569 pInstrHC = pInstrHC + (pSrc - pDisInfo->pInstrGC);
570
571 memcpy(pDest, (void *)pInstrHC, size);
572 }
573
574 return VINF_SUCCESS;
575}
576
577/**
578 * Callback function for RTAvloU32DoWithAll
579 *
580 * Updates all fixups in the patches
581 *
582 * @returns VBox status code.
583 * @param pNode Current node
584 * @param pParam The VM to operate on.
585 */
586static DECLCALLBACK(int) RelocatePatches(PAVLOU32NODECORE pNode, void *pParam)
587{
588 PPATMPATCHREC pPatch = (PPATMPATCHREC)pNode;
589 PVM pVM = (PVM)pParam;
590 RTRCINTPTR delta;
591#ifdef LOG_ENABLED
592 DISCPUSTATE cpu;
593 char szOutput[256];
594 uint32_t opsize;
595 bool disret;
596#endif
597 int rc;
598
599 /* Nothing to do if the patch is not active. */
600 if (pPatch->patch.uState == PATCH_REFUSED)
601 return 0;
602
603#ifdef LOG_ENABLED
604 if (pPatch->patch.flags & PATMFL_PATCHED_GUEST_CODE)
605 {
606 cpu.mode = (pPatch->patch.flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
607 disret = PATMR3DISInstr(pVM, &pPatch->patch, &cpu, pPatch->patch.pPrivInstrGC, NULL, &opsize, szOutput, PATMREAD_RAWCODE);
608 Log(("Org patch jump: %s", szOutput));
609 }
610#endif
611
612 Log(("Nr of fixups %d\n", pPatch->patch.nrFixups));
613 delta = (RTRCINTPTR)pVM->patm.s.deltaReloc;
614
615 /*
616 * Apply fixups
617 */
618 PRELOCREC pRec = 0;
619 AVLPVKEY key = 0;
620
621 while (true)
622 {
623 /* Get the record that's closest from above */
624 pRec = (PRELOCREC)RTAvlPVGetBestFit(&pPatch->patch.FixupTree, key, true);
625 if (pRec == 0)
626 break;
627
628 key = (AVLPVKEY)(pRec->pRelocPos + 1); /* search for the next record during the next round. */
629
630 switch (pRec->uType)
631 {
632 case FIXUP_ABSOLUTE:
633 Log(("Absolute fixup at %RRv %RHv -> %RHv at %RRv\n", pRec->pSource, *(RTRCUINTPTR *)pRec->pRelocPos, *(RTRCINTPTR*)pRec->pRelocPos + delta, pRec->pRelocPos));
634 if ( !pRec->pSource
635 || PATMIsPatchGCAddr(pVM, pRec->pSource))
636 {
637 *(RTRCUINTPTR *)pRec->pRelocPos += delta;
638 }
639 else
640 {
641 uint8_t curInstr[15];
642 uint8_t oldInstr[15];
643 Assert(pRec->pSource && pPatch->patch.cbPrivInstr <= 15);
644
645 Assert(!(pPatch->patch.flags & PATMFL_GLOBAL_FUNCTIONS));
646
647 memcpy(oldInstr, pPatch->patch.aPrivInstr, pPatch->patch.cbPrivInstr);
648 *(RTRCPTR *)&oldInstr[pPatch->patch.cbPrivInstr - sizeof(RTRCPTR)] = pRec->pDest;
649
650 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), curInstr, pPatch->patch.pPrivInstrGC, pPatch->patch.cbPrivInstr);
651 Assert(RT_SUCCESS(rc) || rc == VERR_PAGE_NOT_PRESENT || rc == VERR_PAGE_TABLE_NOT_PRESENT);
652
653 pRec->pDest = (RTRCPTR)((RTRCUINTPTR)pRec->pDest + delta);
654
655 if ( rc == VERR_PAGE_NOT_PRESENT
656 || rc == VERR_PAGE_TABLE_NOT_PRESENT)
657 {
658 RTRCPTR pPage = pPatch->patch.pPrivInstrGC & PAGE_BASE_GC_MASK;
659
660 Log(("PATM: Patch page not present -> check later!\n"));
661 rc = PGMR3HandlerVirtualRegister(pVM, PGMVIRTHANDLERTYPE_ALL, pPage, pPage + (PAGE_SIZE - 1) /* inclusive! */, 0, patmVirtPageHandler, "PATMGCMonitorPage", 0, "PATMMonitorPatchJump");
662 Assert(RT_SUCCESS(rc) || rc == VERR_PGM_HANDLER_VIRTUAL_CONFLICT);
663 }
664 else
665 if (memcmp(curInstr, oldInstr, pPatch->patch.cbPrivInstr))
666 {
667 Log(("PATM: Patch was overwritten -> disabling patch!!\n"));
668 /*
669 * Disable patch; this is not a good solution
670 */
671 /* @todo hopefully it was completely overwritten (if the read was successful)!!!! */
672 pPatch->patch.uState = PATCH_DISABLED;
673 }
674 else
675 if (RT_SUCCESS(rc))
676 {
677 *(RTRCPTR *)&curInstr[pPatch->patch.cbPrivInstr - sizeof(RTRCPTR)] = pRec->pDest;
678 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pRec->pSource, curInstr, pPatch->patch.cbPrivInstr);
679 AssertRC(rc);
680 }
681 }
682 break;
683
684 case FIXUP_REL_JMPTOPATCH:
685 {
686 RTRCPTR pTarget = (RTRCPTR)((RTRCINTPTR)pRec->pDest + delta);
687
688 if ( pPatch->patch.uState == PATCH_ENABLED
689 && (pPatch->patch.flags & PATMFL_PATCHED_GUEST_CODE))
690 {
691 uint8_t oldJump[SIZEOF_NEAR_COND_JUMP32];
692 uint8_t temp[SIZEOF_NEAR_COND_JUMP32];
693 RTRCPTR pJumpOffGC;
694 RTRCINTPTR displ = (RTRCINTPTR)pTarget - (RTRCINTPTR)pRec->pSource;
695 RTRCINTPTR displOld= (RTRCINTPTR)pRec->pDest - (RTRCINTPTR)pRec->pSource;
696
697#if 0 /** @todo '*(int32_t*)pRec->pRelocPos' crashes on restore of an XP VM here. pRelocPos=0x8000dbe2180a (bird) */
698 Log(("Relative fixup (g2p) %08X -> %08X at %08X (source=%08x, target=%08x)\n", *(int32_t*)pRec->pRelocPos, displ, pRec->pRelocPos, pRec->pSource, pRec->pDest));
699#else
700 Log(("Relative fixup (g2p) ???????? -> %08X at %08X (source=%08x, target=%08x)\n", displ, pRec->pRelocPos, pRec->pSource, pRec->pDest));
701#endif
702
703 Assert(pRec->pSource - pPatch->patch.cbPatchJump == pPatch->patch.pPrivInstrGC);
704#ifdef PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES
705 if (pPatch->patch.cbPatchJump == SIZEOF_NEAR_COND_JUMP32)
706 {
707 Assert(pPatch->patch.flags & PATMFL_JUMP_CONFLICT);
708
709 pJumpOffGC = pPatch->patch.pPrivInstrGC + 2; //two byte opcode
710 oldJump[0] = pPatch->patch.aPrivInstr[0];
711 oldJump[1] = pPatch->patch.aPrivInstr[1];
712 *(RTRCUINTPTR *)&oldJump[2] = displOld;
713 }
714 else
715#endif
716 if (pPatch->patch.cbPatchJump == SIZEOF_NEARJUMP32)
717 {
718 pJumpOffGC = pPatch->patch.pPrivInstrGC + 1; //one byte opcode
719 oldJump[0] = 0xE9;
720 *(RTRCUINTPTR *)&oldJump[1] = displOld;
721 }
722 else
723 {
724 AssertMsgFailed(("Invalid patch jump size %d\n", pPatch->patch.cbPatchJump));
725 continue; //this should never happen!!
726 }
727 Assert(pPatch->patch.cbPatchJump <= sizeof(temp));
728
729 /*
730 * Read old patch jump and compare it to the one we previously installed
731 */
732 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), temp, pPatch->patch.pPrivInstrGC, pPatch->patch.cbPatchJump);
733 Assert(RT_SUCCESS(rc) || rc == VERR_PAGE_NOT_PRESENT || rc == VERR_PAGE_TABLE_NOT_PRESENT);
734
735 if ( rc == VERR_PAGE_NOT_PRESENT
736 || rc == VERR_PAGE_TABLE_NOT_PRESENT)
737 {
738 RTRCPTR pPage = pPatch->patch.pPrivInstrGC & PAGE_BASE_GC_MASK;
739
740 rc = PGMR3HandlerVirtualRegister(pVM, PGMVIRTHANDLERTYPE_ALL, pPage, pPage + (PAGE_SIZE - 1) /* inclusive! */, 0, patmVirtPageHandler, "PATMGCMonitorPage", 0, "PATMMonitorPatchJump");
741 Assert(RT_SUCCESS(rc) || rc == VERR_PGM_HANDLER_VIRTUAL_CONFLICT);
742 }
743 else
744 if (memcmp(temp, oldJump, pPatch->patch.cbPatchJump))
745 {
746 Log(("PATM: Patch jump was overwritten -> disabling patch!!\n"));
747 /*
748 * Disable patch; this is not a good solution
749 */
750 /* @todo hopefully it was completely overwritten (if the read was successful)!!!! */
751 pPatch->patch.uState = PATCH_DISABLED;
752 }
753 else
754 if (RT_SUCCESS(rc))
755 {
756 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pJumpOffGC, &displ, sizeof(displ));
757 AssertRC(rc);
758 }
759 else
760 AssertMsgFailed(("Unexpected error %d from MMR3PhysReadGCVirt\n", rc));
761 }
762 else
763 Log(("Skip the guest jump to patch code for this disabled patch %RGv - %08X\n", pPatch->patch.pPrivInstrGC, pRec->pRelocPos));
764
765 pRec->pDest = pTarget;
766 break;
767 }
768
769 case FIXUP_REL_JMPTOGUEST:
770 {
771 RTRCPTR pSource = (RTRCPTR)((RTRCINTPTR)pRec->pSource + delta);
772 RTRCINTPTR displ = (RTRCINTPTR)pRec->pDest - (RTRCINTPTR)pSource;
773
774 Assert(!(pPatch->patch.flags & PATMFL_GLOBAL_FUNCTIONS));
775 Log(("Relative fixup (p2g) %08X -> %08X at %08X (source=%08x, target=%08x)\n", *(int32_t*)pRec->pRelocPos, displ, pRec->pRelocPos, pRec->pSource, pRec->pDest));
776 *(RTRCUINTPTR *)pRec->pRelocPos = displ;
777 pRec->pSource = pSource;
778 break;
779 }
780
781 default:
782 AssertMsg(0, ("Invalid fixup type!!\n"));
783 return VERR_INVALID_PARAMETER;
784 }
785 }
786
787#ifdef LOG_ENABLED
788 if (pPatch->patch.flags & PATMFL_PATCHED_GUEST_CODE)
789 {
790 cpu.mode = (pPatch->patch.flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
791 disret = PATMR3DISInstr(pVM, &pPatch->patch, &cpu, pPatch->patch.pPrivInstrGC, NULL, &opsize, szOutput, PATMREAD_RAWCODE);
792 Log(("Rel patch jump: %s", szOutput));
793 }
794#endif
795 return 0;
796}
797
798/**
799 * \#PF Handler callback for virtual access handler ranges.
800 *
801 * Important to realize that a physical page in a range can have aliases, and
802 * for ALL and WRITE handlers these will also trigger.
803 *
804 * @returns VINF_SUCCESS if the handler have carried out the operation.
805 * @returns VINF_PGM_HANDLER_DO_DEFAULT if the caller should carry out the access operation.
806 * @param pVM VM Handle.
807 * @param GCPtr The virtual address the guest is writing to. (not correct if it's an alias!)
808 * @param pvPtr The HC mapping of that address.
809 * @param pvBuf What the guest is reading/writing.
810 * @param cbBuf How much it's reading/writing.
811 * @param enmAccessType The access type.
812 * @param pvUser User argument.
813 */
814DECLCALLBACK(int) patmVirtPageHandler(PVM pVM, RTGCPTR GCPtr, void *pvPtr, void *pvBuf, size_t cbBuf, PGMACCESSTYPE enmAccessType, void *pvUser)
815{
816 Assert(enmAccessType == PGMACCESSTYPE_WRITE);
817 /** @todo could be the wrong virtual address (alias) */
818 pVM->patm.s.pvFaultMonitor = GCPtr;
819 PATMR3HandleMonitoredPage(pVM);
820 return VINF_PGM_HANDLER_DO_DEFAULT;
821}
822
823
824#ifdef VBOX_WITH_DEBUGGER
825/**
826 * Callback function for RTAvloU32DoWithAll
827 *
828 * Enables the patch that's being enumerated
829 *
830 * @returns 0 (continue enumeration).
831 * @param pNode Current node
832 * @param pVM The VM to operate on.
833 */
834static DECLCALLBACK(int) EnableAllPatches(PAVLOU32NODECORE pNode, void *pVM)
835{
836 PPATMPATCHREC pPatch = (PPATMPATCHREC)pNode;
837
838 PATMR3EnablePatch((PVM)pVM, (RTRCPTR)pPatch->Core.Key);
839 return 0;
840}
841#endif /* VBOX_WITH_DEBUGGER */
842
843
844#ifdef VBOX_WITH_DEBUGGER
845/**
846 * Callback function for RTAvloU32DoWithAll
847 *
848 * Disables the patch that's being enumerated
849 *
850 * @returns 0 (continue enumeration).
851 * @param pNode Current node
852 * @param pVM The VM to operate on.
853 */
854static DECLCALLBACK(int) DisableAllPatches(PAVLOU32NODECORE pNode, void *pVM)
855{
856 PPATMPATCHREC pPatch = (PPATMPATCHREC)pNode;
857
858 PATMR3DisablePatch((PVM)pVM, (RTRCPTR)pPatch->Core.Key);
859 return 0;
860}
861#endif
862
863/**
864 * Returns the host context pointer and size of the patch memory block
865 *
866 * @returns VBox status code.
867 * @param pVM The VM to operate on.
868 * @param pcb Size of the patch memory block
869 */
870VMMR3DECL(void *) PATMR3QueryPatchMemHC(PVM pVM, uint32_t *pcb)
871{
872 if (pcb)
873 *pcb = pVM->patm.s.cbPatchMem;
874
875 return pVM->patm.s.pPatchMemHC;
876}
877
878
879/**
880 * Returns the guest context pointer and size of the patch memory block
881 *
882 * @returns VBox status code.
883 * @param pVM The VM to operate on.
884 * @param pcb Size of the patch memory block
885 */
886VMMR3DECL(RTRCPTR) PATMR3QueryPatchMemGC(PVM pVM, uint32_t *pcb)
887{
888 if (pcb)
889 *pcb = pVM->patm.s.cbPatchMem;
890
891 return pVM->patm.s.pPatchMemGC;
892}
893
894
895/**
896 * Returns the host context pointer of the GC context structure
897 *
898 * @returns VBox status code.
899 * @param pVM The VM to operate on.
900 */
901VMMR3DECL(PPATMGCSTATE) PATMR3QueryGCStateHC(PVM pVM)
902{
903 return pVM->patm.s.pGCStateHC;
904}
905
906
907/**
908 * Checks whether the HC address is part of our patch region
909 *
910 * @returns VBox status code.
911 * @param pVM The VM to operate on.
912 * @param pAddrGC Guest context address
913 */
914VMMR3DECL(bool) PATMR3IsPatchHCAddr(PVM pVM, R3PTRTYPE(uint8_t *) pAddrHC)
915{
916 return (pAddrHC >= pVM->patm.s.pPatchMemHC && pAddrHC < pVM->patm.s.pPatchMemHC + pVM->patm.s.cbPatchMem) ? true : false;
917}
918
919
920/**
921 * Allows or disallow patching of privileged instructions executed by the guest OS
922 *
923 * @returns VBox status code.
924 * @param pVM The VM to operate on.
925 * @param fAllowPatching Allow/disallow patching
926 */
927VMMR3DECL(int) PATMR3AllowPatching(PVM pVM, uint32_t fAllowPatching)
928{
929 pVM->fPATMEnabled = (fAllowPatching) ? true : false;
930 return VINF_SUCCESS;
931}
932
933/**
934 * Convert a GC patch block pointer to a HC patch pointer
935 *
936 * @returns HC pointer or NULL if it's not a GC patch pointer
937 * @param pVM The VM to operate on.
938 * @param pAddrGC GC pointer
939 */
940VMMR3DECL(R3PTRTYPE(void *)) PATMR3GCPtrToHCPtr(PVM pVM, RTRCPTR pAddrGC)
941{
942 if (pVM->patm.s.pPatchMemGC <= pAddrGC && pVM->patm.s.pPatchMemGC + pVM->patm.s.cbPatchMem > pAddrGC)
943 return pVM->patm.s.pPatchMemHC + (pAddrGC - pVM->patm.s.pPatchMemGC);
944 else
945 return NULL;
946}
947
948/**
949 * Query PATM state (enabled/disabled)
950 *
951 * @returns 0 - disabled, 1 - enabled
952 * @param pVM The VM to operate on.
953 */
954VMMR3DECL(int) PATMR3IsEnabled(PVM pVM)
955{
956 return pVM->fPATMEnabled;
957}
958
959
960/**
961 * Convert guest context address to host context pointer
962 *
963 * @returns VBox status code.
964 * @param pVM The VM to operate on.
965 * @param pCacheRec Address conversion cache record
966 * @param pGCPtr Guest context pointer
967 *
968 * @returns Host context pointer or NULL in case of an error
969 *
970 */
971R3PTRTYPE(uint8_t *) PATMGCVirtToHCVirt(PVM pVM, PPATMP2GLOOKUPREC pCacheRec, RCPTRTYPE(uint8_t *) pGCPtr)
972{
973 int rc;
974 R3PTRTYPE(uint8_t *) pHCPtr;
975 uint32_t offset;
976
977 if (PATMIsPatchGCAddr(pVM, pGCPtr))
978 {
979 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
980 Assert(pPatch);
981 return PATCHCODE_PTR_HC(pPatch) + (pGCPtr - PATCHCODE_PTR_GC(pPatch));
982 }
983
984 offset = pGCPtr & PAGE_OFFSET_MASK;
985 if (pCacheRec->pGuestLoc == (pGCPtr & PAGE_BASE_GC_MASK))
986 return pCacheRec->pPageLocStartHC + offset;
987
988 /* Release previous lock if any. */
989 if (pCacheRec->Lock.pvMap)
990 {
991 PGMPhysReleasePageMappingLock(pVM, &pCacheRec->Lock);
992 pCacheRec->Lock.pvMap = NULL;
993 }
994
995 rc = PGMPhysGCPtr2CCPtrReadOnly(VMMGetCpu(pVM), pGCPtr, (const void **)&pHCPtr, &pCacheRec->Lock);
996 if (rc != VINF_SUCCESS)
997 {
998 AssertMsg(rc == VINF_SUCCESS || rc == VERR_PAGE_NOT_PRESENT || rc == VERR_PAGE_TABLE_NOT_PRESENT, ("MMR3PhysGCVirt2HCVirtEx failed for %08X\n", pGCPtr));
999 return NULL;
1000 }
1001 pCacheRec->pPageLocStartHC = (R3PTRTYPE(uint8_t*))((RTHCUINTPTR)pHCPtr & PAGE_BASE_HC_MASK);
1002 pCacheRec->pGuestLoc = pGCPtr & PAGE_BASE_GC_MASK;
1003 return pHCPtr;
1004}
1005
1006
1007/* Calculates and fills in all branch targets
1008 *
1009 * @returns VBox status code.
1010 * @param pVM The VM to operate on.
1011 * @param pPatch Current patch block pointer
1012 *
1013 */
1014static int patmr3SetBranchTargets(PVM pVM, PPATCHINFO pPatch)
1015{
1016 int32_t displ;
1017
1018 PJUMPREC pRec = 0;
1019 unsigned nrJumpRecs = 0;
1020
1021 /*
1022 * Set all branch targets inside the patch block.
1023 * We remove all jump records as they are no longer needed afterwards.
1024 */
1025 while (true)
1026 {
1027 RCPTRTYPE(uint8_t *) pInstrGC;
1028 RCPTRTYPE(uint8_t *) pBranchTargetGC = 0;
1029
1030 pRec = (PJUMPREC)RTAvlPVRemoveBestFit(&pPatch->JumpTree, 0, true);
1031 if (pRec == 0)
1032 break;
1033
1034 nrJumpRecs++;
1035
1036 /* HC in patch block to GC in patch block. */
1037 pInstrGC = patmPatchHCPtr2PatchGCPtr(pVM, pRec->pJumpHC);
1038
1039 if (pRec->opcode == OP_CALL)
1040 {
1041 /* Special case: call function replacement patch from this patch block.
1042 */
1043 PPATMPATCHREC pFunctionRec = PATMQueryFunctionPatch(pVM, pRec->pTargetGC);
1044 if (!pFunctionRec)
1045 {
1046 int rc;
1047
1048 if (PATMR3HasBeenPatched(pVM, pRec->pTargetGC) == false)
1049 rc = PATMR3InstallPatch(pVM, pRec->pTargetGC, PATMFL_CODE32 | PATMFL_DUPLICATE_FUNCTION);
1050 else
1051 rc = VERR_PATCHING_REFUSED; /* exists as a normal patch; can't use it */
1052
1053 if (RT_FAILURE(rc))
1054 {
1055 uint8_t *pPatchHC;
1056 RTRCPTR pPatchGC;
1057 RTRCPTR pOrgInstrGC;
1058
1059 pOrgInstrGC = PATMR3PatchToGCPtr(pVM, pInstrGC, 0);
1060 Assert(pOrgInstrGC);
1061
1062 /* Failure for some reason -> mark exit point with int 3. */
1063 Log(("Failed to install function replacement patch (at %x) for reason %Rrc\n", pOrgInstrGC, rc));
1064
1065 pPatchGC = patmGuestGCPtrToPatchGCPtr(pVM, pPatch, pOrgInstrGC);
1066 Assert(pPatchGC);
1067
1068 pPatchHC = pVM->patm.s.pPatchMemHC + (pPatchGC - pVM->patm.s.pPatchMemGC);
1069
1070 /* Set a breakpoint at the very beginning of the recompiled instruction */
1071 *pPatchHC = 0xCC;
1072
1073 continue;
1074 }
1075 }
1076 else
1077 {
1078 Log(("Patch block %RRv called as function\n", pFunctionRec->patch.pPrivInstrGC));
1079 pFunctionRec->patch.flags |= PATMFL_CODE_REFERENCED;
1080 }
1081
1082 pBranchTargetGC = PATMR3QueryPatchGCPtr(pVM, pRec->pTargetGC);
1083 }
1084 else
1085 pBranchTargetGC = patmGuestGCPtrToPatchGCPtr(pVM, pPatch, pRec->pTargetGC);
1086
1087 if (pBranchTargetGC == 0)
1088 {
1089 AssertMsgFailed(("patmr3SetBranchTargets: patmGuestGCPtrToPatchGCPtr failed for %08X\n", pRec->pTargetGC));
1090 return VERR_PATCHING_REFUSED;
1091 }
1092 /* Our jumps *always* have a dword displacement (to make things easier). */
1093 Assert(sizeof(uint32_t) == sizeof(RTRCPTR));
1094 displ = pBranchTargetGC - (pInstrGC + pRec->offDispl + sizeof(RTRCPTR));
1095 *(RTRCPTR *)(pRec->pJumpHC + pRec->offDispl) = displ;
1096 Log(("Set branch target %d to %08X : %08x - (%08x + %d + %d)\n", nrJumpRecs, displ, pBranchTargetGC, pInstrGC, pRec->offDispl, sizeof(RTRCPTR)));
1097 }
1098 Assert(nrJumpRecs == pPatch->nrJumpRecs);
1099 Assert(pPatch->JumpTree == 0);
1100 return VINF_SUCCESS;
1101}
1102
1103/* Add an illegal instruction record
1104 *
1105 * @param pVM The VM to operate on.
1106 * @param pPatch Patch structure ptr
1107 * @param pInstrGC Guest context pointer to privileged instruction
1108 *
1109 */
1110static void patmAddIllegalInstrRecord(PVM pVM, PPATCHINFO pPatch, RTRCPTR pInstrGC)
1111{
1112 PAVLPVNODECORE pRec;
1113
1114 pRec = (PAVLPVNODECORE)MMR3HeapAllocZ(pVM, MM_TAG_PATM_PATCH, sizeof(*pRec));
1115 Assert(pRec);
1116 pRec->Key = (AVLPVKEY)pInstrGC;
1117
1118 bool ret = RTAvlPVInsert(&pPatch->pTempInfo->IllegalInstrTree, pRec);
1119 Assert(ret); NOREF(ret);
1120 pPatch->pTempInfo->nrIllegalInstr++;
1121}
1122
1123static bool patmIsIllegalInstr(PPATCHINFO pPatch, RTRCPTR pInstrGC)
1124{
1125 PAVLPVNODECORE pRec;
1126
1127 pRec = RTAvlPVGet(&pPatch->pTempInfo->IllegalInstrTree, (AVLPVKEY)pInstrGC);
1128 if (pRec)
1129 return true;
1130 else
1131 return false;
1132}
1133
1134/**
1135 * Add a patch to guest lookup record
1136 *
1137 * @param pVM The VM to operate on.
1138 * @param pPatch Patch structure ptr
1139 * @param pPatchInstrHC Guest context pointer to patch block
1140 * @param pInstrGC Guest context pointer to privileged instruction
1141 * @param enmType Lookup type
1142 * @param fDirty Dirty flag
1143 *
1144 */
1145 /** @note Be extremely careful with this function. Make absolutely sure the guest address is correct! (to avoid executing instructions twice!) */
1146void patmr3AddP2GLookupRecord(PVM pVM, PPATCHINFO pPatch, uint8_t *pPatchInstrHC, RTRCPTR pInstrGC, PATM_LOOKUP_TYPE enmType, bool fDirty)
1147{
1148 bool ret;
1149 PRECPATCHTOGUEST pPatchToGuestRec;
1150 PRECGUESTTOPATCH pGuestToPatchRec;
1151 uint32_t PatchOffset = pPatchInstrHC - pVM->patm.s.pPatchMemHC; /* Offset in memory reserved for PATM. */
1152
1153 if (enmType == PATM_LOOKUP_PATCH2GUEST)
1154 {
1155 pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32Get(&pPatch->Patch2GuestAddrTree, PatchOffset);
1156 if (pPatchToGuestRec && pPatchToGuestRec->Core.Key == PatchOffset)
1157 return; /* already there */
1158
1159 Assert(!pPatchToGuestRec);
1160 }
1161#ifdef VBOX_STRICT
1162 else
1163 {
1164 pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32Get(&pPatch->Patch2GuestAddrTree, PatchOffset);
1165 Assert(!pPatchToGuestRec);
1166 }
1167#endif
1168
1169 pPatchToGuestRec = (PRECPATCHTOGUEST)MMR3HeapAllocZ(pVM, MM_TAG_PATM_PATCH, sizeof(RECPATCHTOGUEST) + sizeof(RECGUESTTOPATCH));
1170 Assert(pPatchToGuestRec);
1171 pPatchToGuestRec->Core.Key = PatchOffset;
1172 pPatchToGuestRec->pOrgInstrGC = pInstrGC;
1173 pPatchToGuestRec->enmType = enmType;
1174 pPatchToGuestRec->fDirty = fDirty;
1175
1176 ret = RTAvlU32Insert(&pPatch->Patch2GuestAddrTree, &pPatchToGuestRec->Core);
1177 Assert(ret);
1178
1179 /* GC to patch address */
1180 if (enmType == PATM_LOOKUP_BOTHDIR)
1181 {
1182 pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32Get(&pPatch->Guest2PatchAddrTree, pInstrGC);
1183 if (!pGuestToPatchRec)
1184 {
1185 pGuestToPatchRec = (PRECGUESTTOPATCH)(pPatchToGuestRec+1);
1186 pGuestToPatchRec->Core.Key = pInstrGC;
1187 pGuestToPatchRec->PatchOffset = PatchOffset;
1188
1189 ret = RTAvlU32Insert(&pPatch->Guest2PatchAddrTree, &pGuestToPatchRec->Core);
1190 Assert(ret);
1191 }
1192 }
1193
1194 pPatch->nrPatch2GuestRecs++;
1195}
1196
1197
1198/**
1199 * Removes a patch to guest lookup record
1200 *
1201 * @param pVM The VM to operate on.
1202 * @param pPatch Patch structure ptr
1203 * @param pPatchInstrGC Guest context pointer to patch block
1204 */
1205void patmr3RemoveP2GLookupRecord(PVM pVM, PPATCHINFO pPatch, RTRCPTR pPatchInstrGC)
1206{
1207 PAVLU32NODECORE pNode;
1208 PAVLU32NODECORE pNode2;
1209 PRECPATCHTOGUEST pPatchToGuestRec;
1210 uint32_t PatchOffset = pPatchInstrGC - pVM->patm.s.pPatchMemGC; /* Offset in memory reserved for PATM. */
1211
1212 pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32Get(&pPatch->Patch2GuestAddrTree, PatchOffset);
1213 Assert(pPatchToGuestRec);
1214 if (pPatchToGuestRec)
1215 {
1216 if (pPatchToGuestRec->enmType == PATM_LOOKUP_BOTHDIR)
1217 {
1218 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)(pPatchToGuestRec+1);
1219
1220 Assert(pGuestToPatchRec->Core.Key);
1221 pNode2 = RTAvlU32Remove(&pPatch->Guest2PatchAddrTree, pGuestToPatchRec->Core.Key);
1222 Assert(pNode2);
1223 }
1224 pNode = RTAvlU32Remove(&pPatch->Patch2GuestAddrTree, pPatchToGuestRec->Core.Key);
1225 Assert(pNode);
1226
1227 MMR3HeapFree(pPatchToGuestRec);
1228 pPatch->nrPatch2GuestRecs--;
1229 }
1230}
1231
1232
1233/**
1234 * RTAvlPVDestroy callback.
1235 */
1236static DECLCALLBACK(int) patmEmptyTreePVCallback(PAVLPVNODECORE pNode, void *)
1237{
1238 MMR3HeapFree(pNode);
1239 return 0;
1240}
1241
1242/**
1243 * Empty the specified tree (PV tree, MMR3 heap)
1244 *
1245 * @param pVM The VM to operate on.
1246 * @param ppTree Tree to empty
1247 */
1248void patmEmptyTree(PVM pVM, PAVLPVNODECORE *ppTree)
1249{
1250 RTAvlPVDestroy(ppTree, patmEmptyTreePVCallback, NULL);
1251}
1252
1253
1254/**
1255 * RTAvlU32Destroy callback.
1256 */
1257static DECLCALLBACK(int) patmEmptyTreeU32Callback(PAVLU32NODECORE pNode, void *)
1258{
1259 MMR3HeapFree(pNode);
1260 return 0;
1261}
1262
1263/**
1264 * Empty the specified tree (U32 tree, MMR3 heap)
1265 *
1266 * @param pVM The VM to operate on.
1267 * @param ppTree Tree to empty
1268 */
1269void patmEmptyTreeU32(PVM pVM, PPAVLU32NODECORE ppTree)
1270{
1271 RTAvlU32Destroy(ppTree, patmEmptyTreeU32Callback, NULL);
1272}
1273
1274
1275/**
1276 * Analyses the instructions following the cli for compliance with our heuristics for cli & pushf
1277 *
1278 * @returns VBox status code.
1279 * @param pVM The VM to operate on.
1280 * @param pCpu CPU disassembly state
1281 * @param pInstrGC Guest context pointer to privileged instruction
1282 * @param pCurInstrGC Guest context pointer to the current instruction
1283 * @param pCacheRec Cache record ptr
1284 *
1285 */
1286static int patmAnalyseBlockCallback(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PPATMP2GLOOKUPREC pCacheRec)
1287{
1288 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
1289 bool fIllegalInstr = false;
1290
1291 //Preliminary heuristics:
1292 //- no call instructions without a fixed displacement between cli and sti/popf
1293 //- no jumps in the instructions following cli (4+ bytes; enough for the replacement jump (5 bytes))
1294 //- no nested pushf/cli
1295 //- sti/popf should be the (eventual) target of all branches
1296 //- no near or far returns; no int xx, no into
1297 //
1298 // Note: Later on we can impose less stricter guidelines if the need arises
1299
1300 /* Bail out if the patch gets too big. */
1301 if (pPatch->cbPatchBlockSize >= MAX_PATCH_SIZE)
1302 {
1303 Log(("Code block too big (%x) for patch at %RRv!!\n", pPatch->cbPatchBlockSize, pCurInstrGC));
1304 fIllegalInstr = true;
1305 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1306 }
1307 else
1308 {
1309 /* No unconditinal jumps or calls without fixed displacements. */
1310 if ( (pCpu->pCurInstr->optype & OPTYPE_CONTROLFLOW)
1311 && (pCpu->pCurInstr->opcode == OP_JMP || pCpu->pCurInstr->opcode == OP_CALL)
1312 )
1313 {
1314 Assert(pCpu->param1.size <= 4 || pCpu->param1.size == 6);
1315 if ( pCpu->param1.size == 6 /* far call/jmp */
1316 || (pCpu->pCurInstr->opcode == OP_CALL && !(pPatch->flags & PATMFL_SUPPORT_CALLS))
1317 || (OP_PARM_VTYPE(pCpu->pCurInstr->param1) != OP_PARM_J && !(pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS))
1318 )
1319 {
1320 fIllegalInstr = true;
1321 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1322 }
1323 }
1324
1325 /* An unconditional (short) jump right after a cli is a potential problem; we will overwrite whichever function comes afterwards */
1326 if (pPatch->opcode == OP_CLI && pCpu->pCurInstr->opcode == OP_JMP)
1327 {
1328 if (pCurInstrGC > pPatch->pPrivInstrGC && pCurInstrGC + pCpu->opsize < pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32) /* hardcoded patch jump size; cbPatchJump is still zero */
1329 {
1330 Log(("Dangerous unconditional jump ends in our generated patch jump!! (%x vs %x)\n", pCurInstrGC, pPatch->pPrivInstrGC));
1331 /* We turn this one into a int 3 callable patch. */
1332 pPatch->flags |= PATMFL_INT3_REPLACEMENT_BLOCK;
1333 }
1334 }
1335 else
1336 /* no nested pushfs just yet; nested cli is allowed for cli patches though. */
1337 if (pPatch->opcode == OP_PUSHF)
1338 {
1339 if (pCurInstrGC != pInstrGC && pCpu->pCurInstr->opcode == OP_PUSHF)
1340 {
1341 fIllegalInstr = true;
1342 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1343 }
1344 }
1345
1346 // no far returns
1347 if (pCpu->pCurInstr->opcode == OP_RETF)
1348 {
1349 pPatch->pTempInfo->nrRetInstr++;
1350 fIllegalInstr = true;
1351 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1352 }
1353 else
1354 // no int xx or into either
1355 if (pCpu->pCurInstr->opcode == OP_INT3 || pCpu->pCurInstr->opcode == OP_INT || pCpu->pCurInstr->opcode == OP_INTO)
1356 {
1357 fIllegalInstr = true;
1358 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1359 }
1360 }
1361
1362 pPatch->cbPatchBlockSize += pCpu->opsize;
1363
1364 /* Illegal instruction -> end of analysis phase for this code block */
1365 if (fIllegalInstr || patmIsIllegalInstr(pPatch, pCurInstrGC))
1366 return VINF_SUCCESS;
1367
1368 /* Check for exit points. */
1369 switch (pCpu->pCurInstr->opcode)
1370 {
1371 case OP_SYSEXIT:
1372 return VINF_SUCCESS; /* duplicate it; will fault or emulated in GC. */
1373
1374 case OP_SYSENTER:
1375 case OP_ILLUD2:
1376 //This appears to be some kind of kernel panic in Linux 2.4; no point to analyse more
1377 Log(("Illegal opcode (0xf 0xb) -> return here\n"));
1378 return VINF_SUCCESS;
1379
1380 case OP_STI:
1381 case OP_POPF:
1382 Assert(!(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION)));
1383 /* If out exit point lies within the generated patch jump, then we have to refuse!! */
1384 if (pCurInstrGC > pPatch->pPrivInstrGC && pCurInstrGC < pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32) /* hardcoded patch jump size; cbPatchJump is still zero */
1385 {
1386 Log(("Exit point within patch jump itself!! (%x vs %x)\n", pCurInstrGC, pPatch->pPrivInstrGC));
1387 return VERR_PATCHING_REFUSED;
1388 }
1389 if (pPatch->opcode == OP_PUSHF)
1390 {
1391 if (pCpu->pCurInstr->opcode == OP_POPF)
1392 {
1393 if (pPatch->cbPatchBlockSize >= SIZEOF_NEARJUMP32)
1394 return VINF_SUCCESS;
1395
1396 /* Or else we need to duplicate more instructions, because we can't jump back yet! */
1397 Log(("WARNING: End of block reached, but we need to duplicate some extra instruction to avoid a conflict with the patch jump\n"));
1398 pPatch->flags |= PATMFL_CHECK_SIZE;
1399 }
1400 break; //sti doesn't mark the end of a pushf block; only popf does
1401 }
1402 //else no break
1403 case OP_RETN: /* exit point for function replacement */
1404 return VINF_SUCCESS;
1405
1406 case OP_IRET:
1407 return VINF_SUCCESS; /* exitpoint */
1408
1409 case OP_CPUID:
1410 case OP_CALL:
1411 case OP_JMP:
1412 break;
1413
1414 default:
1415 if (pCpu->pCurInstr->optype & (OPTYPE_PRIVILEGED_NOTRAP))
1416 {
1417 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1418 return VINF_SUCCESS; /* exit point */
1419 }
1420 break;
1421 }
1422
1423 // If single instruction patch, we've copied enough instructions *and* the current instruction is not a relative jump
1424 if ((pPatch->flags & PATMFL_CHECK_SIZE) && pPatch->cbPatchBlockSize > SIZEOF_NEARJUMP32 && !(pCpu->pCurInstr->optype & OPTYPE_RELATIVE_CONTROLFLOW))
1425 {
1426 // The end marker for this kind of patch is any instruction at a location outside our patch jump
1427 Log(("End of block at %RRv size %d\n", pCurInstrGC, pCpu->opsize));
1428 return VINF_SUCCESS;
1429 }
1430
1431 return VWRN_CONTINUE_ANALYSIS;
1432}
1433
1434/**
1435 * Analyses the instructions inside a function for compliance
1436 *
1437 * @returns VBox status code.
1438 * @param pVM The VM to operate on.
1439 * @param pCpu CPU disassembly state
1440 * @param pInstrGC Guest context pointer to privileged instruction
1441 * @param pCurInstrGC Guest context pointer to the current instruction
1442 * @param pCacheRec Cache record ptr
1443 *
1444 */
1445static int patmAnalyseFunctionCallback(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PPATMP2GLOOKUPREC pCacheRec)
1446{
1447 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
1448 bool fIllegalInstr = false;
1449
1450 //Preliminary heuristics:
1451 //- no call instructions
1452 //- ret ends a block
1453
1454 Assert(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION));
1455
1456 // bail out if the patch gets too big
1457 if (pPatch->cbPatchBlockSize >= MAX_PATCH_SIZE)
1458 {
1459 Log(("Code block too big (%x) for function patch at %RRv!!\n", pPatch->cbPatchBlockSize, pCurInstrGC));
1460 fIllegalInstr = true;
1461 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1462 }
1463 else
1464 {
1465 // no unconditinal jumps or calls without fixed displacements
1466 if ( (pCpu->pCurInstr->optype & OPTYPE_CONTROLFLOW)
1467 && (pCpu->pCurInstr->opcode == OP_JMP || pCpu->pCurInstr->opcode == OP_CALL)
1468 )
1469 {
1470 Assert(pCpu->param1.size <= 4 || pCpu->param1.size == 6);
1471 if ( pCpu->param1.size == 6 /* far call/jmp */
1472 || (pCpu->pCurInstr->opcode == OP_CALL && !(pPatch->flags & PATMFL_SUPPORT_CALLS))
1473 || (OP_PARM_VTYPE(pCpu->pCurInstr->param1) != OP_PARM_J && !(pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS))
1474 )
1475 {
1476 fIllegalInstr = true;
1477 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1478 }
1479 }
1480 else /* no far returns */
1481 if (pCpu->pCurInstr->opcode == OP_RETF)
1482 {
1483 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1484 fIllegalInstr = true;
1485 }
1486 else /* no int xx or into either */
1487 if (pCpu->pCurInstr->opcode == OP_INT3 || pCpu->pCurInstr->opcode == OP_INT || pCpu->pCurInstr->opcode == OP_INTO)
1488 {
1489 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1490 fIllegalInstr = true;
1491 }
1492
1493 #if 0
1494 ///@todo we can handle certain in/out and privileged instructions in the guest context
1495 if (pCpu->pCurInstr->optype & OPTYPE_PRIVILEGED && pCpu->pCurInstr->opcode != OP_STI)
1496 {
1497 Log(("Illegal instructions for function patch!!\n"));
1498 return VERR_PATCHING_REFUSED;
1499 }
1500 #endif
1501 }
1502
1503 pPatch->cbPatchBlockSize += pCpu->opsize;
1504
1505 /* Illegal instruction -> end of analysis phase for this code block */
1506 if (fIllegalInstr || patmIsIllegalInstr(pPatch, pCurInstrGC))
1507 {
1508 return VINF_SUCCESS;
1509 }
1510
1511 // Check for exit points
1512 switch (pCpu->pCurInstr->opcode)
1513 {
1514 case OP_ILLUD2:
1515 //This appears to be some kind of kernel panic in Linux 2.4; no point to analyse more
1516 Log(("Illegal opcode (0xf 0xb) -> return here\n"));
1517 return VINF_SUCCESS;
1518
1519 case OP_IRET:
1520 case OP_SYSEXIT: /* will fault or emulated in GC */
1521 case OP_RETN:
1522 return VINF_SUCCESS;
1523
1524 case OP_POPF:
1525 case OP_STI:
1526 return VWRN_CONTINUE_ANALYSIS;
1527 default:
1528 if (pCpu->pCurInstr->optype & (OPTYPE_PRIVILEGED_NOTRAP))
1529 {
1530 patmAddIllegalInstrRecord(pVM, pPatch, pCurInstrGC);
1531 return VINF_SUCCESS; /* exit point */
1532 }
1533 return VWRN_CONTINUE_ANALYSIS;
1534 }
1535
1536 return VWRN_CONTINUE_ANALYSIS;
1537}
1538
1539/**
1540 * Recompiles the instructions in a code block
1541 *
1542 * @returns VBox status code.
1543 * @param pVM The VM to operate on.
1544 * @param pCpu CPU disassembly state
1545 * @param pInstrGC Guest context pointer to privileged instruction
1546 * @param pCurInstrGC Guest context pointer to the current instruction
1547 * @param pCacheRec Cache record ptr
1548 *
1549 */
1550static int patmRecompileCallback(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PPATMP2GLOOKUPREC pCacheRec)
1551{
1552 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
1553 int rc = VINF_SUCCESS;
1554 bool fInhibitIRQInstr = false; /* did the instruction cause PATMFL_INHIBITIRQS to be set? */
1555
1556 LogFlow(("patmRecompileCallback %RRv %RRv\n", pInstrGC, pCurInstrGC));
1557
1558 if ( patmGuestGCPtrToPatchGCPtr(pVM, pPatch, pCurInstrGC) != 0
1559 && !(pPatch->flags & PATMFL_RECOMPILE_NEXT)) /* do not do this when the next instruction *must* be executed! */
1560 {
1561 /*
1562 * Been there, done that; so insert a jump (we don't want to duplicate code)
1563 * no need to record this instruction as it's glue code that never crashes (it had better not!)
1564 */
1565 Log(("patmRecompileCallback: jump to code we've recompiled before %RRv!\n", pCurInstrGC));
1566 return patmPatchGenRelJump(pVM, pPatch, pCurInstrGC, OP_JMP, !!(pCpu->prefix & PREFIX_OPSIZE));
1567 }
1568
1569 if (pPatch->flags & (PATMFL_DUPLICATE_FUNCTION))
1570 {
1571 rc = patmAnalyseFunctionCallback(pVM, pCpu, pInstrGC, pCurInstrGC, pCacheRec);
1572 }
1573 else
1574 rc = patmAnalyseBlockCallback(pVM, pCpu, pInstrGC, pCurInstrGC, pCacheRec);
1575
1576 if (RT_FAILURE(rc))
1577 return rc;
1578
1579 /* Note: Never do a direct return unless a failure is encountered! */
1580
1581 /* Clear recompilation of next instruction flag; we are doing that right here. */
1582 if (pPatch->flags & PATMFL_RECOMPILE_NEXT)
1583 pPatch->flags &= ~PATMFL_RECOMPILE_NEXT;
1584
1585 /* Add lookup record for patch to guest address translation */
1586 patmr3AddP2GLookupRecord(pVM, pPatch, PATCHCODE_PTR_HC(pPatch) + pPatch->uCurPatchOffset, pCurInstrGC, PATM_LOOKUP_BOTHDIR);
1587
1588 /* Update lowest and highest instruction address for this patch */
1589 if (pCurInstrGC < pPatch->pInstrGCLowest)
1590 pPatch->pInstrGCLowest = pCurInstrGC;
1591 else
1592 if (pCurInstrGC > pPatch->pInstrGCHighest)
1593 pPatch->pInstrGCHighest = pCurInstrGC + pCpu->opsize;
1594
1595 /* Illegal instruction -> end of recompile phase for this code block. */
1596 if (patmIsIllegalInstr(pPatch, pCurInstrGC))
1597 {
1598 Log(("Illegal instruction at %RRv -> mark with int 3\n", pCurInstrGC));
1599 rc = patmPatchGenIllegalInstr(pVM, pPatch);
1600 goto end;
1601 }
1602
1603 /* For our first attempt, we'll handle only simple relative jumps (immediate offset coded in instruction).
1604 * Indirect calls are handled below.
1605 */
1606 if ( (pCpu->pCurInstr->optype & OPTYPE_CONTROLFLOW)
1607 && (pCpu->pCurInstr->opcode != OP_CALL || (pPatch->flags & PATMFL_SUPPORT_CALLS))
1608 && (OP_PARM_VTYPE(pCpu->pCurInstr->param1) == OP_PARM_J))
1609 {
1610 RCPTRTYPE(uint8_t *) pTargetGC = PATMResolveBranch(pCpu, pCurInstrGC);
1611 if (pTargetGC == 0)
1612 {
1613 Log(("We don't support far jumps here!! (%08X)\n", pCpu->param1.flags));
1614 return VERR_PATCHING_REFUSED;
1615 }
1616
1617 if (pCpu->pCurInstr->opcode == OP_CALL)
1618 {
1619 Assert(!PATMIsPatchGCAddr(pVM, pTargetGC));
1620 rc = patmPatchGenCall(pVM, pPatch, pCpu, pCurInstrGC, pTargetGC, false);
1621 if (RT_FAILURE(rc))
1622 goto end;
1623 }
1624 else
1625 rc = patmPatchGenRelJump(pVM, pPatch, pTargetGC, pCpu->pCurInstr->opcode, !!(pCpu->prefix & PREFIX_OPSIZE));
1626
1627 if (RT_SUCCESS(rc))
1628 rc = VWRN_CONTINUE_RECOMPILE;
1629
1630 goto end;
1631 }
1632
1633 switch (pCpu->pCurInstr->opcode)
1634 {
1635 case OP_CLI:
1636 {
1637 /* If a cli is found while duplicating instructions for another patch, then it's of vital importance to continue
1638 * until we've found the proper exit point(s).
1639 */
1640 if ( pCurInstrGC != pInstrGC
1641 && !(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION))
1642 )
1643 {
1644 Log(("cli instruction found in other instruction patch block; force it to continue & find an exit point\n"));
1645 pPatch->flags &= ~(PATMFL_CHECK_SIZE | PATMFL_SINGLE_INSTRUCTION);
1646 }
1647 /* Set by irq inhibition; no longer valid now. */
1648 pPatch->flags &= ~PATMFL_GENERATE_JUMPTOGUEST;
1649
1650 rc = patmPatchGenCli(pVM, pPatch);
1651 if (RT_SUCCESS(rc))
1652 rc = VWRN_CONTINUE_RECOMPILE;
1653 break;
1654 }
1655
1656 case OP_MOV:
1657 if (pCpu->pCurInstr->optype & OPTYPE_POTENTIALLY_DANGEROUS)
1658 {
1659 /* mov ss, src? */
1660 if ( (pCpu->param1.flags & USE_REG_SEG)
1661 && (pCpu->param1.base.reg_seg == DIS_SELREG_SS))
1662 {
1663 Log(("Force recompilation of next instruction for OP_MOV at %RRv\n", pCurInstrGC));
1664 pPatch->flags |= PATMFL_RECOMPILE_NEXT;
1665 /** @todo this could cause a fault (ring 0 selector being loaded in ring 1) */
1666 }
1667#if 0 /* necessary for Haiku */
1668 else
1669 if ( (pCpu->param2.flags & USE_REG_SEG)
1670 && (pCpu->param2.base.reg_seg == USE_REG_SS)
1671 && (pCpu->param1.flags & (USE_REG_GEN32|USE_REG_GEN16))) /** @todo memory operand must in theory be handled too */
1672 {
1673 /* mov GPR, ss */
1674 rc = patmPatchGenMovFromSS(pVM, pPatch, pCpu, pCurInstrGC);
1675 if (RT_SUCCESS(rc))
1676 rc = VWRN_CONTINUE_RECOMPILE;
1677 break;
1678 }
1679#endif
1680 }
1681 goto duplicate_instr;
1682
1683 case OP_POP:
1684 if (pCpu->pCurInstr->param1 == OP_PARM_REG_SS)
1685 {
1686 Assert(pCpu->pCurInstr->optype & OPTYPE_INHIBIT_IRQS);
1687
1688 Log(("Force recompilation of next instruction for OP_MOV at %RRv\n", pCurInstrGC));
1689 pPatch->flags |= PATMFL_RECOMPILE_NEXT;
1690 }
1691 goto duplicate_instr;
1692
1693 case OP_STI:
1694 {
1695 RTRCPTR pNextInstrGC = 0; /* by default no inhibit irq */
1696
1697 /* In a sequence of instructions that inhibit irqs, only the first one actually inhibits irqs. */
1698 if (!(pPatch->flags & PATMFL_INHIBIT_IRQS))
1699 {
1700 pPatch->flags |= PATMFL_INHIBIT_IRQS | PATMFL_GENERATE_JUMPTOGUEST;
1701 fInhibitIRQInstr = true;
1702 pNextInstrGC = pCurInstrGC + pCpu->opsize;
1703 Log(("Inhibit irqs for instruction OP_STI at %RRv\n", pCurInstrGC));
1704 }
1705 rc = patmPatchGenSti(pVM, pPatch, pCurInstrGC, pNextInstrGC);
1706
1707 if (RT_SUCCESS(rc))
1708 {
1709 DISCPUSTATE cpu = *pCpu;
1710 unsigned opsize;
1711 int disret;
1712 RCPTRTYPE(uint8_t *) pReturnInstrGC;
1713
1714 pPatch->flags |= PATMFL_FOUND_PATCHEND;
1715
1716 pNextInstrGC = pCurInstrGC + pCpu->opsize;
1717 { /* Force pNextInstrHC out of scope after using it */
1718 uint8_t *pNextInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pNextInstrGC);
1719 if (pNextInstrHC == NULL)
1720 {
1721 AssertFailed();
1722 return VERR_PATCHING_REFUSED;
1723 }
1724
1725 // Disassemble the next instruction
1726 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pNextInstrGC, pNextInstrHC, &opsize, NULL);
1727 }
1728 if (disret == false)
1729 {
1730 AssertMsgFailed(("STI: Disassembly failed (probably page not present) -> return to caller\n"));
1731 return VERR_PATCHING_REFUSED;
1732 }
1733 pReturnInstrGC = pNextInstrGC + opsize;
1734
1735 if ( (pPatch->flags & (PATMFL_DUPLICATE_FUNCTION))
1736 || pReturnInstrGC <= pInstrGC
1737 || pReturnInstrGC - pInstrGC >= SIZEOF_NEARJUMP32
1738 )
1739 {
1740 /* Not an exit point for function duplication patches */
1741 if ( (pPatch->flags & PATMFL_DUPLICATE_FUNCTION)
1742 && RT_SUCCESS(rc))
1743 {
1744 pPatch->flags &= ~PATMFL_GENERATE_JUMPTOGUEST; /* Don't generate a jump back */
1745 rc = VWRN_CONTINUE_RECOMPILE;
1746 }
1747 else
1748 rc = VINF_SUCCESS; //exit point
1749 }
1750 else {
1751 Log(("PATM: sti occurred too soon; refusing patch!\n"));
1752 rc = VERR_PATCHING_REFUSED; //not allowed!!
1753 }
1754 }
1755 break;
1756 }
1757
1758 case OP_POPF:
1759 {
1760 bool fGenerateJmpBack = (pCurInstrGC + pCpu->opsize - pInstrGC >= SIZEOF_NEARJUMP32);
1761
1762 /* Not an exit point for IDT handler or function replacement patches */
1763 /* Note: keep IOPL in mind when changing any of this!! (see comments in PATMA.asm, PATMPopf32Replacement) */
1764 if (pPatch->flags & (PATMFL_IDTHANDLER|PATMFL_DUPLICATE_FUNCTION))
1765 fGenerateJmpBack = false;
1766
1767 rc = patmPatchGenPopf(pVM, pPatch, pCurInstrGC + pCpu->opsize, !!(pCpu->prefix & PREFIX_OPSIZE), fGenerateJmpBack);
1768 if (RT_SUCCESS(rc))
1769 {
1770 if (fGenerateJmpBack == false)
1771 {
1772 /* Not an exit point for IDT handler or function replacement patches */
1773 rc = VWRN_CONTINUE_RECOMPILE;
1774 }
1775 else
1776 {
1777 pPatch->flags |= PATMFL_FOUND_PATCHEND;
1778 rc = VINF_SUCCESS; /* exit point! */
1779 }
1780 }
1781 break;
1782 }
1783
1784 case OP_PUSHF:
1785 rc = patmPatchGenPushf(pVM, pPatch, !!(pCpu->prefix & PREFIX_OPSIZE));
1786 if (RT_SUCCESS(rc))
1787 rc = VWRN_CONTINUE_RECOMPILE;
1788 break;
1789
1790 case OP_PUSH:
1791 if (pCpu->pCurInstr->param1 == OP_PARM_REG_CS)
1792 {
1793 rc = patmPatchGenPushCS(pVM, pPatch);
1794 if (RT_SUCCESS(rc))
1795 rc = VWRN_CONTINUE_RECOMPILE;
1796 break;
1797 }
1798 goto duplicate_instr;
1799
1800 case OP_IRET:
1801 Log(("IRET at %RRv\n", pCurInstrGC));
1802 rc = patmPatchGenIret(pVM, pPatch, pCurInstrGC, !!(pCpu->prefix & PREFIX_OPSIZE));
1803 if (RT_SUCCESS(rc))
1804 {
1805 pPatch->flags |= PATMFL_FOUND_PATCHEND;
1806 rc = VINF_SUCCESS; /* exit point by definition */
1807 }
1808 break;
1809
1810 case OP_ILLUD2:
1811 /* This appears to be some kind of kernel panic in Linux 2.4; no point to continue */
1812 rc = patmPatchGenIllegalInstr(pVM, pPatch);
1813 if (RT_SUCCESS(rc))
1814 rc = VINF_SUCCESS; /* exit point by definition */
1815 Log(("Illegal opcode (0xf 0xb)\n"));
1816 break;
1817
1818 case OP_CPUID:
1819 rc = patmPatchGenCpuid(pVM, pPatch, pCurInstrGC);
1820 if (RT_SUCCESS(rc))
1821 rc = VWRN_CONTINUE_RECOMPILE;
1822 break;
1823
1824 case OP_STR:
1825 case OP_SLDT:
1826 rc = patmPatchGenSldtStr(pVM, pPatch, pCpu, pCurInstrGC);
1827 if (RT_SUCCESS(rc))
1828 rc = VWRN_CONTINUE_RECOMPILE;
1829 break;
1830
1831 case OP_SGDT:
1832 case OP_SIDT:
1833 rc = patmPatchGenSxDT(pVM, pPatch, pCpu, pCurInstrGC);
1834 if (RT_SUCCESS(rc))
1835 rc = VWRN_CONTINUE_RECOMPILE;
1836 break;
1837
1838 case OP_RETN:
1839 /* retn is an exit point for function patches */
1840 rc = patmPatchGenRet(pVM, pPatch, pCpu, pCurInstrGC);
1841 if (RT_SUCCESS(rc))
1842 rc = VINF_SUCCESS; /* exit point by definition */
1843 break;
1844
1845 case OP_SYSEXIT:
1846 /* Duplicate it, so it can be emulated in GC (or fault). */
1847 rc = patmPatchGenDuplicate(pVM, pPatch, pCpu, pCurInstrGC);
1848 if (RT_SUCCESS(rc))
1849 rc = VINF_SUCCESS; /* exit point by definition */
1850 break;
1851
1852 case OP_CALL:
1853 Assert(pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS);
1854 /* In interrupt gate handlers it's possible to encounter jumps or calls when IF has been enabled again.
1855 * In that case we'll jump to the original instruction and continue from there. Otherwise an int 3 is executed.
1856 */
1857 Assert(pCpu->param1.size == 4 || pCpu->param1.size == 6);
1858 if (pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS && pCpu->param1.size == 4 /* no far calls! */)
1859 {
1860 rc = patmPatchGenCall(pVM, pPatch, pCpu, pCurInstrGC, (RTRCPTR)0xDEADBEEF, true);
1861 if (RT_SUCCESS(rc))
1862 {
1863 rc = VWRN_CONTINUE_RECOMPILE;
1864 }
1865 break;
1866 }
1867 goto gen_illegal_instr;
1868
1869 case OP_JMP:
1870 Assert(pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS);
1871 /* In interrupt gate handlers it's possible to encounter jumps or calls when IF has been enabled again.
1872 * In that case we'll jump to the original instruction and continue from there. Otherwise an int 3 is executed.
1873 */
1874 Assert(pCpu->param1.size == 4 || pCpu->param1.size == 6);
1875 if (pPatch->flags & PATMFL_SUPPORT_INDIRECT_CALLS && pCpu->param1.size == 4 /* no far jumps! */)
1876 {
1877 rc = patmPatchGenJump(pVM, pPatch, pCpu, pCurInstrGC);
1878 if (RT_SUCCESS(rc))
1879 rc = VINF_SUCCESS; /* end of branch */
1880 break;
1881 }
1882 goto gen_illegal_instr;
1883
1884 case OP_INT3:
1885 case OP_INT:
1886 case OP_INTO:
1887 goto gen_illegal_instr;
1888
1889 case OP_MOV_DR:
1890 /* Note: currently we let DRx writes cause a trap d; our trap handler will decide to interpret it or not. */
1891 if (pCpu->pCurInstr->param2 == OP_PARM_Dd)
1892 {
1893 rc = patmPatchGenMovDebug(pVM, pPatch, pCpu);
1894 if (RT_SUCCESS(rc))
1895 rc = VWRN_CONTINUE_RECOMPILE;
1896 break;
1897 }
1898 goto duplicate_instr;
1899
1900 case OP_MOV_CR:
1901 /* Note: currently we let CRx writes cause a trap d; our trap handler will decide to interpret it or not. */
1902 if (pCpu->pCurInstr->param2 == OP_PARM_Cd)
1903 {
1904 rc = patmPatchGenMovControl(pVM, pPatch, pCpu);
1905 if (RT_SUCCESS(rc))
1906 rc = VWRN_CONTINUE_RECOMPILE;
1907 break;
1908 }
1909 goto duplicate_instr;
1910
1911 default:
1912 if (pCpu->pCurInstr->optype & (OPTYPE_CONTROLFLOW | OPTYPE_PRIVILEGED_NOTRAP))
1913 {
1914gen_illegal_instr:
1915 rc = patmPatchGenIllegalInstr(pVM, pPatch);
1916 if (RT_SUCCESS(rc))
1917 rc = VINF_SUCCESS; /* exit point by definition */
1918 }
1919 else
1920 {
1921duplicate_instr:
1922 Log(("patmPatchGenDuplicate\n"));
1923 rc = patmPatchGenDuplicate(pVM, pPatch, pCpu, pCurInstrGC);
1924 if (RT_SUCCESS(rc))
1925 rc = VWRN_CONTINUE_RECOMPILE;
1926 }
1927 break;
1928 }
1929
1930end:
1931
1932 if ( !fInhibitIRQInstr
1933 && (pPatch->flags & PATMFL_INHIBIT_IRQS))
1934 {
1935 int rc2;
1936 RTRCPTR pNextInstrGC = pCurInstrGC + pCpu->opsize;
1937
1938 pPatch->flags &= ~PATMFL_INHIBIT_IRQS;
1939 Log(("Clear inhibit IRQ flag at %RRv\n", pCurInstrGC));
1940 if (pPatch->flags & PATMFL_GENERATE_JUMPTOGUEST)
1941 {
1942 Log(("patmRecompileCallback: generate jump back to guest (%RRv) after fused instruction\n", pNextInstrGC));
1943
1944 rc2 = patmPatchGenJumpToGuest(pVM, pPatch, pNextInstrGC, true /* clear inhibit irq flag */);
1945 pPatch->flags &= ~PATMFL_GENERATE_JUMPTOGUEST;
1946 rc = VINF_SUCCESS; /* end of the line */
1947 }
1948 else
1949 {
1950 rc2 = patmPatchGenClearInhibitIRQ(pVM, pPatch, pNextInstrGC);
1951 }
1952 if (RT_FAILURE(rc2))
1953 rc = rc2;
1954 }
1955
1956 if (RT_SUCCESS(rc))
1957 {
1958 // If single instruction patch, we've copied enough instructions *and* the current instruction is not a relative jump
1959 if ( (pPatch->flags & PATMFL_CHECK_SIZE)
1960 && pCurInstrGC + pCpu->opsize - pInstrGC >= SIZEOF_NEARJUMP32
1961 && !(pCpu->pCurInstr->optype & OPTYPE_RELATIVE_CONTROLFLOW)
1962 && !(pPatch->flags & PATMFL_RECOMPILE_NEXT) /* do not do this when the next instruction *must* be executed! */
1963 )
1964 {
1965 RTRCPTR pNextInstrGC = pCurInstrGC + pCpu->opsize;
1966
1967 // The end marker for this kind of patch is any instruction at a location outside our patch jump
1968 Log(("patmRecompileCallback: end found for single instruction patch at %RRv opsize %d\n", pNextInstrGC, pCpu->opsize));
1969
1970 rc = patmPatchGenJumpToGuest(pVM, pPatch, pNextInstrGC);
1971 AssertRC(rc);
1972 }
1973 }
1974 return rc;
1975}
1976
1977
1978#ifdef LOG_ENABLED
1979
1980/* Add a disasm jump record (temporary for prevent duplicate analysis)
1981 *
1982 * @param pVM The VM to operate on.
1983 * @param pPatch Patch structure ptr
1984 * @param pInstrGC Guest context pointer to privileged instruction
1985 *
1986 */
1987static void patmPatchAddDisasmJump(PVM pVM, PPATCHINFO pPatch, RTRCPTR pInstrGC)
1988{
1989 PAVLPVNODECORE pRec;
1990
1991 pRec = (PAVLPVNODECORE)MMR3HeapAllocZ(pVM, MM_TAG_PATM_PATCH, sizeof(*pRec));
1992 Assert(pRec);
1993 pRec->Key = (AVLPVKEY)pInstrGC;
1994
1995 int ret = RTAvlPVInsert(&pPatch->pTempInfo->DisasmJumpTree, pRec);
1996 Assert(ret);
1997}
1998
1999/**
2000 * Checks if jump target has been analysed before.
2001 *
2002 * @returns VBox status code.
2003 * @param pPatch Patch struct
2004 * @param pInstrGC Jump target
2005 *
2006 */
2007static bool patmIsKnownDisasmJump(PPATCHINFO pPatch, RTRCPTR pInstrGC)
2008{
2009 PAVLPVNODECORE pRec;
2010
2011 pRec = RTAvlPVGet(&pPatch->pTempInfo->DisasmJumpTree, (AVLPVKEY)pInstrGC);
2012 if (pRec)
2013 return true;
2014 return false;
2015}
2016
2017/**
2018 * For proper disassembly of the final patch block
2019 *
2020 * @returns VBox status code.
2021 * @param pVM The VM to operate on.
2022 * @param pCpu CPU disassembly state
2023 * @param pInstrGC Guest context pointer to privileged instruction
2024 * @param pCurInstrGC Guest context pointer to the current instruction
2025 * @param pCacheRec Cache record ptr
2026 *
2027 */
2028int patmr3DisasmCallback(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PPATMP2GLOOKUPREC pCacheRec)
2029{
2030 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
2031
2032 if (pCpu->pCurInstr->opcode == OP_INT3)
2033 {
2034 /* Could be an int3 inserted in a call patch. Check to be sure */
2035 DISCPUSTATE cpu;
2036 RTRCPTR pOrgJumpGC;
2037 uint32_t dummy;
2038
2039 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2040 pOrgJumpGC = patmPatchGCPtr2GuestGCPtr(pVM, pPatch, pCurInstrGC);
2041
2042 { /* Force pOrgJumpHC out of scope after using it */
2043 uint8_t *pOrgJumpHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pOrgJumpGC);
2044
2045 bool disret = PATMR3DISInstr(pVM, pPatch, &cpu, pOrgJumpGC, pOrgJumpHC, &dummy, NULL);
2046 if (!disret || cpu.pCurInstr->opcode != OP_CALL || cpu.param1.size != 4 /* only near calls */)
2047 return VINF_SUCCESS;
2048 }
2049 return VWRN_CONTINUE_ANALYSIS;
2050 }
2051
2052 if ( pCpu->pCurInstr->opcode == OP_ILLUD2
2053 && PATMIsPatchGCAddr(pVM, pCurInstrGC))
2054 {
2055 /* the indirect call patch contains an 0xF/0xB illegal instr to call for assistance; check for this and continue */
2056 return VWRN_CONTINUE_ANALYSIS;
2057 }
2058
2059 if ( (pCpu->pCurInstr->opcode == OP_CALL && !(pPatch->flags & PATMFL_SUPPORT_CALLS))
2060 || pCpu->pCurInstr->opcode == OP_INT
2061 || pCpu->pCurInstr->opcode == OP_IRET
2062 || pCpu->pCurInstr->opcode == OP_RETN
2063 || pCpu->pCurInstr->opcode == OP_RETF
2064 )
2065 {
2066 return VINF_SUCCESS;
2067 }
2068
2069 if (pCpu->pCurInstr->opcode == OP_ILLUD2)
2070 return VINF_SUCCESS;
2071
2072 return VWRN_CONTINUE_ANALYSIS;
2073}
2074
2075
2076/**
2077 * Disassembles the code stream until the callback function detects a failure or decides everything is acceptable
2078 *
2079 * @returns VBox status code.
2080 * @param pVM The VM to operate on.
2081 * @param pInstrGC Guest context pointer to the initial privileged instruction
2082 * @param pCurInstrGC Guest context pointer to the current instruction
2083 * @param pfnPATMR3Disasm Callback for testing the disassembled instruction
2084 * @param pCacheRec Cache record ptr
2085 *
2086 */
2087int patmr3DisasmCode(PVM pVM, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PFN_PATMR3ANALYSE pfnPATMR3Disasm, PPATMP2GLOOKUPREC pCacheRec)
2088{
2089 DISCPUSTATE cpu;
2090 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
2091 int rc = VWRN_CONTINUE_ANALYSIS;
2092 uint32_t opsize, delta;
2093 R3PTRTYPE(uint8_t *) pCurInstrHC = 0;
2094 bool disret;
2095 char szOutput[256];
2096
2097 Assert(pCurInstrHC != PATCHCODE_PTR_HC(pPatch) || pPatch->pTempInfo->DisasmJumpTree == 0);
2098
2099 /* We need this to determine branch targets (and for disassembling). */
2100 delta = pVM->patm.s.pPatchMemGC - (uintptr_t)pVM->patm.s.pPatchMemHC;
2101
2102 while(rc == VWRN_CONTINUE_ANALYSIS)
2103 {
2104 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2105
2106 pCurInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pCurInstrGC);
2107 if (pCurInstrHC == NULL)
2108 {
2109 rc = VERR_PATCHING_REFUSED;
2110 goto end;
2111 }
2112
2113 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pCurInstrGC, pCurInstrHC, &opsize, szOutput, PATMREAD_RAWCODE);
2114 if (PATMIsPatchGCAddr(pVM, pCurInstrGC))
2115 {
2116 RTRCPTR pOrgInstrGC = patmPatchGCPtr2GuestGCPtr(pVM, pPatch, pCurInstrGC);
2117
2118 if (pOrgInstrGC != pPatch->pTempInfo->pLastDisasmInstrGC)
2119 Log(("DIS %RRv<-%s", pOrgInstrGC, szOutput));
2120 else
2121 Log(("DIS %s", szOutput));
2122
2123 pPatch->pTempInfo->pLastDisasmInstrGC = pOrgInstrGC;
2124 if (patmIsIllegalInstr(pPatch, pOrgInstrGC))
2125 {
2126 rc = VINF_SUCCESS;
2127 goto end;
2128 }
2129 }
2130 else
2131 Log(("DIS: %s", szOutput));
2132
2133 if (disret == false)
2134 {
2135 Log(("Disassembly failed (probably page not present) -> return to caller\n"));
2136 rc = VINF_SUCCESS;
2137 goto end;
2138 }
2139
2140 rc = pfnPATMR3Disasm(pVM, &cpu, pInstrGC, pCurInstrGC, pCacheRec);
2141 if (rc != VWRN_CONTINUE_ANALYSIS) {
2142 break; //done!
2143 }
2144
2145 /* For our first attempt, we'll handle only simple relative jumps and calls (immediate offset coded in instruction) */
2146 if ( (cpu.pCurInstr->optype & OPTYPE_CONTROLFLOW)
2147 && (OP_PARM_VTYPE(cpu.pCurInstr->param1) == OP_PARM_J)
2148 && cpu.pCurInstr->opcode != OP_CALL /* complete functions are replaced; don't bother here. */
2149 )
2150 {
2151 RTRCPTR pTargetGC = PATMResolveBranch(&cpu, pCurInstrGC);
2152 RTRCPTR pOrgTargetGC;
2153
2154 if (pTargetGC == 0)
2155 {
2156 Log(("We don't support far jumps here!! (%08X)\n", cpu.param1.flags));
2157 rc = VERR_PATCHING_REFUSED;
2158 break;
2159 }
2160
2161 if (!PATMIsPatchGCAddr(pVM, pTargetGC))
2162 {
2163 //jump back to guest code
2164 rc = VINF_SUCCESS;
2165 goto end;
2166 }
2167 pOrgTargetGC = PATMR3PatchToGCPtr(pVM, pTargetGC, 0);
2168
2169 if (patmIsCommonIDTHandlerPatch(pVM, pOrgTargetGC))
2170 {
2171 rc = VINF_SUCCESS;
2172 goto end;
2173 }
2174
2175 if (patmIsKnownDisasmJump(pPatch, pTargetGC) == false)
2176 {
2177 /* New jump, let's check it. */
2178 patmPatchAddDisasmJump(pVM, pPatch, pTargetGC);
2179
2180 if (cpu.pCurInstr->opcode == OP_CALL) pPatch->pTempInfo->nrCalls++;
2181 rc = patmr3DisasmCode(pVM, pInstrGC, pTargetGC, pfnPATMR3Disasm, pCacheRec);
2182 if (cpu.pCurInstr->opcode == OP_CALL) pPatch->pTempInfo->nrCalls--;
2183
2184 if (rc != VINF_SUCCESS) {
2185 break; //done!
2186 }
2187 }
2188 if (cpu.pCurInstr->opcode == OP_JMP)
2189 {
2190 /* Unconditional jump; return to caller. */
2191 rc = VINF_SUCCESS;
2192 goto end;
2193 }
2194
2195 rc = VWRN_CONTINUE_ANALYSIS;
2196 }
2197 pCurInstrGC += opsize;
2198 }
2199end:
2200 return rc;
2201}
2202
2203/**
2204 * Disassembles the code stream until the callback function detects a failure or decides everything is acceptable
2205 *
2206 * @returns VBox status code.
2207 * @param pVM The VM to operate on.
2208 * @param pInstrGC Guest context pointer to the initial privileged instruction
2209 * @param pCurInstrGC Guest context pointer to the current instruction
2210 * @param pfnPATMR3Disasm Callback for testing the disassembled instruction
2211 * @param pCacheRec Cache record ptr
2212 *
2213 */
2214int patmr3DisasmCodeStream(PVM pVM, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PFN_PATMR3ANALYSE pfnPATMR3Disasm, PPATMP2GLOOKUPREC pCacheRec)
2215{
2216 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
2217
2218 int rc = patmr3DisasmCode(pVM, pInstrGC, pCurInstrGC, pfnPATMR3Disasm, pCacheRec);
2219 /* Free all disasm jump records. */
2220 patmEmptyTree(pVM, &pPatch->pTempInfo->DisasmJumpTree);
2221 return rc;
2222}
2223
2224#endif /* LOG_ENABLED */
2225
2226/**
2227 * Detects it the specified address falls within a 5 byte jump generated for an active patch.
2228 * If so, this patch is permanently disabled.
2229 *
2230 * @param pVM The VM to operate on.
2231 * @param pInstrGC Guest context pointer to instruction
2232 * @param pConflictGC Guest context pointer to check
2233 *
2234 * @note also checks for patch hints to make sure they can never be enabled if a conflict is present.
2235 *
2236 */
2237VMMR3DECL(int) PATMR3DetectConflict(PVM pVM, RTRCPTR pInstrGC, RTRCPTR pConflictGC)
2238{
2239 PPATCHINFO pTargetPatch = PATMFindActivePatchByEntrypoint(pVM, pConflictGC, true /* include patch hints */);
2240 if (pTargetPatch)
2241 {
2242 return patmDisableUnusablePatch(pVM, pInstrGC, pConflictGC, pTargetPatch);
2243 }
2244 return VERR_PATCH_NO_CONFLICT;
2245}
2246
2247/**
2248 * Recompile the code stream until the callback function detects a failure or decides everything is acceptable
2249 *
2250 * @returns VBox status code.
2251 * @param pVM The VM to operate on.
2252 * @param pInstrGC Guest context pointer to privileged instruction
2253 * @param pCurInstrGC Guest context pointer to the current instruction
2254 * @param pfnPATMR3Recompile Callback for testing the disassembled instruction
2255 * @param pCacheRec Cache record ptr
2256 *
2257 */
2258static int patmRecompileCodeStream(PVM pVM, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PFN_PATMR3ANALYSE pfnPATMR3Recompile, PPATMP2GLOOKUPREC pCacheRec)
2259{
2260 DISCPUSTATE cpu;
2261 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
2262 int rc = VWRN_CONTINUE_ANALYSIS;
2263 uint32_t opsize;
2264 R3PTRTYPE(uint8_t *) pCurInstrHC = 0;
2265 bool disret;
2266#ifdef LOG_ENABLED
2267 char szOutput[256];
2268#endif
2269
2270 while (rc == VWRN_CONTINUE_RECOMPILE)
2271 {
2272 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2273
2274 pCurInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pCurInstrGC);
2275 if (pCurInstrHC == NULL)
2276 {
2277 rc = VERR_PATCHING_REFUSED; /* fatal in this case */
2278 goto end;
2279 }
2280#ifdef LOG_ENABLED
2281 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pCurInstrGC, pCurInstrHC, &opsize, szOutput);
2282 Log(("Recompile: %s", szOutput));
2283#else
2284 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pCurInstrGC, pCurInstrHC, &opsize, NULL);
2285#endif
2286 if (disret == false)
2287 {
2288 Log(("Disassembly failed (probably page not present) -> return to caller\n"));
2289
2290 /* Add lookup record for patch to guest address translation */
2291 patmr3AddP2GLookupRecord(pVM, pPatch, PATCHCODE_PTR_HC(pPatch) + pPatch->uCurPatchOffset, pCurInstrGC, PATM_LOOKUP_BOTHDIR);
2292 patmPatchGenIllegalInstr(pVM, pPatch);
2293 rc = VINF_SUCCESS; /* Note: don't fail here; we might refuse an important patch!! */
2294 goto end;
2295 }
2296
2297 rc = pfnPATMR3Recompile(pVM, &cpu, pInstrGC, pCurInstrGC, pCacheRec);
2298 if (rc != VWRN_CONTINUE_RECOMPILE)
2299 {
2300 /* If irqs are inhibited because of the current instruction, then we must make sure the next one is executed! */
2301 if ( rc == VINF_SUCCESS
2302 && (pPatch->flags & PATMFL_INHIBIT_IRQS))
2303 {
2304 DISCPUSTATE cpunext;
2305 uint32_t opsizenext;
2306 uint8_t *pNextInstrHC;
2307 RTRCPTR pNextInstrGC = pCurInstrGC + opsize;
2308
2309 Log(("patmRecompileCodeStream: irqs inhibited by instruction %RRv\n", pNextInstrGC));
2310
2311 /* Certain instructions (e.g. sti) force the next instruction to be executed before any interrupts can occur.
2312 * Recompile the next instruction as well
2313 */
2314 pNextInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pNextInstrGC);
2315 if (pNextInstrHC == NULL)
2316 {
2317 rc = VERR_PATCHING_REFUSED; /* fatal in this case */
2318 goto end;
2319 }
2320 cpunext.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2321 disret = PATMR3DISInstr(pVM, pPatch, &cpunext, pNextInstrGC, pNextInstrHC, &opsizenext, NULL);
2322 if (disret == false)
2323 {
2324 rc = VERR_PATCHING_REFUSED; /* fatal in this case */
2325 goto end;
2326 }
2327 switch(cpunext.pCurInstr->opcode)
2328 {
2329 case OP_IRET: /* inhibit cleared in generated code */
2330 case OP_SYSEXIT: /* faults; inhibit should be cleared in HC handling */
2331 case OP_HLT:
2332 break; /* recompile these */
2333
2334 default:
2335 if (cpunext.pCurInstr->optype & OPTYPE_CONTROLFLOW)
2336 {
2337 Log(("Unexpected control flow instruction after inhibit irq instruction\n"));
2338
2339 rc = patmPatchGenJumpToGuest(pVM, pPatch, pNextInstrGC, true /* clear inhibit irq flag */);
2340 AssertRC(rc);
2341 pPatch->flags &= ~PATMFL_INHIBIT_IRQS;
2342 goto end; /** @todo should be ok to ignore instruction fusing in this case */
2343 }
2344 break;
2345 }
2346
2347 /* Note: after a cli we must continue to a proper exit point */
2348 if (cpunext.pCurInstr->opcode != OP_CLI)
2349 {
2350 rc = pfnPATMR3Recompile(pVM, &cpunext, pInstrGC, pNextInstrGC, pCacheRec);
2351 if (RT_SUCCESS(rc))
2352 {
2353 rc = VINF_SUCCESS;
2354 goto end;
2355 }
2356 break;
2357 }
2358 else
2359 rc = VWRN_CONTINUE_RECOMPILE;
2360 }
2361 else
2362 break; /* done! */
2363 }
2364
2365 /** @todo continue with the instructions following the jump and then recompile the jump target code */
2366
2367
2368 /* For our first attempt, we'll handle only simple relative jumps and calls (immediate offset coded in instruction). */
2369 if ( (cpu.pCurInstr->optype & OPTYPE_CONTROLFLOW)
2370 && (OP_PARM_VTYPE(cpu.pCurInstr->param1) == OP_PARM_J)
2371 && cpu.pCurInstr->opcode != OP_CALL /* complete functions are replaced; don't bother here. */
2372 )
2373 {
2374 RCPTRTYPE(uint8_t *) addr = PATMResolveBranch(&cpu, pCurInstrGC);
2375 if (addr == 0)
2376 {
2377 Log(("We don't support far jumps here!! (%08X)\n", cpu.param1.flags));
2378 rc = VERR_PATCHING_REFUSED;
2379 break;
2380 }
2381
2382 Log(("Jump encountered target %RRv\n", addr));
2383
2384 /* We don't check if the branch target lies in a valid page as we've already done that in the analysis phase. */
2385 if (!(cpu.pCurInstr->optype & OPTYPE_UNCOND_CONTROLFLOW))
2386 {
2387 Log(("patmRecompileCodeStream continue passed conditional jump\n"));
2388 /* First we need to finish this linear code stream until the next exit point. */
2389 rc = patmRecompileCodeStream(pVM, pInstrGC, pCurInstrGC+opsize, pfnPATMR3Recompile, pCacheRec);
2390 if (RT_FAILURE(rc))
2391 {
2392 Log(("patmRecompileCodeStream fatal error %d\n", rc));
2393 break; //fatal error
2394 }
2395 }
2396
2397 if (patmGuestGCPtrToPatchGCPtr(pVM, pPatch, addr) == 0)
2398 {
2399 /* New code; let's recompile it. */
2400 Log(("patmRecompileCodeStream continue with jump\n"));
2401
2402 /*
2403 * If we are jumping to an existing patch (or within 5 bytes of the entrypoint), then we must temporarily disable
2404 * this patch so we can continue our analysis
2405 *
2406 * We rely on CSAM to detect and resolve conflicts
2407 */
2408 PPATCHINFO pTargetPatch = PATMFindActivePatchByEntrypoint(pVM, addr);
2409 if(pTargetPatch)
2410 {
2411 Log(("Found active patch at target %RRv (%RRv) -> temporarily disabling it!!\n", addr, pTargetPatch->pPrivInstrGC));
2412 PATMR3DisablePatch(pVM, pTargetPatch->pPrivInstrGC);
2413 }
2414
2415 if (cpu.pCurInstr->opcode == OP_CALL) pPatch->pTempInfo->nrCalls++;
2416 rc = patmRecompileCodeStream(pVM, pInstrGC, addr, pfnPATMR3Recompile, pCacheRec);
2417 if (cpu.pCurInstr->opcode == OP_CALL) pPatch->pTempInfo->nrCalls--;
2418
2419 if(pTargetPatch)
2420 PATMR3EnablePatch(pVM, pTargetPatch->pPrivInstrGC);
2421
2422 if (RT_FAILURE(rc))
2423 {
2424 Log(("patmRecompileCodeStream fatal error %d\n", rc));
2425 break; //done!
2426 }
2427 }
2428 /* Always return to caller here; we're done! */
2429 rc = VINF_SUCCESS;
2430 goto end;
2431 }
2432 else
2433 if (cpu.pCurInstr->optype & OPTYPE_UNCOND_CONTROLFLOW)
2434 {
2435 rc = VINF_SUCCESS;
2436 goto end;
2437 }
2438 pCurInstrGC += opsize;
2439 }
2440end:
2441 Assert(!(pPatch->flags & PATMFL_RECOMPILE_NEXT));
2442 return rc;
2443}
2444
2445
2446/**
2447 * Generate the jump from guest to patch code
2448 *
2449 * @returns VBox status code.
2450 * @param pVM The VM to operate on.
2451 * @param pPatch Patch record
2452 * @param pCacheRec Guest translation lookup cache record
2453 */
2454static int patmGenJumpToPatch(PVM pVM, PPATCHINFO pPatch, PPATMP2GLOOKUPREC pCacheRec, bool fAddFixup = true)
2455{
2456 uint8_t temp[8];
2457 uint8_t *pPB;
2458 int rc;
2459
2460 Assert(pPatch->cbPatchJump <= sizeof(temp));
2461 Assert(!(pPatch->flags & PATMFL_PATCHED_GUEST_CODE));
2462
2463 pPB = PATMGCVirtToHCVirt(pVM, pCacheRec, pPatch->pPrivInstrGC);
2464 Assert(pPB);
2465
2466#ifdef PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES
2467 if (pPatch->flags & PATMFL_JUMP_CONFLICT)
2468 {
2469 Assert(pPatch->pPatchJumpDestGC);
2470
2471 if (pPatch->cbPatchJump == SIZEOF_NEARJUMP32)
2472 {
2473 // jmp [PatchCode]
2474 if (fAddFixup)
2475 {
2476 if (patmPatchAddReloc32(pVM, pPatch, &pPB[1], FIXUP_REL_JMPTOPATCH, pPatch->pPrivInstrGC + pPatch->cbPatchJump, pPatch->pPatchJumpDestGC) != VINF_SUCCESS)
2477 {
2478 Log(("Relocation failed for the jump in the guest code!!\n"));
2479 return VERR_PATCHING_REFUSED;
2480 }
2481 }
2482
2483 temp[0] = pPatch->aPrivInstr[0]; //jump opcode copied from original instruction
2484 *(uint32_t *)&temp[1] = (uint32_t)pPatch->pPatchJumpDestGC - ((uint32_t)pPatch->pPrivInstrGC + pPatch->cbPatchJump); //return address
2485 }
2486 else
2487 if (pPatch->cbPatchJump == SIZEOF_NEAR_COND_JUMP32)
2488 {
2489 // jmp [PatchCode]
2490 if (fAddFixup)
2491 {
2492 if (patmPatchAddReloc32(pVM, pPatch, &pPB[2], FIXUP_REL_JMPTOPATCH, pPatch->pPrivInstrGC + pPatch->cbPatchJump, pPatch->pPatchJumpDestGC) != VINF_SUCCESS)
2493 {
2494 Log(("Relocation failed for the jump in the guest code!!\n"));
2495 return VERR_PATCHING_REFUSED;
2496 }
2497 }
2498
2499 temp[0] = pPatch->aPrivInstr[0]; //jump opcode copied from original instruction
2500 temp[1] = pPatch->aPrivInstr[1]; //jump opcode copied from original instruction
2501 *(uint32_t *)&temp[2] = (uint32_t)pPatch->pPatchJumpDestGC - ((uint32_t)pPatch->pPrivInstrGC + pPatch->cbPatchJump); //return address
2502 }
2503 else
2504 {
2505 Assert(0);
2506 return VERR_PATCHING_REFUSED;
2507 }
2508 }
2509 else
2510#endif
2511 {
2512 Assert(pPatch->cbPatchJump == SIZEOF_NEARJUMP32);
2513
2514 // jmp [PatchCode]
2515 if (fAddFixup)
2516 {
2517 if (patmPatchAddReloc32(pVM, pPatch, &pPB[1], FIXUP_REL_JMPTOPATCH, pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32, PATCHCODE_PTR_GC(pPatch)) != VINF_SUCCESS)
2518 {
2519 Log(("Relocation failed for the jump in the guest code!!\n"));
2520 return VERR_PATCHING_REFUSED;
2521 }
2522 }
2523 temp[0] = 0xE9; //jmp
2524 *(uint32_t *)&temp[1] = (RTRCUINTPTR)PATCHCODE_PTR_GC(pPatch) - ((RTRCUINTPTR)pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32); //return address
2525 }
2526 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pPatch->pPrivInstrGC, temp, pPatch->cbPatchJump);
2527 AssertRC(rc);
2528
2529 if (rc == VINF_SUCCESS)
2530 pPatch->flags |= PATMFL_PATCHED_GUEST_CODE;
2531
2532 return rc;
2533}
2534
2535/**
2536 * Remove the jump from guest to patch code
2537 *
2538 * @returns VBox status code.
2539 * @param pVM The VM to operate on.
2540 * @param pPatch Patch record
2541 */
2542static int patmRemoveJumpToPatch(PVM pVM, PPATCHINFO pPatch)
2543{
2544#ifdef DEBUG
2545 DISCPUSTATE cpu;
2546 char szOutput[256];
2547 uint32_t opsize, i = 0;
2548 bool disret;
2549
2550 while (i < pPatch->cbPrivInstr)
2551 {
2552 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2553 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC + i, NULL, &opsize, szOutput);
2554 if (disret == false)
2555 break;
2556
2557 Log(("Org patch jump: %s", szOutput));
2558 Assert(opsize);
2559 i += opsize;
2560 }
2561#endif
2562
2563 /* Restore original code (privileged instruction + following instructions that were overwritten because of the 5/6 byte jmp). */
2564 int rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pPatch->pPrivInstrGC, pPatch->aPrivInstr, pPatch->cbPatchJump);
2565#ifdef DEBUG
2566 if (rc == VINF_SUCCESS)
2567 {
2568 i = 0;
2569 while(i < pPatch->cbPrivInstr)
2570 {
2571 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2572 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC + i, NULL, &opsize, szOutput);
2573 if (disret == false)
2574 break;
2575
2576 Log(("Org instr: %s", szOutput));
2577 Assert(opsize);
2578 i += opsize;
2579 }
2580 }
2581#endif
2582 pPatch->flags &= ~PATMFL_PATCHED_GUEST_CODE;
2583 return rc;
2584}
2585
2586/**
2587 * Generate the call from guest to patch code
2588 *
2589 * @returns VBox status code.
2590 * @param pVM The VM to operate on.
2591 * @param pPatch Patch record
2592 * @param pInstrHC HC address where to insert the jump
2593 * @param pCacheRec Guest translation cache record
2594 */
2595static int patmGenCallToPatch(PVM pVM, PPATCHINFO pPatch, RTRCPTR pTargetGC, PPATMP2GLOOKUPREC pCacheRec, bool fAddFixup = true)
2596{
2597 uint8_t temp[8];
2598 uint8_t *pPB;
2599 int rc;
2600
2601 Assert(pPatch->cbPatchJump <= sizeof(temp));
2602
2603 pPB = PATMGCVirtToHCVirt(pVM, pCacheRec, pPatch->pPrivInstrGC);
2604 Assert(pPB);
2605
2606 Assert(pPatch->cbPatchJump == SIZEOF_NEARJUMP32);
2607
2608 // jmp [PatchCode]
2609 if (fAddFixup)
2610 {
2611 if (patmPatchAddReloc32(pVM, pPatch, &pPB[1], FIXUP_REL_JMPTOPATCH, pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32, pTargetGC) != VINF_SUCCESS)
2612 {
2613 Log(("Relocation failed for the jump in the guest code!!\n"));
2614 return VERR_PATCHING_REFUSED;
2615 }
2616 }
2617
2618 Assert(pPatch->aPrivInstr[0] == 0xE8 || pPatch->aPrivInstr[0] == 0xE9); /* call or jmp */
2619 temp[0] = pPatch->aPrivInstr[0];
2620 *(uint32_t *)&temp[1] = (uint32_t)pTargetGC - ((uint32_t)pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32); //return address
2621
2622 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pPatch->pPrivInstrGC, temp, pPatch->cbPatchJump);
2623 AssertRC(rc);
2624
2625 return rc;
2626}
2627
2628
2629/**
2630 * Patch cli/sti pushf/popf instruction block at specified location
2631 *
2632 * @returns VBox status code.
2633 * @param pVM The VM to operate on.
2634 * @param pInstrGC Guest context point to privileged instruction
2635 * @param pInstrHC Host context point to privileged instruction
2636 * @param uOpcode Instruction opcode
2637 * @param uOpSize Size of starting instruction
2638 * @param pPatchRec Patch record
2639 *
2640 * @note returns failure if patching is not allowed or possible
2641 *
2642 */
2643VMMR3DECL(int) PATMR3PatchBlock(PVM pVM, RTRCPTR pInstrGC, R3PTRTYPE(uint8_t *) pInstrHC,
2644 uint32_t uOpcode, uint32_t uOpSize, PPATMPATCHREC pPatchRec)
2645{
2646 PPATCHINFO pPatch = &pPatchRec->patch;
2647 int rc = VERR_PATCHING_REFUSED;
2648 DISCPUSTATE cpu;
2649 uint32_t orgOffsetPatchMem = ~0;
2650 RTRCPTR pInstrStart;
2651#ifdef LOG_ENABLED
2652 uint32_t opsize;
2653 char szOutput[256];
2654 bool disret;
2655#endif
2656
2657 /* Save original offset (in case of failures later on) */
2658 /** @todo use the hypervisor heap (that has quite a few consequences for save/restore though) */
2659 orgOffsetPatchMem = pVM->patm.s.offPatchMem;
2660
2661 Assert(!(pPatch->flags & (PATMFL_GUEST_SPECIFIC|PATMFL_USER_MODE|PATMFL_TRAPHANDLER)));
2662 switch (uOpcode)
2663 {
2664 case OP_MOV:
2665 break;
2666
2667 case OP_CLI:
2668 case OP_PUSHF:
2669 /* We can 'call' a cli or pushf patch. It will either return to the original guest code when IF is set again, or fault. */
2670 /* Note: special precautions are taken when disabling and enabling such patches. */
2671 pPatch->flags |= PATMFL_CALLABLE_AS_FUNCTION;
2672 break;
2673
2674 default:
2675 if (!(pPatch->flags & PATMFL_IDTHANDLER))
2676 {
2677 AssertMsg(0, ("PATMR3PatchBlock: Invalid opcode %x\n", uOpcode));
2678 return VERR_INVALID_PARAMETER;
2679 }
2680 }
2681
2682 if (!(pPatch->flags & (PATMFL_IDTHANDLER|PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT|PATMFL_SYSENTER|PATMFL_INT3_REPLACEMENT_BLOCK)))
2683 pPatch->flags |= PATMFL_MUST_INSTALL_PATCHJMP;
2684
2685 /* If we're going to insert a patch jump, then the jump itself is not allowed to cross a page boundary. */
2686 if ( (pPatch->flags & PATMFL_MUST_INSTALL_PATCHJMP)
2687 && PAGE_ADDRESS(pInstrGC) != PAGE_ADDRESS(pInstrGC + SIZEOF_NEARJUMP32)
2688 )
2689 {
2690 STAM_COUNTER_INC(&pVM->patm.s.StatPageBoundaryCrossed);
2691 Log(("Patch jump would cross page boundary -> refuse!!\n"));
2692 rc = VERR_PATCHING_REFUSED;
2693 goto failure;
2694 }
2695
2696 pPatch->nrPatch2GuestRecs = 0;
2697 pInstrStart = pInstrGC;
2698
2699#ifdef PATM_ENABLE_CALL
2700 pPatch->flags |= PATMFL_SUPPORT_CALLS | PATMFL_SUPPORT_INDIRECT_CALLS;
2701#endif
2702
2703 pPatch->pPatchBlockOffset = pVM->patm.s.offPatchMem;
2704 pPatch->uCurPatchOffset = 0;
2705
2706 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2707
2708 if ((pPatch->flags & (PATMFL_IDTHANDLER|PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT|PATMFL_SYSENTER)) == PATMFL_IDTHANDLER)
2709 {
2710 Assert(pPatch->flags & PATMFL_INTHANDLER);
2711
2712 /* Install fake cli patch (to clear the virtual IF and check int xx parameters) */
2713 rc = patmPatchGenIntEntry(pVM, pPatch, pInstrGC);
2714 if (RT_FAILURE(rc))
2715 goto failure;
2716 }
2717
2718 /***************************************************************************************************************************/
2719 /* Note: We can't insert *any* code before a sysenter handler; some linux guests have an invalid stack at this point!!!!! */
2720 /***************************************************************************************************************************/
2721#ifdef VBOX_WITH_STATISTICS
2722 if (!(pPatch->flags & PATMFL_SYSENTER))
2723 {
2724 rc = patmPatchGenStats(pVM, pPatch, pInstrGC);
2725 if (RT_FAILURE(rc))
2726 goto failure;
2727 }
2728#endif
2729
2730 PATMP2GLOOKUPREC cacheRec;
2731 RT_ZERO(cacheRec);
2732 cacheRec.pPatch = pPatch;
2733
2734 rc = patmRecompileCodeStream(pVM, pInstrGC, pInstrGC, patmRecompileCallback, &cacheRec);
2735 /* Free leftover lock if any. */
2736 if (cacheRec.Lock.pvMap)
2737 {
2738 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
2739 cacheRec.Lock.pvMap = NULL;
2740 }
2741 if (rc != VINF_SUCCESS)
2742 {
2743 Log(("PATMR3PatchCli: patmRecompileCodeStream failed with %d\n", rc));
2744 goto failure;
2745 }
2746
2747 /* Calculated during analysis. */
2748 if (pPatch->cbPatchBlockSize < SIZEOF_NEARJUMP32)
2749 {
2750 /* Most likely cause: we encountered an illegal instruction very early on. */
2751 /** @todo could turn it into an int3 callable patch. */
2752 Log(("PATMR3PatchBlock: patch block too small -> refuse\n"));
2753 rc = VERR_PATCHING_REFUSED;
2754 goto failure;
2755 }
2756
2757 /* size of patch block */
2758 pPatch->cbPatchBlockSize = pPatch->uCurPatchOffset;
2759
2760
2761 /* Update free pointer in patch memory. */
2762 pVM->patm.s.offPatchMem += pPatch->cbPatchBlockSize;
2763 /* Round to next 8 byte boundary. */
2764 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
2765
2766 /*
2767 * Insert into patch to guest lookup tree
2768 */
2769 LogFlow(("Insert %RRv patch offset %RRv\n", pPatchRec->patch.pPrivInstrGC, pPatch->pPatchBlockOffset));
2770 pPatchRec->CoreOffset.Key = pPatch->pPatchBlockOffset;
2771 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, &pPatchRec->CoreOffset);
2772 AssertMsg(rc, ("RTAvlULInsert failed for %x\n", pPatchRec->CoreOffset.Key));
2773 if (!rc)
2774 {
2775 rc = VERR_PATCHING_REFUSED;
2776 goto failure;
2777 }
2778
2779 /* Note that patmr3SetBranchTargets can install additional patches!! */
2780 rc = patmr3SetBranchTargets(pVM, pPatch);
2781 if (rc != VINF_SUCCESS)
2782 {
2783 Log(("PATMR3PatchCli: patmr3SetBranchTargets failed with %d\n", rc));
2784 goto failure;
2785 }
2786
2787#ifdef LOG_ENABLED
2788 Log(("Patch code ----------------------------------------------------------\n"));
2789 patmr3DisasmCodeStream(pVM, PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_GC(pPatch), patmr3DisasmCallback, &cacheRec);
2790 /* Free leftover lock if any. */
2791 if (cacheRec.Lock.pvMap)
2792 {
2793 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
2794 cacheRec.Lock.pvMap = NULL;
2795 }
2796 Log(("Patch code ends -----------------------------------------------------\n"));
2797#endif
2798
2799 /* make a copy of the guest code bytes that will be overwritten */
2800 pPatch->cbPatchJump = SIZEOF_NEARJUMP32;
2801
2802 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pPatch->aPrivInstr, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
2803 AssertRC(rc);
2804
2805 if (pPatch->flags & PATMFL_INT3_REPLACEMENT_BLOCK)
2806 {
2807 /*uint8_t ASMInt3 = 0xCC; - unused */
2808
2809 Log(("PATMR3PatchBlock %RRv -> int 3 callable patch.\n", pPatch->pPrivInstrGC));
2810 /* Replace first opcode byte with 'int 3'. */
2811 rc = patmActivateInt3Patch(pVM, pPatch);
2812 if (RT_FAILURE(rc))
2813 goto failure;
2814
2815 /* normal patch can be turned into an int3 patch -> clear patch jump installation flag. */
2816 pPatch->flags &= ~PATMFL_MUST_INSTALL_PATCHJMP;
2817
2818 pPatch->flags &= ~PATMFL_INSTR_HINT;
2819 STAM_COUNTER_INC(&pVM->patm.s.StatInt3Callable);
2820 }
2821 else
2822 if (pPatch->flags & PATMFL_MUST_INSTALL_PATCHJMP)
2823 {
2824 Assert(!(pPatch->flags & (PATMFL_IDTHANDLER|PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT|PATMFL_SYSENTER|PATMFL_INT3_REPLACEMENT_BLOCK)));
2825 /* now insert a jump in the guest code */
2826 rc = patmGenJumpToPatch(pVM, pPatch, &cacheRec, true);
2827 AssertRC(rc);
2828 if (RT_FAILURE(rc))
2829 goto failure;
2830
2831 }
2832
2833#ifdef LOG_ENABLED
2834 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2835 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput, PATMREAD_RAWCODE);
2836 Log(("%s patch: %s", patmGetInstructionString(pPatch->opcode, pPatch->flags), szOutput));
2837#endif
2838
2839 patmEmptyTree(pVM, &pPatch->pTempInfo->IllegalInstrTree);
2840 pPatch->pTempInfo->nrIllegalInstr = 0;
2841
2842 Log(("Successfully installed %s patch at %RRv\n", patmGetInstructionString(pPatch->opcode, pPatch->flags), pInstrGC));
2843
2844 pPatch->uState = PATCH_ENABLED;
2845 return VINF_SUCCESS;
2846
2847failure:
2848 if (pPatchRec->CoreOffset.Key)
2849 RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, pPatchRec->CoreOffset.Key);
2850
2851 patmEmptyTree(pVM, &pPatch->FixupTree);
2852 pPatch->nrFixups = 0;
2853
2854 patmEmptyTree(pVM, &pPatch->JumpTree);
2855 pPatch->nrJumpRecs = 0;
2856
2857 patmEmptyTree(pVM, &pPatch->pTempInfo->IllegalInstrTree);
2858 pPatch->pTempInfo->nrIllegalInstr = 0;
2859
2860 /* Turn this cli patch into a dummy. */
2861 pPatch->uState = PATCH_REFUSED;
2862 pPatch->pPatchBlockOffset = 0;
2863
2864 // Give back the patch memory we no longer need
2865 Assert(orgOffsetPatchMem != (uint32_t)~0);
2866 pVM->patm.s.offPatchMem = orgOffsetPatchMem;
2867
2868 return rc;
2869}
2870
2871/**
2872 * Patch IDT handler
2873 *
2874 * @returns VBox status code.
2875 * @param pVM The VM to operate on.
2876 * @param pInstrGC Guest context point to privileged instruction
2877 * @param uOpSize Size of starting instruction
2878 * @param pPatchRec Patch record
2879 * @param pCacheRec Cache record ptr
2880 *
2881 * @note returns failure if patching is not allowed or possible
2882 *
2883 */
2884static int patmIdtHandler(PVM pVM, RTRCPTR pInstrGC, uint32_t uOpSize, PPATMPATCHREC pPatchRec, PPATMP2GLOOKUPREC pCacheRec)
2885{
2886 PPATCHINFO pPatch = &pPatchRec->patch;
2887 bool disret;
2888 DISCPUSTATE cpuPush, cpuJmp;
2889 uint32_t opsize;
2890 RTRCPTR pCurInstrGC = pInstrGC;
2891 uint8_t *pCurInstrHC, *pInstrHC;
2892 uint32_t orgOffsetPatchMem = ~0;
2893
2894 pInstrHC = pCurInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pCurInstrGC);
2895 AssertReturn(pCurInstrHC, VERR_PAGE_NOT_PRESENT);
2896
2897 /*
2898 * In Linux it's often the case that many interrupt handlers push a predefined value onto the stack
2899 * and then jump to a common entrypoint. In order not to waste a lot of memory, we will check for this
2900 * condition here and only patch the common entypoint once.
2901 */
2902 cpuPush.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2903 disret = PATMR3DISInstr(pVM, pPatch, &cpuPush, pCurInstrGC, pCurInstrHC, &opsize, NULL);
2904 Assert(disret);
2905 if (disret && cpuPush.pCurInstr->opcode == OP_PUSH)
2906 {
2907 RTRCPTR pJmpInstrGC;
2908 int rc;
2909 pCurInstrGC += opsize;
2910
2911 cpuJmp.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
2912 disret = PATMR3DISInstr(pVM, pPatch, &cpuJmp, pCurInstrGC, pCurInstrHC, &opsize, NULL);
2913 if ( disret
2914 && cpuJmp.pCurInstr->opcode == OP_JMP
2915 && (pJmpInstrGC = PATMResolveBranch(&cpuJmp, pCurInstrGC))
2916 )
2917 {
2918 PPATMPATCHREC pJmpPatch = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pJmpInstrGC);
2919 if (pJmpPatch == 0)
2920 {
2921 /* Patch it first! */
2922 rc = PATMR3InstallPatch(pVM, pJmpInstrGC, pPatch->flags | PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT);
2923 if (rc != VINF_SUCCESS)
2924 goto failure;
2925 pJmpPatch = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pJmpInstrGC);
2926 Assert(pJmpPatch);
2927 }
2928 if (pJmpPatch->patch.uState != PATCH_ENABLED)
2929 goto failure;
2930
2931 /* save original offset (in case of failures later on) */
2932 orgOffsetPatchMem = pVM->patm.s.offPatchMem;
2933
2934 pPatch->pPatchBlockOffset = pVM->patm.s.offPatchMem;
2935 pPatch->uCurPatchOffset = 0;
2936 pPatch->nrPatch2GuestRecs = 0;
2937
2938#ifdef VBOX_WITH_STATISTICS
2939 rc = patmPatchGenStats(pVM, pPatch, pInstrGC);
2940 if (RT_FAILURE(rc))
2941 goto failure;
2942#endif
2943
2944 /* Install fake cli patch (to clear the virtual IF) */
2945 rc = patmPatchGenIntEntry(pVM, pPatch, pInstrGC);
2946 if (RT_FAILURE(rc))
2947 goto failure;
2948
2949 /* Add lookup record for patch to guest address translation (for the push) */
2950 patmr3AddP2GLookupRecord(pVM, pPatch, PATCHCODE_PTR_HC(pPatch) + pPatch->uCurPatchOffset, pInstrGC, PATM_LOOKUP_BOTHDIR);
2951
2952 /* Duplicate push. */
2953 rc = patmPatchGenDuplicate(pVM, pPatch, &cpuPush, pInstrGC);
2954 if (RT_FAILURE(rc))
2955 goto failure;
2956
2957 /* Generate jump to common entrypoint. */
2958 rc = patmPatchGenPatchJump(pVM, pPatch, pCurInstrGC, PATCHCODE_PTR_GC(&pJmpPatch->patch));
2959 if (RT_FAILURE(rc))
2960 goto failure;
2961
2962 /* size of patch block */
2963 pPatch->cbPatchBlockSize = pPatch->uCurPatchOffset;
2964
2965 /* Update free pointer in patch memory. */
2966 pVM->patm.s.offPatchMem += pPatch->cbPatchBlockSize;
2967 /* Round to next 8 byte boundary */
2968 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
2969
2970 /* There's no jump from guest to patch code. */
2971 pPatch->cbPatchJump = 0;
2972
2973
2974#ifdef LOG_ENABLED
2975 Log(("Patch code ----------------------------------------------------------\n"));
2976 patmr3DisasmCodeStream(pVM, PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_GC(pPatch), patmr3DisasmCallback, pCacheRec);
2977 Log(("Patch code ends -----------------------------------------------------\n"));
2978#endif
2979 Log(("Successfully installed IDT handler patch at %RRv\n", pInstrGC));
2980
2981 /*
2982 * Insert into patch to guest lookup tree
2983 */
2984 LogFlow(("Insert %RRv patch offset %RRv\n", pPatchRec->patch.pPrivInstrGC, pPatch->pPatchBlockOffset));
2985 pPatchRec->CoreOffset.Key = pPatch->pPatchBlockOffset;
2986 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, &pPatchRec->CoreOffset);
2987 AssertMsg(rc, ("RTAvlULInsert failed for %x\n", pPatchRec->CoreOffset.Key));
2988
2989 pPatch->uState = PATCH_ENABLED;
2990
2991 return VINF_SUCCESS;
2992 }
2993 }
2994failure:
2995 /* Give back the patch memory we no longer need */
2996 if (orgOffsetPatchMem != (uint32_t)~0)
2997 pVM->patm.s.offPatchMem = orgOffsetPatchMem;
2998
2999 return PATMR3PatchBlock(pVM, pInstrGC, pInstrHC, OP_CLI, uOpSize, pPatchRec);
3000}
3001
3002/**
3003 * Install a trampoline to call a guest trap handler directly
3004 *
3005 * @returns VBox status code.
3006 * @param pVM The VM to operate on.
3007 * @param pInstrGC Guest context point to privileged instruction
3008 * @param pPatchRec Patch record
3009 * @param pCacheRec Cache record ptr
3010 *
3011 */
3012static int patmInstallTrapTrampoline(PVM pVM, RTRCPTR pInstrGC, PPATMPATCHREC pPatchRec, PPATMP2GLOOKUPREC pCacheRec)
3013{
3014 PPATCHINFO pPatch = &pPatchRec->patch;
3015 int rc = VERR_PATCHING_REFUSED;
3016 uint32_t orgOffsetPatchMem = ~0;
3017#ifdef LOG_ENABLED
3018 bool disret;
3019 DISCPUSTATE cpu;
3020 uint32_t opsize;
3021 char szOutput[256];
3022#endif
3023
3024 // save original offset (in case of failures later on)
3025 orgOffsetPatchMem = pVM->patm.s.offPatchMem;
3026
3027 pPatch->pPatchBlockOffset = pVM->patm.s.offPatchMem;
3028 pPatch->uCurPatchOffset = 0;
3029 pPatch->nrPatch2GuestRecs = 0;
3030
3031#ifdef VBOX_WITH_STATISTICS
3032 rc = patmPatchGenStats(pVM, pPatch, pInstrGC);
3033 if (RT_FAILURE(rc))
3034 goto failure;
3035#endif
3036
3037 rc = patmPatchGenTrapEntry(pVM, pPatch, pInstrGC);
3038 if (RT_FAILURE(rc))
3039 goto failure;
3040
3041 /* size of patch block */
3042 pPatch->cbPatchBlockSize = pPatch->uCurPatchOffset;
3043
3044 /* Update free pointer in patch memory. */
3045 pVM->patm.s.offPatchMem += pPatch->cbPatchBlockSize;
3046 /* Round to next 8 byte boundary */
3047 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
3048
3049 /* There's no jump from guest to patch code. */
3050 pPatch->cbPatchJump = 0;
3051
3052#ifdef LOG_ENABLED
3053 Log(("Patch code ----------------------------------------------------------\n"));
3054 patmr3DisasmCodeStream(pVM, PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_GC(pPatch), patmr3DisasmCallback, pCacheRec);
3055 Log(("Patch code ends -----------------------------------------------------\n"));
3056#endif
3057
3058#ifdef LOG_ENABLED
3059 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3060 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput);
3061 Log(("TRAP handler patch: %s", szOutput));
3062#endif
3063 Log(("Successfully installed Trap Trampoline patch at %RRv\n", pInstrGC));
3064
3065 /*
3066 * Insert into patch to guest lookup tree
3067 */
3068 LogFlow(("Insert %RRv patch offset %RRv\n", pPatchRec->patch.pPrivInstrGC, pPatch->pPatchBlockOffset));
3069 pPatchRec->CoreOffset.Key = pPatch->pPatchBlockOffset;
3070 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, &pPatchRec->CoreOffset);
3071 AssertMsg(rc, ("RTAvlULInsert failed for %x\n", pPatchRec->CoreOffset.Key));
3072
3073 pPatch->uState = PATCH_ENABLED;
3074 return VINF_SUCCESS;
3075
3076failure:
3077 AssertMsgFailed(("Failed to install trap handler trampoline!!\n"));
3078
3079 /* Turn this cli patch into a dummy. */
3080 pPatch->uState = PATCH_REFUSED;
3081 pPatch->pPatchBlockOffset = 0;
3082
3083 /* Give back the patch memory we no longer need */
3084 Assert(orgOffsetPatchMem != (uint32_t)~0);
3085 pVM->patm.s.offPatchMem = orgOffsetPatchMem;
3086
3087 return rc;
3088}
3089
3090
3091#ifdef LOG_ENABLED
3092/**
3093 * Check if the instruction is patched as a common idt handler
3094 *
3095 * @returns true or false
3096 * @param pVM The VM to operate on.
3097 * @param pInstrGC Guest context point to the instruction
3098 *
3099 */
3100static bool patmIsCommonIDTHandlerPatch(PVM pVM, RTRCPTR pInstrGC)
3101{
3102 PPATMPATCHREC pRec;
3103
3104 pRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
3105 if (pRec && pRec->patch.flags & PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT)
3106 return true;
3107 return false;
3108}
3109#endif //DEBUG
3110
3111
3112/**
3113 * Duplicates a complete function
3114 *
3115 * @returns VBox status code.
3116 * @param pVM The VM to operate on.
3117 * @param pInstrGC Guest context point to privileged instruction
3118 * @param pPatchRec Patch record
3119 * @param pCacheRec Cache record ptr
3120 *
3121 */
3122static int patmDuplicateFunction(PVM pVM, RTRCPTR pInstrGC, PPATMPATCHREC pPatchRec, PPATMP2GLOOKUPREC pCacheRec)
3123{
3124 PPATCHINFO pPatch = &pPatchRec->patch;
3125 int rc = VERR_PATCHING_REFUSED;
3126 DISCPUSTATE cpu;
3127 uint32_t orgOffsetPatchMem = ~0;
3128
3129 Log(("patmDuplicateFunction %RRv\n", pInstrGC));
3130 /* Save original offset (in case of failures later on). */
3131 orgOffsetPatchMem = pVM->patm.s.offPatchMem;
3132
3133 /* We will not go on indefinitely with call instruction handling. */
3134 if (pVM->patm.s.ulCallDepth > PATM_MAX_CALL_DEPTH)
3135 {
3136 Log(("patmDuplicateFunction: maximum callback depth reached!!\n"));
3137 return VERR_PATCHING_REFUSED;
3138 }
3139
3140 pVM->patm.s.ulCallDepth++;
3141
3142#ifdef PATM_ENABLE_CALL
3143 pPatch->flags |= PATMFL_SUPPORT_CALLS | PATMFL_SUPPORT_INDIRECT_CALLS;
3144#endif
3145
3146 Assert(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION));
3147
3148 pPatch->nrPatch2GuestRecs = 0;
3149 pPatch->pPatchBlockOffset = pVM->patm.s.offPatchMem;
3150 pPatch->uCurPatchOffset = 0;
3151
3152 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3153
3154 /* Note: Set the PATM interrupt flag here; it was cleared before the patched call. (!!!) */
3155 rc = patmPatchGenSetPIF(pVM, pPatch, pInstrGC);
3156 if (RT_FAILURE(rc))
3157 goto failure;
3158
3159#ifdef VBOX_WITH_STATISTICS
3160 rc = patmPatchGenStats(pVM, pPatch, pInstrGC);
3161 if (RT_FAILURE(rc))
3162 goto failure;
3163#endif
3164
3165 rc = patmRecompileCodeStream(pVM, pInstrGC, pInstrGC, patmRecompileCallback, pCacheRec);
3166 if (rc != VINF_SUCCESS)
3167 {
3168 Log(("PATMR3PatchCli: patmRecompileCodeStream failed with %d\n", rc));
3169 goto failure;
3170 }
3171
3172 //size of patch block
3173 pPatch->cbPatchBlockSize = pPatch->uCurPatchOffset;
3174
3175 //update free pointer in patch memory
3176 pVM->patm.s.offPatchMem += pPatch->cbPatchBlockSize;
3177 /* Round to next 8 byte boundary. */
3178 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
3179
3180 pPatch->uState = PATCH_ENABLED;
3181
3182 /*
3183 * Insert into patch to guest lookup tree
3184 */
3185 LogFlow(("Insert %RRv patch offset %RRv\n", pPatchRec->patch.pPrivInstrGC, pPatch->pPatchBlockOffset));
3186 pPatchRec->CoreOffset.Key = pPatch->pPatchBlockOffset;
3187 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, &pPatchRec->CoreOffset);
3188 AssertMsg(rc, ("RTAvloU32Insert failed for %x\n", pPatchRec->CoreOffset.Key));
3189 if (!rc)
3190 {
3191 rc = VERR_PATCHING_REFUSED;
3192 goto failure;
3193 }
3194
3195 /* Note that patmr3SetBranchTargets can install additional patches!! */
3196 rc = patmr3SetBranchTargets(pVM, pPatch);
3197 if (rc != VINF_SUCCESS)
3198 {
3199 Log(("PATMR3PatchCli: patmr3SetBranchTargets failed with %d\n", rc));
3200 goto failure;
3201 }
3202
3203#ifdef LOG_ENABLED
3204 Log(("Patch code ----------------------------------------------------------\n"));
3205 patmr3DisasmCodeStream(pVM, PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_GC(pPatch), patmr3DisasmCallback, pCacheRec);
3206 Log(("Patch code ends -----------------------------------------------------\n"));
3207#endif
3208
3209 Log(("Successfully installed function duplication patch at %RRv\n", pInstrGC));
3210
3211 patmEmptyTree(pVM, &pPatch->pTempInfo->IllegalInstrTree);
3212 pPatch->pTempInfo->nrIllegalInstr = 0;
3213
3214 pVM->patm.s.ulCallDepth--;
3215 STAM_COUNTER_INC(&pVM->patm.s.StatInstalledFunctionPatches);
3216 return VINF_SUCCESS;
3217
3218failure:
3219 if (pPatchRec->CoreOffset.Key)
3220 RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, pPatchRec->CoreOffset.Key);
3221
3222 patmEmptyTree(pVM, &pPatch->FixupTree);
3223 pPatch->nrFixups = 0;
3224
3225 patmEmptyTree(pVM, &pPatch->JumpTree);
3226 pPatch->nrJumpRecs = 0;
3227
3228 patmEmptyTree(pVM, &pPatch->pTempInfo->IllegalInstrTree);
3229 pPatch->pTempInfo->nrIllegalInstr = 0;
3230
3231 /* Turn this cli patch into a dummy. */
3232 pPatch->uState = PATCH_REFUSED;
3233 pPatch->pPatchBlockOffset = 0;
3234
3235 // Give back the patch memory we no longer need
3236 Assert(orgOffsetPatchMem != (uint32_t)~0);
3237 pVM->patm.s.offPatchMem = orgOffsetPatchMem;
3238
3239 pVM->patm.s.ulCallDepth--;
3240 Log(("patmDupicateFunction %RRv failed!!\n", pInstrGC));
3241 return rc;
3242}
3243
3244/**
3245 * Creates trampoline code to jump inside an existing patch
3246 *
3247 * @returns VBox status code.
3248 * @param pVM The VM to operate on.
3249 * @param pInstrGC Guest context point to privileged instruction
3250 * @param pPatchRec Patch record
3251 *
3252 */
3253static int patmCreateTrampoline(PVM pVM, RTRCPTR pInstrGC, PPATMPATCHREC pPatchRec)
3254{
3255 PPATCHINFO pPatch = &pPatchRec->patch;
3256 RTRCPTR pPage, pPatchTargetGC = 0;
3257 uint32_t orgOffsetPatchMem = ~0;
3258 int rc = VERR_PATCHING_REFUSED;
3259
3260 Log(("patmCreateTrampoline %RRv\n", pInstrGC));
3261 /* Save original offset (in case of failures later on). */
3262 orgOffsetPatchMem = pVM->patm.s.offPatchMem;
3263
3264 /* First we check if the duplicate function target lies in some existing function patch already. Will save some space. */
3265 /** @todo we already checked this before */
3266 pPage = pInstrGC & PAGE_BASE_GC_MASK;
3267
3268 PPATMPATCHPAGE pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, (RTRCPTR)pPage);
3269 if (pPatchPage)
3270 {
3271 uint32_t i;
3272
3273 for (i=0;i<pPatchPage->cCount;i++)
3274 {
3275 if (pPatchPage->aPatch[i])
3276 {
3277 PPATCHINFO pPatch2 = pPatchPage->aPatch[i];
3278
3279 if ( (pPatch2->flags & PATMFL_DUPLICATE_FUNCTION)
3280 && pPatch2->uState == PATCH_ENABLED)
3281 {
3282 pPatchTargetGC = patmGuestGCPtrToPatchGCPtr(pVM, pPatch2, pInstrGC);
3283 if (pPatchTargetGC)
3284 {
3285 uint32_t offsetPatch = pPatchTargetGC - pVM->patm.s.pPatchMemGC;
3286 PRECPATCHTOGUEST pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32GetBestFit(&pPatch2->Patch2GuestAddrTree, offsetPatch, false);
3287 Assert(pPatchToGuestRec);
3288
3289 pPatchToGuestRec->fJumpTarget = true;
3290 Assert(pPatchTargetGC != pPatch2->pPrivInstrGC);
3291 Log(("patmCreateTrampoline: generating jump to code inside patch at %RRv\n", pPatch2->pPrivInstrGC));
3292 pPatch2->flags |= PATMFL_EXTERNAL_JUMP_INSIDE;
3293 break;
3294 }
3295 }
3296 }
3297 }
3298 }
3299 AssertReturn(pPatchPage && pPatchTargetGC, VERR_PATCHING_REFUSED);
3300
3301 pPatch->nrPatch2GuestRecs = 0;
3302 pPatch->pPatchBlockOffset = pVM->patm.s.offPatchMem;
3303 pPatch->uCurPatchOffset = 0;
3304
3305 /* Note: Set the PATM interrupt flag here; it was cleared before the patched call. (!!!) */
3306 rc = patmPatchGenSetPIF(pVM, pPatch, pInstrGC);
3307 if (RT_FAILURE(rc))
3308 goto failure;
3309
3310#ifdef VBOX_WITH_STATISTICS
3311 rc = patmPatchGenStats(pVM, pPatch, pInstrGC);
3312 if (RT_FAILURE(rc))
3313 goto failure;
3314#endif
3315
3316 rc = patmPatchGenPatchJump(pVM, pPatch, pInstrGC, pPatchTargetGC);
3317 if (RT_FAILURE(rc))
3318 goto failure;
3319
3320 /*
3321 * Insert into patch to guest lookup tree
3322 */
3323 LogFlow(("Insert %RRv patch offset %RRv\n", pPatchRec->patch.pPrivInstrGC, pPatch->pPatchBlockOffset));
3324 pPatchRec->CoreOffset.Key = pPatch->pPatchBlockOffset;
3325 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, &pPatchRec->CoreOffset);
3326 AssertMsg(rc, ("RTAvloU32Insert failed for %x\n", pPatchRec->CoreOffset.Key));
3327 if (!rc)
3328 {
3329 rc = VERR_PATCHING_REFUSED;
3330 goto failure;
3331 }
3332
3333 /* size of patch block */
3334 pPatch->cbPatchBlockSize = pPatch->uCurPatchOffset;
3335
3336 /* Update free pointer in patch memory. */
3337 pVM->patm.s.offPatchMem += pPatch->cbPatchBlockSize;
3338 /* Round to next 8 byte boundary */
3339 pVM->patm.s.offPatchMem = RT_ALIGN_32(pVM->patm.s.offPatchMem, 8);
3340
3341 /* There's no jump from guest to patch code. */
3342 pPatch->cbPatchJump = 0;
3343
3344 /* Enable the patch. */
3345 pPatch->uState = PATCH_ENABLED;
3346 /* We allow this patch to be called as a function. */
3347 pPatch->flags |= PATMFL_CALLABLE_AS_FUNCTION;
3348 STAM_COUNTER_INC(&pVM->patm.s.StatInstalledTrampoline);
3349 return VINF_SUCCESS;
3350
3351failure:
3352 if (pPatchRec->CoreOffset.Key)
3353 RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, pPatchRec->CoreOffset.Key);
3354
3355 patmEmptyTree(pVM, &pPatch->FixupTree);
3356 pPatch->nrFixups = 0;
3357
3358 patmEmptyTree(pVM, &pPatch->JumpTree);
3359 pPatch->nrJumpRecs = 0;
3360
3361 patmEmptyTree(pVM, &pPatch->pTempInfo->IllegalInstrTree);
3362 pPatch->pTempInfo->nrIllegalInstr = 0;
3363
3364 /* Turn this cli patch into a dummy. */
3365 pPatch->uState = PATCH_REFUSED;
3366 pPatch->pPatchBlockOffset = 0;
3367
3368 // Give back the patch memory we no longer need
3369 Assert(orgOffsetPatchMem != (uint32_t)~0);
3370 pVM->patm.s.offPatchMem = orgOffsetPatchMem;
3371
3372 return rc;
3373}
3374
3375
3376/**
3377 * Patch branch target function for call/jump at specified location.
3378 * (in responds to a VINF_PATM_DUPLICATE_FUNCTION GC exit reason)
3379 *
3380 * @returns VBox status code.
3381 * @param pVM The VM to operate on.
3382 * @param pCtx Guest context
3383 *
3384 */
3385VMMR3DECL(int) PATMR3DuplicateFunctionRequest(PVM pVM, PCPUMCTX pCtx)
3386{
3387 RTRCPTR pBranchTarget, pPage;
3388 int rc;
3389 RTRCPTR pPatchTargetGC = 0;
3390
3391 pBranchTarget = pCtx->edx;
3392 pBranchTarget = SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), pBranchTarget);
3393
3394 /* First we check if the duplicate function target lies in some existing function patch already. Will save some space. */
3395 pPage = pBranchTarget & PAGE_BASE_GC_MASK;
3396
3397 PPATMPATCHPAGE pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, (RTRCPTR)pPage);
3398 if (pPatchPage)
3399 {
3400 uint32_t i;
3401
3402 for (i=0;i<pPatchPage->cCount;i++)
3403 {
3404 if (pPatchPage->aPatch[i])
3405 {
3406 PPATCHINFO pPatch = pPatchPage->aPatch[i];
3407
3408 if ( (pPatch->flags & PATMFL_DUPLICATE_FUNCTION)
3409 && pPatch->uState == PATCH_ENABLED)
3410 {
3411 pPatchTargetGC = patmGuestGCPtrToPatchGCPtr(pVM, pPatch, pBranchTarget);
3412 if (pPatchTargetGC)
3413 {
3414 STAM_COUNTER_INC(&pVM->patm.s.StatDuplicateUseExisting);
3415 break;
3416 }
3417 }
3418 }
3419 }
3420 }
3421
3422 if (pPatchTargetGC)
3423 {
3424 /* Create a trampoline that also sets PATM_INTERRUPTFLAG. */
3425 rc = PATMR3InstallPatch(pVM, pBranchTarget, PATMFL_CODE32 | PATMFL_TRAMPOLINE);
3426 }
3427 else
3428 {
3429 rc = PATMR3InstallPatch(pVM, pBranchTarget, PATMFL_CODE32 | PATMFL_DUPLICATE_FUNCTION);
3430 }
3431
3432 if (rc == VINF_SUCCESS)
3433 {
3434 pPatchTargetGC = PATMR3QueryPatchGCPtr(pVM, pBranchTarget);
3435 Assert(pPatchTargetGC);
3436 }
3437
3438 if (pPatchTargetGC)
3439 {
3440 pCtx->eax = pPatchTargetGC;
3441 pCtx->eax = pCtx->eax - (RTRCUINTPTR)pVM->patm.s.pPatchMemGC; /* make it relative */
3442 }
3443 else
3444 {
3445 /* We add a dummy entry into the lookup cache so we won't get bombarded with the same requests over and over again. */
3446 pCtx->eax = 0;
3447 STAM_COUNTER_INC(&pVM->patm.s.StatDuplicateREQFailed);
3448 }
3449 Assert(PATMIsPatchGCAddr(pVM, pCtx->edi));
3450 rc = PATMAddBranchToLookupCache(pVM, pCtx->edi, pBranchTarget, pCtx->eax);
3451 AssertRC(rc);
3452
3453 pCtx->eip += PATM_ILLEGAL_INSTR_SIZE;
3454 STAM_COUNTER_INC(&pVM->patm.s.StatDuplicateREQSuccess);
3455 return VINF_SUCCESS;
3456}
3457
3458/**
3459 * Replaces a function call by a call to an existing function duplicate (or jmp -> jmp)
3460 *
3461 * @returns VBox status code.
3462 * @param pVM The VM to operate on.
3463 * @param pCpu Disassembly CPU structure ptr
3464 * @param pInstrGC Guest context point to privileged instruction
3465 * @param pCacheRec Cache record ptr
3466 *
3467 */
3468static int patmReplaceFunctionCall(PVM pVM, DISCPUSTATE *pCpu, RTRCPTR pInstrGC, PPATMP2GLOOKUPREC pCacheRec)
3469{
3470 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
3471 int rc = VERR_PATCHING_REFUSED;
3472 DISCPUSTATE cpu;
3473 RTRCPTR pTargetGC;
3474 PPATMPATCHREC pPatchFunction;
3475 uint32_t opsize;
3476 bool disret;
3477#ifdef LOG_ENABLED
3478 char szOutput[256];
3479#endif
3480
3481 Assert(pPatch->flags & PATMFL_REPLACE_FUNCTION_CALL);
3482 Assert((pCpu->pCurInstr->opcode == OP_CALL || pCpu->pCurInstr->opcode == OP_JMP) && pCpu->opsize == SIZEOF_NEARJUMP32);
3483
3484 if ((pCpu->pCurInstr->opcode != OP_CALL && pCpu->pCurInstr->opcode != OP_JMP) || pCpu->opsize != SIZEOF_NEARJUMP32)
3485 {
3486 rc = VERR_PATCHING_REFUSED;
3487 goto failure;
3488 }
3489
3490 pTargetGC = PATMResolveBranch(pCpu, pInstrGC);
3491 if (pTargetGC == 0)
3492 {
3493 Log(("We don't support far jumps here!! (%08X)\n", pCpu->param1.flags));
3494 rc = VERR_PATCHING_REFUSED;
3495 goto failure;
3496 }
3497
3498 pPatchFunction = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pTargetGC);
3499 if (pPatchFunction == NULL)
3500 {
3501 for(;;)
3502 {
3503 /* It could be an indirect call (call -> jmp dest).
3504 * Note that it's dangerous to assume the jump will never change...
3505 */
3506 uint8_t *pTmpInstrHC;
3507
3508 pTmpInstrHC = PATMGCVirtToHCVirt(pVM, pCacheRec, pTargetGC);
3509 Assert(pTmpInstrHC);
3510 if (pTmpInstrHC == 0)
3511 break;
3512
3513 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3514 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pTargetGC, pTmpInstrHC, &opsize, NULL);
3515 if (disret == false || cpu.pCurInstr->opcode != OP_JMP)
3516 break;
3517
3518 pTargetGC = PATMResolveBranch(&cpu, pTargetGC);
3519 if (pTargetGC == 0)
3520 {
3521 break;
3522 }
3523
3524 pPatchFunction = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pTargetGC);
3525 break;
3526 }
3527 if (pPatchFunction == 0)
3528 {
3529 AssertMsgFailed(("Unable to find duplicate function %RRv\n", pTargetGC));
3530 rc = VERR_PATCHING_REFUSED;
3531 goto failure;
3532 }
3533 }
3534
3535 // make a copy of the guest code bytes that will be overwritten
3536 pPatch->cbPatchJump = SIZEOF_NEARJUMP32;
3537
3538 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pPatch->aPrivInstr, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
3539 AssertRC(rc);
3540
3541 /* Now replace the original call in the guest code */
3542 rc = patmGenCallToPatch(pVM, pPatch, PATCHCODE_PTR_GC(&pPatchFunction->patch), pCacheRec, true);
3543 AssertRC(rc);
3544 if (RT_FAILURE(rc))
3545 goto failure;
3546
3547 /* Lowest and highest address for write monitoring. */
3548 pPatch->pInstrGCLowest = pInstrGC;
3549 pPatch->pInstrGCHighest = pInstrGC + pCpu->opsize;
3550
3551#ifdef LOG_ENABLED
3552 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3553 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput);
3554 Log(("Call patch: %s", szOutput));
3555#endif
3556
3557 Log(("Successfully installed function replacement patch at %RRv\n", pInstrGC));
3558
3559 pPatch->uState = PATCH_ENABLED;
3560 return VINF_SUCCESS;
3561
3562failure:
3563 /* Turn this patch into a dummy. */
3564 pPatch->uState = PATCH_REFUSED;
3565
3566 return rc;
3567}
3568
3569/**
3570 * Replace the address in an MMIO instruction with the cached version.
3571 *
3572 * @returns VBox status code.
3573 * @param pVM The VM to operate on.
3574 * @param pInstrGC Guest context point to privileged instruction
3575 * @param pCpu Disassembly CPU structure ptr
3576 * @param pCacheRec Cache record ptr
3577 *
3578 * @note returns failure if patching is not allowed or possible
3579 *
3580 */
3581static int patmPatchMMIOInstr(PVM pVM, RTRCPTR pInstrGC, DISCPUSTATE *pCpu, PPATMP2GLOOKUPREC pCacheRec)
3582{
3583 PPATCHINFO pPatch = (PPATCHINFO)pCacheRec->pPatch;
3584 uint8_t *pPB;
3585 int rc = VERR_PATCHING_REFUSED;
3586#ifdef LOG_ENABLED
3587 DISCPUSTATE cpu;
3588 uint32_t opsize;
3589 bool disret;
3590 char szOutput[256];
3591#endif
3592
3593 Assert(pVM->patm.s.mmio.pCachedData);
3594 if (!pVM->patm.s.mmio.pCachedData)
3595 goto failure;
3596
3597 if (pCpu->param2.flags != USE_DISPLACEMENT32)
3598 goto failure;
3599
3600 pPB = PATMGCVirtToHCVirt(pVM, pCacheRec, pPatch->pPrivInstrGC);
3601 if (pPB == 0)
3602 goto failure;
3603
3604 /* Add relocation record for cached data access. */
3605 if (patmPatchAddReloc32(pVM, pPatch, &pPB[pCpu->opsize - sizeof(RTRCPTR)], FIXUP_ABSOLUTE, pPatch->pPrivInstrGC, pVM->patm.s.mmio.pCachedData) != VINF_SUCCESS)
3606 {
3607 Log(("Relocation failed for cached mmio address!!\n"));
3608 return VERR_PATCHING_REFUSED;
3609 }
3610#ifdef LOG_ENABLED
3611 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3612 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput);
3613 Log(("MMIO patch old instruction: %s", szOutput));
3614#endif
3615
3616 /* Save original instruction. */
3617 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pPatch->aPrivInstr, pPatch->pPrivInstrGC, pPatch->cbPrivInstr);
3618 AssertRC(rc);
3619
3620 pPatch->cbPatchJump = pPatch->cbPrivInstr; /* bit of a misnomer in this case; size of replacement instruction. */
3621
3622 /* Replace address with that of the cached item. */
3623 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pInstrGC + pCpu->opsize - sizeof(RTRCPTR), &pVM->patm.s.mmio.pCachedData, sizeof(RTRCPTR));
3624 AssertRC(rc);
3625 if (RT_FAILURE(rc))
3626 {
3627 goto failure;
3628 }
3629
3630#ifdef LOG_ENABLED
3631 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3632 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput);
3633 Log(("MMIO patch: %s", szOutput));
3634#endif
3635 pVM->patm.s.mmio.pCachedData = 0;
3636 pVM->patm.s.mmio.GCPhys = 0;
3637 pPatch->uState = PATCH_ENABLED;
3638 return VINF_SUCCESS;
3639
3640failure:
3641 /* Turn this patch into a dummy. */
3642 pPatch->uState = PATCH_REFUSED;
3643
3644 return rc;
3645}
3646
3647
3648/**
3649 * Replace the address in an MMIO instruction with the cached version. (instruction is part of an existing patch)
3650 *
3651 * @returns VBox status code.
3652 * @param pVM The VM to operate on.
3653 * @param pInstrGC Guest context point to privileged instruction
3654 * @param pPatch Patch record
3655 *
3656 * @note returns failure if patching is not allowed or possible
3657 *
3658 */
3659static int patmPatchPATMMMIOInstr(PVM pVM, RTRCPTR pInstrGC, PPATCHINFO pPatch)
3660{
3661 DISCPUSTATE cpu;
3662 uint32_t opsize;
3663 bool disret;
3664 uint8_t *pInstrHC;
3665#ifdef LOG_ENABLED
3666 char szOutput[256];
3667#endif
3668
3669 AssertReturn(pVM->patm.s.mmio.pCachedData, VERR_INVALID_PARAMETER);
3670
3671 /* Convert GC to HC address. */
3672 pInstrHC = patmPatchGCPtr2PatchHCPtr(pVM, pInstrGC);
3673 AssertReturn(pInstrHC, VERR_PATCHING_REFUSED);
3674
3675 /* Disassemble mmio instruction. */
3676 cpu.mode = pPatch->uOpMode;
3677 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pInstrGC, pInstrHC, &opsize, NULL);
3678 if (disret == false)
3679 {
3680 Log(("Disassembly failed (probably page not present) -> return to caller\n"));
3681 return VERR_PATCHING_REFUSED;
3682 }
3683
3684 AssertMsg(opsize <= MAX_INSTR_SIZE, ("privileged instruction too big %d!!\n", opsize));
3685 if (opsize > MAX_INSTR_SIZE)
3686 return VERR_PATCHING_REFUSED;
3687 if (cpu.param2.flags != USE_DISPLACEMENT32)
3688 return VERR_PATCHING_REFUSED;
3689
3690 /* Add relocation record for cached data access. */
3691 if (patmPatchAddReloc32(pVM, pPatch, &pInstrHC[cpu.opsize - sizeof(RTRCPTR)], FIXUP_ABSOLUTE) != VINF_SUCCESS)
3692 {
3693 Log(("Relocation failed for cached mmio address!!\n"));
3694 return VERR_PATCHING_REFUSED;
3695 }
3696 /* Replace address with that of the cached item. */
3697 *(RTRCPTR *)&pInstrHC[cpu.opsize - sizeof(RTRCPTR)] = pVM->patm.s.mmio.pCachedData;
3698
3699 /* Lowest and highest address for write monitoring. */
3700 pPatch->pInstrGCLowest = pInstrGC;
3701 pPatch->pInstrGCHighest = pInstrGC + cpu.opsize;
3702
3703#ifdef LOG_ENABLED
3704 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3705 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pInstrGC, pInstrHC, &opsize, szOutput);
3706 Log(("MMIO patch: %s", szOutput));
3707#endif
3708
3709 pVM->patm.s.mmio.pCachedData = 0;
3710 pVM->patm.s.mmio.GCPhys = 0;
3711 return VINF_SUCCESS;
3712}
3713
3714/**
3715 * Activates an int3 patch
3716 *
3717 * @returns VBox status code.
3718 * @param pVM The VM to operate on.
3719 * @param pPatch Patch record
3720 */
3721static int patmActivateInt3Patch(PVM pVM, PPATCHINFO pPatch)
3722{
3723 uint8_t ASMInt3 = 0xCC;
3724 int rc;
3725
3726 Assert(pPatch->flags & (PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK));
3727 Assert(pPatch->uState != PATCH_ENABLED);
3728
3729 /* Replace first opcode byte with 'int 3'. */
3730 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pPatch->pPrivInstrGC, &ASMInt3, sizeof(ASMInt3));
3731 AssertRC(rc);
3732
3733 pPatch->cbPatchJump = sizeof(ASMInt3);
3734
3735 return rc;
3736}
3737
3738/**
3739 * Deactivates an int3 patch
3740 *
3741 * @returns VBox status code.
3742 * @param pVM The VM to operate on.
3743 * @param pPatch Patch record
3744 */
3745static int patmDeactivateInt3Patch(PVM pVM, PPATCHINFO pPatch)
3746{
3747 uint8_t ASMInt3 = 0xCC;
3748 int rc;
3749
3750 Assert(pPatch->flags & (PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK));
3751 Assert(pPatch->uState == PATCH_ENABLED || pPatch->uState == PATCH_DIRTY);
3752
3753 /* Restore first opcode byte. */
3754 rc = PGMPhysSimpleDirtyWriteGCPtr(VMMGetCpu0(pVM), pPatch->pPrivInstrGC, pPatch->aPrivInstr, sizeof(ASMInt3));
3755 AssertRC(rc);
3756 return rc;
3757}
3758
3759/**
3760 * Replace an instruction with a breakpoint (0xCC), that is handled dynamically in the guest context.
3761 *
3762 * @returns VBox status code.
3763 * @param pVM The VM to operate on.
3764 * @param pInstrGC Guest context point to privileged instruction
3765 * @param pInstrHC Host context point to privileged instruction
3766 * @param pCpu Disassembly CPU structure ptr
3767 * @param pPatch Patch record
3768 *
3769 * @note returns failure if patching is not allowed or possible
3770 *
3771 */
3772VMMR3DECL(int) PATMR3PatchInstrInt3(PVM pVM, RTRCPTR pInstrGC, R3PTRTYPE(uint8_t *) pInstrHC, DISCPUSTATE *pCpu, PPATCHINFO pPatch)
3773{
3774 uint8_t ASMInt3 = 0xCC;
3775 int rc;
3776
3777 /* Note: Do not use patch memory here! It might called during patch installation too. */
3778
3779#ifdef LOG_ENABLED
3780 DISCPUSTATE cpu;
3781 char szOutput[256];
3782 uint32_t opsize;
3783
3784 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3785 PATMR3DISInstr(pVM, pPatch, &cpu, pInstrGC, pInstrHC, &opsize, szOutput);
3786 Log(("PATMR3PatchInstrInt3: %s", szOutput));
3787#endif
3788
3789 /* Save the original instruction. */
3790 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pPatch->aPrivInstr, pPatch->pPrivInstrGC, pPatch->cbPrivInstr);
3791 AssertRC(rc);
3792 pPatch->cbPatchJump = sizeof(ASMInt3); /* bit of a misnomer in this case; size of replacement instruction. */
3793
3794 pPatch->flags |= PATMFL_INT3_REPLACEMENT;
3795
3796 /* Replace first opcode byte with 'int 3'. */
3797 rc = patmActivateInt3Patch(pVM, pPatch);
3798 if (RT_FAILURE(rc))
3799 goto failure;
3800
3801 /* Lowest and highest address for write monitoring. */
3802 pPatch->pInstrGCLowest = pInstrGC;
3803 pPatch->pInstrGCHighest = pInstrGC + pCpu->opsize;
3804
3805 pPatch->uState = PATCH_ENABLED;
3806 return VINF_SUCCESS;
3807
3808failure:
3809 /* Turn this patch into a dummy. */
3810 return VERR_PATCHING_REFUSED;
3811}
3812
3813#ifdef PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES
3814/**
3815 * Patch a jump instruction at specified location
3816 *
3817 * @returns VBox status code.
3818 * @param pVM The VM to operate on.
3819 * @param pInstrGC Guest context point to privileged instruction
3820 * @param pInstrHC Host context point to privileged instruction
3821 * @param pCpu Disassembly CPU structure ptr
3822 * @param pPatchRec Patch record
3823 *
3824 * @note returns failure if patching is not allowed or possible
3825 *
3826 */
3827int patmPatchJump(PVM pVM, RTRCPTR pInstrGC, R3PTRTYPE(uint8_t *) pInstrHC, DISCPUSTATE *pCpu, PPATMPATCHREC pPatchRec)
3828{
3829 PPATCHINFO pPatch = &pPatchRec->patch;
3830 int rc = VERR_PATCHING_REFUSED;
3831#ifdef LOG_ENABLED
3832 bool disret;
3833 DISCPUSTATE cpu;
3834 uint32_t opsize;
3835 char szOutput[256];
3836#endif
3837
3838 pPatch->pPatchBlockOffset = 0; /* doesn't use patch memory */
3839 pPatch->uCurPatchOffset = 0;
3840 pPatch->cbPatchBlockSize = 0;
3841 pPatch->flags |= PATMFL_SINGLE_INSTRUCTION;
3842
3843 /*
3844 * Instruction replacements such as these should never be interrupted. I've added code to EM.cpp to
3845 * make sure this never happens. (unless a trap is triggered (intentionally or not))
3846 */
3847 switch (pCpu->pCurInstr->opcode)
3848 {
3849 case OP_JO:
3850 case OP_JNO:
3851 case OP_JC:
3852 case OP_JNC:
3853 case OP_JE:
3854 case OP_JNE:
3855 case OP_JBE:
3856 case OP_JNBE:
3857 case OP_JS:
3858 case OP_JNS:
3859 case OP_JP:
3860 case OP_JNP:
3861 case OP_JL:
3862 case OP_JNL:
3863 case OP_JLE:
3864 case OP_JNLE:
3865 case OP_JMP:
3866 Assert(pPatch->flags & PATMFL_JUMP_CONFLICT);
3867 Assert(pCpu->param1.flags & USE_IMMEDIATE32_REL);
3868 if (!(pCpu->param1.flags & USE_IMMEDIATE32_REL))
3869 goto failure;
3870
3871 Assert(pCpu->opsize == SIZEOF_NEARJUMP32 || pCpu->opsize == SIZEOF_NEAR_COND_JUMP32);
3872 if (pCpu->opsize != SIZEOF_NEARJUMP32 && pCpu->opsize != SIZEOF_NEAR_COND_JUMP32)
3873 goto failure;
3874
3875 if (PAGE_ADDRESS(pInstrGC) != PAGE_ADDRESS(pInstrGC + pCpu->opsize))
3876 {
3877 STAM_COUNTER_INC(&pVM->patm.s.StatPageBoundaryCrossed);
3878 AssertMsgFailed(("Patch jump would cross page boundary -> refuse!!\n"));
3879 rc = VERR_PATCHING_REFUSED;
3880 goto failure;
3881 }
3882
3883 break;
3884
3885 default:
3886 goto failure;
3887 }
3888
3889 // make a copy of the guest code bytes that will be overwritten
3890 Assert(pCpu->opsize <= sizeof(pPatch->aPrivInstr));
3891 Assert(pCpu->opsize >= SIZEOF_NEARJUMP32);
3892 pPatch->cbPatchJump = pCpu->opsize;
3893
3894 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pPatch->aPrivInstr, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
3895 AssertRC(rc);
3896
3897 /* Now insert a jump in the guest code. */
3898 /*
3899 * A conflict jump patch needs to be treated differently; we'll just replace the relative jump address with one that
3900 * references the target instruction in the conflict patch.
3901 */
3902 RTRCPTR pJmpDest = PATMR3GuestGCPtrToPatchGCPtr(pVM, pInstrGC + pCpu->opsize + (int32_t)pCpu->param1.parval);
3903
3904 AssertMsg(pJmpDest, ("PATMR3GuestGCPtrToPatchGCPtr failed for %RRv\n", pInstrGC + pCpu->opsize + (int32_t)pCpu->param1.parval));
3905 pPatch->pPatchJumpDestGC = pJmpDest;
3906
3907 PATMP2GLOOKUPREC cacheRec;
3908 RT_ZERO(cacheRec);
3909 cacheRec.pPatch = pPatch;
3910
3911 rc = patmGenJumpToPatch(pVM, pPatch, &cacherec, true);
3912 /* Free leftover lock if any. */
3913 if (cacheRec.Lock.pvMap)
3914 {
3915 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
3916 cacheRec.Lock.pvMap = NULL;
3917 }
3918 AssertRC(rc);
3919 if (RT_FAILURE(rc))
3920 goto failure;
3921
3922 pPatch->flags |= PATMFL_MUST_INSTALL_PATCHJMP;
3923
3924#ifdef LOG_ENABLED
3925 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
3926 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC, NULL, &opsize, szOutput);
3927 Log(("%s patch: %s", patmGetInstructionString(pPatch->opcode, pPatch->flags), szOutput));
3928#endif
3929
3930 Log(("Successfully installed %s patch at %RRv\n", patmGetInstructionString(pPatch->opcode, pPatch->flags), pInstrGC));
3931
3932 STAM_COUNTER_INC(&pVM->patm.s.StatInstalledJump);
3933
3934 /* Lowest and highest address for write monitoring. */
3935 pPatch->pInstrGCLowest = pInstrGC;
3936 pPatch->pInstrGCHighest = pInstrGC + pPatch->cbPatchJump;
3937
3938 pPatch->uState = PATCH_ENABLED;
3939 return VINF_SUCCESS;
3940
3941failure:
3942 /* Turn this cli patch into a dummy. */
3943 pPatch->uState = PATCH_REFUSED;
3944
3945 return rc;
3946}
3947#endif /* PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES */
3948
3949
3950/**
3951 * Gives hint to PATM about supervisor guest instructions
3952 *
3953 * @returns VBox status code.
3954 * @param pVM The VM to operate on.
3955 * @param pInstr Guest context point to privileged instruction
3956 * @param flags Patch flags
3957 */
3958VMMR3DECL(int) PATMR3AddHint(PVM pVM, RTRCPTR pInstrGC, uint32_t flags)
3959{
3960 Assert(pInstrGC);
3961 Assert(flags == PATMFL_CODE32);
3962
3963 Log(("PATMR3AddHint %RRv\n", pInstrGC));
3964 return PATMR3InstallPatch(pVM, pInstrGC, PATMFL_CODE32 | PATMFL_INSTR_HINT);
3965}
3966
3967/**
3968 * Patch privileged instruction at specified location
3969 *
3970 * @returns VBox status code.
3971 * @param pVM The VM to operate on.
3972 * @param pInstr Guest context point to privileged instruction (0:32 flat address)
3973 * @param flags Patch flags
3974 *
3975 * @note returns failure if patching is not allowed or possible
3976 */
3977VMMR3DECL(int) PATMR3InstallPatch(PVM pVM, RTRCPTR pInstrGC, uint64_t flags)
3978{
3979 DISCPUSTATE cpu;
3980 R3PTRTYPE(uint8_t *) pInstrHC;
3981 uint32_t opsize;
3982 PPATMPATCHREC pPatchRec;
3983 PCPUMCTX pCtx = 0;
3984 bool disret;
3985 int rc;
3986 PVMCPU pVCpu = VMMGetCpu0(pVM);
3987
3988 if ( !pVM
3989 || pInstrGC == 0
3990 || (flags & ~(PATMFL_CODE32|PATMFL_IDTHANDLER|PATMFL_INTHANDLER|PATMFL_SYSENTER|PATMFL_TRAPHANDLER|PATMFL_DUPLICATE_FUNCTION|PATMFL_REPLACE_FUNCTION_CALL|PATMFL_GUEST_SPECIFIC|PATMFL_INT3_REPLACEMENT|PATMFL_TRAPHANDLER_WITH_ERRORCODE|PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT|PATMFL_MMIO_ACCESS|PATMFL_TRAMPOLINE|PATMFL_INSTR_HINT|PATMFL_JUMP_CONFLICT)))
3991 {
3992 AssertFailed();
3993 return VERR_INVALID_PARAMETER;
3994 }
3995
3996 if (PATMIsEnabled(pVM) == false)
3997 return VERR_PATCHING_REFUSED;
3998
3999 /* Test for patch conflict only with patches that actually change guest code. */
4000 if (!(flags & (PATMFL_GUEST_SPECIFIC|PATMFL_IDTHANDLER|PATMFL_INTHANDLER|PATMFL_TRAMPOLINE)))
4001 {
4002 PPATCHINFO pConflictPatch = PATMFindActivePatchByEntrypoint(pVM, pInstrGC);
4003 AssertReleaseMsg(pConflictPatch == 0, ("Unable to patch overwritten instruction at %RRv (%RRv)\n", pInstrGC, pConflictPatch->pPrivInstrGC));
4004 if (pConflictPatch != 0)
4005 return VERR_PATCHING_REFUSED;
4006 }
4007
4008 if (!(flags & PATMFL_CODE32))
4009 {
4010 /** @todo Only 32 bits code right now */
4011 AssertMsgFailed(("PATMR3InstallPatch: We don't support 16 bits code at this moment!!\n"));
4012 return VERR_NOT_IMPLEMENTED;
4013 }
4014
4015 /* We ran out of patch memory; don't bother anymore. */
4016 if (pVM->patm.s.fOutOfMemory == true)
4017 return VERR_PATCHING_REFUSED;
4018
4019 /* Make sure the code selector is wide open; otherwise refuse. */
4020 pCtx = CPUMQueryGuestCtxPtr(pVCpu);
4021 if (CPUMGetGuestCPL(pVCpu, CPUMCTX2CORE(pCtx)) == 0)
4022 {
4023 RTRCPTR pInstrGCFlat = SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), pInstrGC);
4024 if (pInstrGCFlat != pInstrGC)
4025 {
4026 Log(("PATMR3InstallPatch: code selector not wide open: %04x:%RRv != %RRv eflags=%08x\n", pCtx->cs, pInstrGCFlat, pInstrGC, pCtx->eflags.u32));
4027 return VERR_PATCHING_REFUSED;
4028 }
4029 }
4030
4031 /* Note: the OpenBSD specific check will break if we allow additional patches to be installed (int 3)) */
4032 if (!(flags & PATMFL_GUEST_SPECIFIC))
4033 {
4034 /* New code. Make sure CSAM has a go at it first. */
4035 CSAMR3CheckCode(pVM, pInstrGC);
4036 }
4037
4038 /* Note: obsolete */
4039 if ( PATMIsPatchGCAddr(pVM, pInstrGC)
4040 && (flags & PATMFL_MMIO_ACCESS))
4041 {
4042 RTRCUINTPTR offset;
4043 void *pvPatchCoreOffset;
4044
4045 /* Find the patch record. */
4046 offset = pInstrGC - pVM->patm.s.pPatchMemGC;
4047 pvPatchCoreOffset = RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, offset, false);
4048 if (pvPatchCoreOffset == NULL)
4049 {
4050 AssertMsgFailed(("PATMR3InstallPatch: patch not found at address %RRv!!\n", pInstrGC));
4051 return VERR_PATCH_NOT_FOUND; //fatal error
4052 }
4053 pPatchRec = PATM_PATCHREC_FROM_COREOFFSET(pvPatchCoreOffset);
4054
4055 return patmPatchPATMMMIOInstr(pVM, pInstrGC, &pPatchRec->patch);
4056 }
4057
4058 AssertReturn(!PATMIsPatchGCAddr(pVM, pInstrGC), VERR_PATCHING_REFUSED);
4059
4060 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
4061 if (pPatchRec)
4062 {
4063 Assert(!(flags & PATMFL_TRAMPOLINE));
4064
4065 /* Hints about existing patches are ignored. */
4066 if (flags & PATMFL_INSTR_HINT)
4067 return VERR_PATCHING_REFUSED;
4068
4069 if (pPatchRec->patch.uState == PATCH_DISABLE_PENDING)
4070 {
4071 Log(("PATMR3InstallPatch: disable operation is pending for patch at %RRv\n", pPatchRec->patch.pPrivInstrGC));
4072 PATMR3DisablePatch(pVM, pPatchRec->patch.pPrivInstrGC);
4073 Assert(pPatchRec->patch.uState == PATCH_DISABLED);
4074 }
4075
4076 if (pPatchRec->patch.uState == PATCH_DISABLED)
4077 {
4078 /* A patch, for which we previously received a hint, will be enabled and turned into a normal patch. */
4079 if (pPatchRec->patch.flags & PATMFL_INSTR_HINT)
4080 {
4081 Log(("Enabling HINTED patch %RRv\n", pInstrGC));
4082 pPatchRec->patch.flags &= ~PATMFL_INSTR_HINT;
4083 }
4084 else
4085 Log(("Enabling patch %RRv again\n", pInstrGC));
4086
4087 /** @todo we shouldn't disable and enable patches too often (it's relatively cheap, but pointless if it always happens) */
4088 rc = PATMR3EnablePatch(pVM, pInstrGC);
4089 if (RT_SUCCESS(rc))
4090 return VWRN_PATCH_ENABLED;
4091
4092 return rc;
4093 }
4094 if ( pPatchRec->patch.uState == PATCH_ENABLED
4095 || pPatchRec->patch.uState == PATCH_DIRTY)
4096 {
4097 /*
4098 * The patch might have been overwritten.
4099 */
4100 STAM_COUNTER_INC(&pVM->patm.s.StatOverwritten);
4101 if (pPatchRec->patch.uState != PATCH_REFUSED && pPatchRec->patch.uState != PATCH_UNUSABLE)
4102 {
4103 /* Patch must have been overwritten; remove it and pretend nothing happened. */
4104 Log(("Patch an existing patched instruction?!? (%RRv)\n", pInstrGC));
4105 if (pPatchRec->patch.flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER|PATMFL_MMIO_ACCESS|PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK))
4106 {
4107 if (flags & PATMFL_IDTHANDLER)
4108 pPatchRec->patch.flags |= (flags & (PATMFL_IDTHANDLER|PATMFL_TRAPHANDLER|PATMFL_INTHANDLER)); /* update the type */
4109
4110 return VERR_PATM_ALREADY_PATCHED; /* already done once */
4111 }
4112 }
4113 rc = PATMR3RemovePatch(pVM, pInstrGC);
4114 if (RT_FAILURE(rc))
4115 return VERR_PATCHING_REFUSED;
4116 }
4117 else
4118 {
4119 AssertMsg(pPatchRec->patch.uState == PATCH_REFUSED || pPatchRec->patch.uState == PATCH_UNUSABLE, ("Patch an existing patched instruction?!? (%RRv, state=%d)\n", pInstrGC, pPatchRec->patch.uState));
4120 /* already tried it once! */
4121 return VERR_PATCHING_REFUSED;
4122 }
4123 }
4124
4125 RTGCPHYS GCPhys;
4126 rc = PGMGstGetPage(pVCpu, pInstrGC, NULL, &GCPhys);
4127 if (rc != VINF_SUCCESS)
4128 {
4129 Log(("PGMGstGetPage failed with %Rrc\n", rc));
4130 return rc;
4131 }
4132 /* Disallow patching instructions inside ROM code; complete function duplication is allowed though. */
4133 if ( !(flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_TRAMPOLINE))
4134 && !PGMPhysIsGCPhysNormal(pVM, GCPhys))
4135 {
4136 Log(("Code at %RGv (phys %RGp) is in a ROM, MMIO or invalid page - refused\n", pInstrGC, GCPhys));
4137 return VERR_PATCHING_REFUSED;
4138 }
4139
4140 /* Initialize cache record for guest address translations. */
4141 PATMP2GLOOKUPREC cacheRec;
4142 RT_ZERO(cacheRec);
4143
4144 pInstrHC = PATMGCVirtToHCVirt(pVM, &cacheRec, pInstrGC);
4145 AssertReturn(pInstrHC, VERR_PATCHING_REFUSED);
4146
4147 /* Allocate patch record. */
4148 rc = MMHyperAlloc(pVM, sizeof(PATMPATCHREC), 0, MM_TAG_PATM_PATCH, (void **)&pPatchRec);
4149 if (RT_FAILURE(rc))
4150 {
4151 Log(("Out of memory!!!!\n"));
4152 return VERR_NO_MEMORY;
4153 }
4154 pPatchRec->Core.Key = pInstrGC;
4155 pPatchRec->patch.uState = PATCH_REFUSED; /* default value */
4156 /* Insert patch record into the lookup tree. */
4157 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTree, &pPatchRec->Core);
4158 Assert(rc);
4159
4160 pPatchRec->patch.pPrivInstrGC = pInstrGC;
4161 pPatchRec->patch.flags = flags;
4162 pPatchRec->patch.uOpMode = (flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
4163
4164 pPatchRec->patch.pInstrGCLowest = pInstrGC;
4165 pPatchRec->patch.pInstrGCHighest = pInstrGC;
4166
4167 if (!(pPatchRec->patch.flags & (PATMFL_DUPLICATE_FUNCTION | PATMFL_IDTHANDLER | PATMFL_SYSENTER | PATMFL_TRAMPOLINE)))
4168 {
4169 /*
4170 * Close proximity to an unusable patch is a possible hint that this patch would turn out to be dangerous too!
4171 */
4172 PPATMPATCHREC pPatchNear = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, (pInstrGC + SIZEOF_NEARJUMP32 - 1), false);
4173 if (pPatchNear)
4174 {
4175 if (pPatchNear->patch.uState == PATCH_UNUSABLE && pInstrGC < pPatchNear->patch.pPrivInstrGC && pInstrGC + SIZEOF_NEARJUMP32 > pPatchNear->patch.pPrivInstrGC)
4176 {
4177 Log(("Dangerous patch; would overwrite the ususable patch at %RRv\n", pPatchNear->patch.pPrivInstrGC));
4178
4179 pPatchRec->patch.uState = PATCH_UNUSABLE;
4180 /*
4181 * Leave the new patch active as it's marked unusable; to prevent us from checking it over and over again
4182 */
4183 return VERR_PATCHING_REFUSED;
4184 }
4185 }
4186 }
4187
4188 pPatchRec->patch.pTempInfo = (PPATCHINFOTEMP)MMR3HeapAllocZ(pVM, MM_TAG_PATM_PATCH, sizeof(PATCHINFOTEMP));
4189 if (pPatchRec->patch.pTempInfo == 0)
4190 {
4191 Log(("Out of memory!!!!\n"));
4192 return VERR_NO_MEMORY;
4193 }
4194
4195 cpu.mode = pPatchRec->patch.uOpMode;
4196 disret = PATMR3DISInstr(pVM, &pPatchRec->patch, &cpu, pInstrGC, NULL, &opsize, NULL);
4197 if (disret == false)
4198 {
4199 Log(("Disassembly failed (probably page not present) -> return to caller\n"));
4200 return VERR_PATCHING_REFUSED;
4201 }
4202
4203 AssertMsg(opsize <= MAX_INSTR_SIZE, ("privileged instruction too big %d!!\n", opsize));
4204 if (opsize > MAX_INSTR_SIZE)
4205 return VERR_PATCHING_REFUSED;
4206
4207 pPatchRec->patch.cbPrivInstr = opsize;
4208 pPatchRec->patch.opcode = cpu.pCurInstr->opcode;
4209
4210 /* Restricted hinting for now. */
4211 Assert(!(flags & PATMFL_INSTR_HINT) || cpu.pCurInstr->opcode == OP_CLI);
4212
4213 /* Initialize cache record patch pointer. */
4214 cacheRec.pPatch = &pPatchRec->patch;
4215
4216 /* Allocate statistics slot */
4217 if (pVM->patm.s.uCurrentPatchIdx < PATM_STAT_MAX_COUNTERS)
4218 {
4219 pPatchRec->patch.uPatchIdx = pVM->patm.s.uCurrentPatchIdx++;
4220 }
4221 else
4222 {
4223 Log(("WARNING: Patch index wrap around!!\n"));
4224 pPatchRec->patch.uPatchIdx = PATM_STAT_INDEX_DUMMY;
4225 }
4226
4227 if (pPatchRec->patch.flags & PATMFL_TRAPHANDLER)
4228 {
4229 rc = patmInstallTrapTrampoline(pVM, pInstrGC, pPatchRec, &cacheRec);
4230 }
4231 else
4232 if (pPatchRec->patch.flags & (PATMFL_DUPLICATE_FUNCTION ))
4233 {
4234 rc = patmDuplicateFunction(pVM, pInstrGC, pPatchRec, &cacheRec);
4235 }
4236 else
4237 if (pPatchRec->patch.flags & PATMFL_TRAMPOLINE)
4238 {
4239 rc = patmCreateTrampoline(pVM, pInstrGC, pPatchRec);
4240 }
4241 else
4242 if (pPatchRec->patch.flags & PATMFL_REPLACE_FUNCTION_CALL)
4243 {
4244 rc = patmReplaceFunctionCall(pVM, &cpu, pInstrGC, &cacheRec);
4245 }
4246 else
4247 if (pPatchRec->patch.flags & PATMFL_INT3_REPLACEMENT)
4248 {
4249 rc = PATMR3PatchInstrInt3(pVM, pInstrGC, pInstrHC, &cpu, &pPatchRec->patch);
4250 }
4251 else
4252 if (pPatchRec->patch.flags & PATMFL_MMIO_ACCESS)
4253 {
4254 rc = patmPatchMMIOInstr(pVM, pInstrGC, &cpu, &cacheRec);
4255 }
4256 else
4257 if (pPatchRec->patch.flags & (PATMFL_IDTHANDLER|PATMFL_SYSENTER))
4258 {
4259 if (pPatchRec->patch.flags & PATMFL_SYSENTER)
4260 pPatchRec->patch.flags |= PATMFL_IDTHANDLER; /* we treat a sysenter handler as an IDT handler */
4261
4262 rc = patmIdtHandler(pVM, pInstrGC, opsize, pPatchRec, &cacheRec);
4263#ifdef VBOX_WITH_STATISTICS
4264 if ( rc == VINF_SUCCESS
4265 && (pPatchRec->patch.flags & PATMFL_SYSENTER))
4266 {
4267 pVM->patm.s.uSysEnterPatchIdx = pPatchRec->patch.uPatchIdx;
4268 }
4269#endif
4270 }
4271 else
4272 if (pPatchRec->patch.flags & PATMFL_GUEST_SPECIFIC)
4273 {
4274 switch (cpu.pCurInstr->opcode)
4275 {
4276 case OP_SYSENTER:
4277 case OP_PUSH:
4278 rc = PATMInstallGuestSpecificPatch(pVM, &cpu, pInstrGC, pInstrHC, pPatchRec);
4279 if (rc == VINF_SUCCESS)
4280 {
4281 if (rc == VINF_SUCCESS)
4282 Log(("PATMR3InstallPatch GUEST: %s %RRv code32=%d\n", patmGetInstructionString(pPatchRec->patch.opcode, pPatchRec->patch.flags), pInstrGC, (flags & PATMFL_CODE32) ? 1 : 0));
4283 return rc;
4284 }
4285 break;
4286
4287 default:
4288 rc = VERR_NOT_IMPLEMENTED;
4289 break;
4290 }
4291 }
4292 else
4293 {
4294 switch (cpu.pCurInstr->opcode)
4295 {
4296 case OP_SYSENTER:
4297 rc = PATMInstallGuestSpecificPatch(pVM, &cpu, pInstrGC, pInstrHC, pPatchRec);
4298 if (rc == VINF_SUCCESS)
4299 {
4300 Log(("PATMR3InstallPatch GUEST: %s %RRv code32=%d\n", patmGetInstructionString(pPatchRec->patch.opcode, pPatchRec->patch.flags), pInstrGC, (flags & PATMFL_CODE32) ? 1 : 0));
4301 return VINF_SUCCESS;
4302 }
4303 break;
4304
4305#ifdef PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES
4306 case OP_JO:
4307 case OP_JNO:
4308 case OP_JC:
4309 case OP_JNC:
4310 case OP_JE:
4311 case OP_JNE:
4312 case OP_JBE:
4313 case OP_JNBE:
4314 case OP_JS:
4315 case OP_JNS:
4316 case OP_JP:
4317 case OP_JNP:
4318 case OP_JL:
4319 case OP_JNL:
4320 case OP_JLE:
4321 case OP_JNLE:
4322 case OP_JECXZ:
4323 case OP_LOOP:
4324 case OP_LOOPNE:
4325 case OP_LOOPE:
4326 case OP_JMP:
4327 if (pPatchRec->patch.flags & PATMFL_JUMP_CONFLICT)
4328 {
4329 rc = patmPatchJump(pVM, pInstrGC, pInstrHC, &cpu, pPatchRec);
4330 break;
4331 }
4332 return VERR_NOT_IMPLEMENTED;
4333#endif
4334
4335 case OP_PUSHF:
4336 case OP_CLI:
4337 Log(("PATMR3InstallPatch %s %RRv code32=%d\n", patmGetInstructionString(pPatchRec->patch.opcode, pPatchRec->patch.flags), pInstrGC, (flags & PATMFL_CODE32) ? 1 : 0));
4338 rc = PATMR3PatchBlock(pVM, pInstrGC, pInstrHC, cpu.pCurInstr->opcode, opsize, pPatchRec);
4339 break;
4340
4341 case OP_STR:
4342 case OP_SGDT:
4343 case OP_SLDT:
4344 case OP_SIDT:
4345 case OP_CPUID:
4346 case OP_LSL:
4347 case OP_LAR:
4348 case OP_SMSW:
4349 case OP_VERW:
4350 case OP_VERR:
4351 case OP_IRET:
4352 rc = PATMR3PatchInstrInt3(pVM, pInstrGC, pInstrHC, &cpu, &pPatchRec->patch);
4353 break;
4354
4355 default:
4356 return VERR_NOT_IMPLEMENTED;
4357 }
4358 }
4359
4360 if (rc != VINF_SUCCESS)
4361 {
4362 if (pPatchRec && pPatchRec->patch.nrPatch2GuestRecs)
4363 {
4364 patmEmptyTreeU32(pVM, &pPatchRec->patch.Patch2GuestAddrTree);
4365 pPatchRec->patch.nrPatch2GuestRecs = 0;
4366 }
4367 pVM->patm.s.uCurrentPatchIdx--;
4368 }
4369 else
4370 {
4371 rc = patmInsertPatchPages(pVM, &pPatchRec->patch);
4372 AssertRCReturn(rc, rc);
4373
4374 /* Keep track upper and lower boundaries of patched instructions */
4375 if (pPatchRec->patch.pInstrGCLowest < pVM->patm.s.pPatchedInstrGCLowest)
4376 pVM->patm.s.pPatchedInstrGCLowest = pPatchRec->patch.pInstrGCLowest;
4377 if (pPatchRec->patch.pInstrGCHighest > pVM->patm.s.pPatchedInstrGCHighest)
4378 pVM->patm.s.pPatchedInstrGCHighest = pPatchRec->patch.pInstrGCHighest;
4379
4380 Log(("Patch lowest %RRv highest %RRv\n", pPatchRec->patch.pInstrGCLowest, pPatchRec->patch.pInstrGCHighest));
4381 Log(("Global lowest %RRv highest %RRv\n", pVM->patm.s.pPatchedInstrGCLowest, pVM->patm.s.pPatchedInstrGCHighest));
4382
4383 STAM_COUNTER_ADD(&pVM->patm.s.StatInstalled, 1);
4384 STAM_COUNTER_ADD(&pVM->patm.s.StatPATMMemoryUsed, pPatchRec->patch.cbPatchBlockSize);
4385
4386 rc = VINF_SUCCESS;
4387
4388 /* Patch hints are not enabled by default. Only when the are actually encountered. */
4389 if (pPatchRec->patch.flags & PATMFL_INSTR_HINT)
4390 {
4391 rc = PATMR3DisablePatch(pVM, pInstrGC);
4392 AssertRCReturn(rc, rc);
4393 }
4394
4395#ifdef VBOX_WITH_STATISTICS
4396 /* Register statistics counter */
4397 if (PATM_STAT_INDEX_IS_VALID(pPatchRec->patch.uPatchIdx))
4398 {
4399 STAMR3RegisterCallback(pVM, &pPatchRec->patch, STAMVISIBILITY_NOT_GUI, STAMUNIT_GOOD_BAD, patmResetStat, patmPrintStat, "Patch statistics",
4400 "/PATM/Stats/Patch/0x%RRv", pPatchRec->patch.pPrivInstrGC);
4401#ifndef DEBUG_sandervl
4402 /* Full breakdown for the GUI. */
4403 STAMR3RegisterF(pVM, &pVM->patm.s.pStatsHC[pPatchRec->patch.uPatchIdx], STAMTYPE_RATIO_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_GOOD_BAD, PATMPatchType(pVM, &pPatchRec->patch),
4404 "/PATM/Stats/PatchBD/0x%RRv", pPatchRec->patch.pPrivInstrGC);
4405 STAMR3RegisterF(pVM, &pPatchRec->patch.cbPatchBlockSize,STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cbPatchBlockSize", pPatchRec->patch.pPrivInstrGC);
4406 STAMR3RegisterF(pVM, &pPatchRec->patch.cbPatchJump, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cbPatchJump", pPatchRec->patch.pPrivInstrGC);
4407 STAMR3RegisterF(pVM, &pPatchRec->patch.cbPrivInstr, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cbPrivInstr", pPatchRec->patch.pPrivInstrGC);
4408 STAMR3RegisterF(pVM, &pPatchRec->patch.cCodeWrites, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cCodeWrites", pPatchRec->patch.pPrivInstrGC);
4409 STAMR3RegisterF(pVM, &pPatchRec->patch.cInvalidWrites, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cInvalidWrites", pPatchRec->patch.pPrivInstrGC);
4410 STAMR3RegisterF(pVM, &pPatchRec->patch.cTraps, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/cTraps", pPatchRec->patch.pPrivInstrGC);
4411 STAMR3RegisterF(pVM, &pPatchRec->patch.flags, STAMTYPE_X32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE, NULL, "/PATM/Stats/PatchBD/0x%RRv/flags", pPatchRec->patch.pPrivInstrGC);
4412 STAMR3RegisterF(pVM, &pPatchRec->patch.nrJumpRecs, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/nrJumpRecs", pPatchRec->patch.pPrivInstrGC);
4413 STAMR3RegisterF(pVM, &pPatchRec->patch.nrFixups, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/nrFixups", pPatchRec->patch.pPrivInstrGC);
4414 STAMR3RegisterF(pVM, &pPatchRec->patch.opcode, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, NULL, "/PATM/Stats/PatchBD/0x%RRv/opcode", pPatchRec->patch.pPrivInstrGC);
4415 STAMR3RegisterF(pVM, &pPatchRec->patch.uOldState, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE, NULL, "/PATM/Stats/PatchBD/0x%RRv/uOldState", pPatchRec->patch.pPrivInstrGC);
4416 STAMR3RegisterF(pVM, &pPatchRec->patch.uOpMode, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE, NULL, "/PATM/Stats/PatchBD/0x%RRv/uOpMode", pPatchRec->patch.pPrivInstrGC);
4417 /// @todo change the state to be a callback so we can get a state mnemonic instead.
4418 STAMR3RegisterF(pVM, &pPatchRec->patch.uState, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE, NULL, "/PATM/Stats/PatchBD/0x%RRv/uState", pPatchRec->patch.pPrivInstrGC);
4419#endif
4420 }
4421#endif
4422 }
4423 /* Free leftover lock if any. */
4424 if (cacheRec.Lock.pvMap)
4425 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
4426 return rc;
4427}
4428
4429/**
4430 * Query instruction size
4431 *
4432 * @returns VBox status code.
4433 * @param pVM The VM to operate on.
4434 * @param pPatch Patch record
4435 * @param pInstrGC Instruction address
4436 */
4437static uint32_t patmGetInstrSize(PVM pVM, PPATCHINFO pPatch, RTRCPTR pInstrGC)
4438{
4439 uint8_t *pInstrHC;
4440 PGMPAGEMAPLOCK Lock;
4441
4442 int rc = PGMPhysGCPtr2CCPtrReadOnly(VMMGetCpu(pVM), pInstrGC, (const void **)&pInstrHC, &Lock);
4443 if (rc == VINF_SUCCESS)
4444 {
4445 DISCPUSTATE cpu;
4446 bool disret;
4447 uint32_t opsize;
4448
4449 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
4450 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pInstrGC, pInstrHC, &opsize, NULL, PATMREAD_ORGCODE | PATMREAD_NOCHECK);
4451 PGMPhysReleasePageMappingLock(pVM, &Lock);
4452 if (disret)
4453 return opsize;
4454 }
4455 return 0;
4456}
4457
4458/**
4459 * Add patch to page record
4460 *
4461 * @returns VBox status code.
4462 * @param pVM The VM to operate on.
4463 * @param pPage Page address
4464 * @param pPatch Patch record
4465 */
4466int patmAddPatchToPage(PVM pVM, RTRCUINTPTR pPage, PPATCHINFO pPatch)
4467{
4468 PPATMPATCHPAGE pPatchPage;
4469 int rc;
4470
4471 Log(("patmAddPatchToPage: insert patch %RHv to page %RRv\n", pPatch, pPage));
4472
4473 pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, pPage);
4474 if (pPatchPage)
4475 {
4476 Assert(pPatchPage->cCount <= pPatchPage->cMaxPatches);
4477 if (pPatchPage->cCount == pPatchPage->cMaxPatches)
4478 {
4479 uint32_t cMaxPatchesOld = pPatchPage->cMaxPatches;
4480 PPATCHINFO *paPatchOld = pPatchPage->aPatch;
4481
4482 pPatchPage->cMaxPatches += PATMPATCHPAGE_PREALLOC_INCREMENT;
4483 rc = MMHyperAlloc(pVM, sizeof(PPATCHINFO)*pPatchPage->cMaxPatches, 0, MM_TAG_PATM_PATCH, (void **)&pPatchPage->aPatch);
4484 if (RT_FAILURE(rc))
4485 {
4486 Log(("Out of memory!!!!\n"));
4487 return VERR_NO_MEMORY;
4488 }
4489 memcpy(pPatchPage->aPatch, paPatchOld, cMaxPatchesOld*sizeof(PPATCHINFO));
4490 MMHyperFree(pVM, paPatchOld);
4491 }
4492 pPatchPage->aPatch[pPatchPage->cCount] = pPatch;
4493 pPatchPage->cCount++;
4494 }
4495 else
4496 {
4497 rc = MMHyperAlloc(pVM, sizeof(PATMPATCHPAGE), 0, MM_TAG_PATM_PATCH, (void **)&pPatchPage);
4498 if (RT_FAILURE(rc))
4499 {
4500 Log(("Out of memory!!!!\n"));
4501 return VERR_NO_MEMORY;
4502 }
4503 pPatchPage->Core.Key = pPage;
4504 pPatchPage->cCount = 1;
4505 pPatchPage->cMaxPatches = PATMPATCHPAGE_PREALLOC_INCREMENT;
4506
4507 rc = MMHyperAlloc(pVM, sizeof(PPATCHINFO)*PATMPATCHPAGE_PREALLOC_INCREMENT, 0, MM_TAG_PATM_PATCH, (void **)&pPatchPage->aPatch);
4508 if (RT_FAILURE(rc))
4509 {
4510 Log(("Out of memory!!!!\n"));
4511 MMHyperFree(pVM, pPatchPage);
4512 return VERR_NO_MEMORY;
4513 }
4514 pPatchPage->aPatch[0] = pPatch;
4515
4516 rc = RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, &pPatchPage->Core);
4517 Assert(rc);
4518 pVM->patm.s.cPageRecords++;
4519
4520 STAM_COUNTER_INC(&pVM->patm.s.StatPatchPageInserted);
4521 }
4522 CSAMR3MonitorPage(pVM, pPage, CSAM_TAG_PATM);
4523
4524 /* Get the closest guest instruction (from below) */
4525 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pPage, true);
4526 Assert(pGuestToPatchRec);
4527 if (pGuestToPatchRec)
4528 {
4529 LogFlow(("patmAddPatchToPage: lowest patch page address %RRv current lowest %RRv\n", pGuestToPatchRec->Core.Key, pPatchPage->pLowestAddrGC));
4530 if ( pPatchPage->pLowestAddrGC == 0
4531 || pPatchPage->pLowestAddrGC > (RTRCPTR)pGuestToPatchRec->Core.Key)
4532 {
4533 RTRCUINTPTR offset;
4534
4535 pPatchPage->pLowestAddrGC = (RTRCPTR)pGuestToPatchRec->Core.Key;
4536
4537 offset = pPatchPage->pLowestAddrGC & PAGE_OFFSET_MASK;
4538 /* If we're too close to the page boundary, then make sure an instruction from the previous page doesn't cross the boundary itself. */
4539 if (offset && offset < MAX_INSTR_SIZE)
4540 {
4541 /* Get the closest guest instruction (from above) */
4542 pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pPage-1, false);
4543
4544 if (pGuestToPatchRec)
4545 {
4546 uint32_t size = patmGetInstrSize(pVM, pPatch, (RTRCPTR)pGuestToPatchRec->Core.Key);
4547 if ((RTRCUINTPTR)pGuestToPatchRec->Core.Key + size > pPage)
4548 {
4549 pPatchPage->pLowestAddrGC = pPage;
4550 LogFlow(("patmAddPatchToPage: new lowest %RRv\n", pPatchPage->pLowestAddrGC));
4551 }
4552 }
4553 }
4554 }
4555 }
4556
4557 /* Get the closest guest instruction (from above) */
4558 pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pPage+PAGE_SIZE-1, false);
4559 Assert(pGuestToPatchRec);
4560 if (pGuestToPatchRec)
4561 {
4562 LogFlow(("patmAddPatchToPage: highest patch page address %RRv current lowest %RRv\n", pGuestToPatchRec->Core.Key, pPatchPage->pHighestAddrGC));
4563 if ( pPatchPage->pHighestAddrGC == 0
4564 || pPatchPage->pHighestAddrGC <= (RTRCPTR)pGuestToPatchRec->Core.Key)
4565 {
4566 pPatchPage->pHighestAddrGC = (RTRCPTR)pGuestToPatchRec->Core.Key;
4567 /* Increase by instruction size. */
4568 uint32_t size = patmGetInstrSize(pVM, pPatch, pPatchPage->pHighestAddrGC);
4569//// Assert(size);
4570 pPatchPage->pHighestAddrGC += size;
4571 LogFlow(("patmAddPatchToPage: new highest %RRv\n", pPatchPage->pHighestAddrGC));
4572 }
4573 }
4574
4575 return VINF_SUCCESS;
4576}
4577
4578/**
4579 * Remove patch from page record
4580 *
4581 * @returns VBox status code.
4582 * @param pVM The VM to operate on.
4583 * @param pPage Page address
4584 * @param pPatch Patch record
4585 */
4586int patmRemovePatchFromPage(PVM pVM, RTRCUINTPTR pPage, PPATCHINFO pPatch)
4587{
4588 PPATMPATCHPAGE pPatchPage;
4589 int rc;
4590
4591 pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, pPage);
4592 Assert(pPatchPage);
4593
4594 if (!pPatchPage)
4595 return VERR_INVALID_PARAMETER;
4596
4597 Assert(pPatchPage->cCount <= pPatchPage->cMaxPatches);
4598
4599 Log(("patmRemovePatchPage: remove patch %RHv from page %RRv\n", pPatch, pPage));
4600 if (pPatchPage->cCount > 1)
4601 {
4602 uint32_t i;
4603
4604 /* Used by multiple patches */
4605 for (i=0;i<pPatchPage->cCount;i++)
4606 {
4607 if (pPatchPage->aPatch[i] == pPatch)
4608 {
4609 pPatchPage->aPatch[i] = 0;
4610 break;
4611 }
4612 }
4613 /* close the gap between the remaining pointers. */
4614 if (i < pPatchPage->cCount - 1)
4615 {
4616 memcpy(&pPatchPage->aPatch[i], &pPatchPage->aPatch[i+1], sizeof(PPATCHINFO)*(pPatchPage->cCount - (i+1)));
4617 }
4618 AssertMsg(i < pPatchPage->cCount, ("Unable to find patch %RHv in page %RRv\n", pPatch, pPage));
4619
4620 pPatchPage->cCount--;
4621 }
4622 else
4623 {
4624 PPATMPATCHPAGE pPatchNode;
4625
4626 Log(("patmRemovePatchFromPage %RRv\n", pPage));
4627
4628 STAM_COUNTER_INC(&pVM->patm.s.StatPatchPageRemoved);
4629 pPatchNode = (PPATMPATCHPAGE)RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, pPage);
4630 Assert(pPatchNode && pPatchNode == pPatchPage);
4631
4632 Assert(pPatchPage->aPatch);
4633 rc = MMHyperFree(pVM, pPatchPage->aPatch);
4634 AssertRC(rc);
4635 rc = MMHyperFree(pVM, pPatchPage);
4636 AssertRC(rc);
4637 pVM->patm.s.cPageRecords--;
4638 }
4639 return VINF_SUCCESS;
4640}
4641
4642/**
4643 * Insert page records for all guest pages that contain instructions that were recompiled for this patch
4644 *
4645 * @returns VBox status code.
4646 * @param pVM The VM to operate on.
4647 * @param pPatch Patch record
4648 */
4649int patmInsertPatchPages(PVM pVM, PPATCHINFO pPatch)
4650{
4651 int rc;
4652 RTRCUINTPTR pPatchPageStart, pPatchPageEnd, pPage;
4653
4654 /* Insert the pages that contain patched instructions into a lookup tree for detecting self-modifying code. */
4655 pPatchPageStart = (RTRCUINTPTR)pPatch->pInstrGCLowest & PAGE_BASE_GC_MASK;
4656 pPatchPageEnd = (RTRCUINTPTR)pPatch->pInstrGCHighest & PAGE_BASE_GC_MASK;
4657
4658 /** @todo optimize better (large gaps between current and next used page) */
4659 for(pPage = pPatchPageStart; pPage <= pPatchPageEnd; pPage += PAGE_SIZE)
4660 {
4661 /* Get the closest guest instruction (from above) */
4662 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pPage, true);
4663 if ( pGuestToPatchRec
4664 && PAGE_ADDRESS(pGuestToPatchRec->Core.Key) == PAGE_ADDRESS(pPage)
4665 )
4666 {
4667 /* Code in page really patched -> add record */
4668 rc = patmAddPatchToPage(pVM, pPage, pPatch);
4669 AssertRC(rc);
4670 }
4671 }
4672 pPatch->flags |= PATMFL_CODE_MONITORED;
4673 return VINF_SUCCESS;
4674}
4675
4676/**
4677 * Remove page records for all guest pages that contain instructions that were recompiled for this patch
4678 *
4679 * @returns VBox status code.
4680 * @param pVM The VM to operate on.
4681 * @param pPatch Patch record
4682 */
4683int patmRemovePatchPages(PVM pVM, PPATCHINFO pPatch)
4684{
4685 int rc;
4686 RTRCUINTPTR pPatchPageStart, pPatchPageEnd, pPage;
4687
4688 /* Insert the pages that contain patched instructions into a lookup tree for detecting self-modifying code. */
4689 pPatchPageStart = (RTRCUINTPTR)pPatch->pInstrGCLowest & PAGE_BASE_GC_MASK;
4690 pPatchPageEnd = (RTRCUINTPTR)pPatch->pInstrGCHighest & PAGE_BASE_GC_MASK;
4691
4692 for(pPage = pPatchPageStart; pPage <= pPatchPageEnd; pPage += PAGE_SIZE)
4693 {
4694 /* Get the closest guest instruction (from above) */
4695 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pPage, true);
4696 if ( pGuestToPatchRec
4697 && PAGE_ADDRESS(pGuestToPatchRec->Core.Key) == PAGE_ADDRESS(pPage) /** @todo bird: PAGE_ADDRESS is for the current context really. check out these. */
4698 )
4699 {
4700 /* Code in page really patched -> remove record */
4701 rc = patmRemovePatchFromPage(pVM, pPage, pPatch);
4702 AssertRC(rc);
4703 }
4704 }
4705 pPatch->flags &= ~PATMFL_CODE_MONITORED;
4706 return VINF_SUCCESS;
4707}
4708
4709/**
4710 * Notifies PATM about a (potential) write to code that has been patched.
4711 *
4712 * @returns VBox status code.
4713 * @param pVM The VM to operate on.
4714 * @param GCPtr GC pointer to write address
4715 * @param cbWrite Nr of bytes to write
4716 *
4717 */
4718VMMR3DECL(int) PATMR3PatchWrite(PVM pVM, RTRCPTR GCPtr, uint32_t cbWrite)
4719{
4720 RTRCUINTPTR pWritePageStart, pWritePageEnd, pPage;
4721
4722 Log(("PATMR3PatchWrite %RRv %x\n", GCPtr, cbWrite));
4723
4724 Assert(VM_IS_EMT(pVM));
4725
4726 /* Quick boundary check */
4727 if ( GCPtr < pVM->patm.s.pPatchedInstrGCLowest
4728 || GCPtr > pVM->patm.s.pPatchedInstrGCHighest
4729 )
4730 return VINF_SUCCESS;
4731
4732 STAM_PROFILE_ADV_START(&pVM->patm.s.StatPatchWrite, a);
4733
4734 pWritePageStart = (RTRCUINTPTR)GCPtr & PAGE_BASE_GC_MASK;
4735 pWritePageEnd = ((RTRCUINTPTR)GCPtr + cbWrite - 1) & PAGE_BASE_GC_MASK;
4736
4737 for (pPage = pWritePageStart; pPage <= pWritePageEnd; pPage += PAGE_SIZE)
4738 {
4739loop_start:
4740 PPATMPATCHPAGE pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, (RTRCPTR)pPage);
4741 if (pPatchPage)
4742 {
4743 uint32_t i;
4744 bool fValidPatchWrite = false;
4745
4746 /* Quick check to see if the write is in the patched part of the page */
4747 if ( pPatchPage->pLowestAddrGC > (RTRCPTR)((RTRCUINTPTR)GCPtr + cbWrite - 1)
4748 || pPatchPage->pHighestAddrGC < GCPtr)
4749 {
4750 break;
4751 }
4752
4753 for (i=0;i<pPatchPage->cCount;i++)
4754 {
4755 if (pPatchPage->aPatch[i])
4756 {
4757 PPATCHINFO pPatch = pPatchPage->aPatch[i];
4758 RTRCPTR pPatchInstrGC;
4759 //unused: bool fForceBreak = false;
4760
4761 Assert(pPatchPage->aPatch[i]->flags & PATMFL_CODE_MONITORED);
4762 /** @todo inefficient and includes redundant checks for multiple pages. */
4763 for (uint32_t j=0; j<cbWrite; j++)
4764 {
4765 RTRCPTR pGuestPtrGC = (RTRCPTR)((RTRCUINTPTR)GCPtr + j);
4766
4767 if ( pPatch->cbPatchJump
4768 && pGuestPtrGC >= pPatch->pPrivInstrGC
4769 && pGuestPtrGC < pPatch->pPrivInstrGC + pPatch->cbPatchJump)
4770 {
4771 /* The guest is about to overwrite the 5 byte jump to patch code. Remove the patch. */
4772 Log(("PATMR3PatchWrite: overwriting jump to patch code -> remove patch.\n"));
4773 int rc = PATMR3RemovePatch(pVM, pPatch->pPrivInstrGC);
4774 if (rc == VINF_SUCCESS)
4775 /* Note: jump back to the start as the pPatchPage has been deleted or changed */
4776 goto loop_start;
4777
4778 continue;
4779 }
4780
4781 /* Find the closest instruction from below; the above quick check ensured that we are indeed in patched code */
4782 pPatchInstrGC = patmGuestGCPtrToPatchGCPtr(pVM, pPatch, pGuestPtrGC);
4783 if (!pPatchInstrGC)
4784 {
4785 RTRCPTR pClosestInstrGC;
4786 uint32_t size;
4787
4788 pPatchInstrGC = patmGuestGCPtrToClosestPatchGCPtr(pVM, pPatch, pGuestPtrGC);
4789 if (pPatchInstrGC)
4790 {
4791 pClosestInstrGC = patmPatchGCPtr2GuestGCPtr(pVM, pPatch, pPatchInstrGC);
4792 Assert(pClosestInstrGC <= pGuestPtrGC);
4793 size = patmGetInstrSize(pVM, pPatch, pClosestInstrGC);
4794 /* Check if this is not a write into a gap between two patches */
4795 if (pClosestInstrGC + size - 1 < pGuestPtrGC)
4796 pPatchInstrGC = 0;
4797 }
4798 }
4799 if (pPatchInstrGC)
4800 {
4801 uint32_t PatchOffset = pPatchInstrGC - pVM->patm.s.pPatchMemGC; /* Offset in memory reserved for PATM. */
4802
4803 fValidPatchWrite = true;
4804
4805 PRECPATCHTOGUEST pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32Get(&pPatch->Patch2GuestAddrTree, PatchOffset);
4806 Assert(pPatchToGuestRec);
4807 if (pPatchToGuestRec && !pPatchToGuestRec->fDirty)
4808 {
4809 Log(("PATMR3PatchWrite: Found patched instruction %RRv -> %RRv\n", pGuestPtrGC, pPatchInstrGC));
4810
4811 if (++pPatch->cCodeWrites > PATM_MAX_CODE_WRITES)
4812 {
4813 LogRel(("PATM: Disable block at %RRv - write %RRv-%RRv\n", pPatch->pPrivInstrGC, pGuestPtrGC, pGuestPtrGC+cbWrite));
4814
4815 PATMR3MarkDirtyPatch(pVM, pPatch);
4816
4817 /* Note: jump back to the start as the pPatchPage has been deleted or changed */
4818 goto loop_start;
4819 }
4820 else
4821 {
4822 /* Replace the patch instruction with a breakpoint; when it's hit, then we'll attempt to recompile the instruction again. */
4823 uint8_t *pInstrHC = patmPatchGCPtr2PatchHCPtr(pVM, pPatchInstrGC);
4824
4825 pPatchToGuestRec->u8DirtyOpcode = *pInstrHC;
4826 pPatchToGuestRec->fDirty = true;
4827
4828 *pInstrHC = 0xCC;
4829
4830 STAM_COUNTER_INC(&pVM->patm.s.StatInstrDirty);
4831 }
4832 }
4833 /* else already marked dirty */
4834 }
4835 }
4836 }
4837 } /* for each patch */
4838
4839 if (fValidPatchWrite == false)
4840 {
4841 /* Write to a part of the page that either:
4842 * - doesn't contain any code (shared code/data); rather unlikely
4843 * - old code page that's no longer in active use.
4844 */
4845invalid_write_loop_start:
4846 pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, (RTRCPTR)pPage);
4847
4848 if (pPatchPage)
4849 {
4850 for (i=0;i<pPatchPage->cCount;i++)
4851 {
4852 PPATCHINFO pPatch = pPatchPage->aPatch[i];
4853
4854 if (pPatch->cInvalidWrites > PATM_MAX_INVALID_WRITES)
4855 {
4856 /* Note: possibly dangerous assumption that all future writes will be harmless. */
4857 if (pPatch->flags & PATMFL_IDTHANDLER)
4858 {
4859 LogRel(("PATM: Stop monitoring IDT handler pages at %RRv - invalid write %RRv-%RRv (this is not a fatal error)\n", pPatch->pPrivInstrGC, GCPtr, GCPtr+cbWrite));
4860
4861 Assert(pPatch->flags & PATMFL_CODE_MONITORED);
4862 int rc = patmRemovePatchPages(pVM, pPatch);
4863 AssertRC(rc);
4864 }
4865 else
4866 {
4867 LogRel(("PATM: Disable block at %RRv - invalid write %RRv-%RRv \n", pPatch->pPrivInstrGC, GCPtr, GCPtr+cbWrite));
4868 PATMR3MarkDirtyPatch(pVM, pPatch);
4869 }
4870 /* Note: jump back to the start as the pPatchPage has been deleted or changed */
4871 goto invalid_write_loop_start;
4872 }
4873 } /* for */
4874 }
4875 }
4876 }
4877 }
4878 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatPatchWrite, a);
4879 return VINF_SUCCESS;
4880
4881}
4882
4883/**
4884 * Disable all patches in a flushed page
4885 *
4886 * @returns VBox status code
4887 * @param pVM The VM to operate on.
4888 * @param addr GC address of the page to flush
4889 */
4890/** @note Currently only called by CSAMR3FlushPage; optimization to avoid having to double check if the physical address has changed
4891 */
4892VMMR3DECL(int) PATMR3FlushPage(PVM pVM, RTRCPTR addr)
4893{
4894 addr &= PAGE_BASE_GC_MASK;
4895
4896 PPATMPATCHPAGE pPatchPage = (PPATMPATCHPAGE)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPage, addr);
4897 if (pPatchPage)
4898 {
4899 int i;
4900
4901 /* From top to bottom as the array is modified by PATMR3MarkDirtyPatch. */
4902 for (i=(int)pPatchPage->cCount-1;i>=0;i--)
4903 {
4904 if (pPatchPage->aPatch[i])
4905 {
4906 PPATCHINFO pPatch = pPatchPage->aPatch[i];
4907
4908 Log(("PATMR3FlushPage %RRv remove patch at %RRv\n", addr, pPatch->pPrivInstrGC));
4909 PATMR3MarkDirtyPatch(pVM, pPatch);
4910 }
4911 }
4912 STAM_COUNTER_INC(&pVM->patm.s.StatFlushed);
4913 }
4914 return VINF_SUCCESS;
4915}
4916
4917/**
4918 * Checks if the instructions at the specified address has been patched already.
4919 *
4920 * @returns boolean, patched or not
4921 * @param pVM The VM to operate on.
4922 * @param pInstrGC Guest context pointer to instruction
4923 */
4924VMMR3DECL(bool) PATMR3HasBeenPatched(PVM pVM, RTRCPTR pInstrGC)
4925{
4926 PPATMPATCHREC pPatchRec;
4927 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
4928 if (pPatchRec && pPatchRec->patch.uState == PATCH_ENABLED)
4929 return true;
4930 return false;
4931}
4932
4933/**
4934 * Query the opcode of the original code that was overwritten by the 5 bytes patch jump
4935 *
4936 * @returns VBox status code.
4937 * @param pVM The VM to operate on.
4938 * @param pInstrGC GC address of instr
4939 * @param pByte opcode byte pointer (OUT)
4940 *
4941 */
4942VMMR3DECL(int) PATMR3QueryOpcode(PVM pVM, RTRCPTR pInstrGC, uint8_t *pByte)
4943{
4944 PPATMPATCHREC pPatchRec;
4945
4946 /** @todo this will not work for aliased pages! (never has, but so far not a problem for us) */
4947
4948 /* Shortcut. */
4949 if ( !PATMIsEnabled(pVM)
4950 || pInstrGC < pVM->patm.s.pPatchedInstrGCLowest
4951 || pInstrGC > pVM->patm.s.pPatchedInstrGCHighest)
4952 {
4953 return VERR_PATCH_NOT_FOUND;
4954 }
4955
4956 pPatchRec = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC, false);
4957 // if the patch is enabled and the pointer lies within 5 bytes of this priv instr ptr, then we've got a hit!
4958 if ( pPatchRec
4959 && pPatchRec->patch.uState == PATCH_ENABLED
4960 && pInstrGC >= pPatchRec->patch.pPrivInstrGC
4961 && pInstrGC < pPatchRec->patch.pPrivInstrGC + pPatchRec->patch.cbPatchJump)
4962 {
4963 RTRCPTR offset = pInstrGC - pPatchRec->patch.pPrivInstrGC;
4964 *pByte = pPatchRec->patch.aPrivInstr[offset];
4965
4966 if (pPatchRec->patch.cbPatchJump == 1)
4967 {
4968 Log(("PATMR3QueryOpcode: returning opcode %2X for instruction at %RRv\n", *pByte, pInstrGC));
4969 }
4970 STAM_COUNTER_ADD(&pVM->patm.s.StatNrOpcodeRead, 1);
4971 return VINF_SUCCESS;
4972 }
4973 return VERR_PATCH_NOT_FOUND;
4974}
4975
4976/**
4977 * Disable patch for privileged instruction at specified location
4978 *
4979 * @returns VBox status code.
4980 * @param pVM The VM to operate on.
4981 * @param pInstr Guest context point to privileged instruction
4982 *
4983 * @note returns failure if patching is not allowed or possible
4984 *
4985 */
4986VMMR3DECL(int) PATMR3DisablePatch(PVM pVM, RTRCPTR pInstrGC)
4987{
4988 PPATMPATCHREC pPatchRec;
4989 PPATCHINFO pPatch;
4990
4991 Log(("PATMR3DisablePatch: %RRv\n", pInstrGC));
4992 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
4993 if (pPatchRec)
4994 {
4995 int rc = VINF_SUCCESS;
4996
4997 pPatch = &pPatchRec->patch;
4998
4999 /* Already disabled? */
5000 if (pPatch->uState == PATCH_DISABLED)
5001 return VINF_SUCCESS;
5002
5003 /* Clear the IDT entries for the patch we're disabling. */
5004 /* Note: very important as we clear IF in the patch itself */
5005 /** @todo this needs to be changed */
5006 if (pPatch->flags & PATMFL_IDTHANDLER)
5007 {
5008 uint32_t iGate;
5009
5010 iGate = TRPMR3QueryGateByHandler(pVM, PATCHCODE_PTR_GC(pPatch));
5011 if (iGate != (uint32_t)~0)
5012 {
5013 TRPMR3SetGuestTrapHandler(pVM, iGate, TRPM_INVALID_HANDLER);
5014 if (++cIDTHandlersDisabled < 256)
5015 LogRel(("PATM: Disabling IDT %x patch handler %RRv\n", iGate, pInstrGC));
5016 }
5017 }
5018
5019 /* Mark the entry with a breakpoint in case somebody else calls it later on (cli patch used as a function, function, trampoline or idt patches) */
5020 if ( pPatch->pPatchBlockOffset
5021 && pPatch->uState == PATCH_ENABLED)
5022 {
5023 Log(("Invalidate patch at %RRv (HC=%RRv)\n", PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_HC(pPatch)));
5024 pPatch->bDirtyOpcode = *PATCHCODE_PTR_HC(pPatch);
5025 *PATCHCODE_PTR_HC(pPatch) = 0xCC;
5026 }
5027
5028 /* IDT or function patches haven't changed any guest code. */
5029 if (pPatch->flags & PATMFL_PATCHED_GUEST_CODE)
5030 {
5031 Assert(pPatch->flags & PATMFL_MUST_INSTALL_PATCHJMP);
5032 Assert(!(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER|PATMFL_TRAMPOLINE|PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK)));
5033
5034 if (pPatch->uState != PATCH_REFUSED)
5035 {
5036 uint8_t temp[16];
5037
5038 Assert(pPatch->cbPatchJump < sizeof(temp));
5039
5040 /* Let's first check if the guest code is still the same. */
5041 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), temp, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
5042 Assert(rc == VINF_SUCCESS || rc == VERR_PAGE_TABLE_NOT_PRESENT || rc == VERR_PAGE_NOT_PRESENT);
5043 if (rc == VINF_SUCCESS)
5044 {
5045 RTRCINTPTR displ = (RTRCUINTPTR)PATCHCODE_PTR_GC(pPatch) - ((RTRCUINTPTR)pPatch->pPrivInstrGC + SIZEOF_NEARJUMP32);
5046
5047 if ( temp[0] != 0xE9 /* jmp opcode */
5048 || *(RTRCINTPTR *)(&temp[1]) != displ
5049 )
5050 {
5051 Log(("PATMR3DisablePatch: Can't disable a patch who's guest code has changed!!\n"));
5052 STAM_COUNTER_INC(&pVM->patm.s.StatOverwritten);
5053 /* Remove it completely */
5054 pPatch->uState = PATCH_DISABLED; /* don't call PATMR3DisablePatch again */
5055 rc = PATMR3RemovePatch(pVM, pInstrGC);
5056 AssertRC(rc);
5057 return VWRN_PATCH_REMOVED;
5058 }
5059 patmRemoveJumpToPatch(pVM, pPatch);
5060 }
5061 else
5062 {
5063 Log(("PATMR3DisablePatch: unable to disable patch -> mark PATCH_DISABLE_PENDING\n"));
5064 pPatch->uState = PATCH_DISABLE_PENDING;
5065 }
5066 }
5067 else
5068 {
5069 AssertMsgFailed(("Patch was refused!\n"));
5070 return VERR_PATCH_ALREADY_DISABLED;
5071 }
5072 }
5073 else
5074 if (pPatch->flags & (PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK))
5075 {
5076 uint8_t temp[16];
5077
5078 Assert(pPatch->cbPatchJump < sizeof(temp));
5079
5080 /* Let's first check if the guest code is still the same. */
5081 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), temp, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
5082 Assert(rc == VINF_SUCCESS || rc == VERR_PAGE_TABLE_NOT_PRESENT || rc == VERR_PAGE_NOT_PRESENT);
5083 if (rc == VINF_SUCCESS)
5084 {
5085 if (temp[0] != 0xCC)
5086 {
5087 Log(("PATMR3DisablePatch: Can't disable a patch who's guest code has changed!!\n"));
5088 STAM_COUNTER_INC(&pVM->patm.s.StatOverwritten);
5089 /* Remove it completely */
5090 pPatch->uState = PATCH_DISABLED; /* don't call PATMR3DisablePatch again */
5091 rc = PATMR3RemovePatch(pVM, pInstrGC);
5092 AssertRC(rc);
5093 return VWRN_PATCH_REMOVED;
5094 }
5095 patmDeactivateInt3Patch(pVM, pPatch);
5096 }
5097 }
5098
5099 if (rc == VINF_SUCCESS)
5100 {
5101 /* Save old state and mark this one as disabled (so it can be enabled later on). */
5102 if (pPatch->uState == PATCH_DISABLE_PENDING)
5103 {
5104 /* Just to be safe, let's make sure this one can never be reused; the patch might be marked dirty already (int3 at start) */
5105 pPatch->uState = PATCH_UNUSABLE;
5106 }
5107 else
5108 if (pPatch->uState != PATCH_DIRTY)
5109 {
5110 pPatch->uOldState = pPatch->uState;
5111 pPatch->uState = PATCH_DISABLED;
5112 }
5113 STAM_COUNTER_ADD(&pVM->patm.s.StatDisabled, 1);
5114 }
5115
5116 Log(("PATMR3DisablePatch: disabled patch at %RRv\n", pInstrGC));
5117 return VINF_SUCCESS;
5118 }
5119 Log(("Patch not found!\n"));
5120 return VERR_PATCH_NOT_FOUND;
5121}
5122
5123/**
5124 * Permanently disable patch for privileged instruction at specified location
5125 *
5126 * @returns VBox status code.
5127 * @param pVM The VM to operate on.
5128 * @param pInstr Guest context instruction pointer
5129 * @param pConflictAddr Guest context pointer which conflicts with specified patch
5130 * @param pConflictPatch Conflicting patch
5131 *
5132 */
5133static int patmDisableUnusablePatch(PVM pVM, RTRCPTR pInstrGC, RTRCPTR pConflictAddr, PPATCHINFO pConflictPatch)
5134{
5135#ifdef PATM_RESOLVE_CONFLICTS_WITH_JUMP_PATCHES
5136 PATCHINFO patch;
5137 DISCPUSTATE cpu;
5138 R3PTRTYPE(uint8_t *) pInstrHC;
5139 uint32_t opsize;
5140 bool disret;
5141 int rc;
5142
5143 RT_ZERO(patch);
5144 pInstrHC = PATMGCVirtToHCVirt(pVM, &patch, pInstrGC);
5145 cpu.mode = (pConflictPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
5146 disret = PATMR3DISInstr(pVM, &patch, &cpu, pInstrGC, pInstrHC, &opsize, NULL);
5147 /*
5148 * If it's a 5 byte relative jump, then we can work around the problem by replacing the 32 bits relative offset
5149 * with one that jumps right into the conflict patch.
5150 * Otherwise we must disable the conflicting patch to avoid serious problems.
5151 */
5152 if ( disret == true
5153 && (pConflictPatch->flags & PATMFL_CODE32)
5154 && (cpu.pCurInstr->opcode == OP_JMP || (cpu.pCurInstr->optype & OPTYPE_COND_CONTROLFLOW))
5155 && (cpu.param1.flags & USE_IMMEDIATE32_REL))
5156 {
5157 /* Hint patches must be enabled first. */
5158 if (pConflictPatch->flags & PATMFL_INSTR_HINT)
5159 {
5160 Log(("Enabling HINTED patch %RRv\n", pConflictPatch->pPrivInstrGC));
5161 pConflictPatch->flags &= ~PATMFL_INSTR_HINT;
5162 rc = PATMR3EnablePatch(pVM, pConflictPatch->pPrivInstrGC);
5163 Assert(rc == VINF_SUCCESS || rc == VERR_PATCH_NOT_FOUND);
5164 /* Enabling might fail if the patched code has changed in the meantime. */
5165 if (rc != VINF_SUCCESS)
5166 return rc;
5167 }
5168
5169 rc = PATMR3InstallPatch(pVM, pInstrGC, PATMFL_CODE32 | PATMFL_JUMP_CONFLICT);
5170 if (RT_SUCCESS(rc))
5171 {
5172 Log(("PATM -> CONFLICT: Installed JMP patch for patch conflict at %RRv\n", pInstrGC));
5173 STAM_COUNTER_INC(&pVM->patm.s.StatFixedConflicts);
5174 return VINF_SUCCESS;
5175 }
5176 }
5177#endif
5178
5179 if (pConflictPatch->opcode == OP_CLI)
5180 {
5181 /* Turn it into an int3 patch; our GC trap handler will call the generated code manually. */
5182 Log(("PATM -> CONFLICT: Found active patch at instruction %RRv with target %RRv -> turn into int 3 patch!!\n", pInstrGC, pConflictPatch->pPrivInstrGC));
5183 int rc = PATMR3DisablePatch(pVM, pConflictPatch->pPrivInstrGC);
5184 if (rc == VWRN_PATCH_REMOVED)
5185 return VINF_SUCCESS;
5186 if (RT_SUCCESS(rc))
5187 {
5188 pConflictPatch->flags &= ~(PATMFL_MUST_INSTALL_PATCHJMP|PATMFL_INSTR_HINT);
5189 pConflictPatch->flags |= PATMFL_INT3_REPLACEMENT_BLOCK;
5190 rc = PATMR3EnablePatch(pVM, pConflictPatch->pPrivInstrGC);
5191 if (rc == VERR_PATCH_NOT_FOUND)
5192 return VINF_SUCCESS; /* removed already */
5193
5194 AssertRC(rc);
5195 if (RT_SUCCESS(rc))
5196 {
5197 STAM_COUNTER_INC(&pVM->patm.s.StatInt3Callable);
5198 return VINF_SUCCESS;
5199 }
5200 }
5201 /* else turned into unusable patch (see below) */
5202 }
5203 else
5204 {
5205 Log(("PATM -> CONFLICT: Found active patch at instruction %RRv with target %RRv -> DISABLING it!!\n", pInstrGC, pConflictPatch->pPrivInstrGC));
5206 int rc = PATMR3DisablePatch(pVM, pConflictPatch->pPrivInstrGC);
5207 if (rc == VWRN_PATCH_REMOVED)
5208 return VINF_SUCCESS;
5209 }
5210
5211 /* No need to monitor the code anymore. */
5212 if (pConflictPatch->flags & PATMFL_CODE_MONITORED)
5213 {
5214 int rc = patmRemovePatchPages(pVM, pConflictPatch);
5215 AssertRC(rc);
5216 }
5217 pConflictPatch->uState = PATCH_UNUSABLE;
5218 STAM_COUNTER_INC(&pVM->patm.s.StatUnusable);
5219 return VERR_PATCH_DISABLED;
5220}
5221
5222/**
5223 * Enable patch for privileged instruction at specified location
5224 *
5225 * @returns VBox status code.
5226 * @param pVM The VM to operate on.
5227 * @param pInstr Guest context point to privileged instruction
5228 *
5229 * @note returns failure if patching is not allowed or possible
5230 *
5231 */
5232VMMR3DECL(int) PATMR3EnablePatch(PVM pVM, RTRCPTR pInstrGC)
5233{
5234 PPATMPATCHREC pPatchRec;
5235 PPATCHINFO pPatch;
5236
5237 Log(("PATMR3EnablePatch %RRv\n", pInstrGC));
5238 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
5239 if (pPatchRec)
5240 {
5241 int rc = VINF_SUCCESS;
5242
5243 pPatch = &pPatchRec->patch;
5244
5245 if (pPatch->uState == PATCH_DISABLED)
5246 {
5247 if (pPatch->flags & PATMFL_MUST_INSTALL_PATCHJMP)
5248 {
5249 Assert(!(pPatch->flags & PATMFL_PATCHED_GUEST_CODE));
5250 uint8_t temp[16];
5251
5252 Assert(pPatch->cbPatchJump < sizeof(temp));
5253
5254 /* Let's first check if the guest code is still the same. */
5255 int rc2 = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), temp, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
5256 AssertRC(rc2);
5257 if (rc2 == VINF_SUCCESS)
5258 {
5259 if (memcmp(temp, pPatch->aPrivInstr, pPatch->cbPatchJump))
5260 {
5261 Log(("PATMR3EnablePatch: Can't enable a patch who's guest code has changed!!\n"));
5262 STAM_COUNTER_INC(&pVM->patm.s.StatOverwritten);
5263 /* Remove it completely */
5264 rc = PATMR3RemovePatch(pVM, pInstrGC);
5265 AssertRC(rc);
5266 return VERR_PATCH_NOT_FOUND;
5267 }
5268
5269 PATMP2GLOOKUPREC cacheRec;
5270 RT_ZERO(cacheRec);
5271 cacheRec.pPatch = pPatch;
5272
5273 rc2 = patmGenJumpToPatch(pVM, pPatch, &cacheRec, false);
5274 /* Free leftover lock if any. */
5275 if (cacheRec.Lock.pvMap)
5276 {
5277 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
5278 cacheRec.Lock.pvMap = NULL;
5279 }
5280 AssertRC(rc2);
5281 if (RT_FAILURE(rc2))
5282 return rc2;
5283
5284#ifdef DEBUG
5285 {
5286 DISCPUSTATE cpu;
5287 char szOutput[256];
5288 uint32_t opsize, i = 0;
5289 bool disret;
5290 i = 0;
5291 while(i < pPatch->cbPatchJump)
5292 {
5293 cpu.mode = (pPatch->flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
5294 disret = PATMR3DISInstr(pVM, pPatch, &cpu, pPatch->pPrivInstrGC + i, NULL, &opsize, szOutput);
5295 Log(("Renewed patch instr: %s", szOutput));
5296 i += opsize;
5297 }
5298 }
5299#endif
5300 }
5301 }
5302 else
5303 if (pPatch->flags & (PATMFL_INT3_REPLACEMENT|PATMFL_INT3_REPLACEMENT_BLOCK))
5304 {
5305 uint8_t temp[16];
5306
5307 Assert(pPatch->cbPatchJump < sizeof(temp));
5308
5309 /* Let's first check if the guest code is still the same. */
5310 int rc2 = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), temp, pPatch->pPrivInstrGC, pPatch->cbPatchJump);
5311 AssertRC(rc2);
5312
5313 if (memcmp(temp, pPatch->aPrivInstr, pPatch->cbPatchJump))
5314 {
5315 Log(("PATMR3EnablePatch: Can't enable a patch who's guest code has changed!!\n"));
5316 STAM_COUNTER_INC(&pVM->patm.s.StatOverwritten);
5317 rc = PATMR3RemovePatch(pVM, pInstrGC);
5318 AssertRC(rc);
5319 return VERR_PATCH_NOT_FOUND;
5320 }
5321
5322 rc2 = patmActivateInt3Patch(pVM, pPatch);
5323 if (RT_FAILURE(rc2))
5324 return rc2;
5325 }
5326
5327 pPatch->uState = pPatch->uOldState; //restore state
5328
5329 /* Restore the entry breakpoint with the original opcode (see PATMR3DisablePatch). */
5330 if (pPatch->pPatchBlockOffset)
5331 *PATCHCODE_PTR_HC(pPatch) = pPatch->bDirtyOpcode;
5332
5333 STAM_COUNTER_ADD(&pVM->patm.s.StatEnabled, 1);
5334 }
5335 else
5336 Log(("PATMR3EnablePatch: Unable to enable patch %RRv with state %d\n", pInstrGC, pPatch->uState));
5337
5338 return rc;
5339 }
5340 return VERR_PATCH_NOT_FOUND;
5341}
5342
5343/**
5344 * Remove patch for privileged instruction at specified location
5345 *
5346 * @returns VBox status code.
5347 * @param pVM The VM to operate on.
5348 * @param pPatchRec Patch record
5349 * @param fForceRemove Remove *all* patches
5350 */
5351int PATMRemovePatch(PVM pVM, PPATMPATCHREC pPatchRec, bool fForceRemove)
5352{
5353 PPATCHINFO pPatch;
5354
5355 pPatch = &pPatchRec->patch;
5356
5357 /* Strictly forbidden to remove such patches. There can be dependencies!! */
5358 if (!fForceRemove && (pPatch->flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_CODE_REFERENCED)))
5359 {
5360 Log(("PATMRemovePatch %RRv REFUSED!\n", pPatch->pPrivInstrGC));
5361 return VERR_ACCESS_DENIED;
5362 }
5363 Log(("PATMRemovePatch %RRv\n", pPatch->pPrivInstrGC));
5364
5365 /* Note: NEVER EVER REUSE PATCH MEMORY */
5366 /* Note: PATMR3DisablePatch puts a breakpoint (0xCC) at the entry of this patch */
5367
5368 if (pPatchRec->patch.pPatchBlockOffset)
5369 {
5370 PAVLOU32NODECORE pNode;
5371
5372 pNode = RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, pPatchRec->patch.pPatchBlockOffset);
5373 Assert(pNode);
5374 }
5375
5376 if (pPatchRec->patch.flags & PATMFL_CODE_MONITORED)
5377 {
5378 int rc = patmRemovePatchPages(pVM, &pPatchRec->patch);
5379 AssertRC(rc);
5380 }
5381
5382#ifdef VBOX_WITH_STATISTICS
5383 if (PATM_STAT_INDEX_IS_VALID(pPatchRec->patch.uPatchIdx))
5384 {
5385 STAMR3Deregister(pVM, &pPatchRec->patch);
5386#ifndef DEBUG_sandervl
5387 STAMR3Deregister(pVM, &pVM->patm.s.pStatsHC[pPatchRec->patch.uPatchIdx]);
5388 STAMR3Deregister(pVM, &pPatchRec->patch.cbPatchBlockSize);
5389 STAMR3Deregister(pVM, &pPatchRec->patch.cbPatchJump);
5390 STAMR3Deregister(pVM, &pPatchRec->patch.cbPrivInstr);
5391 STAMR3Deregister(pVM, &pPatchRec->patch.cCodeWrites);
5392 STAMR3Deregister(pVM, &pPatchRec->patch.cInvalidWrites);
5393 STAMR3Deregister(pVM, &pPatchRec->patch.cTraps);
5394 STAMR3Deregister(pVM, &pPatchRec->patch.flags);
5395 STAMR3Deregister(pVM, &pPatchRec->patch.nrJumpRecs);
5396 STAMR3Deregister(pVM, &pPatchRec->patch.nrFixups);
5397 STAMR3Deregister(pVM, &pPatchRec->patch.opcode);
5398 STAMR3Deregister(pVM, &pPatchRec->patch.uState);
5399 STAMR3Deregister(pVM, &pPatchRec->patch.uOldState);
5400 STAMR3Deregister(pVM, &pPatchRec->patch.uOpMode);
5401#endif
5402 }
5403#endif
5404
5405 /* Note: no need to free Guest2PatchAddrTree as those records share memory with Patch2GuestAddrTree records. */
5406 patmEmptyTreeU32(pVM, &pPatch->Patch2GuestAddrTree);
5407 pPatch->nrPatch2GuestRecs = 0;
5408 Assert(pPatch->Patch2GuestAddrTree == 0);
5409
5410 patmEmptyTree(pVM, &pPatch->FixupTree);
5411 pPatch->nrFixups = 0;
5412 Assert(pPatch->FixupTree == 0);
5413
5414 if (pPatchRec->patch.pTempInfo)
5415 MMR3HeapFree(pPatchRec->patch.pTempInfo);
5416
5417 /* Note: might fail, because it has already been removed (e.g. during reset). */
5418 RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pPatchRec->Core.Key);
5419
5420 /* Free the patch record */
5421 MMHyperFree(pVM, pPatchRec);
5422 return VINF_SUCCESS;
5423}
5424
5425/**
5426 * Attempt to refresh the patch by recompiling its entire code block
5427 *
5428 * @returns VBox status code.
5429 * @param pVM The VM to operate on.
5430 * @param pPatchRec Patch record
5431 */
5432int patmR3RefreshPatch(PVM pVM, PPATMPATCHREC pPatchRec)
5433{
5434 PPATCHINFO pPatch;
5435 int rc;
5436 RTRCPTR pInstrGC = pPatchRec->patch.pPrivInstrGC;
5437
5438 Log(("patmR3RefreshPatch: attempt to refresh patch at %RRv\n", pInstrGC));
5439
5440 pPatch = &pPatchRec->patch;
5441 AssertReturn(pPatch->flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER|PATMFL_TRAPHANDLER), VERR_PATCHING_REFUSED);
5442 if (pPatch->flags & PATMFL_EXTERNAL_JUMP_INSIDE)
5443 {
5444 Log(("patmR3RefreshPatch: refused because external jumps to this patch exist\n"));
5445 return VERR_PATCHING_REFUSED;
5446 }
5447
5448 /** Note: quite ugly to enable/disable/remove/insert old and new patches, but there's no easy way around it. */
5449
5450 rc = PATMR3DisablePatch(pVM, pInstrGC);
5451 AssertRC(rc);
5452
5453 /** Kick it out of the lookup tree to make sure PATMR3InstallPatch doesn't fail (hack alert) */
5454 RTAvloU32Remove(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pPatchRec->Core.Key);
5455#ifdef VBOX_WITH_STATISTICS
5456 if (PATM_STAT_INDEX_IS_VALID(pPatchRec->patch.uPatchIdx))
5457 {
5458 STAMR3Deregister(pVM, &pPatchRec->patch);
5459#ifndef DEBUG_sandervl
5460 STAMR3Deregister(pVM, &pVM->patm.s.pStatsHC[pPatchRec->patch.uPatchIdx]);
5461 STAMR3Deregister(pVM, &pPatchRec->patch.cbPatchBlockSize);
5462 STAMR3Deregister(pVM, &pPatchRec->patch.cbPatchJump);
5463 STAMR3Deregister(pVM, &pPatchRec->patch.cbPrivInstr);
5464 STAMR3Deregister(pVM, &pPatchRec->patch.cCodeWrites);
5465 STAMR3Deregister(pVM, &pPatchRec->patch.cInvalidWrites);
5466 STAMR3Deregister(pVM, &pPatchRec->patch.cTraps);
5467 STAMR3Deregister(pVM, &pPatchRec->patch.flags);
5468 STAMR3Deregister(pVM, &pPatchRec->patch.nrJumpRecs);
5469 STAMR3Deregister(pVM, &pPatchRec->patch.nrFixups);
5470 STAMR3Deregister(pVM, &pPatchRec->patch.opcode);
5471 STAMR3Deregister(pVM, &pPatchRec->patch.uState);
5472 STAMR3Deregister(pVM, &pPatchRec->patch.uOldState);
5473 STAMR3Deregister(pVM, &pPatchRec->patch.uOpMode);
5474#endif
5475 }
5476#endif
5477
5478 /** Note: We don't attempt to reuse patch memory here as it's quite common that the new code block requires more memory. */
5479
5480 /* Attempt to install a new patch. */
5481 rc = PATMR3InstallPatch(pVM, pInstrGC, pPatch->flags & (PATMFL_CODE32|PATMFL_IDTHANDLER|PATMFL_INTHANDLER|PATMFL_TRAPHANDLER|PATMFL_DUPLICATE_FUNCTION|PATMFL_TRAPHANDLER_WITH_ERRORCODE|PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT));
5482 if (RT_SUCCESS(rc))
5483 {
5484 RTRCPTR pPatchTargetGC;
5485 PPATMPATCHREC pNewPatchRec;
5486
5487 /* Determine target address in new patch */
5488 pPatchTargetGC = PATMR3QueryPatchGCPtr(pVM, pInstrGC);
5489 Assert(pPatchTargetGC);
5490 if (!pPatchTargetGC)
5491 {
5492 rc = VERR_PATCHING_REFUSED;
5493 goto failure;
5494 }
5495
5496 /* Reset offset into patch memory to put the next code blocks right at the beginning. */
5497 pPatch->uCurPatchOffset = 0;
5498
5499 /* insert jump to new patch in old patch block */
5500 rc = patmPatchGenPatchJump(pVM, pPatch, pInstrGC, pPatchTargetGC, false /* no lookup record */);
5501 if (RT_FAILURE(rc))
5502 goto failure;
5503
5504 pNewPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
5505 Assert(pNewPatchRec); /* can't fail */
5506
5507 /* Remove old patch (only do that when everything is finished) */
5508 int rc2 = PATMRemovePatch(pVM, pPatchRec, true /* force removal */);
5509 AssertRC(rc2);
5510
5511 /* Put the new patch back into the tree, because removing the old one kicked this one out. (hack alert) */
5512 RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTree, &pNewPatchRec->Core);
5513
5514 LogRel(("PATM: patmR3RefreshPatch: succeeded to refresh patch at %RRv \n", pInstrGC));
5515 STAM_COUNTER_INC(&pVM->patm.s.StatPatchRefreshSuccess);
5516
5517 /* Used by another patch, so don't remove it! */
5518 pNewPatchRec->patch.flags |= PATMFL_CODE_REFERENCED;
5519 }
5520
5521failure:
5522 if (RT_FAILURE(rc))
5523 {
5524 LogRel(("PATM: patmR3RefreshPatch: failed to refresh patch at %RRv. Reactiving old one. \n", pInstrGC));
5525
5526 /* Remove the new inactive patch */
5527 rc = PATMR3RemovePatch(pVM, pInstrGC);
5528 AssertRC(rc);
5529
5530 /* Put the old patch back into the tree (or else it won't be saved) (hack alert) */
5531 RTAvloU32Insert(&pVM->patm.s.PatchLookupTreeHC->PatchTree, &pPatchRec->Core);
5532
5533 /* Enable again in case the dirty instruction is near the end and there are safe code paths. */
5534 int rc2 = PATMR3EnablePatch(pVM, pInstrGC);
5535 AssertRC(rc2);
5536
5537 STAM_COUNTER_INC(&pVM->patm.s.StatPatchRefreshFailed);
5538 }
5539 return rc;
5540}
5541
5542/**
5543 * Find patch for privileged instruction at specified location
5544 *
5545 * @returns Patch structure pointer if found; else NULL
5546 * @param pVM The VM to operate on.
5547 * @param pInstr Guest context point to instruction that might lie within 5 bytes of an existing patch jump
5548 * @param fIncludeHints Include hinted patches or not
5549 *
5550 */
5551PPATCHINFO PATMFindActivePatchByEntrypoint(PVM pVM, RTRCPTR pInstrGC, bool fIncludeHints)
5552{
5553 PPATMPATCHREC pPatchRec = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC, false);
5554 /* if the patch is enabled, the pointer is not indentical to the privileged patch ptr and it lies within 5 bytes of this priv instr ptr, then we've got a hit! */
5555 if (pPatchRec)
5556 {
5557 if ( pPatchRec->patch.uState == PATCH_ENABLED
5558 && (pPatchRec->patch.flags & PATMFL_PATCHED_GUEST_CODE)
5559 && pInstrGC > pPatchRec->patch.pPrivInstrGC
5560 && pInstrGC < pPatchRec->patch.pPrivInstrGC + pPatchRec->patch.cbPatchJump)
5561 {
5562 Log(("Found active patch at %RRv (org %RRv)\n", pInstrGC, pPatchRec->patch.pPrivInstrGC));
5563 return &pPatchRec->patch;
5564 }
5565 else
5566 if ( fIncludeHints
5567 && pPatchRec->patch.uState == PATCH_DISABLED
5568 && (pPatchRec->patch.flags & PATMFL_INSTR_HINT)
5569 && pInstrGC > pPatchRec->patch.pPrivInstrGC
5570 && pInstrGC < pPatchRec->patch.pPrivInstrGC + pPatchRec->patch.cbPatchJump)
5571 {
5572 Log(("Found HINT patch at %RRv (org %RRv)\n", pInstrGC, pPatchRec->patch.pPrivInstrGC));
5573 return &pPatchRec->patch;
5574 }
5575 }
5576 return NULL;
5577}
5578
5579/**
5580 * Checks whether the GC address is inside a generated patch jump
5581 *
5582 * @returns true -> yes, false -> no
5583 * @param pVM The VM to operate on.
5584 * @param pAddr Guest context address
5585 * @param pPatchAddr Guest context patch address (if true)
5586 */
5587VMMR3DECL(bool) PATMR3IsInsidePatchJump(PVM pVM, RTRCPTR pAddr, PRTGCPTR32 pPatchAddr)
5588{
5589 RTRCPTR addr;
5590 PPATCHINFO pPatch;
5591
5592 if (PATMIsEnabled(pVM) == false)
5593 return false;
5594
5595 if (pPatchAddr == NULL)
5596 pPatchAddr = &addr;
5597
5598 *pPatchAddr = 0;
5599
5600 pPatch = PATMFindActivePatchByEntrypoint(pVM, pAddr);
5601 if (pPatch)
5602 *pPatchAddr = pPatch->pPrivInstrGC;
5603
5604 return *pPatchAddr == 0 ? false : true;
5605}
5606
5607/**
5608 * Remove patch for privileged instruction at specified location
5609 *
5610 * @returns VBox status code.
5611 * @param pVM The VM to operate on.
5612 * @param pInstr Guest context point to privileged instruction
5613 *
5614 * @note returns failure if patching is not allowed or possible
5615 *
5616 */
5617VMMR3DECL(int) PATMR3RemovePatch(PVM pVM, RTRCPTR pInstrGC)
5618{
5619 PPATMPATCHREC pPatchRec;
5620
5621 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC);
5622 if (pPatchRec)
5623 {
5624 int rc = PATMR3DisablePatch(pVM, pInstrGC);
5625 if (rc == VWRN_PATCH_REMOVED)
5626 return VINF_SUCCESS;
5627
5628 return PATMRemovePatch(pVM, pPatchRec, false);
5629 }
5630 AssertFailed();
5631 return VERR_PATCH_NOT_FOUND;
5632}
5633
5634/**
5635 * Mark patch as dirty
5636 *
5637 * @returns VBox status code.
5638 * @param pVM The VM to operate on.
5639 * @param pPatch Patch record
5640 *
5641 * @note returns failure if patching is not allowed or possible
5642 *
5643 */
5644VMMR3DECL(int) PATMR3MarkDirtyPatch(PVM pVM, PPATCHINFO pPatch)
5645{
5646 if (pPatch->pPatchBlockOffset)
5647 {
5648 Log(("Invalidate patch at %RRv (HC=%RRv)\n", PATCHCODE_PTR_GC(pPatch), PATCHCODE_PTR_HC(pPatch)));
5649 pPatch->bDirtyOpcode = *PATCHCODE_PTR_HC(pPatch);
5650 *PATCHCODE_PTR_HC(pPatch) = 0xCC;
5651 }
5652
5653 STAM_COUNTER_INC(&pVM->patm.s.StatDirty);
5654 /* Put back the replaced instruction. */
5655 int rc = PATMR3DisablePatch(pVM, pPatch->pPrivInstrGC);
5656 if (rc == VWRN_PATCH_REMOVED)
5657 return VINF_SUCCESS;
5658
5659 /* Note: we don't restore patch pages for patches that are not enabled! */
5660 /* Note: be careful when changing this behaviour!! */
5661
5662 /* The patch pages are no longer marked for self-modifying code detection */
5663 if (pPatch->flags & PATMFL_CODE_MONITORED)
5664 {
5665 rc = patmRemovePatchPages(pVM, pPatch);
5666 AssertRCReturn(rc, rc);
5667 }
5668 pPatch->uState = PATCH_DIRTY;
5669
5670 /* Paranoia; make sure this patch is not somewhere in the callchain, so prevent ret instructions from succeeding. */
5671 CTXSUFF(pVM->patm.s.pGCState)->Psp = PATM_STACK_SIZE;
5672
5673 return VINF_SUCCESS;
5674}
5675
5676/**
5677 * Query the corresponding GC instruction pointer from a pointer inside the patch block itself
5678 *
5679 * @returns VBox status code.
5680 * @param pVM The VM to operate on.
5681 * @param pPatch Patch block structure pointer
5682 * @param pPatchGC GC address in patch block
5683 */
5684RTRCPTR patmPatchGCPtr2GuestGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t *) pPatchGC)
5685{
5686 Assert(pPatch->Patch2GuestAddrTree);
5687 /* Get the closest record from below. */
5688 PRECPATCHTOGUEST pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32GetBestFit(&pPatch->Patch2GuestAddrTree, pPatchGC - pVM->patm.s.pPatchMemGC, false);
5689 if (pPatchToGuestRec)
5690 return pPatchToGuestRec->pOrgInstrGC;
5691
5692 return 0;
5693}
5694
5695/* Converts Guest code GC ptr to Patch code GC ptr (if found)
5696 *
5697 * @returns corresponding GC pointer in patch block
5698 * @param pVM The VM to operate on.
5699 * @param pPatch Current patch block pointer
5700 * @param pInstrGC Guest context pointer to privileged instruction
5701 *
5702 */
5703RTRCPTR patmGuestGCPtrToPatchGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t*) pInstrGC)
5704{
5705 if (pPatch->Guest2PatchAddrTree)
5706 {
5707 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32Get(&pPatch->Guest2PatchAddrTree, pInstrGC);
5708 if (pGuestToPatchRec)
5709 return pVM->patm.s.pPatchMemGC + pGuestToPatchRec->PatchOffset;
5710 }
5711
5712 return 0;
5713}
5714
5715/* Converts Guest code GC ptr to Patch code GC ptr (or nearest from below if no identical match)
5716 *
5717 * @returns corresponding GC pointer in patch block
5718 * @param pVM The VM to operate on.
5719 * @param pPatch Current patch block pointer
5720 * @param pInstrGC Guest context pointer to privileged instruction
5721 *
5722 */
5723RTRCPTR patmGuestGCPtrToClosestPatchGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t*) pInstrGC)
5724{
5725 PRECGUESTTOPATCH pGuestToPatchRec = (PRECGUESTTOPATCH)RTAvlU32GetBestFit(&pPatch->Guest2PatchAddrTree, pInstrGC, false);
5726 if (pGuestToPatchRec)
5727 return pVM->patm.s.pPatchMemGC + pGuestToPatchRec->PatchOffset;
5728
5729 return 0;
5730}
5731
5732/* Converts Guest code GC ptr to Patch code GC ptr (if found)
5733 *
5734 * @returns corresponding GC pointer in patch block
5735 * @param pVM The VM to operate on.
5736 * @param pInstrGC Guest context pointer to privileged instruction
5737 *
5738 */
5739VMMR3DECL(RTRCPTR) PATMR3GuestGCPtrToPatchGCPtr(PVM pVM, RCPTRTYPE(uint8_t*) pInstrGC)
5740{
5741 PPATMPATCHREC pPatchRec = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pInstrGC, false);
5742 if (pPatchRec && pPatchRec->patch.uState == PATCH_ENABLED && pInstrGC >= pPatchRec->patch.pPrivInstrGC)
5743 return patmGuestGCPtrToPatchGCPtr(pVM, &pPatchRec->patch, pInstrGC);
5744 else
5745 return 0;
5746}
5747
5748/**
5749 * Query the corresponding GC instruction pointer from a pointer inside the patch block itself
5750 *
5751 * @returns original GC instruction pointer or 0 if not found
5752 * @param pVM The VM to operate on.
5753 * @param pPatchGC GC address in patch block
5754 * @param pEnmState State of the translated address (out)
5755 *
5756 */
5757VMMR3DECL(RTRCPTR) PATMR3PatchToGCPtr(PVM pVM, RTRCPTR pPatchGC, PATMTRANSSTATE *pEnmState)
5758{
5759 PPATMPATCHREC pPatchRec;
5760 void *pvPatchCoreOffset;
5761 RTRCPTR pPrivInstrGC;
5762
5763 Assert(PATMIsPatchGCAddr(pVM, pPatchGC));
5764 pvPatchCoreOffset = RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, pPatchGC - pVM->patm.s.pPatchMemGC, false);
5765 if (pvPatchCoreOffset == 0)
5766 {
5767 Log(("PATMR3PatchToGCPtr failed for %RRv offset %x\n", pPatchGC, pPatchGC - pVM->patm.s.pPatchMemGC));
5768 return 0;
5769 }
5770 pPatchRec = PATM_PATCHREC_FROM_COREOFFSET(pvPatchCoreOffset);
5771 pPrivInstrGC = patmPatchGCPtr2GuestGCPtr(pVM, &pPatchRec->patch, pPatchGC);
5772 if (pEnmState)
5773 {
5774 AssertMsg(pPrivInstrGC && ( pPatchRec->patch.uState == PATCH_ENABLED
5775 || pPatchRec->patch.uState == PATCH_DIRTY
5776 || pPatchRec->patch.uState == PATCH_DISABLE_PENDING
5777 || pPatchRec->patch.uState == PATCH_UNUSABLE),
5778 ("pPrivInstrGC=%RRv uState=%d\n", pPrivInstrGC, pPatchRec->patch.uState));
5779
5780 if ( !pPrivInstrGC
5781 || pPatchRec->patch.uState == PATCH_UNUSABLE
5782 || pPatchRec->patch.uState == PATCH_REFUSED)
5783 {
5784 pPrivInstrGC = 0;
5785 *pEnmState = PATMTRANS_FAILED;
5786 }
5787 else
5788 if (pVM->patm.s.pGCStateHC->GCPtrInhibitInterrupts == pPrivInstrGC)
5789 {
5790 *pEnmState = PATMTRANS_INHIBITIRQ;
5791 }
5792 else
5793 if ( pPatchRec->patch.uState == PATCH_ENABLED
5794 && !(pPatchRec->patch.flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER|PATMFL_TRAMPOLINE))
5795 && pPrivInstrGC > pPatchRec->patch.pPrivInstrGC
5796 && pPrivInstrGC < pPatchRec->patch.pPrivInstrGC + pPatchRec->patch.cbPatchJump)
5797 {
5798 *pEnmState = PATMTRANS_OVERWRITTEN;
5799 }
5800 else
5801 if (PATMFindActivePatchByEntrypoint(pVM, pPrivInstrGC))
5802 {
5803 *pEnmState = PATMTRANS_OVERWRITTEN;
5804 }
5805 else
5806 if (pPrivInstrGC == pPatchRec->patch.pPrivInstrGC)
5807 {
5808 *pEnmState = PATMTRANS_PATCHSTART;
5809 }
5810 else
5811 *pEnmState = PATMTRANS_SAFE;
5812 }
5813 return pPrivInstrGC;
5814}
5815
5816/**
5817 * Returns the GC pointer of the patch for the specified GC address
5818 *
5819 * @returns VBox status code.
5820 * @param pVM The VM to operate on.
5821 * @param pAddrGC Guest context address
5822 */
5823VMMR3DECL(RTRCPTR) PATMR3QueryPatchGCPtr(PVM pVM, RTRCPTR pAddrGC)
5824{
5825 PPATMPATCHREC pPatchRec;
5826
5827 /* Find the patch record. */
5828 pPatchRec = (PPATMPATCHREC)RTAvloU32Get(&pVM->patm.s.PatchLookupTreeHC->PatchTree, pAddrGC);
5829 /** @todo we should only use patches that are enabled! always did this, but it's incorrect! */
5830 if (pPatchRec && (pPatchRec->patch.uState == PATCH_ENABLED || pPatchRec->patch.uState == PATCH_DIRTY))
5831 return PATCHCODE_PTR_GC(&pPatchRec->patch);
5832 else
5833 return 0;
5834}
5835
5836/**
5837 * Attempt to recover dirty instructions
5838 *
5839 * @returns VBox status code.
5840 * @param pVM The VM to operate on.
5841 * @param pCtx CPU context
5842 * @param pPatch Patch record
5843 * @param pPatchToGuestRec Patch to guest address record
5844 * @param pEip GC pointer of trapping instruction
5845 */
5846static int patmR3HandleDirtyInstr(PVM pVM, PCPUMCTX pCtx, PPATMPATCHREC pPatch, PRECPATCHTOGUEST pPatchToGuestRec, RTRCPTR pEip)
5847{
5848 DISCPUSTATE CpuOld, CpuNew;
5849 uint8_t *pPatchInstrHC, *pCurPatchInstrHC;
5850 int rc;
5851 RTRCPTR pCurInstrGC, pCurPatchInstrGC;
5852 uint32_t cbDirty;
5853 PRECPATCHTOGUEST pRec;
5854 RTRCPTR const pOrgInstrGC = pPatchToGuestRec->pOrgInstrGC;
5855 PVMCPU pVCpu = VMMGetCpu0(pVM);
5856 Log(("patmR3HandleDirtyInstr: dirty instruction at %RRv (%RRv)\n", pEip, pOrgInstrGC));
5857
5858 pRec = pPatchToGuestRec;
5859 pCurInstrGC = pOrgInstrGC;
5860 pCurPatchInstrGC = pEip;
5861 cbDirty = 0;
5862 pPatchInstrHC = patmPatchGCPtr2PatchHCPtr(pVM, pCurPatchInstrGC);
5863
5864 /* Find all adjacent dirty instructions */
5865 while (true)
5866 {
5867 if (pRec->fJumpTarget)
5868 {
5869 LogRel(("PATM: patmR3HandleDirtyInstr: dirty instruction at %RRv (%RRv) ignored, because instruction in function was reused as target of jump\n", pEip, pOrgInstrGC));
5870 pRec->fDirty = false;
5871 return VERR_PATCHING_REFUSED;
5872 }
5873
5874 /* Restore original instruction opcode byte so we can check if the write was indeed safe. */
5875 pCurPatchInstrHC = patmPatchGCPtr2PatchHCPtr(pVM, pCurPatchInstrGC);
5876 *pCurPatchInstrHC = pRec->u8DirtyOpcode;
5877
5878 /* Only harmless instructions are acceptable. */
5879 rc = CPUMR3DisasmInstrCPU(pVM, pVCpu, pCtx, pCurPatchInstrGC, &CpuOld, 0);
5880 if ( RT_FAILURE(rc)
5881 || !(CpuOld.pCurInstr->optype & OPTYPE_HARMLESS))
5882 {
5883 if (RT_SUCCESS(rc))
5884 cbDirty += CpuOld.opsize;
5885 else
5886 if (!cbDirty)
5887 cbDirty = 1;
5888 break;
5889 }
5890
5891#ifdef DEBUG
5892 char szBuf[256];
5893 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pCurPatchInstrGC, DBGF_DISAS_FLAGS_DEFAULT_MODE,
5894 szBuf, sizeof(szBuf), NULL);
5895 Log(("DIRTY: %s\n", szBuf));
5896#endif
5897 /* Mark as clean; if we fail we'll let it always fault. */
5898 pRec->fDirty = false;
5899
5900 /* Remove old lookup record. */
5901 patmr3RemoveP2GLookupRecord(pVM, &pPatch->patch, pCurPatchInstrGC);
5902 pPatchToGuestRec = NULL;
5903
5904 pCurPatchInstrGC += CpuOld.opsize;
5905 cbDirty += CpuOld.opsize;
5906
5907 /* Let's see if there's another dirty instruction right after. */
5908 pRec = (PRECPATCHTOGUEST)RTAvlU32GetBestFit(&pPatch->patch.Patch2GuestAddrTree, pCurPatchInstrGC - pVM->patm.s.pPatchMemGC, true);
5909 if (!pRec || !pRec->fDirty)
5910 break; /* no more dirty instructions */
5911
5912 /* In case of complex instructions the next guest instruction could be quite far off. */
5913 pCurPatchInstrGC = pRec->Core.Key + pVM->patm.s.pPatchMemGC;
5914 }
5915
5916 if ( RT_SUCCESS(rc)
5917 && (CpuOld.pCurInstr->optype & OPTYPE_HARMLESS)
5918 )
5919 {
5920 uint32_t cbLeft;
5921
5922 pCurPatchInstrHC = pPatchInstrHC;
5923 pCurPatchInstrGC = pEip;
5924 cbLeft = cbDirty;
5925
5926 while (cbLeft && RT_SUCCESS(rc))
5927 {
5928 bool fValidInstr;
5929
5930 rc = CPUMR3DisasmInstrCPU(pVM, pVCpu, pCtx, pCurInstrGC, &CpuNew, 0);
5931
5932 fValidInstr = !!(CpuNew.pCurInstr->optype & OPTYPE_HARMLESS);
5933 if ( !fValidInstr
5934 && (CpuNew.pCurInstr->optype & OPTYPE_RELATIVE_CONTROLFLOW)
5935 )
5936 {
5937 RTRCPTR pTargetGC = PATMResolveBranch(&CpuNew, pCurInstrGC);
5938
5939 if ( pTargetGC >= pOrgInstrGC
5940 && pTargetGC <= pOrgInstrGC + cbDirty
5941 )
5942 {
5943 /* A relative jump to an instruction inside or to the end of the dirty block is acceptable. */
5944 fValidInstr = true;
5945 }
5946 }
5947
5948 /* If the instruction is completely harmless (which implies a 1:1 patch copy). */
5949 if ( rc == VINF_SUCCESS
5950 && CpuNew.opsize <= cbLeft /* must still fit */
5951 && fValidInstr
5952 )
5953 {
5954#ifdef DEBUG
5955 char szBuf[256];
5956 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pCurInstrGC, DBGF_DISAS_FLAGS_DEFAULT_MODE,
5957 szBuf, sizeof(szBuf), NULL);
5958 Log(("NEW: %s\n", szBuf));
5959#endif
5960
5961 /* Copy the new instruction. */
5962 rc = PGMPhysSimpleReadGCPtr(VMMGetCpu0(pVM), pCurPatchInstrHC, pCurInstrGC, CpuNew.opsize);
5963 AssertRC(rc);
5964
5965 /* Add a new lookup record for the duplicated instruction. */
5966 patmr3AddP2GLookupRecord(pVM, &pPatch->patch, pCurPatchInstrHC, pCurInstrGC, PATM_LOOKUP_BOTHDIR);
5967 }
5968 else
5969 {
5970#ifdef DEBUG
5971 char szBuf[256];
5972 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pCurInstrGC, DBGF_DISAS_FLAGS_DEFAULT_MODE,
5973 szBuf, sizeof(szBuf), NULL);
5974 Log(("NEW: %s (FAILED)\n", szBuf));
5975#endif
5976 /* Restore the old lookup record for the duplicated instruction. */
5977 patmr3AddP2GLookupRecord(pVM, &pPatch->patch, pCurPatchInstrHC, pCurInstrGC, PATM_LOOKUP_BOTHDIR);
5978
5979 /** @todo in theory we need to restore the lookup records for the remaining dirty instructions too! */
5980 rc = VERR_PATCHING_REFUSED;
5981 break;
5982 }
5983 pCurInstrGC += CpuNew.opsize;
5984 pCurPatchInstrHC += CpuNew.opsize;
5985 pCurPatchInstrGC += CpuNew.opsize;
5986 cbLeft -= CpuNew.opsize;
5987
5988 /* Check if we expanded a complex guest instruction into a patch stream (e.g. call) */
5989 if (!cbLeft)
5990 {
5991 /* If the next patch instruction doesn't correspond to the next guest instruction, then we have some extra room to fill. */
5992 if (RTAvlU32Get(&pPatch->patch.Patch2GuestAddrTree, pCurPatchInstrGC - pVM->patm.s.pPatchMemGC) == NULL)
5993 {
5994 pRec = (PRECPATCHTOGUEST)RTAvlU32GetBestFit(&pPatch->patch.Patch2GuestAddrTree, pCurPatchInstrGC - pVM->patm.s.pPatchMemGC, true);
5995 if (pRec)
5996 {
5997 unsigned cbFiller = pRec->Core.Key + pVM->patm.s.pPatchMemGC - pCurPatchInstrGC;
5998 uint8_t *pPatchFillHC = patmPatchGCPtr2PatchHCPtr(pVM, pCurPatchInstrGC);
5999
6000 Assert(!pRec->fDirty);
6001
6002 Log(("Room left in patched instruction stream (%d bytes)\n", cbFiller));
6003 if (cbFiller >= SIZEOF_NEARJUMP32)
6004 {
6005 pPatchFillHC[0] = 0xE9;
6006 *(uint32_t *)&pPatchFillHC[1] = cbFiller - SIZEOF_NEARJUMP32;
6007#ifdef DEBUG
6008 char szBuf[256];
6009 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pCurPatchInstrGC, DBGF_DISAS_FLAGS_DEFAULT_MODE,
6010 szBuf, sizeof(szBuf), NULL);
6011 Log(("FILL: %s\n", szBuf));
6012#endif
6013 }
6014 else
6015 {
6016 for (unsigned i = 0; i < cbFiller; i++)
6017 {
6018 pPatchFillHC[i] = 0x90; /* NOP */
6019#ifdef DEBUG
6020 char szBuf[256];
6021 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pCurPatchInstrGC + i,
6022 DBGF_DISAS_FLAGS_DEFAULT_MODE, szBuf, sizeof(szBuf), NULL);
6023 Log(("FILL: %s\n", szBuf));
6024#endif
6025 }
6026 }
6027 }
6028 }
6029 }
6030 }
6031 }
6032 else
6033 rc = VERR_PATCHING_REFUSED;
6034
6035 if (RT_SUCCESS(rc))
6036 {
6037 STAM_COUNTER_INC(&pVM->patm.s.StatInstrDirtyGood);
6038 }
6039 else
6040 {
6041 STAM_COUNTER_INC(&pVM->patm.s.StatInstrDirtyBad);
6042 Assert(cbDirty);
6043
6044 /* Mark the whole instruction stream with breakpoints. */
6045 if (cbDirty)
6046 memset(pPatchInstrHC, 0xCC, cbDirty);
6047
6048 if ( pVM->patm.s.fOutOfMemory == false
6049 && (pPatch->patch.flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_IDTHANDLER|PATMFL_TRAPHANDLER)))
6050 {
6051 rc = patmR3RefreshPatch(pVM, pPatch);
6052 if (RT_FAILURE(rc))
6053 {
6054 LogRel(("PATM: Failed to refresh dirty patch at %RRv. Disabling it.\n", pPatch->patch.pPrivInstrGC));
6055 }
6056 /* Even if we succeed, we must go back to the original instruction as the patched one could be invalid. */
6057 rc = VERR_PATCHING_REFUSED;
6058 }
6059 }
6060 return rc;
6061}
6062
6063/**
6064 * Handle trap inside patch code
6065 *
6066 * @returns VBox status code.
6067 * @param pVM The VM to operate on.
6068 * @param pCtx CPU context
6069 * @param pEip GC pointer of trapping instruction
6070 * @param ppNewEip GC pointer to new instruction
6071 */
6072VMMR3DECL(int) PATMR3HandleTrap(PVM pVM, PCPUMCTX pCtx, RTRCPTR pEip, RTGCPTR *ppNewEip)
6073{
6074 PPATMPATCHREC pPatch = 0;
6075 void *pvPatchCoreOffset;
6076 RTRCUINTPTR offset;
6077 RTRCPTR pNewEip;
6078 int rc ;
6079 PRECPATCHTOGUEST pPatchToGuestRec = 0;
6080 PVMCPU pVCpu = VMMGetCpu0(pVM);
6081
6082 Assert(pVM->cCpus == 1);
6083
6084 pNewEip = 0;
6085 *ppNewEip = 0;
6086
6087 STAM_PROFILE_ADV_START(&pVM->patm.s.StatHandleTrap, a);
6088
6089 /* Find the patch record. */
6090 /* Note: there might not be a patch to guest translation record (global function) */
6091 offset = pEip - pVM->patm.s.pPatchMemGC;
6092 pvPatchCoreOffset = RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTreeByPatchAddr, offset, false);
6093 if (pvPatchCoreOffset)
6094 {
6095 pPatch = PATM_PATCHREC_FROM_COREOFFSET(pvPatchCoreOffset);
6096
6097 Assert(offset >= pPatch->patch.pPatchBlockOffset && offset < pPatch->patch.pPatchBlockOffset + pPatch->patch.cbPatchBlockSize);
6098
6099 if (pPatch->patch.uState == PATCH_DIRTY)
6100 {
6101 Log(("PATMR3HandleTrap: trap in dirty patch at %RRv\n", pEip));
6102 if (pPatch->patch.flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_CODE_REFERENCED))
6103 {
6104 /* Function duplication patches set fPIF to 1 on entry */
6105 pVM->patm.s.pGCStateHC->fPIF = 1;
6106 }
6107 }
6108 else
6109 if (pPatch->patch.uState == PATCH_DISABLED)
6110 {
6111 Log(("PATMR3HandleTrap: trap in disabled patch at %RRv\n", pEip));
6112 if (pPatch->patch.flags & (PATMFL_DUPLICATE_FUNCTION|PATMFL_CODE_REFERENCED))
6113 {
6114 /* Function duplication patches set fPIF to 1 on entry */
6115 pVM->patm.s.pGCStateHC->fPIF = 1;
6116 }
6117 }
6118 else
6119 if (pPatch->patch.uState == PATCH_DISABLE_PENDING)
6120 {
6121 RTRCPTR pPrivInstrGC = pPatch->patch.pPrivInstrGC;
6122
6123 Log(("PATMR3HandleTrap: disable operation is pending for patch at %RRv\n", pPatch->patch.pPrivInstrGC));
6124 rc = PATMR3DisablePatch(pVM, pPatch->patch.pPrivInstrGC);
6125 AssertReleaseMsg(rc != VWRN_PATCH_REMOVED, ("PATMR3DisablePatch removed patch at %RRv\n", pPrivInstrGC));
6126 AssertMsg(pPatch->patch.uState == PATCH_DISABLED || pPatch->patch.uState == PATCH_UNUSABLE, ("Unexpected failure to disable patch state=%d rc=%Rrc\n", pPatch->patch.uState, rc));
6127 }
6128
6129 pPatchToGuestRec = (PRECPATCHTOGUEST)RTAvlU32GetBestFit(&pPatch->patch.Patch2GuestAddrTree, offset, false);
6130 AssertReleaseMsg(pPatchToGuestRec, ("PATMR3HandleTrap: Unable to find corresponding guest address for %RRv (offset %x)\n", pEip, offset));
6131
6132 pNewEip = pPatchToGuestRec->pOrgInstrGC;
6133 pPatch->patch.cTraps++;
6134 PATM_STAT_FAULT_INC(&pPatch->patch);
6135 }
6136 else
6137 AssertReleaseMsg(pVM->patm.s.pGCStateHC->fPIF == 0, ("PATMR3HandleTrap: Unable to find translation record for %RRv (PIF=0)\n", pEip));
6138
6139 /* Check if we were interrupted in PATM generated instruction code. */
6140 if (pVM->patm.s.pGCStateHC->fPIF == 0)
6141 {
6142 DISCPUSTATE Cpu;
6143 rc = CPUMR3DisasmInstrCPU(pVM, pVCpu, pCtx, pEip, &Cpu, "PIF Trap: ");
6144 AssertRC(rc);
6145
6146 if ( rc == VINF_SUCCESS
6147 && ( Cpu.pCurInstr->opcode == OP_PUSHF
6148 || Cpu.pCurInstr->opcode == OP_PUSH
6149 || Cpu.pCurInstr->opcode == OP_CALL)
6150 )
6151 {
6152 uint64_t fFlags;
6153
6154 STAM_COUNTER_INC(&pVM->patm.s.StatPushTrap);
6155
6156 if (Cpu.pCurInstr->opcode == OP_PUSH)
6157 {
6158 rc = PGMShwGetPage(pVCpu, pCtx->esp, &fFlags, NULL);
6159 if ( rc == VINF_SUCCESS
6160 && ((fFlags & (X86_PTE_P|X86_PTE_RW)) == (X86_PTE_P|X86_PTE_RW)) )
6161 {
6162 /* The stack address is fine, so the push argument is a pointer -> emulate this instruction */
6163
6164 /* Reset the PATM stack. */
6165 CTXSUFF(pVM->patm.s.pGCState)->Psp = PATM_STACK_SIZE;
6166
6167 pVM->patm.s.pGCStateHC->fPIF = 1;
6168
6169 Log(("Faulting push -> go back to the original instruction\n"));
6170
6171 /* continue at the original instruction */
6172 *ppNewEip = pNewEip - SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), 0);
6173 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6174 return VINF_SUCCESS;
6175 }
6176 }
6177
6178 /* Typical pushf (most patches)/push (call patch) trap because of a monitored page. */
6179 rc = PGMShwMakePageWritable(pVCpu, pCtx->esp, 0 /*fFlags*/);
6180 AssertMsgRC(rc, ("PGMShwModifyPage -> rc=%Rrc\n", rc));
6181 if (rc == VINF_SUCCESS)
6182 {
6183 /* The guest page *must* be present. */
6184 rc = PGMGstGetPage(pVCpu, pCtx->esp, &fFlags, NULL);
6185 if ( rc == VINF_SUCCESS
6186 && (fFlags & X86_PTE_P))
6187 {
6188 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6189 return VINF_PATCH_CONTINUE;
6190 }
6191 }
6192 }
6193 else
6194 if (pPatch->patch.pPrivInstrGC == pNewEip)
6195 {
6196 /* Invalidated patch or first instruction overwritten.
6197 * We can ignore the fPIF state in this case.
6198 */
6199 /* Reset the PATM stack. */
6200 CTXSUFF(pVM->patm.s.pGCState)->Psp = PATM_STACK_SIZE;
6201
6202 Log(("Call to invalidated patch -> go back to the original instruction\n"));
6203
6204 pVM->patm.s.pGCStateHC->fPIF = 1;
6205
6206 /* continue at the original instruction */
6207 *ppNewEip = pNewEip - SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), 0);
6208 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6209 return VINF_SUCCESS;
6210 }
6211
6212 char szBuf[256];
6213 DBGFR3DisasInstrEx(pVM, pVCpu->idCpu, pCtx->cs, pEip, DBGF_DISAS_FLAGS_DEFAULT_MODE, szBuf, sizeof(szBuf), NULL);
6214
6215 /* Very bad. We crashed in emitted code. Probably stack? */
6216 if (pPatch)
6217 {
6218 AssertReleaseMsg(pVM->patm.s.pGCStateHC->fPIF == 1,
6219 ("Crash in patch code %RRv (%RRv) esp=%RX32\nPatch state=%x flags=%RX64 fDirty=%d\n%s\n", pEip, pNewEip, CPUMGetGuestESP(pVCpu), pPatch->patch.uState, pPatch->patch.flags, pPatchToGuestRec->fDirty, szBuf));
6220 }
6221 else
6222 AssertReleaseMsg(pVM->patm.s.pGCStateHC->fPIF == 1,
6223 ("Crash in patch code %RRv (%RRv) esp=%RX32\n%s\n", pEip, pNewEip, CPUMGetGuestESP(pVCpu), szBuf));
6224 EMR3FatalError(pVCpu, VERR_INTERNAL_ERROR);
6225 }
6226
6227 /* From here on, we must have a valid patch to guest translation. */
6228 if (pvPatchCoreOffset == 0)
6229 {
6230 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6231 AssertMsgFailed(("PATMR3HandleTrap: patch not found at address %RRv!!\n", pEip));
6232 return VERR_PATCH_NOT_FOUND;
6233 }
6234
6235 /* Take care of dirty/changed instructions. */
6236 if (pPatchToGuestRec->fDirty)
6237 {
6238 Assert(pPatchToGuestRec->Core.Key == offset);
6239 Assert(pVM->patm.s.pGCStateHC->fPIF == 1);
6240
6241 rc = patmR3HandleDirtyInstr(pVM, pCtx, pPatch, pPatchToGuestRec, pEip);
6242 if (RT_SUCCESS(rc))
6243 {
6244 /* Retry the current instruction. */
6245 pNewEip = pEip;
6246 rc = VINF_PATCH_CONTINUE; /* Continue at current patch instruction. */
6247 }
6248 else
6249 {
6250 /* Reset the PATM stack. */
6251 CTXSUFF(pVM->patm.s.pGCState)->Psp = PATM_STACK_SIZE;
6252
6253 rc = VINF_SUCCESS; /* Continue at original instruction. */
6254 }
6255
6256 *ppNewEip = pNewEip - SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), 0);
6257 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6258 return rc;
6259 }
6260
6261#ifdef VBOX_STRICT
6262 if (pPatch->patch.flags & PATMFL_DUPLICATE_FUNCTION)
6263 {
6264 DISCPUSTATE cpu;
6265 bool disret;
6266 uint32_t opsize;
6267 PATMP2GLOOKUPREC cacheRec;
6268 RT_ZERO(cacheRec);
6269 cacheRec.pPatch = &pPatch->patch;
6270
6271 cpu.mode = (pPatch->patch.flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
6272 disret = PATMR3DISInstr(pVM, &pPatch->patch, &cpu, pNewEip, PATMGCVirtToHCVirt(pVM, &cacheRec, pNewEip), &opsize, NULL, PATMREAD_RAWCODE);
6273 if (cacheRec.Lock.pvMap)
6274 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
6275
6276 if (disret && cpu.pCurInstr->opcode == OP_RETN)
6277 {
6278 RTRCPTR retaddr;
6279 PCPUMCTX pCtx2;
6280
6281 pCtx2 = CPUMQueryGuestCtxPtr(pVCpu);
6282
6283 rc = PGMPhysSimpleReadGCPtr(pVCpu, &retaddr, pCtx2->esp, sizeof(retaddr));
6284 AssertRC(rc);
6285
6286 Log(("Return failed at %RRv (%RRv)\n", pEip, pNewEip));
6287 Log(("Expected return address %RRv found address %RRv Psp=%x\n", pVM->patm.s.pGCStackHC[(pVM->patm.s.pGCStateHC->Psp+PATM_STACK_SIZE)/sizeof(RTRCPTR)], retaddr, pVM->patm.s.pGCStateHC->Psp));
6288 }
6289 }
6290#endif
6291
6292 /* Return original address, correct by subtracting the CS base address. */
6293 *ppNewEip = pNewEip - SELMToFlat(pVM, DIS_SELREG_CS, CPUMCTX2CORE(pCtx), 0);
6294
6295 /* Reset the PATM stack. */
6296 CTXSUFF(pVM->patm.s.pGCState)->Psp = PATM_STACK_SIZE;
6297
6298 if (pVM->patm.s.pGCStateHC->GCPtrInhibitInterrupts == pNewEip)
6299 {
6300 /* Must be a faulting instruction after sti; currently only sysexit, hlt or iret */
6301 Log(("PATMR3HandleTrap %RRv -> inhibit irqs set!\n", pEip));
6302#ifdef VBOX_STRICT
6303 DISCPUSTATE cpu;
6304 bool disret;
6305 uint32_t opsize;
6306 PATMP2GLOOKUPREC cacheRec;
6307 RT_ZERO(cacheRec);
6308 cacheRec.pPatch = &pPatch->patch;
6309
6310 cpu.mode = (pPatch->patch.flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
6311 disret = PATMR3DISInstr(pVM, &pPatch->patch, &cpu, pNewEip, PATMGCVirtToHCVirt(pVM, &cacheRec, pNewEip), &opsize, NULL, PATMREAD_ORGCODE);
6312 if (cacheRec.Lock.pvMap)
6313 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
6314
6315 if (disret && (cpu.pCurInstr->opcode == OP_SYSEXIT || cpu.pCurInstr->opcode == OP_HLT || cpu.pCurInstr->opcode == OP_INT3))
6316 {
6317 cpu.mode = (pPatch->patch.flags & PATMFL_CODE32) ? CPUMODE_32BIT : CPUMODE_16BIT;
6318 disret = PATMR3DISInstr(pVM, &pPatch->patch, &cpu, pNewEip, PATMGCVirtToHCVirt(pVM, &cacheRec, pNewEip), &opsize, NULL, PATMREAD_RAWCODE);
6319 if (cacheRec.Lock.pvMap)
6320 PGMPhysReleasePageMappingLock(pVM, &cacheRec.Lock);
6321
6322 Assert(cpu.pCurInstr->opcode == OP_SYSEXIT || cpu.pCurInstr->opcode == OP_HLT || cpu.pCurInstr->opcode == OP_IRET);
6323 }
6324#endif
6325 EMSetInhibitInterruptsPC(pVCpu, pNewEip);
6326 pVM->patm.s.pGCStateHC->GCPtrInhibitInterrupts = 0;
6327 }
6328
6329 Log2(("pPatchBlockGC %RRv - pEip %RRv corresponding GC address %RRv\n", PATCHCODE_PTR_GC(&pPatch->patch), pEip, pNewEip));
6330#ifdef LOG_ENABLED
6331 CPUMR3DisasmInstr(pVM, pVCpu, pCtx, pNewEip, "PATCHRET: ");
6332#endif
6333 if (pNewEip >= pPatch->patch.pPrivInstrGC && pNewEip < pPatch->patch.pPrivInstrGC + pPatch->patch.cbPatchJump)
6334 {
6335 /* We can't jump back to code that we've overwritten with a 5 byte jump! */
6336 Log(("Disabling patch at location %RRv due to trap too close to the privileged instruction \n", pPatch->patch.pPrivInstrGC));
6337 PATMR3DisablePatch(pVM, pPatch->patch.pPrivInstrGC);
6338 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6339 return VERR_PATCH_DISABLED;
6340 }
6341
6342#ifdef PATM_REMOVE_PATCH_ON_TOO_MANY_TRAPS
6343 /** @todo compare to nr of successful runs. add some aging algorithm and determine the best time to disable the patch */
6344 if (pPatch->patch.cTraps > MAX_PATCH_TRAPS)
6345 {
6346 Log(("Disabling patch at location %RRv due to too many traps inside patch code\n", pPatch->patch.pPrivInstrGC));
6347 //we are only wasting time, back out the patch
6348 PATMR3DisablePatch(pVM, pPatch->patch.pPrivInstrGC);
6349 pTrapRec->pNextPatchInstr = 0;
6350 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6351 return VERR_PATCH_DISABLED;
6352 }
6353#endif
6354
6355 STAM_PROFILE_ADV_STOP(&pVM->patm.s.StatHandleTrap, a);
6356 return VINF_SUCCESS;
6357}
6358
6359
6360/**
6361 * Handle page-fault in monitored page
6362 *
6363 * @returns VBox status code.
6364 * @param pVM The VM to operate on.
6365 */
6366VMMR3DECL(int) PATMR3HandleMonitoredPage(PVM pVM)
6367{
6368 RTRCPTR addr = pVM->patm.s.pvFaultMonitor;
6369
6370 addr &= PAGE_BASE_GC_MASK;
6371
6372 int rc = PGMHandlerVirtualDeregister(pVM, addr);
6373 AssertRC(rc); NOREF(rc);
6374
6375 PPATMPATCHREC pPatchRec = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, addr, false);
6376 if (pPatchRec && pPatchRec->patch.uState == PATCH_ENABLED && PAGE_ADDRESS(pPatchRec->patch.pPrivInstrGC) == PAGE_ADDRESS(addr))
6377 {
6378 STAM_COUNTER_INC(&pVM->patm.s.StatMonitored);
6379 Log(("Renewing patch at %RRv\n", pPatchRec->patch.pPrivInstrGC));
6380 rc = PATMR3DisablePatch(pVM, pPatchRec->patch.pPrivInstrGC);
6381 if (rc == VWRN_PATCH_REMOVED)
6382 return VINF_SUCCESS;
6383
6384 PATMR3EnablePatch(pVM, pPatchRec->patch.pPrivInstrGC);
6385
6386 if (addr == pPatchRec->patch.pPrivInstrGC)
6387 addr++;
6388 }
6389
6390 for(;;)
6391 {
6392 pPatchRec = (PPATMPATCHREC)RTAvloU32GetBestFit(&pVM->patm.s.PatchLookupTreeHC->PatchTree, addr, true);
6393
6394 if (!pPatchRec || PAGE_ADDRESS(pPatchRec->patch.pPrivInstrGC) != PAGE_ADDRESS(addr))
6395 break;
6396
6397 if (pPatchRec && pPatchRec->patch.uState == PATCH_ENABLED)
6398 {
6399 STAM_COUNTER_INC(&pVM->patm.s.StatMonitored);
6400 Log(("Renewing patch at %RRv\n", pPatchRec->patch.pPrivInstrGC));
6401 PATMR3DisablePatch(pVM, pPatchRec->patch.pPrivInstrGC);
6402 PATMR3EnablePatch(pVM, pPatchRec->patch.pPrivInstrGC);
6403 }
6404 addr = pPatchRec->patch.pPrivInstrGC + 1;
6405 }
6406
6407 pVM->patm.s.pvFaultMonitor = 0;
6408 return VINF_SUCCESS;
6409}
6410
6411
6412#ifdef VBOX_WITH_STATISTICS
6413
6414static const char *PATMPatchType(PVM pVM, PPATCHINFO pPatch)
6415{
6416 if (pPatch->flags & PATMFL_SYSENTER)
6417 {
6418 return "SYSENT";
6419 }
6420 else
6421 if (pPatch->flags & (PATMFL_TRAPHANDLER|PATMFL_INTHANDLER))
6422 {
6423 static char szTrap[16];
6424 uint32_t iGate;
6425
6426 iGate = TRPMR3QueryGateByHandler(pVM, PATCHCODE_PTR_GC(pPatch));
6427 if (iGate < 256)
6428 RTStrPrintf(szTrap, sizeof(szTrap), (pPatch->flags & PATMFL_INTHANDLER) ? "INT-%2X" : "TRAP-%2X", iGate);
6429 else
6430 RTStrPrintf(szTrap, sizeof(szTrap), (pPatch->flags & PATMFL_INTHANDLER) ? "INT-??" : "TRAP-??");
6431 return szTrap;
6432 }
6433 else
6434 if (pPatch->flags & (PATMFL_DUPLICATE_FUNCTION))
6435 return "DUPFUNC";
6436 else
6437 if (pPatch->flags & PATMFL_REPLACE_FUNCTION_CALL)
6438 return "FUNCCALL";
6439 else
6440 if (pPatch->flags & PATMFL_TRAMPOLINE)
6441 return "TRAMP";
6442 else
6443 return patmGetInstructionString(pPatch->opcode, pPatch->flags);
6444}
6445
6446static const char *PATMPatchState(PVM pVM, PPATCHINFO pPatch)
6447{
6448 switch(pPatch->uState)
6449 {
6450 case PATCH_ENABLED:
6451 return "ENA";
6452 case PATCH_DISABLED:
6453 return "DIS";
6454 case PATCH_DIRTY:
6455 return "DIR";
6456 case PATCH_UNUSABLE:
6457 return "UNU";
6458 case PATCH_REFUSED:
6459 return "REF";
6460 case PATCH_DISABLE_PENDING:
6461 return "DIP";
6462 default:
6463 AssertFailed();
6464 return " ";
6465 }
6466}
6467
6468/**
6469 * Resets the sample.
6470 * @param pVM The VM handle.
6471 * @param pvSample The sample registered using STAMR3RegisterCallback.
6472 */
6473static void patmResetStat(PVM pVM, void *pvSample)
6474{
6475 PPATCHINFO pPatch = (PPATCHINFO)pvSample;
6476 Assert(pPatch);
6477
6478 pVM->patm.s.pStatsHC[pPatch->uPatchIdx].u32A = 0;
6479 pVM->patm.s.pStatsHC[pPatch->uPatchIdx].u32B = 0;
6480}
6481
6482/**
6483 * Prints the sample into the buffer.
6484 *
6485 * @param pVM The VM handle.
6486 * @param pvSample The sample registered using STAMR3RegisterCallback.
6487 * @param pszBuf The buffer to print into.
6488 * @param cchBuf The size of the buffer.
6489 */
6490static void patmPrintStat(PVM pVM, void *pvSample, char *pszBuf, size_t cchBuf)
6491{
6492 PPATCHINFO pPatch = (PPATCHINFO)pvSample;
6493 Assert(pPatch);
6494
6495 Assert(pPatch->uState != PATCH_REFUSED);
6496 Assert(!(pPatch->flags & (PATMFL_REPLACE_FUNCTION_CALL|PATMFL_MMIO_ACCESS)));
6497
6498 RTStrPrintf(pszBuf, cchBuf, "size %04x ->%3s %8s - %08d - %08d",
6499 pPatch->cbPatchBlockSize, PATMPatchState(pVM, pPatch), PATMPatchType(pVM, pPatch),
6500 pVM->patm.s.pStatsHC[pPatch->uPatchIdx].u32A, pVM->patm.s.pStatsHC[pPatch->uPatchIdx].u32B);
6501}
6502
6503/**
6504 * Returns the GC address of the corresponding patch statistics counter
6505 *
6506 * @returns Stat address
6507 * @param pVM The VM to operate on.
6508 * @param pPatch Patch structure
6509 */
6510RTRCPTR patmPatchQueryStatAddress(PVM pVM, PPATCHINFO pPatch)
6511{
6512 Assert(pPatch->uPatchIdx != PATM_STAT_INDEX_NONE);
6513 return pVM->patm.s.pStatsGC + sizeof(STAMRATIOU32) * pPatch->uPatchIdx + RT_OFFSETOF(STAMRATIOU32, u32A);
6514}
6515
6516#endif /* VBOX_WITH_STATISTICS */
6517
6518#ifdef VBOX_WITH_DEBUGGER
6519/**
6520 * The '.patmoff' command.
6521 *
6522 * @returns VBox status.
6523 * @param pCmd Pointer to the command descriptor (as registered).
6524 * @param pCmdHlp Pointer to command helper functions.
6525 * @param pVM Pointer to the current VM (if any).
6526 * @param paArgs Pointer to (readonly) array of arguments.
6527 * @param cArgs Number of arguments in the array.
6528 */
6529static DECLCALLBACK(int) patmr3CmdOff(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
6530{
6531 /*
6532 * Validate input.
6533 */
6534 if (!pVM)
6535 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: The command requires VM to be selected.\n");
6536
6537 RTAvloU32DoWithAll(&pVM->patm.s.PatchLookupTreeHC->PatchTree, true, DisableAllPatches, pVM);
6538 PATMR3AllowPatching(pVM, false);
6539 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Patching disabled\n");
6540}
6541
6542/**
6543 * The '.patmon' command.
6544 *
6545 * @returns VBox status.
6546 * @param pCmd Pointer to the command descriptor (as registered).
6547 * @param pCmdHlp Pointer to command helper functions.
6548 * @param pVM Pointer to the current VM (if any).
6549 * @param paArgs Pointer to (readonly) array of arguments.
6550 * @param cArgs Number of arguments in the array.
6551 */
6552static DECLCALLBACK(int) patmr3CmdOn(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
6553{
6554 /*
6555 * Validate input.
6556 */
6557 if (!pVM)
6558 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: The command requires VM to be selected.\n");
6559
6560 PATMR3AllowPatching(pVM, true);
6561 RTAvloU32DoWithAll(&pVM->patm.s.PatchLookupTreeHC->PatchTree, true, EnableAllPatches, pVM);
6562 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Patching enabled\n");
6563}
6564#endif
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette