VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/alloc-ef.cpp@ 50874

Last change on this file since 50874 was 48935, checked in by vboxsync, 11 years ago

Runtime: Whitespace and svn:keyword cleanups by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 22.0 KB
Line 
1/* $Id: alloc-ef.cpp 48935 2013-10-07 21:19:37Z vboxsync $ */
2/** @file
3 * IPRT - Memory Allocation, electric fence.
4 */
5
6/*
7 * Copyright (C) 2006-2011 Oracle Corporation
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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include "alloc-ef.h"
32#include <iprt/mem.h>
33#include <iprt/log.h>
34#include <iprt/asm.h>
35#include <iprt/thread.h>
36#include <VBox/sup.h>
37#include <iprt/err.h>
38#include <errno.h>
39#include <stdio.h>
40#include <stdlib.h>
41
42#include <iprt/alloc.h>
43#include <iprt/assert.h>
44#include <iprt/param.h>
45#include <iprt/string.h>
46
47
48/*******************************************************************************
49* Global Variables *
50*******************************************************************************/
51#ifdef RTALLOC_EFENCE_TRACE
52/** Spinlock protecting the all the block's globals. */
53static volatile uint32_t g_BlocksLock;
54/** Tree tracking the allocations. */
55static AVLPVTREE g_BlocksTree;
56# ifdef RTALLOC_EFENCE_FREE_DELAYED
57/** Tail of the delayed blocks. */
58static volatile PRTMEMBLOCK g_pBlocksDelayHead;
59/** Tail of the delayed blocks. */
60static volatile PRTMEMBLOCK g_pBlocksDelayTail;
61/** Number of bytes in the delay list (includes fences). */
62static volatile size_t g_cbBlocksDelay;
63# endif /* RTALLOC_EFENCE_FREE_DELAYED */
64#endif /* RTALLOC_EFENCE_TRACE */
65/** Array of pointers free watches for. */
66void *gapvRTMemFreeWatch[4] = {NULL, NULL, NULL, NULL};
67/** Enable logging of all freed memory. */
68bool gfRTMemFreeLog = false;
69
70
71/*******************************************************************************
72* Internal Functions *
73*******************************************************************************/
74/**
75 * Complains about something.
76 */
77static void rtmemComplain(const char *pszOp, const char *pszFormat, ...)
78{
79 va_list args;
80 fprintf(stderr, "RTMem error: %s: ", pszOp);
81 va_start(args, pszFormat);
82 vfprintf(stderr, pszFormat, args);
83 va_end(args);
84 RTAssertDoPanic();
85}
86
87/**
88 * Log an event.
89 */
90DECLINLINE(void) rtmemLog(const char *pszOp, const char *pszFormat, ...)
91{
92#if 0
93 va_list args;
94 fprintf(stderr, "RTMem info: %s: ", pszOp);
95 va_start(args, pszFormat);
96 vfprintf(stderr, pszFormat, args);
97 va_end(args);
98#else
99 NOREF(pszOp); NOREF(pszFormat);
100#endif
101}
102
103
104#ifdef RTALLOC_EFENCE_TRACE
105
106/**
107 * Acquires the lock.
108 */
109DECLINLINE(void) rtmemBlockLock(void)
110{
111 unsigned c = 0;
112 while (!ASMAtomicCmpXchgU32(&g_BlocksLock, 1, 0))
113 RTThreadSleepNoLog(((++c) >> 2) & 31);
114}
115
116
117/**
118 * Releases the lock.
119 */
120DECLINLINE(void) rtmemBlockUnlock(void)
121{
122 Assert(g_BlocksLock == 1);
123 ASMAtomicXchgU32(&g_BlocksLock, 0);
124}
125
126
127/**
128 * Creates a block.
129 */
130DECLINLINE(PRTMEMBLOCK) rtmemBlockCreate(RTMEMTYPE enmType, size_t cbUnaligned, size_t cbAligned,
131 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
132{
133 PRTMEMBLOCK pBlock = (PRTMEMBLOCK)malloc(sizeof(*pBlock));
134 if (pBlock)
135 {
136 pBlock->enmType = enmType;
137 pBlock->cbUnaligned = cbUnaligned;
138 pBlock->cbAligned = cbAligned;
139 pBlock->pszTag = pszTag;
140 pBlock->pvCaller = pvCaller;
141 pBlock->iLine = iLine;
142 pBlock->pszFile = pszFile;
143 pBlock->pszFunction = pszFunction;
144 }
145 return pBlock;
146}
147
148
149/**
150 * Frees a block.
151 */
152DECLINLINE(void) rtmemBlockFree(PRTMEMBLOCK pBlock)
153{
154 free(pBlock);
155}
156
157
158/**
159 * Insert a block from the tree.
160 */
161DECLINLINE(void) rtmemBlockInsert(PRTMEMBLOCK pBlock, void *pv)
162{
163 pBlock->Core.Key = pv;
164 rtmemBlockLock();
165 bool fRc = RTAvlPVInsert(&g_BlocksTree, &pBlock->Core);
166 rtmemBlockUnlock();
167 AssertRelease(fRc);
168}
169
170
171/**
172 * Remove a block from the tree and returns it to the caller.
173 */
174DECLINLINE(PRTMEMBLOCK) rtmemBlockRemove(void *pv)
175{
176 rtmemBlockLock();
177 PRTMEMBLOCK pBlock = (PRTMEMBLOCK)RTAvlPVRemove(&g_BlocksTree, pv);
178 rtmemBlockUnlock();
179 return pBlock;
180}
181
182/**
183 * Gets a block.
184 */
185DECLINLINE(PRTMEMBLOCK) rtmemBlockGet(void *pv)
186{
187 rtmemBlockLock();
188 PRTMEMBLOCK pBlock = (PRTMEMBLOCK)RTAvlPVGet(&g_BlocksTree, pv);
189 rtmemBlockUnlock();
190 return pBlock;
191}
192
193/**
194 * Dumps one allocation.
195 */
196static DECLCALLBACK(int) RTMemDumpOne(PAVLPVNODECORE pNode, void *pvUser)
197{
198 PRTMEMBLOCK pBlock = (PRTMEMBLOCK)pNode;
199 fprintf(stderr, "%p %08lx(+%02lx) %p\n",
200 pBlock->Core.Key,
201 (unsigned long)pBlock->cbUnaligned,
202 (unsigned long)(pBlock->cbAligned - pBlock->cbUnaligned),
203 pBlock->pvCaller);
204 NOREF(pvUser);
205 return 0;
206}
207
208/**
209 * Dumps the allocated blocks.
210 * This is something which you should call from gdb.
211 */
212extern "C" void RTMemDump(void);
213void RTMemDump(void)
214{
215 fprintf(stderr, "address size(alg) caller\n");
216 RTAvlPVDoWithAll(&g_BlocksTree, true, RTMemDumpOne, NULL);
217}
218
219# ifdef RTALLOC_EFENCE_FREE_DELAYED
220
221/**
222 * Insert a delayed block.
223 */
224DECLINLINE(void) rtmemBlockDelayInsert(PRTMEMBLOCK pBlock)
225{
226 size_t cbBlock = RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE;
227 pBlock->Core.pRight = NULL;
228 pBlock->Core.pLeft = NULL;
229 rtmemBlockLock();
230 if (g_pBlocksDelayHead)
231 {
232 g_pBlocksDelayHead->Core.pLeft = (PAVLPVNODECORE)pBlock;
233 pBlock->Core.pRight = (PAVLPVNODECORE)g_pBlocksDelayHead;
234 g_pBlocksDelayHead = pBlock;
235 }
236 else
237 {
238 g_pBlocksDelayTail = pBlock;
239 g_pBlocksDelayHead = pBlock;
240 }
241 g_cbBlocksDelay += cbBlock;
242 rtmemBlockUnlock();
243}
244
245/**
246 * Removes a delayed block.
247 */
248DECLINLINE(PRTMEMBLOCK) rtmemBlockDelayRemove(void)
249{
250 PRTMEMBLOCK pBlock = NULL;
251 rtmemBlockLock();
252 if (g_cbBlocksDelay > RTALLOC_EFENCE_FREE_DELAYED)
253 {
254 pBlock = g_pBlocksDelayTail;
255 if (pBlock)
256 {
257 g_pBlocksDelayTail = (PRTMEMBLOCK)pBlock->Core.pLeft;
258 if (pBlock->Core.pLeft)
259 pBlock->Core.pLeft->pRight = NULL;
260 else
261 g_pBlocksDelayHead = NULL;
262 g_cbBlocksDelay -= RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE;
263 }
264 }
265 rtmemBlockUnlock();
266 return pBlock;
267}
268
269# endif /* RTALLOC_EFENCE_FREE_DELAYED */
270
271#endif /* RTALLOC_EFENCE_TRACE */
272
273
274/**
275 * Internal allocator.
276 */
277RTDECL(void *) rtR3MemAlloc(const char *pszOp, RTMEMTYPE enmType, size_t cbUnaligned, size_t cbAligned,
278 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
279{
280 /*
281 * Sanity.
282 */
283 if ( RT_ALIGN_Z(RTALLOC_EFENCE_SIZE, PAGE_SIZE) != RTALLOC_EFENCE_SIZE
284 && RTALLOC_EFENCE_SIZE <= 0)
285 {
286 rtmemComplain(pszOp, "Invalid E-fence size! %#x\n", RTALLOC_EFENCE_SIZE);
287 return NULL;
288 }
289 if (!cbUnaligned)
290 {
291#if 0
292 rtmemComplain(pszOp, "Request of ZERO bytes allocation!\n");
293 return NULL;
294#else
295 cbAligned = cbUnaligned = 1;
296#endif
297 }
298
299#ifndef RTALLOC_EFENCE_IN_FRONT
300 /* Alignment decreases fence accuracy, but this is at least partially
301 * counteracted by filling and checking the alignment padding. When the
302 * fence is in front then then no extra alignment is needed. */
303 cbAligned = RT_ALIGN_Z(cbAligned, RTALLOC_EFENCE_ALIGNMENT);
304#endif
305
306#ifdef RTALLOC_EFENCE_TRACE
307 /*
308 * Allocate the trace block.
309 */
310 PRTMEMBLOCK pBlock = rtmemBlockCreate(enmType, cbUnaligned, cbAligned, pszTag, pvCaller, RT_SRC_POS_ARGS);
311 if (!pBlock)
312 {
313 rtmemComplain(pszOp, "Failed to allocate trace block!\n");
314 return NULL;
315 }
316#endif
317
318 /*
319 * Allocate a block with page alignment space + the size of the E-fence.
320 */
321 size_t cbBlock = RT_ALIGN_Z(cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE;
322 void *pvBlock = RTMemPageAlloc(cbBlock);
323 if (pvBlock)
324 {
325 /*
326 * Calc the start of the fence and the user block
327 * and then change the page protection of the fence.
328 */
329#ifdef RTALLOC_EFENCE_IN_FRONT
330 void *pvEFence = pvBlock;
331 void *pv = (char *)pvEFence + RTALLOC_EFENCE_SIZE;
332# ifdef RTALLOC_EFENCE_NOMAN_FILLER
333 memset((char *)pv + cbUnaligned, RTALLOC_EFENCE_NOMAN_FILLER, cbBlock - RTALLOC_EFENCE_SIZE - cbUnaligned);
334# endif
335#else
336 void *pvEFence = (char *)pvBlock + (cbBlock - RTALLOC_EFENCE_SIZE);
337 void *pv = (char *)pvEFence - cbAligned;
338# ifdef RTALLOC_EFENCE_NOMAN_FILLER
339 memset(pvBlock, RTALLOC_EFENCE_NOMAN_FILLER, cbBlock - RTALLOC_EFENCE_SIZE - cbAligned);
340 memset((char *)pv + cbUnaligned, RTALLOC_EFENCE_NOMAN_FILLER, cbAligned - cbUnaligned);
341# endif
342#endif
343
344#ifdef RTALLOC_EFENCE_FENCE_FILLER
345 memset(pvEFence, RTALLOC_EFENCE_FENCE_FILLER, RTALLOC_EFENCE_SIZE);
346#endif
347 int rc = RTMemProtect(pvEFence, RTALLOC_EFENCE_SIZE, RTMEM_PROT_NONE);
348 if (!rc)
349 {
350#ifdef RTALLOC_EFENCE_TRACE
351 rtmemBlockInsert(pBlock, pv);
352#endif
353 if (enmType == RTMEMTYPE_RTMEMALLOCZ)
354 memset(pv, 0, cbUnaligned);
355#ifdef RTALLOC_EFENCE_FILLER
356 else
357 memset(pv, RTALLOC_EFENCE_FILLER, cbUnaligned);
358#endif
359
360 rtmemLog(pszOp, "returns %p (pvBlock=%p cbBlock=%#x pvEFence=%p cbUnaligned=%#x)\n", pv, pvBlock, cbBlock, pvEFence, cbUnaligned);
361 return pv;
362 }
363 rtmemComplain(pszOp, "RTMemProtect failed, pvEFence=%p size %d, rc=%d\n", pvEFence, RTALLOC_EFENCE_SIZE, rc);
364 RTMemPageFree(pvBlock, cbBlock);
365 }
366 else
367 rtmemComplain(pszOp, "Failed to allocated %lu (%lu) bytes.\n", (unsigned long)cbBlock, (unsigned long)cbUnaligned);
368
369#ifdef RTALLOC_EFENCE_TRACE
370 rtmemBlockFree(pBlock);
371#endif
372 return NULL;
373}
374
375
376/**
377 * Internal free.
378 */
379RTDECL(void) rtR3MemFree(const char *pszOp, RTMEMTYPE enmType, void *pv, void *pvCaller, RT_SRC_POS_DECL)
380{
381 NOREF(enmType); RT_SRC_POS_NOREF();
382
383 /*
384 * Simple case.
385 */
386 if (!pv)
387 return;
388
389 /*
390 * Check watch points.
391 */
392 for (unsigned i = 0; i < RT_ELEMENTS(gapvRTMemFreeWatch); i++)
393 if (gapvRTMemFreeWatch[i] == pv)
394 RTAssertDoPanic();
395
396#ifdef RTALLOC_EFENCE_TRACE
397 /*
398 * Find the block.
399 */
400 PRTMEMBLOCK pBlock = rtmemBlockRemove(pv);
401 if (pBlock)
402 {
403 if (gfRTMemFreeLog)
404 RTLogPrintf("RTMem %s: pv=%p pvCaller=%p cbUnaligned=%#x\n", pszOp, pv, pvCaller, pBlock->cbUnaligned);
405
406# ifdef RTALLOC_EFENCE_NOMAN_FILLER
407 /*
408 * Check whether the no man's land is untouched.
409 */
410# ifdef RTALLOC_EFENCE_IN_FRONT
411 void *pvWrong = ASMMemIsAll8((char *)pv + pBlock->cbUnaligned,
412 RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) - pBlock->cbUnaligned,
413 RTALLOC_EFENCE_NOMAN_FILLER);
414# else
415 /* Alignment must match allocation alignment in rtMemAlloc(). */
416 void *pvWrong = ASMMemIsAll8((char *)pv + pBlock->cbUnaligned,
417 pBlock->cbAligned - pBlock->cbUnaligned,
418 RTALLOC_EFENCE_NOMAN_FILLER);
419 if (pvWrong)
420 RTAssertDoPanic();
421 pvWrong = ASMMemIsAll8((void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK),
422 RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) - pBlock->cbAligned,
423 RTALLOC_EFENCE_NOMAN_FILLER);
424# endif
425 if (pvWrong)
426 RTAssertDoPanic();
427# endif
428
429# ifdef RTALLOC_EFENCE_FREE_FILL
430 /*
431 * Fill the user part of the block.
432 */
433 memset(pv, RTALLOC_EFENCE_FREE_FILL, pBlock->cbUnaligned);
434# endif
435
436# if defined(RTALLOC_EFENCE_FREE_DELAYED) && RTALLOC_EFENCE_FREE_DELAYED > 0
437 /*
438 * We're doing delayed freeing.
439 * That means we'll expand the E-fence to cover the entire block.
440 */
441 int rc = RTMemProtect(pv, pBlock->cbAligned, RTMEM_PROT_NONE);
442 if (RT_SUCCESS(rc))
443 {
444 /*
445 * Insert it into the free list and process pending frees.
446 */
447 rtmemBlockDelayInsert(pBlock);
448 while ((pBlock = rtmemBlockDelayRemove()) != NULL)
449 {
450 pv = pBlock->Core.Key;
451# ifdef RTALLOC_EFENCE_IN_FRONT
452 void *pvBlock = (char *)pv - RTALLOC_EFENCE_SIZE;
453# else
454 void *pvBlock = (void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK);
455# endif
456 size_t cbBlock = RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE;
457 rc = RTMemProtect(pvBlock, cbBlock, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
458 if (RT_SUCCESS(rc))
459 RTMemPageFree(pvBlock, RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE);
460 else
461 rtmemComplain(pszOp, "RTMemProtect(%p, %#x, RTMEM_PROT_READ | RTMEM_PROT_WRITE) -> %d\n", pvBlock, cbBlock, rc);
462 rtmemBlockFree(pBlock);
463 }
464 }
465 else
466 rtmemComplain(pszOp, "Failed to expand the efence of pv=%p cb=%d, rc=%d.\n", pv, pBlock, rc);
467
468# else /* !RTALLOC_EFENCE_FREE_DELAYED */
469
470 /*
471 * Turn of the E-fence and free it.
472 */
473# ifdef RTALLOC_EFENCE_IN_FRONT
474 void *pvBlock = (char *)pv - RTALLOC_EFENCE_SIZE;
475 void *pvEFence = pvBlock;
476# else
477 void *pvBlock = (void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK);
478 void *pvEFence = (char *)pv + pBlock->cb;
479# endif
480 int rc = RTMemProtect(pvEFence, RTALLOC_EFENCE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
481 if (RT_SUCCESS(rc))
482 RTMemPageFree(pvBlock, RT_ALIGN_Z(pBlock->cbAligned, PAGE_SIZE) + RTALLOC_EFENCE_SIZE);
483 else
484 rtmemComplain(pszOp, "RTMemProtect(%p, %#x, RTMEM_PROT_READ | RTMEM_PROT_WRITE) -> %d\n", pvEFence, RTALLOC_EFENCE_SIZE, rc);
485 rtmemBlockFree(pBlock);
486
487# endif /* !RTALLOC_EFENCE_FREE_DELAYED */
488 }
489 else
490 rtmemComplain(pszOp, "pv=%p not found! Incorrect free!\n", pv);
491
492#else /* !RTALLOC_EFENCE_TRACE */
493
494 /*
495 * We have no size tracking, so we're not doing any freeing because
496 * we cannot if the E-fence is after the block.
497 * Let's just expand the E-fence to the first page of the user bit
498 * since we know that it's around.
499 */
500 int rc = RTMemProtect((void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK), PAGE_SIZE, RTMEM_PROT_NONE);
501 if (RT_FAILURE(rc))
502 rtmemComplain(pszOp, "RTMemProtect(%p, PAGE_SIZE, RTMEM_PROT_NONE) -> %d\n", (void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK), rc);
503#endif /* !RTALLOC_EFENCE_TRACE */
504}
505
506
507/**
508 * Internal realloc.
509 */
510RTDECL(void *) rtR3MemRealloc(const char *pszOp, RTMEMTYPE enmType, void *pvOld, size_t cbNew,
511 const char *pszTag, void *pvCaller, RT_SRC_POS_DECL)
512{
513 /*
514 * Allocate new and copy.
515 */
516 if (!pvOld)
517 return rtR3MemAlloc(pszOp, enmType, cbNew, cbNew, pszTag, pvCaller, RT_SRC_POS_ARGS);
518 if (!cbNew)
519 {
520 rtR3MemFree(pszOp, RTMEMTYPE_RTMEMREALLOC, pvOld, pvCaller, RT_SRC_POS_ARGS);
521 return NULL;
522 }
523
524#ifdef RTALLOC_EFENCE_TRACE
525
526 /*
527 * Get the block, allocate the new, copy the data, free the old one.
528 */
529 PRTMEMBLOCK pBlock = rtmemBlockGet(pvOld);
530 if (pBlock)
531 {
532 void *pvRet = rtR3MemAlloc(pszOp, enmType, cbNew, cbNew, pszTag, pvCaller, RT_SRC_POS_ARGS);
533 if (pvRet)
534 {
535 memcpy(pvRet, pvOld, RT_MIN(cbNew, pBlock->cbUnaligned));
536 rtR3MemFree(pszOp, RTMEMTYPE_RTMEMREALLOC, pvOld, pvCaller, RT_SRC_POS_ARGS);
537 }
538 return pvRet;
539 }
540 else
541 rtmemComplain(pszOp, "pvOld=%p was not found!\n", pvOld);
542 return NULL;
543
544#else /* !RTALLOC_EFENCE_TRACE */
545
546 rtmemComplain(pszOp, "Not supported if RTALLOC_EFENCE_TRACE isn't defined!\n");
547 return NULL;
548
549#endif /* !RTALLOC_EFENCE_TRACE */
550}
551
552
553
554
555RTDECL(void *) RTMemEfTmpAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
556{
557 return rtR3MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
558}
559
560
561RTDECL(void *) RTMemEfTmpAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
562{
563 return rtR3MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
564}
565
566
567RTDECL(void) RTMemEfTmpFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW
568{
569 if (pv)
570 rtR3MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, ASMReturnAddress(), RT_SRC_POS_ARGS);
571}
572
573
574RTDECL(void *) RTMemEfAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
575{
576 return rtR3MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
577}
578
579
580RTDECL(void *) RTMemEfAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
581{
582 return rtR3MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
583}
584
585
586RTDECL(void *) RTMemEfAllocVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
587{
588 size_t cbAligned;
589 if (cbUnaligned >= 16)
590 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
591 else
592 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
593 return rtR3MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
594}
595
596
597RTDECL(void *) RTMemEfAllocZVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
598{
599 size_t cbAligned;
600 if (cbUnaligned >= 16)
601 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
602 else
603 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
604 return rtR3MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
605}
606
607
608RTDECL(void *) RTMemEfRealloc(void *pvOld, size_t cbNew, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
609{
610 return rtR3MemRealloc("Realloc", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), RT_SRC_POS_ARGS);
611}
612
613
614RTDECL(void) RTMemEfFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW
615{
616 if (pv)
617 rtR3MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, ASMReturnAddress(), RT_SRC_POS_ARGS);
618}
619
620
621RTDECL(void *) RTMemEfDup(const void *pvSrc, size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
622{
623 void *pvDst = RTMemEfAlloc(cb, pszTag, RT_SRC_POS_ARGS);
624 if (pvDst)
625 memcpy(pvDst, pvSrc, cb);
626 return pvDst;
627}
628
629
630RTDECL(void *) RTMemEfDupEx(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW
631{
632 void *pvDst = RTMemEfAlloc(cbSrc + cbExtra, pszTag, RT_SRC_POS_ARGS);
633 if (pvDst)
634 {
635 memcpy(pvDst, pvSrc, cbSrc);
636 memset((uint8_t *)pvDst + cbSrc, 0, cbExtra);
637 }
638 return pvDst;
639}
640
641
642
643
644/*
645 *
646 * The NP (no position) versions.
647 *
648 */
649
650
651
652RTDECL(void *) RTMemEfTmpAllocNP(size_t cb, const char *pszTag) RT_NO_THROW
653{
654 return rtR3MemAlloc("TmpAlloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
655}
656
657
658RTDECL(void *) RTMemEfTmpAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW
659{
660 return rtR3MemAlloc("TmpAllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
661}
662
663
664RTDECL(void) RTMemEfTmpFreeNP(void *pv) RT_NO_THROW
665{
666 if (pv)
667 rtR3MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, ASMReturnAddress(), NULL, 0, NULL);
668}
669
670
671RTDECL(void *) RTMemEfAllocNP(size_t cb, const char *pszTag) RT_NO_THROW
672{
673 return rtR3MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
674}
675
676
677RTDECL(void *) RTMemEfAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW
678{
679 return rtR3MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cb, cb, pszTag, ASMReturnAddress(), NULL, 0, NULL);
680}
681
682
683RTDECL(void *) RTMemEfAllocVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW
684{
685 size_t cbAligned;
686 if (cbUnaligned >= 16)
687 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
688 else
689 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
690 return rtR3MemAlloc("Alloc", RTMEMTYPE_RTMEMALLOC, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), NULL, 0, NULL);
691}
692
693
694RTDECL(void *) RTMemEfAllocZVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW
695{
696 size_t cbAligned;
697 if (cbUnaligned >= 16)
698 cbAligned = RT_ALIGN_Z(cbUnaligned, 16);
699 else
700 cbAligned = RT_ALIGN_Z(cbUnaligned, sizeof(void *));
701 return rtR3MemAlloc("AllocZ", RTMEMTYPE_RTMEMALLOCZ, cbUnaligned, cbAligned, pszTag, ASMReturnAddress(), NULL, 0, NULL);
702}
703
704
705RTDECL(void *) RTMemEfReallocNP(void *pvOld, size_t cbNew, const char *pszTag) RT_NO_THROW
706{
707 return rtR3MemRealloc("Realloc", RTMEMTYPE_RTMEMREALLOC, pvOld, cbNew, pszTag, ASMReturnAddress(), NULL, 0, NULL);
708}
709
710
711RTDECL(void) RTMemEfFreeNP(void *pv) RT_NO_THROW
712{
713 if (pv)
714 rtR3MemFree("Free", RTMEMTYPE_RTMEMFREE, pv, ASMReturnAddress(), NULL, 0, NULL);
715}
716
717
718RTDECL(void *) RTMemEfDupNP(const void *pvSrc, size_t cb, const char *pszTag) RT_NO_THROW
719{
720 void *pvDst = RTMemEfAlloc(cb, pszTag, NULL, 0, NULL);
721 if (pvDst)
722 memcpy(pvDst, pvSrc, cb);
723 return pvDst;
724}
725
726
727RTDECL(void *) RTMemEfDupExNP(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag) RT_NO_THROW
728{
729 void *pvDst = RTMemEfAlloc(cbSrc + cbExtra, pszTag, NULL, 0, NULL);
730 if (pvDst)
731 {
732 memcpy(pvDst, pvSrc, cbSrc);
733 memset((uint8_t *)pvDst + cbSrc, 0, cbExtra);
734 }
735 return pvDst;
736}
737
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