VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/MMAll.cpp@ 1345

Last change on this file since 1345 was 1309, checked in by vboxsync, 18 years ago

r=bird: RTR0UINTREG -> RTR0UINTPTR. created defect.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 19.1 KB
Line 
1/* $Id: MMAll.cpp 1309 2007-03-07 20:05:32Z vboxsync $ */
2/** @file
3 * MM - Memory Monitor(/Manager) - Any Context.
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_MM_HYPER
27#include <VBox/mm.h>
28#include "MMInternal.h"
29#include <VBox/vm.h>
30#include <VBox/log.h>
31#include <iprt/assert.h>
32
33
34
35/**
36 * Lookup a host context ring-3 address.
37 *
38 * @returns Pointer to the corresponding lookup record.
39 * @returns NULL on failure.
40 * @param pVM The VM handle.
41 * @param R3Ptr The host context ring-3 address to lookup.
42 * @param poff Where to store the offset into the HMA memory chunk.
43 */
44DECLINLINE(PMMLOOKUPHYPER) mmHyperLookupR3(PVM pVM, RTR3PTR R3Ptr, uint32_t *poff)
45{
46 /** @todo cache last lookup this stuff ain't cheap! */
47 PMMLOOKUPHYPER pLookup = (PMMLOOKUPHYPER)((char*)CTXSUFF(pVM->mm.s.pHyperHeap) + pVM->mm.s.offLookupHyper);
48 for (;;)
49 {
50 switch (pLookup->enmType)
51 {
52 case MMLOOKUPHYPERTYPE_LOCKED:
53 {
54 const uint32_t off = (RTR3UINTPTR)R3Ptr - (RTR3UINTPTR)pLookup->u.Locked.pvHC;
55 if (off < pLookup->cb)
56 {
57 *poff = off;
58 return pLookup;
59 }
60 break;
61 }
62
63 case MMLOOKUPHYPERTYPE_HCPHYS:
64 {
65 const uint32_t off = (RTR3UINTPTR)R3Ptr - (RTR3UINTPTR)pLookup->u.HCPhys.pvHC;
66 if (off < pLookup->cb)
67 {
68 *poff = off;
69 return pLookup;
70 }
71 break;
72 }
73
74 case MMLOOKUPHYPERTYPE_GCPHYS: /* (for now we'll not allow these kind of conversions) */
75 case MMLOOKUPHYPERTYPE_DYNAMIC:
76 break;
77
78 default:
79 AssertMsgFailed(("enmType=%d\n", pLookup->enmType));
80 break;
81 }
82
83 /* next */
84 if (pLookup->offNext == (int32_t)NIL_OFFSET)
85 break;
86 pLookup = (PMMLOOKUPHYPER)((char *)pLookup + pLookup->offNext);
87 }
88
89 AssertMsgFailed(("R3Ptr=%p is not inside the hypervisor memory area!\n", R3Ptr));
90 return NULL;
91}
92
93
94/**
95 * Lookup a host context ring-0 address.
96 *
97 * @returns Pointer to the corresponding lookup record.
98 * @returns NULL on failure.
99 * @param pVM The VM handle.
100 * @param R0Ptr The host context ring-0 address to lookup.
101 * @param poff Where to store the offset into the HMA memory chunk.
102 */
103DECLINLINE(PMMLOOKUPHYPER) mmHyperLookupR0(PVM pVM, RTR0PTR R0Ptr, uint32_t *poff)
104{
105 AssertCompile(sizeof(RTR0PTR) == sizeof(RTR3PTR));
106
107 /*
108 * Translate Ring-0 VM addresses into Ring-3 VM addresses before feeding it to mmHyperLookupR3.
109 */
110 /** @todo fix this properly; the ring 0 pVM address differs from the R3 one. (#1865) */
111 RTR0UINTPTR offVM = (RTR0UINTPTR)R0Ptr - (RTR0UINTPTR)pVM->pVMR0;
112 RTR3PTR R3Ptr = offVM < sizeof(*pVM)
113 ? (RTR3PTR)((RTR3UINTPTR)pVM->pVMR3 + offVM)
114 : (RTR3PTR)R0Ptr;
115
116 return mmHyperLookupR3(pVM, R3Ptr, poff);
117}
118
119
120/**
121 * Lookup a guest context address.
122 *
123 * @returns Pointer to the corresponding lookup record.
124 * @returns NULL on failure.
125 * @param pVM The VM handle.
126 * @param GCPtr The guest context address to lookup.
127 * @param poff Where to store the offset into the HMA memory chunk.
128 */
129DECLINLINE(PMMLOOKUPHYPER) mmHyperLookupGC(PVM pVM, RTGCPTR GCPtr, uint32_t *poff)
130{
131 /** @todo cache last lookup this stuff ain't cheap! */
132 unsigned offGC = (RTGCUINTPTR)GCPtr - (RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC;
133 PMMLOOKUPHYPER pLookup = (PMMLOOKUPHYPER)((char*)CTXSUFF(pVM->mm.s.pHyperHeap) + pVM->mm.s.offLookupHyper);
134 for (;;)
135 {
136 const uint32_t off = offGC - pLookup->off;
137 if (off < pLookup->cb)
138 {
139 switch (pLookup->enmType)
140 {
141 case MMLOOKUPHYPERTYPE_LOCKED:
142 case MMLOOKUPHYPERTYPE_HCPHYS:
143 *poff = off;
144 return pLookup;
145 default:
146 break;
147 }
148 AssertMsgFailed(("enmType=%d\n", pLookup->enmType));
149 return NULL;
150 }
151
152 /* next */
153 if (pLookup->offNext == (int32_t)NIL_OFFSET)
154 break;
155 pLookup = (PMMLOOKUPHYPER)((char *)pLookup + pLookup->offNext);
156 }
157
158 AssertMsgFailed(("GCPtr=%p is not inside the hypervisor memory area!\n", GCPtr));
159 return NULL;
160}
161
162
163/**
164 * Lookup a current context address.
165 *
166 * @returns Pointer to the corresponding lookup record.
167 * @returns NULL on failure.
168 * @param pVM The VM handle.
169 * @param pv The current context address to lookup.
170 * @param poff Where to store the offset into the HMA memory chunk.
171 */
172DECLINLINE(PMMLOOKUPHYPER) mmHyperLookupCC(PVM pVM, void *pv, uint32_t *poff)
173{
174#ifdef IN_GC
175 return mmHyperLookupGC(pVM, pv, poff);
176#elif defined(IN_RING0)
177 return mmHyperLookupR0(pVM, pv, poff);
178#else
179 return mmHyperLookupR3(pVM, pv, poff);
180#endif
181}
182
183
184/**
185 * Calculate the host context ring-3 address of an offset into the HMA memory chunk.
186 *
187 * @returns the host context ring-3 address.
188 * @param pLookup The HMA lookup record.
189 * @param off The offset into the HMA memory chunk.
190 */
191DECLINLINE(RTR3PTR) mmHyperLookupCalcR3(PMMLOOKUPHYPER pLookup, uint32_t off)
192{
193 switch (pLookup->enmType)
194 {
195 case MMLOOKUPHYPERTYPE_LOCKED:
196 return (RTR3PTR)((RTR3UINTPTR)pLookup->u.Locked.pvHC + off);
197 case MMLOOKUPHYPERTYPE_HCPHYS:
198 return (RTR3PTR)((RTR3UINTPTR)pLookup->u.HCPhys.pvHC + off);
199 default:
200 AssertMsgFailed(("enmType=%d\n", pLookup->enmType));
201 return NIL_RTR3PTR;
202 }
203}
204
205
206/**
207 * Calculate the host context ring-0 address of an offset into the HMA memory chunk.
208 *
209 * @returns the host context ring-0 address.
210 * @param pLookup The HMA lookup record.
211 * @param off The offset into the HMA memory chunk.
212 */
213DECLINLINE(RTR0PTR) mmHyperLookupCalcR0(PMMLOOKUPHYPER pLookup, uint32_t off)
214{
215 return (RTR0PTR)mmHyperLookupCalcR3(pLookup, off);
216}
217
218
219/**
220 * Calculate the guest context address of an offset into the HMA memory chunk.
221 *
222 * @returns the guest context base address.
223 * @param pVM The the VM handle.
224 * @param pLookup The HMA lookup record.
225 * @param off The offset into the HMA memory chunk.
226 */
227DECLINLINE(RTGCPTR) mmHyperLookupCalcGC(PVM pVM, PMMLOOKUPHYPER pLookup, uint32_t off)
228{
229 return (RTGCPTR)((RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC + pLookup->off + off);
230}
231
232
233/**
234 * Calculate the guest context address of an offset into the HMA memory chunk.
235 *
236 * @returns the guest context base address.
237 * @param pVM The the VM handle.
238 * @param pLookup The HMA lookup record.
239 * @param off The offset into the HMA memory chunk.
240 */
241DECLINLINE(void *) mmHyperLookupCalcCC(PVM pVM, PMMLOOKUPHYPER pLookup, uint32_t off)
242{
243#ifdef IN_GC
244 return mmHyperLookupCalcGC(pVM, pLookup, off);
245#elif defined(IN_RING0)
246 return mmHyperLookupCalcR0(pLookup, off);
247#else
248 return mmHyperLookupCalcR3(pLookup, off);
249#endif
250}
251
252
253/**
254 * Converts a ring-0 host context address in the Hypervisor memory region to a ring-3 host context address.
255 *
256 * @returns ring-3 host context address.
257 * @param pVM The VM to operate on.
258 * @param R0Ptr The ring-0 host context address.
259 * You'll be damned if this is not in the HMA! :-)
260 * @thread The Emulation Thread.
261 */
262MMDECL(RTR3PTR) MMHyperR0ToR3(PVM pVM, RTR0PTR R0Ptr)
263{
264 uint32_t off;
265 PMMLOOKUPHYPER pLookup = mmHyperLookupR0(pVM, R0Ptr, &off);
266 if (pLookup)
267 return mmHyperLookupCalcR3(pLookup, off);
268 return NIL_RTR3PTR;
269}
270
271
272/**
273 * Converts a ring-0 host context address in the Hypervisor memory region to a guest context address.
274 *
275 * @returns guest context address.
276 * @param pVM The VM to operate on.
277 * @param R0Ptr The ring-0 host context address.
278 * You'll be damned if this is not in the HMA! :-)
279 * @thread The Emulation Thread.
280 */
281MMDECL(RTGCPTR) MMHyperR0ToGC(PVM pVM, RTR0PTR R0Ptr)
282{
283 uint32_t off;
284 PMMLOOKUPHYPER pLookup = mmHyperLookupR0(pVM, R0Ptr, &off);
285 if (pLookup)
286 return mmHyperLookupCalcGC(pVM, pLookup, off);
287 return NIL_RTGCPTR;
288}
289
290
291/**
292 * Converts a ring-0 host context address in the Hypervisor memory region to a current context address.
293 *
294 * @returns current context address.
295 * @param pVM The VM to operate on.
296 * @param R0Ptr The ring-0 host context address.
297 * You'll be damned if this is not in the HMA! :-)
298 * @thread The Emulation Thread.
299 */
300#ifndef IN_RING0
301MMDECL(void *) MMHyperR0ToCC(PVM pVM, RTR0PTR R0Ptr)
302{
303 uint32_t off;
304 PMMLOOKUPHYPER pLookup = mmHyperLookupR0(pVM, R0Ptr, &off);
305 if (pLookup)
306 return mmHyperLookupCalcCC(pVM, pLookup, off);
307 return NULL;
308}
309#endif
310
311
312/**
313 * Converts a ring-3 host context address in the Hypervisor memory region to a ring-0 host context address.
314 *
315 * @returns ring-0 host context address.
316 * @param pVM The VM to operate on.
317 * @param R3Ptr The ring-3 host context address.
318 * You'll be damned if this is not in the HMA! :-)
319 * @thread The Emulation Thread.
320 */
321MMDECL(RTR0PTR) MMHyperR3ToR0(PVM pVM, RTR3PTR R3Ptr)
322{
323 uint32_t off;
324 PMMLOOKUPHYPER pLookup = mmHyperLookupR3(pVM, R3Ptr, &off);
325 if (pLookup)
326 return mmHyperLookupCalcR0(pLookup, off);
327 return NIL_RTR0PTR;
328}
329
330
331/**
332 * Converts a ring-3 host context address in the Hypervisor memory region to a guest context address.
333 *
334 * @returns guest context address.
335 * @param pVM The VM to operate on.
336 * @param R3Ptr The ring-3 host context address.
337 * You'll be damned if this is not in the HMA! :-)
338 * @thread The Emulation Thread.
339 */
340MMDECL(RTGCPTR) MMHyperR3ToGC(PVM pVM, RTR3PTR R3Ptr)
341{
342 uint32_t off;
343 PMMLOOKUPHYPER pLookup = mmHyperLookupR3(pVM, R3Ptr, &off);
344 if (pLookup)
345 return mmHyperLookupCalcGC(pVM, pLookup, off);
346 return NIL_RTGCPTR;
347}
348
349
350/**
351 * Converts a ring-3 host context address in the Hypervisor memory region to a current context address.
352 *
353 * @returns current context address.
354 * @param pVM The VM to operate on.
355 * @param R3Ptr The ring-3 host context address.
356 * You'll be damned if this is not in the HMA! :-)
357 * @thread The Emulation Thread.
358 */
359#ifndef IN_RING3
360MMDECL(void *) MMHyperR3ToCC(PVM pVM, RTR3PTR R3Ptr)
361{
362 uint32_t off;
363 PMMLOOKUPHYPER pLookup = mmHyperLookupR3(pVM, R3Ptr, &off);
364 if (pLookup)
365 return mmHyperLookupCalcCC(pVM, pLookup, off);
366 return NULL;
367}
368#endif
369
370
371/**
372 * Converts a guest context address in the Hypervisor memory region to a ring-3 context address.
373 *
374 * @returns ring-3 host context address.
375 * @param pVM The VM to operate on.
376 * @param GCPtr The guest context address.
377 * You'll be damned if this is not in the HMA! :-)
378 * @thread The Emulation Thread.
379 */
380MMDECL(RTR3PTR) MMHyperGCToR3(PVM pVM, RTGCPTR GCPtr)
381{
382 uint32_t off;
383 PMMLOOKUPHYPER pLookup = mmHyperLookupGC(pVM, GCPtr, &off);
384 if (pLookup)
385 return mmHyperLookupCalcR3(pLookup, off);
386 return NIL_RTR3PTR;
387}
388
389
390/**
391 * Converts a guest context address in the Hypervisor memory region to a ring-0 host context address.
392 *
393 * @returns ring-0 host context address.
394 * @param pVM The VM to operate on.
395 * @param GCPtr The guest context address.
396 * You'll be damned if this is not in the HMA! :-)
397 * @thread The Emulation Thread.
398 */
399MMDECL(RTR0PTR) MMHyperGCToR0(PVM pVM, RTGCPTR GCPtr)
400{
401 uint32_t off;
402 PMMLOOKUPHYPER pLookup = mmHyperLookupGC(pVM, GCPtr, &off);
403 if (pLookup)
404 return mmHyperLookupCalcR0(pLookup, off);
405 return NIL_RTR0PTR;
406}
407
408
409/**
410 * Converts a guest context address in the Hypervisor memory region to a current context address.
411 *
412 * @returns current context address.
413 * @param pVM The VM to operate on.
414 * @param GCPtr The guest host context address.
415 * You'll be damned if this is not in the HMA! :-)
416 * @thread The Emulation Thread.
417 */
418#ifndef IN_GC
419MMDECL(void *) MMHyperGCToCC(PVM pVM, RTGCPTR GCPtr)
420{
421 uint32_t off;
422 PMMLOOKUPHYPER pLookup = mmHyperLookupGC(pVM, GCPtr, &off);
423 if (pLookup)
424 return mmHyperLookupCalcCC(pVM, pLookup, off);
425 return NULL;
426}
427#endif
428
429
430
431/**
432 * Converts a current context address in the Hypervisor memory region to a ring-3 host context address.
433 *
434 * @returns ring-3 host context address.
435 * @param pVM The VM to operate on.
436 * @param pv The current context address.
437 * You'll be damned if this is not in the HMA! :-)
438 * @thread The Emulation Thread.
439 */
440#ifndef IN_RING3
441MMDECL(RTR3PTR) MMHyperCCToR3(PVM pVM, void *pv)
442{
443 uint32_t off;
444 PMMLOOKUPHYPER pLookup = mmHyperLookupCC(pVM, pv, &off);
445 if (pLookup)
446 return mmHyperLookupCalcR3(pLookup, off);
447 return NIL_RTR3PTR;
448}
449#endif
450
451/**
452 * Converts a current context address in the Hypervisor memory region to a ring-0 host context address.
453 *
454 * @returns ring-0 host context address.
455 * @param pVM The VM to operate on.
456 * @param pv The current context address.
457 * You'll be damned if this is not in the HMA! :-)
458 * @thread The Emulation Thread.
459 */
460#ifndef IN_RING0
461MMDECL(RTR0PTR) MMHyperCCToR0(PVM pVM, void *pv)
462{
463 uint32_t off;
464 PMMLOOKUPHYPER pLookup = mmHyperLookupCC(pVM, pv, &off);
465 if (pLookup)
466 return mmHyperLookupCalcR0(pLookup, off);
467 return NIL_RTR0PTR;
468}
469#endif
470
471
472/**
473 * Converts a current context address in the Hypervisor memory region to a guest context address.
474 *
475 * @returns guest context address.
476 * @param pVM The VM to operate on.
477 * @param pv The current context address.
478 * You'll be damned if this is not in the HMA! :-)
479 * @thread The Emulation Thread.
480 */
481#ifndef IN_GC
482MMDECL(RTGCPTR) MMHyperCCToGC(PVM pVM, void *pv)
483{
484 uint32_t off;
485 PMMLOOKUPHYPER pLookup = mmHyperLookupCC(pVM, pv, &off);
486 if (pLookup)
487 return mmHyperLookupCalcGC(pVM, pLookup, off);
488 return NIL_RTGCPTR;
489}
490#endif
491
492
493
494/**
495 * Converts a HC address in the Hypervisor memory region to a GC address.
496 * The memory must have been allocated with MMGCHyperAlloc() or MMR3HyperAlloc().
497 *
498 * @returns GC address.
499 * @param pVM The VM to operate on.
500 * @param HCPtr The host context address.
501 * You'll be damed if this is not in the hypervisor region! :-)
502 * @deprecated
503 */
504MMDECL(RTGCPTR) MMHyperHC2GC(PVM pVM, RTHCPTR HCPtr)
505{
506 PMMLOOKUPHYPER pLookup = (PMMLOOKUPHYPER)((char*)CTXSUFF(pVM->mm.s.pHyperHeap) + pVM->mm.s.offLookupHyper);
507 for (;;)
508 {
509 switch (pLookup->enmType)
510 {
511 case MMLOOKUPHYPERTYPE_LOCKED:
512 {
513 unsigned off = (RTHCUINTPTR)HCPtr - (RTHCUINTPTR)pLookup->u.Locked.pvHC;
514 if (off < pLookup->cb)
515 return (RTGCPTR)((RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC + pLookup->off + off);
516 break;
517 }
518
519 case MMLOOKUPHYPERTYPE_HCPHYS:
520 {
521 unsigned off = (RTHCUINTPTR)HCPtr - (RTHCUINTPTR)pLookup->u.HCPhys.pvHC;
522 if (off < pLookup->cb)
523 return (RTGCPTR)((RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC + pLookup->off + off);
524 break;
525 }
526
527 case MMLOOKUPHYPERTYPE_GCPHYS: /* (for now we'll not allow these kind of conversions) */
528 case MMLOOKUPHYPERTYPE_DYNAMIC:
529 break;
530
531 default:
532 AssertMsgFailed(("enmType=%d\n", pLookup->enmType));
533 break;
534 }
535
536 /* next */
537 if ((unsigned)pLookup->offNext == NIL_OFFSET)
538 break;
539 pLookup = (PMMLOOKUPHYPER)((char *)pLookup + pLookup->offNext);
540 }
541
542 AssertMsgFailed(("HCPtr=%p is not inside the hypervisor memory area!\n", HCPtr));
543 return (RTGCPTR)0;
544}
545
546
547/**
548 * Converts a GC address in the Hypervisor memory region to a HC address.
549 * The memory must have been allocated with MMHyperAlloc().
550 *
551 * @returns HC address.
552 * @param pVM The VM to operate on.
553 * @param GCPtr The guest context address.
554 * You'll be damed if this is not in the hypervisor region! :-)
555 * @deprecated
556 */
557MMDECL(RTHCPTR) MMHyperGC2HC(PVM pVM, RTGCPTR GCPtr)
558{
559 unsigned offGC = (RTGCUINTPTR)GCPtr - (RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC;
560 PMMLOOKUPHYPER pLookup = (PMMLOOKUPHYPER)((char*)CTXSUFF(pVM->mm.s.pHyperHeap) + pVM->mm.s.offLookupHyper);
561 for (;;)
562 {
563 unsigned off = offGC - pLookup->off;
564 if (off < pLookup->cb)
565 {
566 switch (pLookup->enmType)
567 {
568 case MMLOOKUPHYPERTYPE_LOCKED:
569 return (RTHCPTR)((RTHCUINTPTR)pLookup->u.Locked.pvHC + off);
570 case MMLOOKUPHYPERTYPE_HCPHYS:
571 return (RTHCPTR)((RTHCUINTPTR)pLookup->u.HCPhys.pvHC + off);
572 default:
573 break;
574 }
575 AssertMsgFailed(("enmType=%d\n", pLookup->enmType));
576 return (RTHCPTR)0;
577 }
578
579 /* next */
580 if ((unsigned)pLookup->offNext == NIL_OFFSET)
581 break;
582 pLookup = (PMMLOOKUPHYPER)((char *)pLookup + pLookup->offNext);
583 }
584
585 AssertMsgFailed(("GCPtr=%p is not inside the hypervisor memory area!\n", GCPtr));
586 return (RTHCPTR)0;
587}
588
589
590#ifdef IN_GC
591/**
592 * Converts a current context address in the Hypervisor memory region to a HC address.
593 * The memory must have been allocated with MMGCHyperAlloc() or MMR3HyperAlloc().
594 *
595 * @returns HC address.
596 * @param pVM The VM to operate on.
597 * @param Ptr The current context address.
598 * @deprecated
599 */
600MMDECL(RTHCPTR) MMHyper2HC(PVM pVM, uintptr_t Ptr)
601{
602 return MMHyperGC2HC(pVM, (RTGCPTR)Ptr);
603}
604
605#else /* !IN_GC */
606
607/**
608 * Converts a current context address in the Hypervisor memory region to a GC address.
609 * The memory must have been allocated with MMHyperAlloc().
610 *
611 * @returns HC address.
612 * @param pVM The VM to operate on.
613 * @param Ptr The current context address.
614 * @thread The Emulation Thread.
615 * @deprecated
616 */
617MMDECL(RTGCPTR) MMHyper2GC(PVM pVM, uintptr_t Ptr)
618{
619 return MMHyperHC2GC(pVM, (RTHCPTR)Ptr);
620}
621#endif /* !IN_GC */
622
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