VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/MMAllHyper.cpp@ 21258

Last change on this file since 21258 was 20874, checked in by vboxsync, 16 years ago

VMMR0CallHost -> VMMRZCallRing3[NoCpu]; VMMCALLHOST -> VMMCALLRING3.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 47.2 KB
Line 
1/* $Id: MMAllHyper.cpp 20874 2009-06-24 02:19:29Z vboxsync $ */
2/** @file
3 * MM - Memory Manager - Hypervisor Memory Area, All Contexts.
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
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_MM_HYPER_HEAP
27#include <VBox/mm.h>
28#include <VBox/stam.h>
29#include "MMInternal.h"
30#include <VBox/vm.h>
31
32#include <VBox/err.h>
33#include <VBox/param.h>
34#include <iprt/assert.h>
35#include <VBox/log.h>
36#include <iprt/asm.h>
37#include <iprt/string.h>
38
39
40/*******************************************************************************
41* Defined Constants And Macros *
42*******************************************************************************/
43#define ASSERT_L(u1, u2) AssertMsg((u1) < (u2), ("u1=%#x u2=%#x\n", u1, u2))
44#define ASSERT_LE(u1, u2) AssertMsg((u1) <= (u2), ("u1=%#x u2=%#x\n", u1, u2))
45#define ASSERT_GE(u1, u2) AssertMsg((u1) >= (u2), ("u1=%#x u2=%#x\n", u1, u2))
46#define ASSERT_ALIGN(u1) AssertMsg(!((u1) & (MMHYPER_HEAP_ALIGN_MIN - 1)), ("u1=%#x (%d)\n", u1, u1))
47
48#define ASSERT_OFFPREV(pHeap, pChunk) \
49 do { Assert(MMHYPERCHUNK_GET_OFFPREV(pChunk) <= 0); \
50 Assert(MMHYPERCHUNK_GET_OFFPREV(pChunk) >= (intptr_t)(pHeap)->CTX_SUFF(pbHeap) - (intptr_t)(pChunk)); \
51 AssertMsg( MMHYPERCHUNK_GET_OFFPREV(pChunk) != 0 \
52 || (uint8_t *)(pChunk) == (pHeap)->CTX_SUFF(pbHeap), \
53 ("pChunk=%p pvHyperHeap=%p\n", (pChunk), (pHeap)->CTX_SUFF(pbHeap))); \
54 } while (0)
55
56#define ASSERT_OFFNEXT(pHeap, pChunk) \
57 do { ASSERT_ALIGN((pChunk)->offNext); \
58 ASSERT_L((pChunk)->offNext, (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) + (pHeap)->offPageAligned - (uintptr_t)(pChunk)); \
59 } while (0)
60
61#define ASSERT_OFFHEAP(pHeap, pChunk) \
62 do { Assert((pChunk)->offHeap); \
63 AssertMsg((PMMHYPERHEAP)((pChunk)->offHeap + (uintptr_t)pChunk) == (pHeap), \
64 ("offHeap=%RX32 pChunk=%p pHeap=%p\n", (pChunk)->offHeap, (pChunk), (pHeap))); \
65 Assert((pHeap)->u32Magic == MMHYPERHEAP_MAGIC); \
66 } while (0)
67
68#ifdef VBOX_WITH_STATISTICS
69#define ASSERT_OFFSTAT(pHeap, pChunk) \
70 do { if (MMHYPERCHUNK_ISFREE(pChunk)) \
71 Assert(!(pChunk)->offStat); \
72 else if ((pChunk)->offStat) \
73 { \
74 Assert((pChunk)->offStat); \
75 AssertMsg(!((pChunk)->offStat & (MMHYPER_HEAP_ALIGN_MIN - 1)), ("offStat=%RX32\n", (pChunk)->offStat)); \
76 uintptr_t uPtr = (uintptr_t)(pChunk)->offStat + (uintptr_t)pChunk; NOREF(uPtr); \
77 AssertMsg(uPtr - (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) < (pHeap)->offPageAligned, \
78 ("%p - %p < %RX32\n", uPtr, (pHeap)->CTX_SUFF(pbHeap), (pHeap)->offPageAligned)); \
79 } \
80 } while (0)
81#else
82#define ASSERT_OFFSTAT(pHeap, pChunk) \
83 do { Assert(!(pChunk)->offStat); \
84 } while (0)
85#endif
86
87#define ASSERT_CHUNK(pHeap, pChunk) \
88 do { ASSERT_OFFNEXT(pHeap, pChunk); \
89 ASSERT_OFFPREV(pHeap, pChunk); \
90 ASSERT_OFFHEAP(pHeap, pChunk); \
91 ASSERT_OFFSTAT(pHeap, pChunk); \
92 } while (0)
93#define ASSERT_CHUNK_USED(pHeap, pChunk) \
94 do { ASSERT_OFFNEXT(pHeap, pChunk); \
95 ASSERT_OFFPREV(pHeap, pChunk); \
96 Assert(MMHYPERCHUNK_ISUSED(pChunk)); \
97 } while (0)
98
99#define ASSERT_FREE_OFFPREV(pHeap, pChunk) \
100 do { ASSERT_ALIGN((pChunk)->offPrev); \
101 ASSERT_GE(((pChunk)->offPrev & (MMHYPER_HEAP_ALIGN_MIN - 1)), (intptr_t)(pHeap)->CTX_SUFF(pbHeap) - (intptr_t)(pChunk)); \
102 Assert((pChunk)->offPrev != MMHYPERCHUNK_GET_OFFPREV(&(pChunk)->core) || !(pChunk)->offPrev); \
103 AssertMsg( (pChunk)->offPrev \
104 || (uintptr_t)(pChunk) - (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) == (pHeap)->offFreeHead, \
105 ("pChunk=%p offChunk=%#x offFreeHead=%#x\n", (pChunk), (uintptr_t)(pChunk) - (uintptr_t)(pHeap)->CTX_SUFF(pbHeap),\
106 (pHeap)->offFreeHead)); \
107 } while (0)
108
109#define ASSERT_FREE_OFFNEXT(pHeap, pChunk) \
110 do { ASSERT_ALIGN((pChunk)->offNext); \
111 ASSERT_L((pChunk)->offNext, (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) + (pHeap)->offPageAligned - (uintptr_t)(pChunk)); \
112 Assert((pChunk)->offNext != (pChunk)->core.offNext || !(pChunk)->offNext); \
113 AssertMsg( (pChunk)->offNext \
114 || (uintptr_t)(pChunk) - (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) == (pHeap)->offFreeTail, \
115 ("pChunk=%p offChunk=%#x offFreeTail=%#x\n", (pChunk), (uintptr_t)(pChunk) - (uintptr_t)(pHeap)->CTX_SUFF(pbHeap), \
116 (pHeap)->offFreeTail)); \
117 } while (0)
118
119#define ASSERT_FREE_CB(pHeap, pChunk) \
120 do { ASSERT_ALIGN((pChunk)->cb); \
121 Assert((pChunk)->cb > 0); \
122 if ((pChunk)->core.offNext) \
123 AssertMsg((pChunk)->cb == ((pChunk)->core.offNext - sizeof(MMHYPERCHUNK)), \
124 ("cb=%d offNext=%d\n", (pChunk)->cb, (pChunk)->core.offNext)); \
125 else \
126 ASSERT_LE((pChunk)->cb, (uintptr_t)(pHeap)->CTX_SUFF(pbHeap) + (pHeap)->offPageAligned - (uintptr_t)(pChunk)); \
127 } while (0)
128
129#define ASSERT_CHUNK_FREE(pHeap, pChunk) \
130 do { ASSERT_CHUNK(pHeap, &(pChunk)->core); \
131 Assert(MMHYPERCHUNK_ISFREE(pChunk)); \
132 ASSERT_FREE_OFFNEXT(pHeap, pChunk); \
133 ASSERT_FREE_OFFPREV(pHeap, pChunk); \
134 ASSERT_FREE_CB(pHeap, pChunk); \
135 } while (0)
136
137
138/*******************************************************************************
139* Internal Functions *
140*******************************************************************************/
141static PMMHYPERCHUNK mmHyperAllocChunk(PMMHYPERHEAP pHeap, uint32_t cb, unsigned uAlignment);
142static void *mmHyperAllocPages(PMMHYPERHEAP pHeap, uint32_t cb);
143#ifdef VBOX_WITH_STATISTICS
144static PMMHYPERSTAT mmHyperStat(PMMHYPERHEAP pHeap, MMTAG enmTag);
145#ifdef IN_RING3
146static void mmR3HyperStatRegisterOne(PVM pVM, PMMHYPERSTAT pStat);
147#endif
148#endif
149static int mmHyperFree(PMMHYPERHEAP pHeap, PMMHYPERCHUNK pChunk);
150#ifdef MMHYPER_HEAP_STRICT
151static void mmHyperHeapCheck(PMMHYPERHEAP pHeap);
152#endif
153
154/**
155 * Locks the hypervisor heap.
156 * This might call back to Ring-3 in order to deal with lock contention in GC and R3.
157 *
158 * @param pVM The VM handle.
159 */
160static int mmHyperLock(PVM pVM)
161{
162 PMMHYPERHEAP pHeap = pVM->mm.s.CTX_SUFF(pHyperHeap);
163
164#ifdef IN_RING3
165 if (!PDMCritSectIsInitialized(&pHeap->Lock))
166 return VINF_SUCCESS; /* early init */
167#else
168 Assert(PDMCritSectIsInitialized(&pHeap->Lock));
169#endif
170 int rc = PDMCritSectEnter(&pHeap->Lock, VERR_SEM_BUSY);
171#if defined(IN_RC) || defined(IN_RING0)
172 if (rc == VERR_SEM_BUSY)
173 rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_MMHYPER_LOCK, 0);
174#endif
175 AssertRC(rc);
176 return rc;
177}
178
179
180/**
181 * Unlocks the hypervisor heap.
182 *
183 * @param pVM The VM handle.
184 */
185static void mmHyperUnlock(PVM pVM)
186{
187 PMMHYPERHEAP pHeap = pVM->mm.s.CTX_SUFF(pHyperHeap);
188
189#ifdef IN_RING3
190 if (!PDMCritSectIsInitialized(&pHeap->Lock))
191 return; /* early init */
192#endif
193 Assert(PDMCritSectIsInitialized(&pHeap->Lock));
194 PDMCritSectLeave(&pHeap->Lock);
195}
196
197/**
198 * Allocates memory in the Hypervisor (RC VMM) area.
199 * The returned memory is of course zeroed.
200 *
201 * @returns VBox status code.
202 * @param pVM The VM to operate on.
203 * @param cb Number of bytes to allocate.
204 * @param uAlignment Required memory alignment in bytes.
205 * Values are 0,8,16,32 and PAGE_SIZE.
206 * 0 -> default alignment, i.e. 8 bytes.
207 * @param enmTag The statistics tag.
208 * @param ppv Where to store the address to the allocated
209 * memory.
210 */
211static int mmHyperAllocInternal(PVM pVM, size_t cb, unsigned uAlignment, MMTAG enmTag, void **ppv)
212{
213 AssertMsg(cb >= 8, ("Hey! Do you really mean to allocate less than 8 bytes?! cb=%d\n", cb));
214
215 /*
216 * Validate input and adjust it to reasonable values.
217 */
218 if (!uAlignment || uAlignment < MMHYPER_HEAP_ALIGN_MIN)
219 uAlignment = MMHYPER_HEAP_ALIGN_MIN;
220 uint32_t cbAligned;
221 switch (uAlignment)
222 {
223 case 8:
224 case 16:
225 case 32:
226 cbAligned = RT_ALIGN_32(cb, MMHYPER_HEAP_ALIGN_MIN);
227 if (!cbAligned || cbAligned < cb)
228 {
229 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VERR_INVALID_PARAMETER\n", cb, uAlignment));
230 AssertMsgFailed(("Nice try.\n"));
231 return VERR_INVALID_PARAMETER;
232 }
233 break;
234
235 case PAGE_SIZE:
236 AssertMsg(RT_ALIGN_32(cb, PAGE_SIZE) == cb, ("The size isn't page aligned. (cb=%#x)\n", cb));
237 cbAligned = RT_ALIGN_32(cb, PAGE_SIZE);
238 if (!cbAligned)
239 {
240 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VERR_INVALID_PARAMETER\n", cb, uAlignment));
241 AssertMsgFailed(("Nice try.\n"));
242 return VERR_INVALID_PARAMETER;
243 }
244 break;
245
246 default:
247 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VERR_INVALID_PARAMETER\n", cb, uAlignment));
248 AssertMsgFailed(("Invalid alignment %u\n", uAlignment));
249 return VERR_INVALID_PARAMETER;
250 }
251
252
253 /*
254 * Get heap and statisticsStatistics.
255 */
256 PMMHYPERHEAP pHeap = pVM->mm.s.CTX_SUFF(pHyperHeap);
257#ifdef VBOX_WITH_STATISTICS
258 PMMHYPERSTAT pStat = mmHyperStat(pHeap, enmTag);
259 if (!pStat)
260 {
261 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VERR_MM_HYPER_NO_MEMORY\n", cb, uAlignment));
262 AssertMsgFailed(("Failed to allocate statistics!\n"));
263 return VERR_MM_HYPER_NO_MEMORY;
264 }
265#endif
266 if (uAlignment < PAGE_SIZE)
267 {
268 /*
269 * Allocate a chunk.
270 */
271 PMMHYPERCHUNK pChunk = mmHyperAllocChunk(pHeap, cbAligned, uAlignment);
272 if (pChunk)
273 {
274#ifdef VBOX_WITH_STATISTICS
275 const uint32_t cbChunk = pChunk->offNext
276 ? pChunk->offNext
277 : pHeap->CTX_SUFF(pbHeap) + pHeap->offPageAligned - (uint8_t *)pChunk;
278 pStat->cbAllocated += (uint32_t)cbChunk;
279 pStat->cbCurAllocated += (uint32_t)cbChunk;
280 if (pStat->cbCurAllocated > pStat->cbMaxAllocated)
281 pStat->cbMaxAllocated = pStat->cbCurAllocated;
282 pStat->cAllocations++;
283 pChunk->offStat = (uintptr_t)pStat - (uintptr_t)pChunk;
284#else
285 pChunk->offStat = 0;
286#endif
287 void *pv = pChunk + 1;
288 *ppv = pv;
289 ASMMemZero32(pv, cbAligned);
290 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VINF_SUCCESS and *ppv=%p\n", cb, uAlignment, pv));
291 return VINF_SUCCESS;
292 }
293 }
294 else
295 {
296 /*
297 * Allocate page aligned memory.
298 */
299 void *pv = mmHyperAllocPages(pHeap, cbAligned);
300 if (pv)
301 {
302#ifdef VBOX_WITH_STATISTICS
303 pStat->cbAllocated += cbAligned;
304 pStat->cbCurAllocated += cbAligned;
305 if (pStat->cbCurAllocated > pStat->cbMaxAllocated)
306 pStat->cbMaxAllocated = pStat->cbCurAllocated;
307 pStat->cAllocations++;
308#endif
309 *ppv = pv;
310 /* ASMMemZero32(pv, cbAligned); - not required since memory is alloc-only and SUPR3PageAlloc zeros it. */
311 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VINF_SUCCESS and *ppv=%p\n", cb, uAlignment, ppv));
312 return VINF_SUCCESS;
313 }
314 }
315
316#ifdef VBOX_WITH_STATISTICS
317 pStat->cAllocations++;
318 pStat->cFailures++;
319#endif
320 Log2(("MMHyperAlloc: cb=%#x uAlignment=%#x returns VERR_MM_HYPER_NO_MEMORY\n", cb, uAlignment));
321 AssertMsgFailed(("Failed to allocate %d bytes!\n", cb));
322 return VERR_MM_HYPER_NO_MEMORY;
323}
324
325/**
326 * Wrapper for mmHyperAllocInternal
327 */
328VMMDECL(int) MMHyperAlloc(PVM pVM, size_t cb, unsigned uAlignment, MMTAG enmTag, void **ppv)
329{
330 int rc;
331
332 rc = mmHyperLock(pVM);
333 AssertRCReturn(rc, rc);
334
335 LogFlow(("MMHyperAlloc %x align=%x tag=%s\n", cb, uAlignment, mmGetTagName(enmTag)));
336
337 rc = mmHyperAllocInternal(pVM, cb, uAlignment, enmTag, ppv);
338
339 mmHyperUnlock(pVM);
340 return rc;
341}
342
343/**
344 * Allocates a chunk of memory from the specified heap.
345 * The caller validates the parameters of this request.
346 *
347 * @returns Pointer to the allocated chunk.
348 * @returns NULL on failure.
349 * @param pHeap The heap.
350 * @param cb Size of the memory block to allocate.
351 * @param uAlignment The alignment specifications for the allocated block.
352 * @internal
353 */
354static PMMHYPERCHUNK mmHyperAllocChunk(PMMHYPERHEAP pHeap, uint32_t cb, unsigned uAlignment)
355{
356 Log3(("mmHyperAllocChunk: Enter cb=%#x uAlignment=%#x\n", cb, uAlignment));
357#ifdef MMHYPER_HEAP_STRICT
358 mmHyperHeapCheck(pHeap);
359#endif
360#ifdef MMHYPER_HEAP_STRICT_FENCE
361 uint32_t cbFence = RT_MAX(MMHYPER_HEAP_STRICT_FENCE_SIZE, uAlignment);
362 cb += cbFence;
363#endif
364
365 /*
366 * Check if there are any free chunks. (NIL_OFFSET use/not-use forces this check)
367 */
368 if (pHeap->offFreeHead == NIL_OFFSET)
369 return NULL;
370
371 /*
372 * Small alignments - from the front of the heap.
373 *
374 * Must split off free chunks at the end to prevent messing up the
375 * last free node which we take the page aligned memory from the top of.
376 */
377 PMMHYPERCHUNK pRet = NULL;
378 PMMHYPERCHUNKFREE pFree = (PMMHYPERCHUNKFREE)((char *)pHeap->CTX_SUFF(pbHeap) + pHeap->offFreeHead);
379 while (pFree)
380 {
381 ASSERT_CHUNK_FREE(pHeap, pFree);
382 if (pFree->cb >= cb)
383 {
384 unsigned offAlign = (uintptr_t)(&pFree->core + 1) & (uAlignment - 1);
385 if (offAlign)
386 offAlign = uAlignment - offAlign;
387 if (!offAlign || pFree->cb - offAlign >= cb)
388 {
389 Log3(("mmHyperAllocChunk: Using pFree=%p pFree->cb=%d offAlign=%d\n", pFree, pFree->cb, offAlign));
390
391 /*
392 * Adjust the node in front.
393 * Because of multiple alignments we need to special case allocation of the first block.
394 */
395 if (offAlign)
396 {
397 MMHYPERCHUNKFREE Free = *pFree;
398 if (MMHYPERCHUNK_GET_OFFPREV(&pFree->core))
399 {
400 /* just add a bit of memory to it. */
401 PMMHYPERCHUNKFREE pPrev = (PMMHYPERCHUNKFREE)((char *)pFree + MMHYPERCHUNK_GET_OFFPREV(&Free.core));
402 pPrev->core.offNext += offAlign;
403 AssertMsg(!MMHYPERCHUNK_ISFREE(&pPrev->core), ("Impossible!\n"));
404 Log3(("mmHyperAllocChunk: Added %d bytes to %p\n", offAlign, pPrev));
405 }
406 else
407 {
408 /* make new head node, mark it USED for simplisity. */
409 PMMHYPERCHUNK pPrev = (PMMHYPERCHUNK)pHeap->CTX_SUFF(pbHeap);
410 Assert(pPrev == &pFree->core);
411 pPrev->offPrev = 0;
412 MMHYPERCHUNK_SET_TYPE(pPrev, MMHYPERCHUNK_FLAGS_USED);
413 pPrev->offNext = offAlign;
414 Log3(("mmHyperAllocChunk: Created new first node of %d bytes\n", offAlign));
415
416 }
417 Log3(("mmHyperAllocChunk: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree - offAlign, -(int)offAlign));
418 pHeap->cbFree -= offAlign;
419
420 /* Recreate pFree node and adjusting everything... */
421 pFree = (PMMHYPERCHUNKFREE)((char *)pFree + offAlign);
422 *pFree = Free;
423
424 pFree->cb -= offAlign;
425 if (pFree->core.offNext)
426 {
427 pFree->core.offNext -= offAlign;
428 PMMHYPERCHUNK pNext = (PMMHYPERCHUNK)((char *)pFree + pFree->core.offNext);
429 MMHYPERCHUNK_SET_OFFPREV(pNext, -(int32_t)pFree->core.offNext);
430 ASSERT_CHUNK(pHeap, pNext);
431 }
432 if (MMHYPERCHUNK_GET_OFFPREV(&pFree->core))
433 MMHYPERCHUNK_SET_OFFPREV(&pFree->core, MMHYPERCHUNK_GET_OFFPREV(&pFree->core) - offAlign);
434
435 if (pFree->offNext)
436 {
437 pFree->offNext -= offAlign;
438 PMMHYPERCHUNKFREE pNext = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offNext);
439 pNext->offPrev = -(int32_t)pFree->offNext;
440 ASSERT_CHUNK_FREE(pHeap, pNext);
441 }
442 else
443 pHeap->offFreeTail += offAlign;
444 if (pFree->offPrev)
445 {
446 pFree->offPrev -= offAlign;
447 PMMHYPERCHUNKFREE pPrev = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offPrev);
448 pPrev->offNext = -pFree->offPrev;
449 ASSERT_CHUNK_FREE(pHeap, pPrev);
450 }
451 else
452 pHeap->offFreeHead += offAlign;
453 pFree->core.offHeap = (uintptr_t)pHeap - (uintptr_t)pFree;
454 pFree->core.offStat = 0;
455 ASSERT_CHUNK_FREE(pHeap, pFree);
456 Log3(("mmHyperAllocChunk: Realigned pFree=%p\n", pFree));
457 }
458
459 /*
460 * Split off a new FREE chunk?
461 */
462 if (pFree->cb >= cb + RT_ALIGN(sizeof(MMHYPERCHUNKFREE), MMHYPER_HEAP_ALIGN_MIN))
463 {
464 /*
465 * Move the FREE chunk up to make room for the new USED chunk.
466 */
467 const int off = cb + sizeof(MMHYPERCHUNK);
468 PMMHYPERCHUNKFREE pNew = (PMMHYPERCHUNKFREE)((char *)&pFree->core + off);
469 *pNew = *pFree;
470 pNew->cb -= off;
471 if (pNew->core.offNext)
472 {
473 pNew->core.offNext -= off;
474 PMMHYPERCHUNK pNext = (PMMHYPERCHUNK)((char *)pNew + pNew->core.offNext);
475 MMHYPERCHUNK_SET_OFFPREV(pNext, -(int32_t)pNew->core.offNext);
476 ASSERT_CHUNK(pHeap, pNext);
477 }
478 pNew->core.offPrev = -off;
479 MMHYPERCHUNK_SET_TYPE(pNew, MMHYPERCHUNK_FLAGS_FREE);
480
481 if (pNew->offNext)
482 {
483 pNew->offNext -= off;
484 PMMHYPERCHUNKFREE pNext = (PMMHYPERCHUNKFREE)((char *)pNew + pNew->offNext);
485 pNext->offPrev = -(int32_t)pNew->offNext;
486 ASSERT_CHUNK_FREE(pHeap, pNext);
487 }
488 else
489 pHeap->offFreeTail += off;
490 if (pNew->offPrev)
491 {
492 pNew->offPrev -= off;
493 PMMHYPERCHUNKFREE pPrev = (PMMHYPERCHUNKFREE)((char *)pNew + pNew->offPrev);
494 pPrev->offNext = -pNew->offPrev;
495 ASSERT_CHUNK_FREE(pHeap, pPrev);
496 }
497 else
498 pHeap->offFreeHead += off;
499 pNew->core.offHeap = (uintptr_t)pHeap - (uintptr_t)pNew;
500 pNew->core.offStat = 0;
501 ASSERT_CHUNK_FREE(pHeap, pNew);
502
503 /*
504 * Update the old FREE node making it a USED node.
505 */
506 pFree->core.offNext = off;
507 MMHYPERCHUNK_SET_TYPE(&pFree->core, MMHYPERCHUNK_FLAGS_USED);
508
509
510 Log3(("mmHyperAllocChunk: cbFree %d -> %d (%d)\n", pHeap->cbFree,
511 pHeap->cbFree - (cb + sizeof(MMHYPERCHUNK)), -(int)(cb + sizeof(MMHYPERCHUNK))));
512 pHeap->cbFree -= (uint32_t)(cb + sizeof(MMHYPERCHUNK));
513 pRet = &pFree->core;
514 ASSERT_CHUNK(pHeap, &pFree->core);
515 Log3(("mmHyperAllocChunk: Created free chunk pNew=%p cb=%d\n", pNew, pNew->cb));
516 }
517 else
518 {
519 /*
520 * Link out of free list.
521 */
522 if (pFree->offNext)
523 {
524 PMMHYPERCHUNKFREE pNext = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offNext);
525 if (pFree->offPrev)
526 {
527 pNext->offPrev += pFree->offPrev;
528 PMMHYPERCHUNKFREE pPrev = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offPrev);
529 pPrev->offNext += pFree->offNext;
530 ASSERT_CHUNK_FREE(pHeap, pPrev);
531 }
532 else
533 {
534 pHeap->offFreeHead += pFree->offNext;
535 pNext->offPrev = 0;
536 }
537 ASSERT_CHUNK_FREE(pHeap, pNext);
538 }
539 else
540 {
541 if (pFree->offPrev)
542 {
543 pHeap->offFreeTail += pFree->offPrev;
544 PMMHYPERCHUNKFREE pPrev = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offPrev);
545 pPrev->offNext = 0;
546 ASSERT_CHUNK_FREE(pHeap, pPrev);
547 }
548 else
549 {
550 pHeap->offFreeHead = NIL_OFFSET;
551 pHeap->offFreeTail = NIL_OFFSET;
552 }
553 }
554
555 Log3(("mmHyperAllocChunk: cbFree %d -> %d (%d)\n", pHeap->cbFree,
556 pHeap->cbFree - pFree->cb, -(int32_t)pFree->cb));
557 pHeap->cbFree -= pFree->cb;
558 MMHYPERCHUNK_SET_TYPE(&pFree->core, MMHYPERCHUNK_FLAGS_USED);
559 pRet = &pFree->core;
560 ASSERT_CHUNK(pHeap, &pFree->core);
561 Log3(("mmHyperAllocChunk: Converted free chunk %p to used chunk.\n", pFree));
562 }
563 Log3(("mmHyperAllocChunk: Returning %p\n", pRet));
564 break;
565 }
566 }
567
568 /* next */
569 pFree = pFree->offNext ? (PMMHYPERCHUNKFREE)((char *)pFree + pFree->offNext) : NULL;
570 }
571
572#ifdef MMHYPER_HEAP_STRICT_FENCE
573 uint32_t *pu32End = (uint32_t *)((uint8_t *)(pRet + 1) + cb);
574 uint32_t *pu32EndReal = pRet->offNext
575 ? (uint32_t *)((uint8_t *)pRet + pRet->offNext)
576 : (uint32_t *)(pHeap->CTX_SUFF(pbHeap) + pHeap->cbHeap);
577 cbFence += (uintptr_t)pu32EndReal - (uintptr_t)pu32End; Assert(!(cbFence & 0x3));
578 ASMMemFill32((uint8_t *)pu32EndReal - cbFence, cbFence, MMHYPER_HEAP_STRICT_FENCE_U32);
579 pu32EndReal[-1] = cbFence;
580#endif
581#ifdef MMHYPER_HEAP_STRICT
582 mmHyperHeapCheck(pHeap);
583#endif
584 return pRet;
585}
586
587
588/**
589 * Allocates one or more pages of memory from the specified heap.
590 * The caller validates the parameters of this request.
591 *
592 * @returns Pointer to the allocated chunk.
593 * @returns NULL on failure.
594 * @param pHeap The heap.
595 * @param cb Size of the memory block to allocate.
596 * @internal
597 */
598static void *mmHyperAllocPages(PMMHYPERHEAP pHeap, uint32_t cb)
599{
600 Log3(("mmHyperAllocPages: Enter cb=%#x\n", cb));
601
602#ifdef MMHYPER_HEAP_STRICT
603 mmHyperHeapCheck(pHeap);
604#endif
605
606 /*
607 * Check if there are any free chunks. (NIL_OFFSET use/not-use forces this check)
608 */
609 if (pHeap->offFreeHead == NIL_OFFSET)
610 return NULL;
611
612 /*
613 * Page aligned chunks.
614 *
615 * Page aligned chunks can only be allocated from the last FREE chunk.
616 * This is for reasons of simplicity and fragmentation. Page aligned memory
617 * must also be allocated in page aligned sizes. Page aligned memory cannot
618 * be freed either.
619 *
620 * So, for this to work, the last FREE chunk needs to end on a page aligned
621 * boundrary.
622 */
623 PMMHYPERCHUNKFREE pFree = (PMMHYPERCHUNKFREE)((char *)pHeap->CTX_SUFF(pbHeap) + pHeap->offFreeTail);
624 ASSERT_CHUNK_FREE(pHeap, pFree);
625 if ( (((uintptr_t)(&pFree->core + 1) + pFree->cb) & (PAGE_OFFSET_MASK - 1))
626 || pFree->cb + sizeof(MMHYPERCHUNK) < cb)
627 {
628 Log3(("mmHyperAllocPages: Not enough/no page aligned memory!\n"));
629 return NULL;
630 }
631
632 void *pvRet;
633 if (pFree->cb > cb)
634 {
635 /*
636 * Simple, just cut the top of the free node and return it.
637 */
638 pFree->cb -= cb;
639 pvRet = (char *)(&pFree->core + 1) + pFree->cb;
640 AssertMsg(RT_ALIGN_P(pvRet, PAGE_SIZE) == pvRet, ("pvRet=%p cb=%#x pFree=%p pFree->cb=%#x\n", pvRet, cb, pFree, pFree->cb));
641 Log3(("mmHyperAllocPages: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree - cb, -(int)cb));
642 pHeap->cbFree -= cb;
643 ASSERT_CHUNK_FREE(pHeap, pFree);
644 Log3(("mmHyperAllocPages: Allocated from pFree=%p new pFree->cb=%d\n", pFree, pFree->cb));
645 }
646 else
647 {
648 /*
649 * Unlink the FREE node.
650 */
651 pvRet = (char *)(&pFree->core + 1) + pFree->cb - cb;
652 Log3(("mmHyperAllocPages: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree - pFree->cb, -(int32_t)pFree->cb));
653 pHeap->cbFree -= pFree->cb;
654
655 /* a scrap of spare memory (unlikely)? add it to the sprevious chunk. */
656 if (pvRet != (void *)pFree)
657 {
658 AssertMsg(MMHYPERCHUNK_GET_OFFPREV(&pFree->core), ("How the *beep* did someone manage to allocated up all the heap with page aligned memory?!?\n"));
659 PMMHYPERCHUNK pPrev = (PMMHYPERCHUNK)((char *)pFree + MMHYPERCHUNK_GET_OFFPREV(&pFree->core));
660 pPrev->offNext += (uintptr_t)pvRet - (uintptr_t)pFree;
661 AssertMsg(!MMHYPERCHUNK_ISFREE(pPrev), ("Free bug?\n"));
662#ifdef VBOX_WITH_STATISTICS
663 PMMHYPERSTAT pStat = (PMMHYPERSTAT)((uintptr_t)pPrev + pPrev->offStat);
664 pStat->cbAllocated += (uintptr_t)pvRet - (uintptr_t)pFree;
665 pStat->cbCurAllocated += (uintptr_t)pvRet - (uintptr_t)pFree;
666#endif
667 Log3(("mmHyperAllocPages: Added %d to %p (page align)\n", (uintptr_t)pvRet - (uintptr_t)pFree, pFree));
668 }
669
670 /* unlink from FREE chain. */
671 if (pFree->offPrev)
672 {
673 pHeap->offFreeTail += pFree->offPrev;
674 ((PMMHYPERCHUNKFREE)((char *)pFree + pFree->offPrev))->offNext = 0;
675 }
676 else
677 {
678 pHeap->offFreeTail = NIL_OFFSET;
679 pHeap->offFreeHead = NIL_OFFSET;
680 }
681 Log3(("mmHyperAllocPages: Unlinked pFree=%d\n", pFree));
682 }
683 pHeap->offPageAligned = (uintptr_t)pvRet - (uintptr_t)pHeap->CTX_SUFF(pbHeap);
684 Log3(("mmHyperAllocPages: Returning %p (page aligned)\n", pvRet));
685
686#ifdef MMHYPER_HEAP_STRICT
687 mmHyperHeapCheck(pHeap);
688#endif
689 return pvRet;
690}
691
692#ifdef VBOX_WITH_STATISTICS
693
694/**
695 * Get the statistic record for a tag.
696 *
697 * @returns Pointer to a stat record.
698 * @returns NULL on failure.
699 * @param pHeap The heap.
700 * @param enmTag The tag.
701 */
702static PMMHYPERSTAT mmHyperStat(PMMHYPERHEAP pHeap, MMTAG enmTag)
703{
704 /* try look it up first. */
705 PMMHYPERSTAT pStat = (PMMHYPERSTAT)RTAvloGCPhysGet(&pHeap->HyperHeapStatTree, enmTag);
706 if (!pStat)
707 {
708 /* try allocate a new one */
709 PMMHYPERCHUNK pChunk = mmHyperAllocChunk(pHeap, RT_ALIGN(sizeof(*pStat), MMHYPER_HEAP_ALIGN_MIN), MMHYPER_HEAP_ALIGN_MIN);
710 if (!pChunk)
711 return NULL;
712 pStat = (PMMHYPERSTAT)(pChunk + 1);
713 pChunk->offStat = (uintptr_t)pStat - (uintptr_t)pChunk;
714
715 ASMMemZero32(pStat, sizeof(*pStat));
716 pStat->Core.Key = enmTag;
717 RTAvloGCPhysInsert(&pHeap->HyperHeapStatTree, &pStat->Core);
718 }
719 if (!pStat->fRegistered)
720 {
721# ifdef IN_RING3
722 mmR3HyperStatRegisterOne(pHeap->pVMR3, pStat);
723# else
724 /** @todo schedule a R3 action. */
725# endif
726 }
727 return pStat;
728}
729
730
731# ifdef IN_RING3
732/**
733 * Registers statistics with STAM.
734 *
735 */
736static void mmR3HyperStatRegisterOne(PVM pVM, PMMHYPERSTAT pStat)
737{
738 if (pStat->fRegistered)
739 return;
740 const char *pszTag = mmGetTagName((MMTAG)pStat->Core.Key);
741 STAMR3RegisterF(pVM, &pStat->cbCurAllocated, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Number of bytes currently allocated.", "/MM/HyperHeap/%s", pszTag);
742 STAMR3RegisterF(pVM, &pStat->cAllocations, STAMTYPE_U64, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of alloc calls.", "/MM/HyperHeap/%s/cAllocations", pszTag);
743 STAMR3RegisterF(pVM, &pStat->cFrees, STAMTYPE_U64, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of free calls.", "/MM/HyperHeap/%s/cFrees", pszTag);
744 STAMR3RegisterF(pVM, &pStat->cFailures, STAMTYPE_U64, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of failures.", "/MM/HyperHeap/%s/cFailures", pszTag);
745 STAMR3RegisterF(pVM, &pStat->cbAllocated, STAMTYPE_U64, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Total number of allocated bytes.", "/MM/HyperHeap/%s/cbAllocated", pszTag);
746 STAMR3RegisterF(pVM, &pStat->cbFreed, STAMTYPE_U64, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Total number of freed bytes.", "/MM/HyperHeap/%s/cbFreed", pszTag);
747 STAMR3RegisterF(pVM, &pStat->cbMaxAllocated, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Max number of bytes allocated at the same time.","/MM/HyperHeap/%s/cbMaxAllocated", pszTag);
748 pStat->fRegistered = true;
749}
750# endif /* IN_RING3 */
751
752#endif /* VBOX_WITH_STATISTICS */
753
754
755/**
756 * Free memory allocated using MMHyperAlloc().
757 * The caller validates the parameters of this request.
758 *
759 * @returns VBox status code.
760 * @param pVM The VM to operate on.
761 * @param pv The memory to free.
762 * @remark Try avoid free hyper memory.
763 */
764static int mmHyperFreeInternal(PVM pVM, void *pv)
765{
766 Log2(("MMHyperFree: pv=%p\n", pv));
767 if (!pv)
768 return VINF_SUCCESS;
769 AssertMsgReturn(RT_ALIGN_P(pv, MMHYPER_HEAP_ALIGN_MIN) == pv,
770 ("Invalid pointer %p!\n", pv),
771 VERR_INVALID_POINTER);
772
773 /*
774 * Get the heap and stats.
775 * Validate the chunk at the same time.
776 */
777 PMMHYPERCHUNK pChunk = (PMMHYPERCHUNK)((PMMHYPERCHUNK)pv - 1);
778
779 AssertMsgReturn( (uintptr_t)pChunk + pChunk->offNext >= (uintptr_t)pChunk
780 || RT_ALIGN_32(pChunk->offNext, MMHYPER_HEAP_ALIGN_MIN) != pChunk->offNext,
781 ("%p: offNext=%#RX32\n", pv, pChunk->offNext),
782 VERR_INVALID_POINTER);
783
784 AssertMsgReturn(MMHYPERCHUNK_ISUSED(pChunk),
785 ("%p: Not used!\n", pv),
786 VERR_INVALID_POINTER);
787
788 int32_t offPrev = MMHYPERCHUNK_GET_OFFPREV(pChunk);
789 AssertMsgReturn( (uintptr_t)pChunk + offPrev <= (uintptr_t)pChunk
790 && !((uint32_t)-offPrev & (MMHYPER_HEAP_ALIGN_MIN - 1)),
791 ("%p: offPrev=%#RX32!\n", pv, offPrev),
792 VERR_INVALID_POINTER);
793
794 /* statistics */
795#ifdef VBOX_WITH_STATISTICS
796 PMMHYPERSTAT pStat = (PMMHYPERSTAT)((uintptr_t)pChunk + pChunk->offStat);
797 AssertMsgReturn( RT_ALIGN_P(pStat, MMHYPER_HEAP_ALIGN_MIN) == (void *)pStat
798 && pChunk->offStat,
799 ("%p: offStat=%#RX32!\n", pv, pChunk->offStat),
800 VERR_INVALID_POINTER);
801#else
802 AssertMsgReturn(!pChunk->offStat,
803 ("%p: offStat=%#RX32!\n", pv, pChunk->offStat),
804 VERR_INVALID_POINTER);
805#endif
806
807 /* The heap structure. */
808 PMMHYPERHEAP pHeap = (PMMHYPERHEAP)((uintptr_t)pChunk + pChunk->offHeap);
809 AssertMsgReturn( !((uintptr_t)pHeap & PAGE_OFFSET_MASK)
810 && pChunk->offHeap,
811 ("%p: pHeap=%#x offHeap=%RX32\n", pv, pHeap->u32Magic, pChunk->offHeap),
812 VERR_INVALID_POINTER);
813
814 AssertMsgReturn(pHeap->u32Magic == MMHYPERHEAP_MAGIC,
815 ("%p: u32Magic=%#x\n", pv, pHeap->u32Magic),
816 VERR_INVALID_POINTER);
817 Assert(pHeap == pVM->mm.s.CTX_SUFF(pHyperHeap));
818
819 /* Some more verifications using additional info from pHeap. */
820 AssertMsgReturn((uintptr_t)pChunk + offPrev >= (uintptr_t)pHeap->CTX_SUFF(pbHeap),
821 ("%p: offPrev=%#RX32!\n", pv, offPrev),
822 VERR_INVALID_POINTER);
823
824 AssertMsgReturn(pChunk->offNext < pHeap->cbHeap,
825 ("%p: offNext=%#RX32!\n", pv, pChunk->offNext),
826 VERR_INVALID_POINTER);
827
828 AssertMsgReturn( (uintptr_t)pv - (uintptr_t)pHeap->CTX_SUFF(pbHeap) <= pHeap->offPageAligned,
829 ("Invalid pointer %p! (heap: %p-%p)\n", pv, pHeap->CTX_SUFF(pbHeap),
830 (char *)pHeap->CTX_SUFF(pbHeap) + pHeap->offPageAligned),
831 VERR_INVALID_POINTER);
832
833#ifdef MMHYPER_HEAP_STRICT
834 mmHyperHeapCheck(pHeap);
835#endif
836
837#if defined(VBOX_WITH_STATISTICS) || defined(MMHYPER_HEAP_FREE_POISON)
838 /* calc block size. */
839 const uint32_t cbChunk = pChunk->offNext
840 ? pChunk->offNext
841 : pHeap->CTX_SUFF(pbHeap) + pHeap->offPageAligned - (uint8_t *)pChunk;
842#endif
843#ifdef MMHYPER_HEAP_FREE_POISON
844 /* poison the block */
845 memset(pChunk + 1, MMHYPER_HEAP_FREE_POISON, cbChunk - sizeof(*pChunk));
846#endif
847
848#ifdef MMHYPER_HEAP_FREE_DELAY
849# ifdef MMHYPER_HEAP_FREE_POISON
850 /*
851 * Check poison.
852 */
853 unsigned i = RT_ELEMENTS(pHeap->aDelayedFrees);
854 while (i-- > 0)
855 if (pHeap->aDelayedFrees[i].offChunk)
856 {
857 PMMHYPERCHUNK pCur = (PMMHYPERCHUNK)((uintptr_t)pHeap + pHeap->aDelayedFrees[i].offChunk);
858 const size_t cb = pCur->offNext
859 ? pCur->offNext - sizeof(*pCur)
860 : pHeap->CTX_SUFF(pbHeap) + pHeap->offPageAligned - (uint8_t *)pCur - sizeof(*pCur);
861 uint8_t *pab = (uint8_t *)(pCur + 1);
862 for (unsigned off = 0; off < cb; off++)
863 AssertReleaseMsg(pab[off] == 0xCB,
864 ("caller=%RTptr cb=%#zx off=%#x: %.*Rhxs\n",
865 pHeap->aDelayedFrees[i].uCaller, cb, off, RT_MIN(cb - off, 32), &pab[off]));
866 }
867# endif /* MMHYPER_HEAP_FREE_POISON */
868
869 /*
870 * Delayed freeing.
871 */
872 int rc = VINF_SUCCESS;
873 if (pHeap->aDelayedFrees[pHeap->iDelayedFree].offChunk)
874 {
875 PMMHYPERCHUNK pChunkFree = (PMMHYPERCHUNK)((uintptr_t)pHeap + pHeap->aDelayedFrees[pHeap->iDelayedFree].offChunk);
876 rc = mmHyperFree(pHeap, pChunkFree);
877 }
878 pHeap->aDelayedFrees[pHeap->iDelayedFree].offChunk = (uintptr_t)pChunk - (uintptr_t)pHeap;
879 pHeap->aDelayedFrees[pHeap->iDelayedFree].uCaller = (uintptr_t)ASMReturnAddress();
880 pHeap->iDelayedFree = (pHeap->iDelayedFree + 1) % RT_ELEMENTS(pHeap->aDelayedFrees);
881
882#else /* !MMHYPER_HEAP_FREE_POISON */
883 /*
884 * Call the worker.
885 */
886 int rc = mmHyperFree(pHeap, pChunk);
887#endif /* !MMHYPER_HEAP_FREE_POISON */
888
889 /*
890 * Update statistics.
891 */
892#ifdef VBOX_WITH_STATISTICS
893 pStat->cFrees++;
894 if (RT_SUCCESS(rc))
895 {
896 pStat->cbFreed += cbChunk;
897 pStat->cbCurAllocated -= cbChunk;
898 }
899 else
900 pStat->cFailures++;
901#endif
902
903 return rc;
904}
905
906
907/**
908 * Wrapper for mmHyperFreeInternal
909 */
910VMMDECL(int) MMHyperFree(PVM pVM, void *pv)
911{
912 int rc;
913
914 rc = mmHyperLock(pVM);
915 AssertRCReturn(rc, rc);
916
917 LogFlow(("MMHyperFree %p\n", pv));
918
919 rc = mmHyperFreeInternal(pVM, pv);
920
921 mmHyperUnlock(pVM);
922 return rc;
923}
924
925
926/**
927 * Free memory a memory chunk.
928 *
929 * @returns VBox status code.
930 * @param pHeap The heap.
931 * @param pChunk The memory chunk to free.
932 */
933static int mmHyperFree(PMMHYPERHEAP pHeap, PMMHYPERCHUNK pChunk)
934{
935 Log3(("mmHyperFree: Enter pHeap=%p pChunk=%p\n", pHeap, pChunk));
936 PMMHYPERCHUNKFREE pFree = (PMMHYPERCHUNKFREE)pChunk;
937
938 /*
939 * Insert into the free list (which is sorted on address).
940 *
941 * We'll search towards the end of the heap to locate the
942 * closest FREE chunk.
943 */
944 PMMHYPERCHUNKFREE pLeft = NULL;
945 PMMHYPERCHUNKFREE pRight = NULL;
946 if (pHeap->offFreeTail != NIL_OFFSET)
947 {
948 if (pFree->core.offNext)
949 {
950 pRight = (PMMHYPERCHUNKFREE)((char *)pFree + pFree->core.offNext);
951 ASSERT_CHUNK(pHeap, &pRight->core);
952 while (!MMHYPERCHUNK_ISFREE(&pRight->core))
953 {
954 if (!pRight->core.offNext)
955 {
956 pRight = NULL;
957 break;
958 }
959 pRight = (PMMHYPERCHUNKFREE)((char *)pRight + pRight->core.offNext);
960 ASSERT_CHUNK(pHeap, &pRight->core);
961 }
962 }
963 if (!pRight)
964 pRight = (PMMHYPERCHUNKFREE)((char *)pHeap->CTX_SUFF(pbHeap) + pHeap->offFreeTail); /** @todo this can't be correct! 'pLeft = .. ; else' I think */
965 if (pRight)
966 {
967 ASSERT_CHUNK_FREE(pHeap, pRight);
968 if (pRight->offPrev)
969 {
970 pLeft = (PMMHYPERCHUNKFREE)((char *)pRight + pRight->offPrev);
971 ASSERT_CHUNK_FREE(pHeap, pLeft);
972 }
973 }
974 }
975 if (pLeft == pFree)
976 {
977 AssertMsgFailed(("Freed twice! pv=%p (pChunk=%p)\n", pChunk + 1, pChunk));
978 return VERR_INVALID_POINTER;
979 }
980 pChunk->offStat = 0;
981
982 /*
983 * Head free chunk list?
984 */
985 if (!pLeft)
986 {
987 MMHYPERCHUNK_SET_TYPE(&pFree->core, MMHYPERCHUNK_FLAGS_FREE);
988 pFree->offPrev = 0;
989 pHeap->offFreeHead = (uintptr_t)pFree - (uintptr_t)pHeap->CTX_SUFF(pbHeap);
990 if (pRight)
991 {
992 pFree->offNext = (uintptr_t)pRight - (uintptr_t)pFree;
993 pRight->offPrev = -(int32_t)pFree->offNext;
994 }
995 else
996 {
997 pFree->offNext = 0;
998 pHeap->offFreeTail = pHeap->offFreeHead;
999 }
1000 Log3(("mmHyperFree: Inserted %p at head of free chain.\n", pFree));
1001 }
1002 else
1003 {
1004 /*
1005 * Can we merge with left hand free chunk?
1006 */
1007 if ((char *)pLeft + pLeft->core.offNext == (char *)pFree)
1008 {
1009 if (pFree->core.offNext)
1010 {
1011 pLeft->core.offNext = pLeft->core.offNext + pFree->core.offNext;
1012 MMHYPERCHUNK_SET_OFFPREV(((PMMHYPERCHUNK)((char *)pLeft + pLeft->core.offNext)), -(int32_t)pLeft->core.offNext);
1013 }
1014 else
1015 pLeft->core.offNext = 0;
1016 pFree = pLeft;
1017 Log3(("mmHyperFree: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree - pLeft->cb, -(int32_t)pLeft->cb));
1018 pHeap->cbFree -= pLeft->cb;
1019 Log3(("mmHyperFree: Merging %p into %p (cb=%d).\n", pFree, pLeft, pLeft->cb));
1020 }
1021 /*
1022 * No, just link it into the free list then.
1023 */
1024 else
1025 {
1026 MMHYPERCHUNK_SET_TYPE(&pFree->core, MMHYPERCHUNK_FLAGS_FREE);
1027 pFree->offPrev = (uintptr_t)pLeft - (uintptr_t)pFree;
1028 pLeft->offNext = -pFree->offPrev;
1029 if (pRight)
1030 {
1031 pFree->offNext = (uintptr_t)pRight - (uintptr_t)pFree;
1032 pRight->offPrev = -(int32_t)pFree->offNext;
1033 }
1034 else
1035 {
1036 pFree->offNext = 0;
1037 pHeap->offFreeTail = (uintptr_t)pFree - (uintptr_t)pHeap->CTX_SUFF(pbHeap);
1038 }
1039 Log3(("mmHyperFree: Inserted %p after %p in free list.\n", pFree, pLeft));
1040 }
1041 }
1042
1043 /*
1044 * Can we merge with right hand free chunk?
1045 */
1046 if (pRight && (char *)pRight == (char *)pFree + pFree->core.offNext)
1047 {
1048 /* core */
1049 if (pRight->core.offNext)
1050 {
1051 pFree->core.offNext += pRight->core.offNext;
1052 PMMHYPERCHUNK pNext = (PMMHYPERCHUNK)((char *)pFree + pFree->core.offNext);
1053 MMHYPERCHUNK_SET_OFFPREV(pNext, -(int32_t)pFree->core.offNext);
1054 ASSERT_CHUNK(pHeap, pNext);
1055 }
1056 else
1057 pFree->core.offNext = 0;
1058
1059 /* free */
1060 if (pRight->offNext)
1061 {
1062 pFree->offNext += pRight->offNext;
1063 ((PMMHYPERCHUNKFREE)((char *)pFree + pFree->offNext))->offPrev = -(int32_t)pFree->offNext;
1064 }
1065 else
1066 {
1067 pFree->offNext = 0;
1068 pHeap->offFreeTail = (uintptr_t)pFree - (uintptr_t)pHeap->CTX_SUFF(pbHeap);
1069 }
1070 Log3(("mmHyperFree: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree - pRight->cb, -(int32_t)pRight->cb));
1071 pHeap->cbFree -= pRight->cb;
1072 Log3(("mmHyperFree: Merged %p (cb=%d) into %p.\n", pRight, pRight->cb, pFree));
1073 }
1074
1075 /* calculate the size. */
1076 if (pFree->core.offNext)
1077 pFree->cb = pFree->core.offNext - sizeof(MMHYPERCHUNK);
1078 else
1079 pFree->cb = pHeap->offPageAligned - ((uintptr_t)pFree - (uintptr_t)pHeap->CTX_SUFF(pbHeap)) - sizeof(MMHYPERCHUNK);
1080 Log3(("mmHyperFree: cbFree %d -> %d (%d)\n", pHeap->cbFree, pHeap->cbFree + pFree->cb, pFree->cb));
1081 pHeap->cbFree += pFree->cb;
1082 ASSERT_CHUNK_FREE(pHeap, pFree);
1083
1084#ifdef MMHYPER_HEAP_STRICT
1085 mmHyperHeapCheck(pHeap);
1086#endif
1087 return VINF_SUCCESS;
1088}
1089
1090
1091#if defined(DEBUG) || defined(MMHYPER_HEAP_STRICT)
1092/**
1093 * Dumps a heap chunk to the log.
1094 *
1095 * @param pHeap Pointer to the heap.
1096 * @param pCur Pointer to the chunk.
1097 */
1098static void mmHyperHeapDumpOne(PMMHYPERHEAP pHeap, PMMHYPERCHUNKFREE pCur)
1099{
1100 if (MMHYPERCHUNK_ISUSED(&pCur->core))
1101 {
1102 if (pCur->core.offStat)
1103 {
1104 PMMHYPERSTAT pStat = (PMMHYPERSTAT)((uintptr_t)pCur + pCur->core.offStat);
1105 const char *pszSelf = pCur->core.offStat == sizeof(MMHYPERCHUNK) ? " stat record" : "";
1106#ifdef IN_RING3
1107 Log(("%p %06x USED offNext=%06x offPrev=-%06x %s%s\n",
1108 pCur, (uintptr_t)pCur - (uintptr_t)pHeap->CTX_SUFF(pbHeap),
1109 pCur->core.offNext, -MMHYPERCHUNK_GET_OFFPREV(&pCur->core),
1110 mmGetTagName((MMTAG)pStat->Core.Key), pszSelf));
1111#else
1112 Log(("%p %06x USED offNext=%06x offPrev=-%06x %d%s\n",
1113 pCur, (uintptr_t)pCur - (uintptr_t)pHeap->CTX_SUFF(pbHeap),
1114 pCur->core.offNext, -MMHYPERCHUNK_GET_OFFPREV(&pCur->core),
1115 (MMTAG)pStat->Core.Key, pszSelf));
1116#endif
1117 }
1118 else
1119 Log(("%p %06x USED offNext=%06x offPrev=-%06x\n",
1120 pCur, (uintptr_t)pCur - (uintptr_t)pHeap->CTX_SUFF(pbHeap),
1121 pCur->core.offNext, -MMHYPERCHUNK_GET_OFFPREV(&pCur->core)));
1122 }
1123 else
1124 Log(("%p %06x FREE offNext=%06x offPrev=-%06x : cb=%06x offNext=%06x offPrev=-%06x\n",
1125 pCur, (uintptr_t)pCur - (uintptr_t)pHeap->CTX_SUFF(pbHeap),
1126 pCur->core.offNext, -MMHYPERCHUNK_GET_OFFPREV(&pCur->core), pCur->cb, pCur->offNext, pCur->offPrev));
1127}
1128#endif /* DEBUG || MMHYPER_HEAP_STRICT */
1129
1130
1131#ifdef MMHYPER_HEAP_STRICT
1132/**
1133 * Internal consitency check.
1134 */
1135static void mmHyperHeapCheck(PMMHYPERHEAP pHeap)
1136{
1137 PMMHYPERCHUNKFREE pPrev = NULL;
1138 PMMHYPERCHUNKFREE pCur = (PMMHYPERCHUNKFREE)pHeap->CTX_SUFF(pbHeap);
1139 for (;;)
1140 {
1141 if (MMHYPERCHUNK_ISUSED(&pCur->core))
1142 ASSERT_CHUNK_USED(pHeap, &pCur->core);
1143 else
1144 ASSERT_CHUNK_FREE(pHeap, pCur);
1145 if (pPrev)
1146 AssertMsg((int32_t)pPrev->core.offNext == -MMHYPERCHUNK_GET_OFFPREV(&pCur->core),
1147 ("pPrev->core.offNext=%d offPrev=%d\n", pPrev->core.offNext, MMHYPERCHUNK_GET_OFFPREV(&pCur->core)));
1148
1149# ifdef MMHYPER_HEAP_STRICT_FENCE
1150 uint32_t off = (uint8_t *)pCur - pHeap->CTX_SUFF(pbHeap);
1151 if ( MMHYPERCHUNK_ISUSED(&pCur->core)
1152 && off < pHeap->offPageAligned)
1153 {
1154 uint32_t cbCur = pCur->core.offNext
1155 ? pCur->core.offNext
1156 : pHeap->cbHeap - off;
1157 uint32_t *pu32End = ((uint32_t *)((uint8_t *)pCur + cbCur));
1158 uint32_t cbFence = pu32End[-1];
1159 if (RT_UNLIKELY( cbFence >= cbCur - sizeof(*pCur)
1160 || cbFence < MMHYPER_HEAP_STRICT_FENCE_SIZE))
1161 {
1162 mmHyperHeapDumpOne(pHeap, pCur);
1163 Assert(cbFence < cbCur - sizeof(*pCur));
1164 Assert(cbFence >= MMHYPER_HEAP_STRICT_FENCE_SIZE);
1165 }
1166
1167 uint32_t *pu32Bad = ASMMemIsAllU32((uint8_t *)pu32End - cbFence, cbFence - sizeof(uint32_t), MMHYPER_HEAP_STRICT_FENCE_U32);
1168 if (RT_UNLIKELY(pu32Bad))
1169 {
1170 mmHyperHeapDumpOne(pHeap, pCur);
1171 Assert(!pu32Bad);
1172 }
1173 }
1174# endif
1175
1176 /* next */
1177 if (!pCur->core.offNext)
1178 break;
1179 pPrev = pCur;
1180 pCur = (PMMHYPERCHUNKFREE)((char *)pCur + pCur->core.offNext);
1181 }
1182}
1183#endif
1184
1185
1186/**
1187 * Performs consistency checks on the heap if MMHYPER_HEAP_STRICT was
1188 * defined at build time.
1189 *
1190 * @param pVM Pointer to the shared VM structure.
1191 */
1192VMMDECL(void) MMHyperHeapCheck(PVM pVM)
1193{
1194#ifdef MMHYPER_HEAP_STRICT
1195 int rc;
1196
1197 rc = mmHyperLock(pVM);
1198 AssertRC(rc);
1199 mmHyperHeapCheck(pVM->mm.s.CTX_SUFF(pHyperHeap));
1200 mmHyperUnlock(pVM);
1201#endif
1202}
1203
1204
1205#ifdef DEBUG
1206/**
1207 * Dumps the hypervisor heap to Log.
1208 * @param pVM VM Handle.
1209 */
1210VMMDECL(void) MMHyperHeapDump(PVM pVM)
1211{
1212 Log(("MMHyperHeapDump: *** heap dump - start ***\n"));
1213 PMMHYPERHEAP pHeap = pVM->mm.s.CTX_SUFF(pHyperHeap);
1214 PMMHYPERCHUNKFREE pCur = (PMMHYPERCHUNKFREE)pHeap->CTX_SUFF(pbHeap);
1215 for (;;)
1216 {
1217 mmHyperHeapDumpOne(pHeap, pCur);
1218
1219 /* next */
1220 if (!pCur->core.offNext)
1221 break;
1222 pCur = (PMMHYPERCHUNKFREE)((char *)pCur + pCur->core.offNext);
1223 }
1224 Log(("MMHyperHeapDump: *** heap dump - end ***\n"));
1225}
1226#endif
1227
1228
1229/**
1230 * Query the amount of free memory in the hypervisor heap.
1231 *
1232 * @returns Number of free bytes in the hypervisor heap.
1233 */
1234VMMDECL(size_t) MMHyperHeapGetFreeSize(PVM pVM)
1235{
1236 return pVM->mm.s.CTX_SUFF(pHyperHeap)->cbFree;
1237}
1238
1239/**
1240 * Query the size the hypervisor heap.
1241 *
1242 * @returns The size of the hypervisor heap in bytes.
1243 */
1244VMMDECL(size_t) MMHyperHeapGetSize(PVM pVM)
1245{
1246 return pVM->mm.s.CTX_SUFF(pHyperHeap)->cbHeap;
1247}
1248
1249
1250/**
1251 * Query the address and size the hypervisor memory area.
1252 *
1253 * @returns Base address of the hypervisor area.
1254 * @param pVM VM Handle.
1255 * @param pcb Where to store the size of the hypervisor area. (out)
1256 */
1257VMMDECL(RTGCPTR) MMHyperGetArea(PVM pVM, size_t *pcb)
1258{
1259 if (pcb)
1260 *pcb = pVM->mm.s.cbHyperArea;
1261 return pVM->mm.s.pvHyperAreaGC;
1262}
1263
1264
1265/**
1266 * Checks if an address is within the hypervisor memory area.
1267 *
1268 * @returns true if inside.
1269 * @returns false if outside.
1270 * @param pVM VM handle.
1271 * @param GCPtr The pointer to check.
1272 */
1273VMMDECL(bool) MMHyperIsInsideArea(PVM pVM, RTGCPTR GCPtr)
1274{
1275 return (RTGCUINTPTR)GCPtr - (RTGCUINTPTR)pVM->mm.s.pvHyperAreaGC < pVM->mm.s.cbHyperArea;
1276}
1277
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