VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/alloc-ef-r0drv.cpp@ 107378

Last change on this file since 107378 was 106433, checked in by vboxsync, 3 months ago

Runtime/r0drv: Work on getting it building on win.arm64 (the easy bits), bugref:10734

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 31.4 KB
Line 
1/* $Id: alloc-ef-r0drv.cpp 106433 2024-10-17 11:37:24Z vboxsync $ */
2/** @file
3 * IPRT - Memory Allocation, electric fence for ring-0 drivers.
4 */
5
6/*
7 * Copyright (C) 2006-2024 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#define RTMEM_NO_WRAP_TO_EF_APIS
42#include "internal/iprt.h"
43#include <iprt/mem.h>
44
45#include <iprt/alloc.h>
46#include <iprt/asm-mem.h>
47#include <iprt/asm.h>
48#include <iprt/assert.h>
49#include <iprt/errcore.h>
50#include <iprt/log.h>
51#include <iprt/memobj.h>
52#include <iprt/param.h>
53#include <iprt/string.h>
54#include <iprt/thread.h>
55
56#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
57# include <iprt/asm-amd64-x86.h>
58#elif defined(RT_ARCH_ARM64) || defined(RT_ARCH_ARM32)
59# include <iprt/asm-arm.h>
60#else
61# error "Port me"
62#endif
63
64
65#include "internal/mem.h"
66
67
68/*********************************************************************************************************************************
69* Defined Constants And Macros *
70*********************************************************************************************************************************/
71#if defined(DOXYGEN_RUNNING)
72# define RTR0MEM_EF_IN_FRONT
73#endif
74
75/** @def RTR0MEM_EF_SIZE
76 * The size of the fence. This must be page aligned.
77 */
78#define RTR0MEM_EF_SIZE PAGE_SIZE
79
80/** @def RTR0MEM_EF_ALIGNMENT
81 * The allocation alignment, power of two of course.
82 *
83 * Use this for working around misaligned sizes, usually stemming from
84 * allocating a string or something after the main structure. When you
85 * encounter this, please fix the allocation to RTMemAllocVar or RTMemAllocZVar.
86 */
87#if 0
88# define RTR0MEM_EF_ALIGNMENT (ARCH_BITS / 8)
89#else
90# define RTR0MEM_EF_ALIGNMENT 1
91#endif
92
93/** @def RTR0MEM_EF_IN_FRONT
94 * Define this to put the fence up in front of the block.
95 * The default (when this isn't defined) is to up it up after the block.
96 */
97//# define RTR0MEM_EF_IN_FRONT
98
99/** @def RTR0MEM_EF_FREE_DELAYED
100 * This define will enable free() delay and protection of the freed data
101 * while it's being delayed. The value of RTR0MEM_EF_FREE_DELAYED defines
102 * the threshold of the delayed blocks.
103 * Delayed blocks does not consume any physical memory, only virtual address space.
104 */
105#define RTR0MEM_EF_FREE_DELAYED (20 * _1M)
106
107/** @def RTR0MEM_EF_FREE_FILL
108 * This define will enable memset(,RTR0MEM_EF_FREE_FILL,)'ing the user memory
109 * in the block before freeing/decommitting it. This is useful in GDB since GDB
110 * appears to be able to read the content of the page even after it's been
111 * decommitted.
112 */
113#define RTR0MEM_EF_FREE_FILL 'f'
114
115/** @def RTR0MEM_EF_FILLER
116 * This define will enable memset(,RTR0MEM_EF_FILLER,)'ing the allocated
117 * memory when the API doesn't require it to be zero'd.
118 */
119#define RTR0MEM_EF_FILLER 0xef
120
121/** @def RTR0MEM_EF_NOMAN_FILLER
122 * This define will enable memset(,RTR0MEM_EF_NOMAN_FILLER,)'ing the
123 * unprotected but not allocated area of memory, the so called no man's land.
124 */
125#define RTR0MEM_EF_NOMAN_FILLER 0xaa
126
127/** @def RTR0MEM_EF_FENCE_FILLER
128 * This define will enable memset(,RTR0MEM_EF_FENCE_FILLER,)'ing the
129 * fence itself, as debuggers can usually read them.
130 */
131#define RTR0MEM_EF_FENCE_FILLER 0xcc
132
133
134/*********************************************************************************************************************************
135* Header Files *
136*********************************************************************************************************************************/
137#ifdef RT_OS_WINDOWS
138# include <iprt/win/windows.h>
139#elif !defined(RT_OS_FREEBSD)
140# include <sys/mman.h>
141#endif
142#include <iprt/avl.h>
143#include <iprt/thread.h>
144
145
146/*********************************************************************************************************************************
147* Structures and Typedefs *
148*********************************************************************************************************************************/
149/**
150 * Allocation types.
151 */
152typedef enum RTMEMTYPE
153{
154 RTMEMTYPE_RTMEMALLOC,
155 RTMEMTYPE_RTMEMALLOCZ,
156 RTMEMTYPE_RTMEMREALLOC,
157 RTMEMTYPE_RTMEMFREE,
158 RTMEMTYPE_RTMEMFREEZ,
159
160 RTMEMTYPE_NEW,
161 RTMEMTYPE_NEW_ARRAY,
162 RTMEMTYPE_DELETE,
163 RTMEMTYPE_DELETE_ARRAY
164} RTMEMTYPE;
165
166/**
167 * Node tracking a memory allocation.
168 */
169typedef struct RTR0MEMEFBLOCK
170{
171 /** Avl node code, key is the user block pointer. */
172 AVLPVNODECORE Core;
173 /** Allocation type. */
174 RTMEMTYPE enmType;
175 /** The memory object. */
176 RTR0MEMOBJ hMemObj;
177 /** The unaligned size of the block. */
178 size_t cbUnaligned;
179 /** The aligned size of the block. */
180 size_t cbAligned;
181 /** The allocation tag (read-only string). */
182 const char *pszTag;
183 /** The return address of the allocator function. */
184 void *pvCaller;
185 /** Line number of the alloc call. */
186 unsigned iLine;
187 /** File from within the allocation was made. */
188 const char *pszFile;
189 /** Function from within the allocation was made. */
190 const char *pszFunction;
191} RTR0MEMEFBLOCK, *PRTR0MEMEFBLOCK;
192
193
194
195/*********************************************************************************************************************************
196* Global Variables *
197*********************************************************************************************************************************/
198/** Spinlock protecting the all the block's globals. */
199static volatile uint32_t g_BlocksLock;
200/** Tree tracking the allocations. */
201static AVLPVTREE g_BlocksTree;
202
203#ifdef RTR0MEM_EF_FREE_DELAYED
204/** Tail of the delayed blocks. */
205static volatile PRTR0MEMEFBLOCK g_pBlocksDelayHead;
206/** Tail of the delayed blocks. */
207static volatile PRTR0MEMEFBLOCK g_pBlocksDelayTail;
208/** Number of bytes in the delay list (includes fences). */
209static volatile size_t g_cbBlocksDelay;
210#endif /* RTR0MEM_EF_FREE_DELAYED */
211
212/** Array of pointers free watches for. */
213void *gapvRTMemFreeWatch[4] = {NULL, NULL, NULL, NULL};
214/** Enable logging of all freed memory. */
215bool gfRTMemFreeLog = false;
216
217
218/*********************************************************************************************************************************
219* Internal Functions *
220*********************************************************************************************************************************/
221
222
223/**
224 * @callback_method_impl{FNRTSTROUTPUT}
225 */
226static DECLCALLBACK(size_t) rtR0MemEfWrite(void *pvArg, const char *pachChars, size_t cbChars)
227{
228 RT_NOREF1(pvArg);
229 if (cbChars)
230 {
231 RTLogWriteDebugger(pachChars, cbChars);
232 RTLogWriteStdOut(pachChars, cbChars);
233 RTLogWriteUser(pachChars, cbChars);
234 }
235 return cbChars;
236}
237
238
239/**
240 * Complains about something.
241 */
242static void rtR0MemComplain(const char *pszOp, const char *pszFormat, ...)
243{
244 va_list args;
245 RTStrFormat(rtR0MemEfWrite, NULL, NULL, NULL, "RTMem error: %s: ", pszOp);
246 va_start(args, pszFormat);
247 RTStrFormatV(rtR0MemEfWrite, NULL, NULL, NULL, pszFormat, args);
248 va_end(args);
249 RTAssertDoPanic();
250}
251
252/**
253 * Log an event.
254 */
255DECLINLINE(void) rtR0MemLog(const char *pszOp, const char *pszFormat, ...)
256{
257#if 0
258 va_list args;
259 RTStrFormat(rtR0MemEfWrite, NULL, NULL, NULL, "RTMem info: %s: ", pszOp);
260 va_start(args, pszFormat);
261 RTStrFormatV(rtR0MemEfWrite, NULL, NULL, NULL, pszFormat, args);
262 va_end(args);
263#else
264 NOREF(pszOp); NOREF(pszFormat);
265#endif
266}
267
268
269
270/**
271 * Acquires the lock.
272 */
273DECLINLINE(RTCCUINTREG) rtR0MemBlockLock(void)
274{
275 RTCCUINTREG uRet;
276 unsigned c = 0;
277 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD))
278 {
279 for (;;)
280 {
281 uRet = ASMIntDisableFlags();
282 if (ASMAtomicCmpXchgU32(&g_BlocksLock, 1, 0))
283 break;
284 ASMSetFlags(uRet);
285 RTThreadSleepNoLog(((++c) >> 2) & 31);
286 }
287 }
288 else
289 {
290 for (;;)
291 {
292 uRet = ASMIntDisableFlags();
293 if (ASMAtomicCmpXchgU32(&g_BlocksLock, 1, 0))
294 break;
295 ASMSetFlags(uRet);
296 ASMNopPause();
297 if (++c & 3)
298 ASMNopPause();
299 }
300 }
301 return uRet;
302}
303
304
305/**
306 * Releases the lock.
307 */
308DECLINLINE(void) rtR0MemBlockUnlock(RTCCUINTREG fSavedIntFlags)
309{
310 Assert(g_BlocksLock == 1);
311 ASMAtomicXchgU32(&g_BlocksLock, 0);
312 ASMSetFlags(fSavedIntFlags);
313}
314
315
316/**
317 * Creates a block.
318 */
319DECLINLINE(PRTR0MEMEFBLOCK) rtR0MemBlockCreate(RTMEMTYPE enmType, size_t cbUnaligned, size_t cbAligned,
320 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
321{
322 PRTR0MEMEFBLOCK pBlock = (PRTR0MEMEFBLOCK)RTMemAlloc(sizeof(*pBlock));
323 if (pBlock)
324 {
325 pBlock->enmType = enmType;
326 pBlock->cbUnaligned = cbUnaligned;
327 pBlock->cbAligned = cbAligned;
328 pBlock->pszTag = pszTag;
329 pBlock->pvCaller = pvCaller;
330 pBlock->iLine = iLine;
331 pBlock->pszFile = pszFile;
332 pBlock->pszFunction = pszFunction;
333 }
334 return pBlock;
335}
336
337
338/**
339 * Frees a block.
340 */
341DECLINLINE(void) rtR0MemBlockFree(PRTR0MEMEFBLOCK pBlock)
342{
343 RTMemFree(pBlock);
344}
345
346
347/**
348 * Insert a block from the tree.
349 */
350DECLINLINE(void) rtR0MemBlockInsert(PRTR0MEMEFBLOCK pBlock, void *pv, RTR0MEMOBJ hMemObj)
351{
352 pBlock->Core.Key = pv;
353 pBlock->hMemObj = hMemObj;
354 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
355 bool fRc = RTAvlPVInsert(&g_BlocksTree, &pBlock->Core);
356 rtR0MemBlockUnlock(fSavedIntFlags);
357 AssertRelease(fRc);
358}
359
360
361/**
362 * Remove a block from the tree and returns it to the caller.
363 */
364DECLINLINE(PRTR0MEMEFBLOCK) rtR0MemBlockRemove(void *pv)
365{
366 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
367 PRTR0MEMEFBLOCK pBlock = (PRTR0MEMEFBLOCK)RTAvlPVRemove(&g_BlocksTree, pv);
368 rtR0MemBlockUnlock(fSavedIntFlags);
369 return pBlock;
370}
371
372
373/**
374 * Gets a block.
375 */
376DECLINLINE(PRTR0MEMEFBLOCK) rtR0MemBlockGet(void *pv)
377{
378 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
379 PRTR0MEMEFBLOCK pBlock = (PRTR0MEMEFBLOCK)RTAvlPVGet(&g_BlocksTree, pv);
380 rtR0MemBlockUnlock(fSavedIntFlags);
381 return pBlock;
382}
383
384
385/**
386 * Dumps one allocation.
387 */
388static DECLCALLBACK(int) RTMemDumpOne(PAVLPVNODECORE pNode, void *pvUser)
389{
390 PRTR0MEMEFBLOCK pBlock = (PRTR0MEMEFBLOCK)pNode;
391 RTStrFormat(rtR0MemEfWrite, NULL, NULL, NULL, "%p %08lx(+%02lx) %p\n",
392 pBlock->Core.Key,
393 (unsigned long)pBlock->cbUnaligned,
394 (unsigned long)(pBlock->cbAligned - pBlock->cbUnaligned),
395 pBlock->pvCaller);
396 NOREF(pvUser);
397 return 0;
398}
399
400
401/**
402 * Dumps the allocated blocks.
403 * This is something which you should call from gdb.
404 */
405RT_C_DECLS_BEGIN
406void RTMemDump(void);
407RT_C_DECLS_END
408
409void RTMemDump(void)
410{
411 RTStrFormat(rtR0MemEfWrite, NULL, NULL, NULL, "address size(alg) caller\n");
412 RTAvlPVDoWithAll(&g_BlocksTree, true, RTMemDumpOne, NULL);
413}
414
415#ifdef RTR0MEM_EF_FREE_DELAYED
416
417/**
418 * Insert a delayed block.
419 */
420DECLINLINE(void) rtR0MemBlockDelayInsert(PRTR0MEMEFBLOCK pBlock)
421{
422 size_t cbBlock = RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTR0MEM_EF_SIZE;
423 pBlock->Core.pRight = NULL;
424 pBlock->Core.pLeft = NULL;
425 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
426 if (g_pBlocksDelayHead)
427 {
428 g_pBlocksDelayHead->Core.pLeft = (PAVLPVNODECORE)pBlock;
429 pBlock->Core.pRight = (PAVLPVNODECORE)g_pBlocksDelayHead;
430 g_pBlocksDelayHead = pBlock;
431 }
432 else
433 {
434 g_pBlocksDelayTail = pBlock;
435 g_pBlocksDelayHead = pBlock;
436 }
437 g_cbBlocksDelay += cbBlock;
438 rtR0MemBlockUnlock(fSavedIntFlags);
439}
440
441/**
442 * Removes a delayed block.
443 */
444DECLINLINE(PRTR0MEMEFBLOCK) rtR0MemBlockDelayRemove(void)
445{
446 PRTR0MEMEFBLOCK pBlock = NULL;
447 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
448 if (g_cbBlocksDelay > RTR0MEM_EF_FREE_DELAYED)
449 {
450 pBlock = g_pBlocksDelayTail;
451 if (pBlock)
452 {
453 g_pBlocksDelayTail = (PRTR0MEMEFBLOCK)pBlock->Core.pLeft;
454 if (pBlock->Core.pLeft)
455 pBlock->Core.pLeft->pRight = NULL;
456 else
457 g_pBlocksDelayHead = NULL;
458 g_cbBlocksDelay -= RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTR0MEM_EF_SIZE;
459 }
460 }
461 rtR0MemBlockUnlock(fSavedIntFlags);
462 return pBlock;
463}
464
465#endif /* RTR0MEM_EF_FREE_DELAYED */
466
467
468static void rtR0MemFreeBlock(PRTR0MEMEFBLOCK pBlock, const char *pszOp)
469{
470 void *pv = pBlock->Core.Key;
471# ifdef RTR0MEM_EF_IN_FRONT
472 void *pvBlock = (char *)pv - RTR0MEM_EF_SIZE;
473# else
474 void *pvBlock = (void *)((uintptr_t)pv & ~(uintptr_t)PAGE_OFFSET_MASK);
475# endif
476 size_t cbBlock = RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTR0MEM_EF_SIZE;
477
478 int rc = RTR0MemObjProtect(pBlock->hMemObj, 0 /*offSub*/, RT_ALIGN_Z(cbBlock, PAGE_SIZE), RTMEM_PROT_READ | RTMEM_PROT_WRITE);
479 if (RT_FAILURE(rc))
480 rtR0MemComplain(pszOp, "RTR0MemObjProtect([%p], 0, %#x, RTMEM_PROT_READ | RTMEM_PROT_WRITE) -> %Rrc\n",
481 pvBlock, cbBlock, rc);
482
483 rc = RTR0MemObjFree(pBlock->hMemObj, true /*fFreeMappings*/);
484 if (RT_FAILURE(rc))
485 rtR0MemComplain(pszOp, "RTR0MemObjFree([%p LB %#x]) -> %Rrc\n", pvBlock, cbBlock, rc);
486 pBlock->hMemObj = NIL_RTR0MEMOBJ;
487
488 rtR0MemBlockFree(pBlock);
489}
490
491
492/**
493 * Initialize call, we shouldn't fail here.
494 */
495void rtR0MemEfInit(void)
496{
497
498}
499
500/**
501 * @callback_method_impl{AVLPVCALLBACK}
502 */
503static DECLCALLBACK(int) rtR0MemEfDestroyBlock(PAVLPVNODECORE pNode, void *pvUser)
504{
505 PRTR0MEMEFBLOCK pBlock = (PRTR0MEMEFBLOCK)pNode;
506
507 /* Note! pszFile and pszFunction may be invalid at this point. */
508 rtR0MemComplain("rtR0MemEfDestroyBlock", "Leaking %zu bytes at %p (iLine=%u pvCaller=%p)\n",
509 pBlock->cbAligned, pBlock->Core.Key, pBlock->iLine, pBlock->pvCaller);
510
511 rtR0MemFreeBlock(pBlock, "rtR0MemEfDestroyBlock");
512
513 NOREF(pvUser);
514 return VINF_SUCCESS;
515}
516
517
518/**
519 * Termination call.
520 *
521 * Will check and free memory.
522 */
523void rtR0MemEfTerm(void)
524{
525#ifdef RTR0MEM_EF_FREE_DELAYED
526 /*
527 * Release delayed frees.
528 */
529 RTCCUINTREG fSavedIntFlags = rtR0MemBlockLock();
530 for (;;)
531 {
532 PRTR0MEMEFBLOCK pBlock = g_pBlocksDelayTail;
533 if (pBlock)
534 {
535 g_pBlocksDelayTail = (PRTR0MEMEFBLOCK)pBlock->Core.pLeft;
536 if (pBlock->Core.pLeft)
537 pBlock->Core.pLeft->pRight = NULL;
538 else
539 g_pBlocksDelayHead = NULL;
540 rtR0MemBlockUnlock(fSavedIntFlags);
541
542 rtR0MemFreeBlock(pBlock, "rtR0MemEfTerm");
543
544 rtR0MemBlockLock();
545 }
546 else
547 break;
548 }
549 g_cbBlocksDelay = 0;
550 rtR0MemBlockUnlock(fSavedIntFlags);
551#endif
552
553 /*
554 * Complain about leaks. Then release them.
555 */
556 RTAvlPVDestroy(&g_BlocksTree, rtR0MemEfDestroyBlock, NULL);
557}
558
559
560/**
561 * Internal allocator.
562 */
563static void * rtR0MemAlloc(const char *pszOp, RTMEMTYPE enmType, size_t cbUnaligned, size_t cbAligned,
564 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
565{
566 /*
567 * Sanity.
568 */
569 if ( RT_ALIGN_Z(RTR0MEM_EF_SIZE, PAGE_SIZE) != RTR0MEM_EF_SIZE
570 && RTR0MEM_EF_SIZE <= 0)
571 {
572 rtR0MemComplain(pszOp, "Invalid E-fence size! %#x\n", RTR0MEM_EF_SIZE);
573 return NULL;
574 }
575 if (!cbUnaligned)
576 {
577#if 1
578 rtR0MemComplain(pszOp, "Request of ZERO bytes allocation!\n");
579 return NULL;
580#else
581 cbAligned = cbUnaligned = 1;
582#endif
583 }
584
585#ifndef RTR0MEM_EF_IN_FRONT
586 /* Alignment decreases fence accuracy, but this is at least partially
587 * counteracted by filling and checking the alignment padding. When the
588 * fence is in front then then no extra alignment is needed. */
589 cbAligned = RT_ALIGN_Z(cbAligned, RTR0MEM_EF_ALIGNMENT);
590#endif
591
592 /*
593 * Allocate the trace block.
594 */
595 PRTR0MEMEFBLOCK pBlock = rtR0MemBlockCreate(enmType, cbUnaligned, cbAligned, pszTag, pvCaller, RT_SRC_POS_ARGS);
596 if (!pBlock)
597 {
598 rtR0MemComplain(pszOp, "Failed to allocate trace block!\n");
599 return NULL;
600 }
601
602 /*
603 * Allocate a block with page alignment space + the size of the E-fence.
604 */
605 void *pvBlock = NULL;
606 RTR0MEMOBJ hMemObj;
607 size_t cbBlock = RT_ALIGN_Z(cbAligned, PAGE_SIZE) + RTR0MEM_EF_SIZE;
608 int rc = RTR0MemObjAllocPage(&hMemObj, cbBlock, false /*fExecutable*/);
609 if (RT_SUCCESS(rc))
610 pvBlock = RTR0MemObjAddress(hMemObj);
611 if (pvBlock)
612 {
613 /*
614 * Calc the start of the fence and the user block
615 * and then change the page protection of the fence.
616 */
617#ifdef RTR0MEM_EF_IN_FRONT
618 void *pvEFence = pvBlock;
619 void *pv = (char *)pvEFence + RTR0MEM_EF_SIZE;
620# ifdef RTR0MEM_EF_NOMAN_FILLER
621 memset((char *)pv + cbUnaligned, RTR0MEM_EF_NOMAN_FILLER, cbBlock - RTR0MEM_EF_SIZE - cbUnaligned);
622# endif
623#else
624 void *pvEFence = (char *)pvBlock + (cbBlock - RTR0MEM_EF_SIZE);
625 void *pv = (char *)pvEFence - cbAligned;
626# ifdef RTR0MEM_EF_NOMAN_FILLER
627 memset(pvBlock, RTR0MEM_EF_NOMAN_FILLER, cbBlock - RTR0MEM_EF_SIZE - cbAligned);
628 memset((char *)pv + cbUnaligned, RTR0MEM_EF_NOMAN_FILLER, cbAligned - cbUnaligned);
629# endif
630#endif
631
632#ifdef RTR0MEM_EF_FENCE_FILLER
633 memset(pvEFence, RTR0MEM_EF_FENCE_FILLER, RTR0MEM_EF_SIZE);
634#endif
635 rc = RTR0MemObjProtect(hMemObj, (uint8_t *)pvEFence - (uint8_t *)pvBlock, RTR0MEM_EF_SIZE, RTMEM_PROT_NONE);
636 if (!rc)
637 {
638 rtR0MemBlockInsert(pBlock, pv, hMemObj);
639 if (enmType == RTMEMTYPE_RTMEMALLOCZ)
640 memset(pv, 0, cbUnaligned);
641#ifdef RTR0MEM_EF_FILLER
642 else
643 memset(pv, RTR0MEM_EF_FILLER, cbUnaligned);
644#endif
645
646 rtR0MemLog(pszOp, "returns %p (pvBlock=%p cbBlock=%#x pvEFence=%p cbUnaligned=%#x)\n", pv, pvBlock, cbBlock, pvEFence, cbUnaligned);
647 return pv;
648 }
649 rtR0MemComplain(pszOp, "RTMemProtect failed, pvEFence=%p size %d, rc=%d\n", pvEFence, RTR0MEM_EF_SIZE, rc);
650 RTR0MemObjFree(hMemObj, true /*fFreeMappings*/);
651 }
652 else
653 {
654 rtR0MemComplain(pszOp, "Failed to allocated %zu (%zu) bytes (rc=%Rrc).\n", cbBlock, cbUnaligned, rc);
655 if (RT_SUCCESS(rc))
656 RTR0MemObjFree(hMemObj, true /*fFreeMappings*/);
657 }
658
659 rtR0MemBlockFree(pBlock);
660 return NULL;
661}
662
663
664/**
665 * Internal free.
666 */
667static void rtR0MemFree(const char *pszOp, RTMEMTYPE enmType, void *pv, size_t cbUser, void *pvCaller, RT_SRC_POS_DECL)
668{
669 NOREF(enmType); RT_SRC_POS_NOREF();
670
671 /*
672 * Simple case.
673 */
674 if (!pv)
675 return;
676
677 /*
678 * Check watch points.
679 */
680 for (unsigned i = 0; i < RT_ELEMENTS(gapvRTMemFreeWatch); i++)
681 if (gapvRTMemFreeWatch[i] == pv)
682 RTAssertDoPanic();
683
684 /*
685 * Find the block.
686 */
687 PRTR0MEMEFBLOCK pBlock = rtR0MemBlockRemove(pv);
688 if (pBlock)
689 {
690 if (gfRTMemFreeLog)
691 RTLogPrintf("RTMem %s: pv=%p pvCaller=%p cbUnaligned=%#x\n", pszOp, pv, pvCaller, pBlock->cbUnaligned);
692
693#ifdef RTR0MEM_EF_NOMAN_FILLER
694 /*
695 * Check whether the no man's land is untouched.
696 */
697# ifdef RTR0MEM_EF_IN_FRONT
698 void *pvWrong = ASMMemFirstMismatchingU8((char *)pv + pBlock->cbUnaligned,
699 RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) - pBlock->cbUnaligned,
700 RTR0MEM_EF_NOMAN_FILLER);
701# else
702 /* Alignment must match allocation alignment in rtMemAlloc(). */
703 void *pvWrong = ASMMemFirstMismatchingU8((char *)pv + pBlock->cbUnaligned,
704 pBlock->cbAligned - pBlock->cbUnaligned,
705 RTR0MEM_EF_NOMAN_FILLER);
706 if (pvWrong)
707 RTAssertDoPanic();
708 pvWrong = ASMMemFirstMismatchingU8((void *)((uintptr_t)pv & ~(uintptr_t)PAGE_OFFSET_MASK),
709 RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) - pBlock->cbAligned,
710 RTR0MEM_EF_NOMAN_FILLER);
711# endif
712 if (pvWrong)
713 RTAssertDoPanic();
714#endif
715
716 /*
717 * Fill the user part of the block.
718 */
719 AssertMsg(enmType != RTMEMTYPE_RTMEMFREEZ || cbUser == pBlock->cbUnaligned,
720 ("cbUser=%#zx cbUnaligned=%#zx\n", cbUser, pBlock->cbUnaligned));
721 RT_NOREF(cbUser);
722 if (enmType == RTMEMTYPE_RTMEMFREEZ)
723 RT_BZERO(pv, pBlock->cbUnaligned);
724#ifdef RTR0MEM_EF_FREE_FILL
725 else
726 memset(pv, RTR0MEM_EF_FREE_FILL, pBlock->cbUnaligned);
727#endif
728
729#if defined(RTR0MEM_EF_FREE_DELAYED) && RTR0MEM_EF_FREE_DELAYED > 0
730 /*
731 * We're doing delayed freeing.
732 * That means we'll expand the E-fence to cover the entire block.
733 */
734 int rc = RTR0MemObjProtect(pBlock->hMemObj,
735# ifdef RTR0MEM_EF_IN_FRONT
736 RTR0MEM_EF_SIZE,
737# else
738 0 /*offSub*/,
739# endif
740 RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE),
741 RTMEM_PROT_NONE);
742 if (RT_SUCCESS(rc))
743 {
744 /*
745 * Insert it into the free list and process pending frees.
746 */
747 rtR0MemBlockDelayInsert(pBlock);
748 while ((pBlock = rtR0MemBlockDelayRemove()) != NULL)
749 rtR0MemFreeBlock(pBlock, pszOp);
750 }
751 else
752 rtR0MemComplain(pszOp, "Failed to expand the efence of pv=%p cb=%d, rc=%d.\n", pv, pBlock, rc);
753
754#else /* !RTR0MEM_EF_FREE_DELAYED */
755 rtR0MemFreeBlock(pBlock, pszOp);
756#endif /* !RTR0MEM_EF_FREE_DELAYED */
757 }
758 else
759 rtR0MemComplain(pszOp, "pv=%p not found! Incorrect free!\n", pv);
760}
761
762
763/**
764 * Internal realloc.
765 */
766static void *rtR0MemRealloc(const char *pszOp, RTMEMTYPE enmType, void *pvOld, size_t cbNew,
767 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
768{
769 /*
770 * Allocate new and copy.
771 */
772 if (!pvOld)
773 return rtR0MemAlloc(pszOp, enmType, cbNew, cbNew, pszTag, pvCaller, RT_SRC_POS_ARGS);
774 if (!cbNew)
775 {
776 rtR0MemFree(pszOp, RTMEMTYPE_RTMEMREALLOC, pvOld, 0, pvCaller, RT_SRC_POS_ARGS);
777 return NULL;
778 }
779
780 /*
781 * Get the block, allocate the new, copy the data, free the old one.
782 */
783 PRTR0MEMEFBLOCK pBlock = rtR0MemBlockGet(pvOld);
784 if (pBlock)
785 {
786 void *pvRet = rtR0MemAlloc(pszOp, enmType, cbNew, cbNew, pszTag, pvCaller, RT_SRC_POS_ARGS);
787 if (pvRet)
788 {
789 memcpy(pvRet, pvOld, RT_MIN(cbNew, pBlock->cbUnaligned));
790 rtR0MemFree(pszOp, RTMEMTYPE_RTMEMREALLOC, pvOld, 0, pvCaller, RT_SRC_POS_ARGS);
791 }
792 return pvRet;
793 }
794 rtR0MemComplain(pszOp, "pvOld=%p was not found!\n", pvOld);
795 return NULL;
796}
797
798
799
800
801RTDECL(void *) RTMemEfTmpAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
802{
803 return rtR0MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
804}
805
806
807RTDECL(void *) RTMemEfTmpAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
808{
809 return rtR0MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
810}
811
812
813RTDECL(void) RTMemEfTmpFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW_DEF
814{
815 if (pv)
816 rtR0MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, 0, ASMReturnAddress(), RT_SRC_POS_ARGS);
817}
818
819
820RTDECL(void) RTMemEfTmpFreeZ(void *pv, size_t cb, RT_SRC_POS_DECL) RT_NO_THROW_DEF
821{
822 if (pv)
823 rtR0MemFree("FreeZ", RTMEMTYPE_RTMEMFREEZ, pv, cb, ASMReturnAddress(), RT_SRC_POS_ARGS);
824}
825
826
827RTDECL(void *) RTMemEfAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
828{
829 return rtR0MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
830}
831
832
833RTDECL(void *) RTMemEfAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
834{
835 return rtR0MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
836}
837
838
839RTDECL(void *) RTMemEfAllocVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
840{
841 size_t cbAligned;
842 if (cbUnaligned >= 16)
843 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
844 else
845 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
846 return rtR0MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
847}
848
849
850RTDECL(void *) RTMemEfAllocZVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
851{
852 size_t cbAligned;
853 if (cbUnaligned >= 16)
854 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
855 else
856 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
857 return rtR0MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
858}
859
860
861RTDECL(void *) RTMemEfRealloc(void *pvOld, size_t cbNew, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
862{
863 return rtR0MemRealloc("Realloc", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
864}
865
866RTDECL(void *) RTMemEfReallocZ(void *pvOld, size_t cbOld, size_t cbNew, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
867{
868 void *pvDst = rtR0MemRealloc("Realloc", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
869 if (pvDst && cbNew > cbOld)
870 memset((uint8_t *)pvDst + cbOld, 0, cbNew - cbOld);
871 return pvDst;
872}
873
874
875RTDECL(void) RTMemEfFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW_DEF
876{
877 if (pv)
878 rtR0MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, 0, ASMReturnAddress(), RT_SRC_POS_ARGS);
879}
880
881
882RTDECL(void) RTMemEfFreeZ(void *pv, size_t cb, RT_SRC_POS_DECL) RT_NO_THROW_DEF
883{
884 if (pv)
885 rtR0MemFree("Free", RTMEMTYPE_RTMEMFREEZ, pv, cb, ASMReturnAddress(), RT_SRC_POS_ARGS);
886}
887
888
889RTDECL(void *) RTMemEfDup(const void *pvSrc, size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
890{
891 void *pvDst = RTMemEfAlloc(cb, pszTag, RT_SRC_POS_ARGS);
892 if (pvDst)
893 memcpy(pvDst, pvSrc, cb);
894 return pvDst;
895}
896
897
898RTDECL(void *) RTMemEfDupEx(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW_DEF
899{
900 void *pvDst = RTMemEfAlloc(cbSrc + cbExtra, pszTag, RT_SRC_POS_ARGS);
901 if (pvDst)
902 {
903 memcpy(pvDst, pvSrc, cbSrc);
904 memset((uint8_t *)pvDst + cbSrc, 0, cbExtra);
905 }
906 return pvDst;
907}
908
909
910
911
912/*
913 *
914 * The NP (no position) versions.
915 *
916 */
917
918
919
920RTDECL(void *) RTMemEfTmpAllocNP(size_t cb, const char *pszTag) RT_NO_THROW_DEF
921{
922 return rtR0MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
923}
924
925
926RTDECL(void *) RTMemEfTmpAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW_DEF
927{
928 return rtR0MemAlloc("TmpAllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
929}
930
931
932RTDECL(void) RTMemEfTmpFreeNP(void *pv) RT_NO_THROW_DEF
933{
934 if (pv)
935 rtR0MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, 0, ASMReturnAddress(), NULL, 0, NULL);
936}
937
938
939RTDECL(void) RTMemEfTmpFreeZNP(void *pv, size_t cb) RT_NO_THROW_DEF
940{
941 if (pv)
942 rtR0MemFree("FreeZ", RTMEMTYPE_RTMEMFREEZ, pv, cb, ASMReturnAddress(), NULL, 0, NULL);
943}
944
945
946RTDECL(void *) RTMemEfAllocNP(size_t cb, const char *pszTag) RT_NO_THROW_DEF
947{
948 return rtR0MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
949}
950
951
952RTDECL(void *) RTMemEfAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW_DEF
953{
954 return rtR0MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
955}
956
957
958RTDECL(void *) RTMemEfAllocVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW_DEF
959{
960 size_t cbAligned;
961 if (cbUnaligned >= 16)
962 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
963 else
964 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
965 return rtR0MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), NULL, 0, NULL);
966}
967
968
969RTDECL(void *) RTMemEfAllocZVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW_DEF
970{
971 size_t cbAligned;
972 if (cbUnaligned >= 16)
973 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
974 else
975 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
976 return rtR0MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), NULL, 0, NULL);
977}
978
979
980RTDECL(void *) RTMemEfReallocNP(void *pvOld, size_t cbNew, const char *pszTag) RT_NO_THROW_DEF
981{
982 return rtR0MemRealloc("Realloc", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), NULL, 0, NULL);
983}
984
985
986RTDECL(void *) RTMemEfReallocZNP(void *pvOld, size_t cbOld, size_t cbNew, const char *pszTag) RT_NO_THROW_DEF
987{
988 void *pvDst = rtR0MemRealloc("ReallocZ", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), NULL, 0, NULL);
989 if (pvDst && cbNew > cbOld)
990 memset((uint8_t *)pvDst + cbOld, 0, cbNew - cbOld);
991 return pvDst;
992}
993
994
995RTDECL(void) RTMemEfFreeNP(void *pv) RT_NO_THROW_DEF
996{
997 if (pv)
998 rtR0MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, 0, ASMReturnAddress(), NULL, 0, NULL);
999}
1000
1001
1002RTDECL(void) RTMemEfFreeZNP(void *pv, size_t cb) RT_NO_THROW_DEF
1003{
1004 if (pv)
1005 rtR0MemFree("FreeZ", RTMEMTYPE_RTMEMFREEZ, pv, cb, ASMReturnAddress(), NULL, 0, NULL);
1006}
1007
1008
1009RTDECL(void *) RTMemEfDupNP(const void *pvSrc, size_t cb, const char *pszTag) RT_NO_THROW_DEF
1010{
1011 void *pvDst = RTMemEfAlloc(cb, pszTag, NULL, 0, NULL);
1012 if (pvDst)
1013 memcpy(pvDst, pvSrc, cb);
1014 return pvDst;
1015}
1016
1017
1018RTDECL(void *) RTMemEfDupExNP(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag) RT_NO_THROW_DEF
1019{
1020 void *pvDst = RTMemEfAlloc(cbSrc + cbExtra, pszTag, NULL, 0, NULL);
1021 if (pvDst)
1022 {
1023 memcpy(pvDst, pvSrc, cbSrc);
1024 memset((uint8_t *)pvDst + cbSrc, 0, cbExtra);
1025 }
1026 return pvDst;
1027}
1028
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