VirtualBox

source: vbox/trunk/src/VBox/VMM/PATM/PATMInternal.h@ 12069

Last change on this file since 12069 was 11967, checked in by vboxsync, 16 years ago

Correct CPUMCTX offset in 1.6 saved states.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 28.7 KB
Line 
1/* $Id: PATMInternal.h 11967 2008-09-02 11:06:07Z vboxsync $ */
2/** @file
3 * PATM - Internal header file.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#ifndef ___PATMInternal_h
23#define ___PATMInternal_h
24
25#include <VBox/cdefs.h>
26#include <VBox/types.h>
27#include <VBox/patm.h>
28#include <VBox/stam.h>
29#include <VBox/dis.h>
30#include <VBox/pgm.h>
31#include <iprt/avl.h>
32#include <iprt/param.h>
33#include <VBox/log.h>
34
35#if !defined(IN_PATM_R3) && !defined(IN_PATM_R0) && !defined(IN_PATM_GC)
36# error "Not in PATM! This is an internal header!"
37#endif
38
39
40#ifdef PATM_WITH_NEW_SSM
41# define PATM_SSM_VERSION 54
42/* Last version to use SSMR3Put/GetMem */
43# define PATM_SSM_VERSION_GETPUTMEM 53
44#else
45# define PATM_SSM_VERSION 54
46# define PATM_SSM_VERSION_VER16 53
47#endif
48
49/* Enable for call patching. */
50#define PATM_ENABLE_CALL
51#define PATCH_MEMORY_SIZE (2*1024*1024)
52#define MAX_PATCH_SIZE (1024*4)
53
54/*
55 * Internal patch type flags (starts at RT_BIT(11))
56 */
57
58#define PATMFL_CHECK_SIZE RT_BIT_64(11)
59#define PATMFL_FOUND_PATCHEND RT_BIT_64(12)
60#define PATMFL_SINGLE_INSTRUCTION RT_BIT_64(13)
61#define PATMFL_SYSENTER_XP RT_BIT_64(14)
62#define PATMFL_JUMP_CONFLICT RT_BIT_64(15)
63#define PATMFL_READ_ORIGINAL_BYTES RT_BIT_64(16) /** opcode might have already been patched */
64#define PATMFL_INT3_REPLACEMENT RT_BIT_64(17)
65#define PATMFL_SUPPORT_CALLS RT_BIT_64(18)
66#define PATMFL_SUPPORT_INDIRECT_CALLS RT_BIT_64(19)
67#define PATMFL_IDTHANDLER_WITHOUT_ENTRYPOINT RT_BIT_64(20) /** internal flag to avoid duplicate entrypoints */
68#define PATMFL_INHIBIT_IRQS RT_BIT_64(21) /** temporary internal flag */
69#define PATMFL_GENERATE_JUMPTOGUEST RT_BIT_64(22) /** temporary internal flag */
70#define PATMFL_RECOMPILE_NEXT RT_BIT_64(23) /** for recompilation of the next instruction */
71#define PATMFL_CODE_MONITORED RT_BIT_64(24) /** code pages of guest monitored for self-modifying code. */
72#define PATMFL_CALLABLE_AS_FUNCTION RT_BIT_64(25) /** cli and pushf blocks can be used as callable functions. */
73#define PATMFL_GLOBAL_FUNCTIONS RT_BIT_64(26) /** fake patch for global patm functions. */
74#define PATMFL_TRAMPOLINE RT_BIT_64(27) /** trampoline patch that clears PATM_INTERRUPTFLAG and jumps to patch destination */
75#define PATMFL_GENERATE_SETPIF RT_BIT_64(28) /** generate set PIF for the next instruction */
76#define PATMFL_INSTR_HINT RT_BIT_64(29) /** Generate patch, but don't activate it. */
77#define PATMFL_PATCHED_GUEST_CODE RT_BIT_64(30) /** Patched guest code. */
78#define PATMFL_MUST_INSTALL_PATCHJMP RT_BIT_64(31) /** Need to patch guest code in order to activate patch. */
79#define PATMFL_INT3_REPLACEMENT_BLOCK RT_BIT_64(32) /** int 3 replacement block */
80#define PATMFL_EXTERNAL_JUMP_INSIDE RT_BIT_64(33) /** A trampoline patch was created that jumps to an instruction in the patch block */
81
82#define SIZEOF_NEARJUMP8 2 //opcode byte + 1 byte relative offset
83#define SIZEOF_NEARJUMP16 3 //opcode byte + 2 byte relative offset
84#define SIZEOF_NEARJUMP32 5 //opcode byte + 4 byte relative offset
85#define SIZEOF_NEAR_COND_JUMP32 6 //0xF + opcode byte + 4 byte relative offset
86
87#define MAX_INSTR_SIZE 16
88
89//Patch states
90#define PATCH_REFUSED 1
91#define PATCH_DISABLED 2
92#define PATCH_ENABLED 4
93#define PATCH_UNUSABLE 8
94#define PATCH_DIRTY 16
95#define PATCH_DISABLE_PENDING 32
96
97
98#define MAX_PATCH_TRAPS 4
99#define PATM_MAX_CALL_DEPTH 32
100/* Maximum nr of writes before a patch is marked dirty. (disabled) */
101#define PATM_MAX_CODE_WRITES 32
102/* Maximum nr of invalid writes before a patch is disabled. */
103#define PATM_MAX_INVALID_WRITES 16384
104
105#define FIXUP_ABSOLUTE 0
106#define FIXUP_REL_JMPTOPATCH 1
107#define FIXUP_REL_JMPTOGUEST 2
108
109#define PATM_ILLEGAL_DESTINATION 0xDEADBEEF
110
111/** Size of the instruction that's used for requests from patch code (currently only call) */
112#define PATM_ILLEGAL_INSTR_SIZE 2
113
114
115/** No statistics counter index allocated just yet */
116#define PATM_STAT_INDEX_NONE (uint32_t)-1
117/** Dummy counter to handle overflows */
118#define PATM_STAT_INDEX_DUMMY 0
119#define PATM_STAT_INDEX_IS_VALID(a) (a != PATM_STAT_INDEX_DUMMY && a != PATM_STAT_INDEX_NONE)
120
121#ifdef VBOX_WITH_STATISTICS
122#define PATM_STAT_RUN_INC(pPatch) \
123 if (PATM_STAT_INDEX_IS_VALID((pPatch)->uPatchIdx)) \
124 CTXSUFF(pVM->patm.s.pStats)[(pPatch)->uPatchIdx].u32A++;
125#define PATM_STAT_FAULT_INC(pPatch) \
126 if (PATM_STAT_INDEX_IS_VALID((pPatch)->uPatchIdx)) \
127 CTXSUFF(pVM->patm.s.pStats)[(pPatch)->uPatchIdx].u32B++;
128#else
129#define PATM_STAT_RUN_INC(pPatch) do { } while (0)
130#define PATM_STAT_FAULT_INC(pPatch) do { } while (0)
131#endif
132
133/** Maximum number of stat counters. */
134#define PATM_STAT_MAX_COUNTERS 1024
135/** Size of memory allocated for patch statistics. */
136#define PATM_STAT_MEMSIZE (PATM_STAT_MAX_COUNTERS*sizeof(STAMRATIOU32))
137
138
139typedef struct
140{
141 /** The key is a HC virtual address. */
142 AVLPVNODECORE Core;
143
144 uint32_t uType;
145 R3PTRTYPE(uint8_t *) pRelocPos;
146 RTRCPTR pSource;
147 RTRCPTR pDest;
148} RELOCREC, *PRELOCREC;
149
150typedef struct
151{
152 R3PTRTYPE(uint8_t *) pPatchLocStartHC;
153 R3PTRTYPE(uint8_t *) pPatchLocEndHC;
154 RCPTRTYPE(uint8_t *) pGuestLoc;
155 uint32_t opsize;
156} P2GLOOKUPREC, *PP2GLOOKUPREC;
157
158typedef struct
159{
160 /** The key is a pointer to a JUMPREC structure. */
161 AVLPVNODECORE Core;
162
163 R3PTRTYPE(uint8_t *) pJumpHC;
164 RCPTRTYPE(uint8_t *) pTargetGC;
165 uint32_t offDispl;
166 uint32_t opcode;
167} JUMPREC, *PJUMPREC;
168
169/**
170 * Patch to guest lookup type (single or both direction)
171 */
172typedef enum
173{
174 PATM_LOOKUP_PATCH2GUEST, /* patch to guest */
175 PATM_LOOKUP_BOTHDIR /* guest to patch + patch to guest */
176} PATM_LOOKUP_TYPE;
177
178/**
179 * Patch to guest address lookup record
180 */
181typedef struct RECPATCHTOGUEST
182{
183 /** The key is an offset inside the patch memory block. */
184 AVLU32NODECORE Core;
185
186 RTRCPTR pOrgInstrGC;
187 PATM_LOOKUP_TYPE enmType;
188 bool fDirty;
189 bool fJumpTarget;
190 uint8_t u8DirtyOpcode; /* original opcode before writing 0xCC there to mark it dirty */
191} RECPATCHTOGUEST, *PRECPATCHTOGUEST;
192
193/**
194 * Guest to patch address lookup record
195 */
196typedef struct RECGUESTTOPATCH
197{
198 /** The key is a GC virtual address. */
199 AVLU32NODECORE Core;
200
201 /** Patch offset (relative to PATM::pPatchMemGC / PATM::pPatchMemHC). */
202 uint32_t PatchOffset;
203} RECGUESTTOPATCH, *PRECGUESTTOPATCH;
204
205/**
206 * Temporary information used in ring 3 only; no need to waste memory in the patch record itself.
207 */
208typedef struct
209{
210 /* Temporary tree for storing the addresses of illegal instructions. */
211 R3PTRTYPE(PAVLPVNODECORE) IllegalInstrTree;
212 uint32_t nrIllegalInstr;
213
214 int32_t nrJumps;
215 uint32_t nrRetInstr;
216
217 /* Temporary tree of encountered jumps. (debug only) */
218 R3PTRTYPE(PAVLPVNODECORE) DisasmJumpTree;
219
220 int32_t nrCalls;
221
222 /** Last original guest instruction pointer; used for disassmebly log. */
223 RTRCPTR pLastDisasmInstrGC;
224
225 /** Keeping track of multiple ret instructions. */
226 RTRCPTR pPatchRetInstrGC;
227 uint32_t uPatchRetParam1;
228} PATCHINFOTEMP, *PPATCHINFOTEMP;
229
230typedef struct _PATCHINFO
231{
232 uint32_t uState;
233 uint32_t uOldState;
234 DISCPUMODE uOpMode;
235
236 RCPTRTYPE(uint8_t *) pPrivInstrGC; //GC pointer of privileged instruction
237 R3PTRTYPE(uint8_t *) pPrivInstrHC; //HC pointer of privileged instruction
238 uint8_t aPrivInstr[MAX_INSTR_SIZE];
239 uint32_t cbPrivInstr;
240 uint32_t opcode; //opcode for priv instr (OP_*)
241 uint32_t cbPatchJump; //patch jump size
242
243 /* Only valid for PATMFL_JUMP_CONFLICT patches */
244 RTRCPTR pPatchJumpDestGC;
245
246 RTGCUINTPTR32 pPatchBlockOffset;
247 uint32_t cbPatchBlockSize;
248 uint32_t uCurPatchOffset;
249#if HC_ARCH_BITS == 64
250 uint32_t Alignment0; /**< Align flags correctly. */
251#endif
252
253 uint64_t flags;
254
255 /**
256 * Lowest and highest patched GC instruction address. To optimize searches.
257 */
258 RTRCPTR pInstrGCLowest;
259 RTRCPTR pInstrGCHighest;
260
261 /* Tree of fixup records for the patch. */
262 R3PTRTYPE(PAVLPVNODECORE) FixupTree;
263 int32_t nrFixups;
264
265 /* Tree of jumps inside the generated patch code. */
266 int32_t nrJumpRecs;
267 R3PTRTYPE(PAVLPVNODECORE) JumpTree;
268
269 /**
270 * Lookup trees for determining the corresponding guest address of an
271 * instruction in the patch block.
272 */
273 R3PTRTYPE(PAVLU32NODECORE) Patch2GuestAddrTree;
274 R3PTRTYPE(PAVLU32NODECORE) Guest2PatchAddrTree;
275 uint32_t nrPatch2GuestRecs;
276#if HC_ARCH_BITS == 64
277 uint32_t Alignment1;
278#endif
279
280 // Cache record for PATMGCVirtToHCVirt
281 P2GLOOKUPREC cacheRec;
282
283 /* Temporary information during patch creation. Don't waste hypervisor memory for this. */
284 R3PTRTYPE(PPATCHINFOTEMP) pTempInfo;
285
286 /* Count the number of writes to the corresponding guest code. */
287 uint32_t cCodeWrites;
288
289 /* Count the number of invalid writes to pages monitored for the patch. */
290 //some statistics to determine if we should keep this patch activated
291 uint32_t cTraps;
292
293 uint32_t cInvalidWrites;
294
295 // Index into the uPatchRun and uPatchTrap arrays (0..MAX_PATCHES-1)
296 uint32_t uPatchIdx;
297
298 /* First opcode byte, that's overwritten when a patch is marked dirty. */
299 uint8_t bDirtyOpcode;
300 uint8_t Alignment2[7]; /**< Align the structure size on a 8-byte boundrary. */
301} PATCHINFO, *PPATCHINFO;
302
303#define PATCHCODE_PTR_GC(pPatch) (RTRCPTR) (pVM->patm.s.pPatchMemGC + (pPatch)->pPatchBlockOffset)
304#define PATCHCODE_PTR_HC(pPatch) (uint8_t *)(pVM->patm.s.pPatchMemHC + (pPatch)->pPatchBlockOffset)
305
306/**
307 * Lookup record for patches
308 */
309typedef struct PATMPATCHREC
310{
311 /** The key is a GC virtual address. */
312 AVLOU32NODECORE Core;
313 /** The key is a patch offset. */
314 AVLOU32NODECORE CoreOffset;
315
316 PATCHINFO patch;
317} PATMPATCHREC, *PPATMPATCHREC;
318
319/** Increment for allocating room for pointer array */
320#define PATMPATCHPAGE_PREALLOC_INCREMENT 16
321
322/**
323 * Lookup record for patch pages
324 */
325typedef struct PATMPATCHPAGE
326{
327 /** The key is a GC virtual address. */
328 AVLOU32NODECORE Core;
329 /** Region to monitor. */
330 RTRCPTR pLowestAddrGC;
331 RTRCPTR pHighestAddrGC;
332 /** Number of patches for this page. */
333 uint32_t cCount;
334 /** Maximum nr of pointers in the array. */
335 uint32_t cMaxPatches;
336 /** Array of patch pointers for this page. */
337 R3PTRTYPE(PPATCHINFO *) aPatch;
338} PATMPATCHPAGE, *PPATMPATCHPAGE;
339
340#define PATM_PATCHREC_FROM_COREOFFSET(a) (PPATMPATCHREC)((uintptr_t)a - RT_OFFSETOF(PATMPATCHREC, CoreOffset))
341#define PATM_PATCHREC_FROM_PATCHINFO(a) (PPATMPATCHREC)((uintptr_t)a - RT_OFFSETOF(PATMPATCHREC, patch))
342
343typedef struct PATMTREES
344{
345 /**
346 * AVL tree with all patches (active or disabled) sorted by guest instruction address
347 */
348 AVLOU32TREE PatchTree;
349
350 /**
351 * AVL tree with all patches sorted by patch address (offset actually)
352 */
353 AVLOU32TREE PatchTreeByPatchAddr;
354
355 /**
356 * AVL tree with all pages which were (partly) patched
357 */
358 AVLOU32TREE PatchTreeByPage;
359
360 uint32_t align[1];
361} PATMTREES, *PPATMTREES;
362
363/**
364 * PATM VM Instance data.
365 * Changes to this must checked against the padding of the patm union in VM!
366 */
367typedef struct PATM
368{
369 /** Offset to the VM structure.
370 * See PATM2VM(). */
371 RTINT offVM;
372
373 RCPTRTYPE(uint8_t *) pPatchMemGC;
374 R3PTRTYPE(uint8_t *) pPatchMemHC;
375 uint32_t cbPatchMem;
376 uint32_t offPatchMem;
377 bool fOutOfMemory;
378
379 int32_t deltaReloc;
380
381 /* GC PATM state pointers */
382 R3PTRTYPE(PPATMGCSTATE) pGCStateHC;
383 RCPTRTYPE(PPATMGCSTATE) pGCStateGC;
384
385 /** PATM stack page for call instruction execution. (2 parts: one for our private stack and one to store the original return address */
386 RCPTRTYPE(RTRCPTR *) pGCStackGC;
387 R3PTRTYPE(RTRCPTR *) pGCStackHC;
388
389 /** GC pointer to CPUMCTX structure. */
390 RCPTRTYPE(PCPUMCTX) pCPUMCtxGC;
391
392 /* GC statistics pointers */
393 RCPTRTYPE(PSTAMRATIOU32) pStatsGC;
394 R3PTRTYPE(PSTAMRATIOU32) pStatsHC;
395
396 /* Current free index value (uPatchRun/uPatchTrap arrays). */
397 uint32_t uCurrentPatchIdx;
398
399 /* Temporary counter for patch installation call depth. (in order not to go on forever) */
400 uint32_t ulCallDepth;
401
402 /** Number of page lookup records. */
403 uint32_t cPageRecords;
404
405 /**
406 * Lowest and highest patched GC instruction addresses. To optimize searches.
407 */
408 RTRCPTR pPatchedInstrGCLowest;
409 RTRCPTR pPatchedInstrGCHighest;
410
411 /** Pointer to the patch tree for instructions replaced by 'int 3'. */
412 RCPTRTYPE(PPATMTREES) PatchLookupTreeGC;
413 R3PTRTYPE(PPATMTREES) PatchLookupTreeHC;
414
415 /** Global PATM lookup and call function (used by call patches). */
416 RTRCPTR pfnHelperCallGC;
417 /** Global PATM return function (used by ret patches). */
418 RTRCPTR pfnHelperRetGC;
419 /** Global PATM jump function (used by indirect jmp patches). */
420 RTRCPTR pfnHelperJumpGC;
421 /** Global PATM return function (used by iret patches). */
422 RTRCPTR pfnHelperIretGC;
423
424 /** Fake patch record for global functions. */
425 R3PTRTYPE(PPATMPATCHREC) pGlobalPatchRec;
426
427 /** Pointer to original sysenter handler */
428 RTRCPTR pfnSysEnterGC;
429 /** Pointer to sysenter handler trampoline */
430 RTRCPTR pfnSysEnterPatchGC;
431 /** Sysenter patch index (for stats only) */
432 uint32_t uSysEnterPatchIdx;
433
434 // GC address of fault in monitored page (set by PATMGCMonitorPage, used by PATMR3HandleMonitoredPage)
435 RTRCPTR pvFaultMonitor;
436
437 /* Temporary information for pending MMIO patch. Set in GC or R0 context. */
438 struct
439 {
440 RTGCPHYS GCPhys;
441 RTRCPTR pCachedData;
442 RTRCPTR Alignment0; /**< Align the structure size on a 8-byte boundrary. */
443 } mmio;
444
445 /* Temporary storage during load/save state */
446 struct
447 {
448 R3PTRTYPE(PSSMHANDLE) pSSM;
449 uint32_t cPatches;
450#if HC_ARCH_BITS == 64
451 uint32_t Alignment0; /**< Align the structure size on a 8-byte boundrary. */
452#endif
453 } savedstate;
454
455 STAMCOUNTER StatNrOpcodeRead;
456 STAMCOUNTER StatDisabled;
457 STAMCOUNTER StatUnusable;
458 STAMCOUNTER StatEnabled;
459 STAMCOUNTER StatInstalled;
460 STAMCOUNTER StatInstalledFunctionPatches;
461 STAMCOUNTER StatInstalledTrampoline;
462 STAMCOUNTER StatInstalledJump;
463 STAMCOUNTER StatInt3Callable;
464 STAMCOUNTER StatInt3BlockRun;
465 STAMCOUNTER StatOverwritten;
466 STAMCOUNTER StatFixedConflicts;
467 STAMCOUNTER StatFlushed;
468 STAMCOUNTER StatPageBoundaryCrossed;
469 STAMCOUNTER StatMonitored;
470 STAMPROFILEADV StatHandleTrap;
471 STAMCOUNTER StatSwitchBack;
472 STAMCOUNTER StatSwitchBackFail;
473 STAMCOUNTER StatPATMMemoryUsed;
474 STAMCOUNTER StatDuplicateREQSuccess;
475 STAMCOUNTER StatDuplicateREQFailed;
476 STAMCOUNTER StatDuplicateUseExisting;
477 STAMCOUNTER StatFunctionFound;
478 STAMCOUNTER StatFunctionNotFound;
479 STAMPROFILEADV StatPatchWrite;
480 STAMPROFILEADV StatPatchWriteDetect;
481 STAMCOUNTER StatDirty;
482 STAMCOUNTER StatPushTrap;
483 STAMCOUNTER StatPatchWriteInterpreted;
484 STAMCOUNTER StatPatchWriteInterpretedFailed;
485
486 STAMCOUNTER StatSysEnter;
487 STAMCOUNTER StatSysExit;
488 STAMCOUNTER StatEmulIret;
489 STAMCOUNTER StatEmulIretFailed;
490
491 STAMCOUNTER StatInstrDirty;
492 STAMCOUNTER StatInstrDirtyGood;
493 STAMCOUNTER StatInstrDirtyBad;
494
495 STAMCOUNTER StatPatchPageInserted;
496 STAMCOUNTER StatPatchPageRemoved;
497
498 STAMCOUNTER StatPatchRefreshSuccess;
499 STAMCOUNTER StatPatchRefreshFailed;
500
501 STAMCOUNTER StatGenRet;
502 STAMCOUNTER StatGenRetReused;
503 STAMCOUNTER StatGenJump;
504 STAMCOUNTER StatGenCall;
505 STAMCOUNTER StatGenPopf;
506
507 STAMCOUNTER StatCheckPendingIRQ;
508
509 STAMCOUNTER StatFunctionLookupReplace;
510 STAMCOUNTER StatFunctionLookupInsert;
511 uint32_t StatU32FunctionMaxSlotsUsed;
512 uint32_t Alignment0; /**< Align the structure size on a 8-byte boundrary. */
513} PATM, *PPATM;
514
515
516/**
517 * Execute state save operation.
518 *
519 * @returns VBox status code.
520 * @param pVM VM Handle.
521 * @param pSSM SSM operation handle.
522 */
523DECLCALLBACK(int) patmr3Save(PVM pVM, PSSMHANDLE pSSM);
524
525DECLCALLBACK(int) patmVirtPageHandler(PVM pVM, RTGCPTR GCPtr, void *pvPtr, void *pvBuf, size_t cbBuf, PGMACCESSTYPE enmAccessType, void *pvUser);
526
527/**
528 * Execute state load operation.
529 *
530 * @returns VBox status code.
531 * @param pVM VM Handle.
532 * @param pSSM SSM operation handle.
533 * @param u32Version Data layout version.
534 */
535DECLCALLBACK(int) patmr3Load(PVM pVM, PSSMHANDLE pSSM, uint32_t u32Version);
536
537#ifdef IN_RING3
538RTRCPTR patmPatchGCPtr2GuestGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t *) pPatchGC);
539RTRCPTR patmGuestGCPtrToPatchGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t*) pInstrGC);
540RTRCPTR patmGuestGCPtrToClosestPatchGCPtr(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t*) pInstrGC);
541#endif
542
543/* Add a patch to guest lookup record
544 *
545 * @param pVM The VM to operate on.
546 * @param pPatch Patch structure ptr
547 * @param pPatchInstrHC Guest context pointer to patch block
548 * @param pInstrGC Guest context pointer to privileged instruction
549 * @param enmType Lookup type
550 * @param fDirty Dirty flag
551 *
552 */
553void patmr3AddP2GLookupRecord(PVM pVM, PPATCHINFO pPatch, uint8_t *pPatchInstrHC, RTRCPTR pInstrGC, PATM_LOOKUP_TYPE enmType, bool fDirty=false);
554
555/**
556 * Insert page records for all guest pages that contain instructions that were recompiled for this patch
557 *
558 * @returns VBox status code.
559 * @param pVM The VM to operate on.
560 * @param pPatch Patch record
561 */
562int patmInsertPatchPages(PVM pVM, PPATCHINFO pPatch);
563
564/**
565 * Remove page records for all guest pages that contain instructions that were recompiled for this patch
566 *
567 * @returns VBox status code.
568 * @param pVM The VM to operate on.
569 * @param pPatch Patch record
570 */
571int patmRemovePatchPages(PVM pVM, PPATCHINFO pPatch);
572
573/**
574 * Returns the GC address of the corresponding patch statistics counter
575 *
576 * @returns Stat address
577 * @param pVM The VM to operate on.
578 * @param pPatch Patch structure
579 */
580RTRCPTR patmPatchQueryStatAddress(PVM pVM, PPATCHINFO pPatch);
581
582/**
583 * Remove patch for privileged instruction at specified location
584 *
585 * @returns VBox status code.
586 * @param pVM The VM to operate on.
587 * @param pPatchRec Patch record
588 * @param fForceRemove Remove *all* patches
589 */
590int PATMRemovePatch(PVM pVM, PPATMPATCHREC pPatchRec, bool fForceRemove);
591
592/**
593 * Call for analysing the instructions following the privileged instr. for compliance with our heuristics
594 *
595 * @returns VBox status code.
596 * @param pVM The VM to operate on.
597 * @param pCpu CPU disassembly state
598 * @param pInstrHC Guest context pointer to privileged instruction
599 * @param pCurInstrHC Guest context pointer to current instruction
600 * @param pUserData User pointer
601 *
602 */
603typedef int (VBOXCALL *PFN_PATMR3ANALYSE)(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, void *pUserData);
604
605/**
606 * Install guest OS specific patch
607 *
608 * @returns VBox status code.
609 * @param pVM The VM to operate on
610 * @param pCpu Disassembly state of instruction.
611 * @param pInstrGC GC Instruction pointer for instruction
612 * @param pInstrHC GC Instruction pointer for instruction
613 * @param pPatchRec Patch structure
614 *
615 */
616int PATMInstallGuestSpecificPatch(PVM pVM, PDISCPUSTATE pCpu, RTRCPTR pInstrGC, uint8_t *pInstrHC, PPATMPATCHREC pPatchRec);
617
618/**
619 * Convert guest context address to host context pointer
620 *
621 * @returns VBox status code.
622 * @param pVM The VM to operate on.
623 * @param pPatch Patch block structure pointer
624 * @param pGCPtr Guest context pointer
625 *
626 * @returns Host context pointer or NULL in case of an error
627 *
628 */
629R3PTRTYPE(uint8_t *) PATMGCVirtToHCVirt(PVM pVM, PPATCHINFO pPatch, RCPTRTYPE(uint8_t *) pGCPtr);
630
631
632/**
633 * Check if the instruction is patched as a duplicated function
634 *
635 * @returns patch record
636 * @param pVM The VM to operate on.
637 * @param pInstrGC Guest context point to the instruction
638 *
639 */
640PATMDECL(PPATMPATCHREC) PATMQueryFunctionPatch(PVM pVM, RTRCPTR pInstrGC);
641
642
643/**
644 * Empty the specified tree (PV tree, MMR3 heap)
645 *
646 * @param pVM The VM to operate on.
647 * @param ppTree Tree to empty
648 */
649void patmEmptyTree(PVM pVM, PPAVLPVNODECORE ppTree);
650
651
652/**
653 * Empty the specified tree (U32 tree, MMR3 heap)
654 *
655 * @param pVM The VM to operate on.
656 * @param ppTree Tree to empty
657 */
658void patmEmptyTreeU32(PVM pVM, PPAVLU32NODECORE ppTree);
659
660
661/**
662 * Return the name of the patched instruction
663 *
664 * @returns instruction name
665 *
666 * @param opcode DIS instruction opcode
667 * @param fPatchFlags Patch flags
668 */
669PATMDECL(const char *) patmGetInstructionString(uint32_t opcode, uint32_t fPatchFlags);
670
671
672/**
673 * Read callback for disassembly function; supports reading bytes that cross a page boundary
674 *
675 * @returns VBox status code.
676 * @param pSrc GC source pointer
677 * @param pDest HC destination pointer
678 * @param size Number of bytes to read
679 * @param pvUserdata Callback specific user data (pCpu)
680 *
681 */
682int patmReadBytes(RTUINTPTR pSrc, uint8_t *pDest, unsigned size, void *pvUserdata);
683
684
685#ifndef IN_GC
686
687#define PATMREAD_RAWCODE 1 /* read code as-is */
688#define PATMREAD_ORGCODE 2 /* read original guest opcode bytes; not the patched bytes */
689#define PATMREAD_NOCHECK 4 /* don't check for patch conflicts */
690
691/*
692 * Private structure used during disassembly
693 */
694typedef struct
695{
696 PVM pVM;
697 PPATCHINFO pPatchInfo;
698 R3PTRTYPE(uint8_t *) pInstrHC;
699 RTRCPTR pInstrGC;
700 uint32_t fReadFlags;
701} PATMDISASM, *PPATMDISASM;
702
703inline bool PATMR3DISInstr(PVM pVM, PPATCHINFO pPatch, DISCPUSTATE *pCpu, RTRCPTR InstrGC,
704 uint8_t *InstrHC, uint32_t *pOpsize, char *pszOutput,
705 uint32_t fReadFlags = PATMREAD_ORGCODE)
706{
707 PATMDISASM disinfo;
708 disinfo.pVM = pVM;
709 disinfo.pPatchInfo = pPatch;
710 disinfo.pInstrHC = InstrHC;
711 disinfo.pInstrGC = InstrGC;
712 disinfo.fReadFlags = fReadFlags;
713 (pCpu)->pfnReadBytes = patmReadBytes;
714 (pCpu)->apvUserData[0] = &disinfo;
715 return VBOX_SUCCESS(DISInstr(pCpu, InstrGC, 0, pOpsize, pszOutput));
716}
717#endif /* !IN_GC */
718
719__BEGIN_DECLS
720/**
721 * #PF Virtual Handler callback for Guest access a page monitored by PATM
722 *
723 * @returns VBox status code (appropritate for trap handling and GC return).
724 * @param pVM VM Handle.
725 * @param uErrorCode CPU Error code.
726 * @param pRegFrame Trap register frame.
727 * @param pvFault The fault address (cr2).
728 * @param pvRange The base address of the handled virtual range.
729 * @param offRange The offset of the access into this range.
730 * (If it's a EIP range this's the EIP, if not it's pvFault.)
731 */
732PATMGCDECL(int) PATMGCMonitorPage(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPTR pvRange, uintptr_t offRange);
733
734/**
735 * Find patch for privileged instruction at specified location
736 *
737 * @returns Patch structure pointer if found; else NULL
738 * @param pVM The VM to operate on.
739 * @param pInstr Guest context point to instruction that might lie within 5 bytes of an existing patch jump
740 * @param fIncludeHints Include hinted patches or not
741 *
742 */
743PPATCHINFO PATMFindActivePatchByEntrypoint(PVM pVM, RTRCPTR pInstrGC, bool fIncludeHints=false);
744
745/**
746 * Patch cli/sti pushf/popf instruction block at specified location
747 *
748 * @returns VBox status code.
749 * @param pVM The VM to operate on.
750 * @param pInstrGC Guest context point to privileged instruction
751 * @param pInstrHC Host context point to privileged instruction
752 * @param uOpcode Instruction opcodee
753 * @param uOpSize Size of starting instruction
754 * @param pPatchRec Patch record
755 *
756 * @note returns failure if patching is not allowed or possible
757 *
758 */
759PATMR3DECL(int) PATMR3PatchBlock(PVM pVM, RTRCPTR pInstrGC, R3PTRTYPE(uint8_t *) pInstrHC,
760 uint32_t uOpcode, uint32_t uOpSize, PPATMPATCHREC pPatchRec);
761
762
763/**
764 * Replace an instruction with a breakpoint (0xCC), that is handled dynamically in the guest context.
765 *
766 * @returns VBox status code.
767 * @param pVM The VM to operate on.
768 * @param pInstrGC Guest context point to privileged instruction
769 * @param pInstrHC Host context point to privileged instruction
770 * @param pCpu Disassembly CPU structure ptr
771 * @param pPatch Patch record
772 *
773 * @note returns failure if patching is not allowed or possible
774 *
775 */
776PATMR3DECL(int) PATMR3PatchInstrInt3(PVM pVM, RTRCPTR pInstrGC, R3PTRTYPE(uint8_t *) pInstrHC, DISCPUSTATE *pCpu, PPATCHINFO pPatch);
777
778/**
779 * Mark patch as dirty
780 *
781 * @returns VBox status code.
782 * @param pVM The VM to operate on.
783 * @param pPatch Patch record
784 *
785 * @note returns failure if patching is not allowed or possible
786 *
787 */
788PATMR3DECL(int) PATMR3MarkDirtyPatch(PVM pVM, PPATCHINFO pPatch);
789
790/**
791 * Calculate the branch destination
792 *
793 * @returns branch destination or 0 if failed
794 * @param pCpu Disassembly state of instruction.
795 * @param pBranchInstrGC GC pointer of branch instruction
796 */
797inline RTRCPTR PATMResolveBranch(PDISCPUSTATE pCpu, RTRCPTR pBranchInstrGC)
798{
799 uint32_t disp;
800 if (pCpu->param1.flags & USE_IMMEDIATE8_REL)
801 {
802 disp = (int32_t)(char)pCpu->param1.parval;
803 }
804 else
805 if (pCpu->param1.flags & USE_IMMEDIATE16_REL)
806 {
807 disp = (int32_t)(uint16_t)pCpu->param1.parval;
808 }
809 else
810 if (pCpu->param1.flags & USE_IMMEDIATE32_REL)
811 {
812 disp = (int32_t)pCpu->param1.parval;
813 }
814 else
815 {
816 Log(("We don't support far jumps here!! (%08X)\n", pCpu->param1.flags));
817 return 0;
818 }
819#ifdef IN_GC
820 return (RTRCPTR)((uint8_t *)pBranchInstrGC + pCpu->opsize + disp);
821#else
822 return pBranchInstrGC + pCpu->opsize + disp;
823#endif
824}
825
826__END_DECLS
827
828#ifdef DEBUG
829int patmr3DisasmCallback(PVM pVM, DISCPUSTATE *pCpu, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, void *pUserData);
830int patmr3DisasmCodeStream(PVM pVM, RCPTRTYPE(uint8_t *) pInstrGC, RCPTRTYPE(uint8_t *) pCurInstrGC, PFN_PATMR3ANALYSE pfnPATMR3Analyse, void *pUserData);
831#endif
832
833#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