VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/memsafer-r3.cpp@ 52334

Last change on this file since 52334 was 52332, checked in by vboxsync, 10 years ago

memsafer-r3.cpp: Fixed bug in the SUP allocator that caused memory outside the allocation to be protected or triggering the ring-3 fallback if vboxdrv was opened.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.1 KB
Line 
1/* $Id: memsafer-r3.cpp 52332 2014-08-11 12:12:59Z vboxsync $ */
2/** @file
3 * IPRT - Memory Allocate for Sensitive Data, generic heap-based implementation.
4 */
5
6/*
7 * Copyright (C) 2006-2014 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 "internal/iprt.h"
32#include <iprt/memsafer.h>
33
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/avl.h>
37#include <iprt/critsect.h>
38#include <iprt/mem.h>
39#include <iprt/once.h>
40#include <iprt/rand.h>
41#include <iprt/param.h>
42#include <iprt/string.h>
43#ifdef IN_SUP_R3
44# include <VBox/sup.h>
45#endif
46
47
48/*******************************************************************************
49* Defined Constants And Macros *
50*******************************************************************************/
51/** Allocation size alignment (power of two). */
52#define RTMEMSAFER_ALIGN 16
53
54
55/*******************************************************************************
56* Structures and Typedefs *
57*******************************************************************************/
58/**
59 * Allocators.
60 */
61typedef enum RTMEMSAFERALLOCATOR
62{
63 /** Invalid method. */
64 RTMEMSAFERALLOCATOR_INVALID = 0,
65 /** RTMemPageAlloc. */
66 RTMEMSAFERALLOCATOR_RTMEMPAGE,
67 /** SUPR3PageAllocEx. */
68 RTMEMSAFERALLOCATOR_SUPR3
69} RTMEMSAFERALLOCATOR;
70
71/**
72 * Tracking node (lives on normal heap).
73 */
74typedef struct RTMEMSAFERNODE
75{
76 /** Node core.
77 * The core key is a scrambled pointer the user memory. */
78 AVLPVNODECORE Core;
79 /** The allocation flags. */
80 uint32_t fFlags;
81 /** The offset into the allocation of the user memory. */
82 uint32_t offUser;
83 /** The requested allocation size. */
84 size_t cbUser;
85 /** The allocation size in pages, this includes the two guard pages. */
86 uint32_t cPages;
87 /** The allocator used for this node. */
88 RTMEMSAFERALLOCATOR enmAllocator;
89} RTMEMSAFERNODE;
90/** Pointer to an allocation tracking node. */
91typedef RTMEMSAFERNODE *PRTMEMSAFERNODE;
92
93
94/*******************************************************************************
95* Global Variables *
96*******************************************************************************/
97/** Init once structure for this module. */
98static RTONCE g_MemSaferOnce = RTONCE_INITIALIZER;
99/** Critical section protecting the allocation tree. */
100static RTCRITSECTRW g_MemSaferCritSect;
101/** Tree of allocation nodes. */
102static AVLPVTREE g_pMemSaferTree;
103/** XOR scrambler value for memory. */
104static uintptr_t g_uMemSaferScramblerXor;
105/** XOR scrambler value pointers. */
106static uintptr_t g_uMemSaferPtrScramblerXor;
107/** Pointer rotate shift count.*/
108static uintptr_t g_cMemSaferPtrScramblerRotate;
109
110
111/**
112 * @callback_method_impl{FNRTONCE, Inits globals.}
113 */
114static DECLCALLBACK(int32_t) rtMemSaferOnceInit(void *pvUserIgnore)
115{
116 g_uMemSaferScramblerXor = (uintptr_t)RTRandU64();
117 g_uMemSaferPtrScramblerXor = (uintptr_t)RTRandU64();
118 g_cMemSaferPtrScramblerRotate = RTRandU32Ex(0, ARCH_BITS - 1);
119 return RTCritSectRwInit(&g_MemSaferCritSect);
120}
121
122
123/**
124 * @callback_method_impl{PFNRTONCECLEANUP, Cleans up globals.}
125 */
126static DECLCALLBACK(void) rtMemSaferOnceTerm(void *pvUser, bool fLazyCleanUpOk)
127{
128 if (!fLazyCleanUpOk)
129 {
130 RTCritSectRwDelete(&g_MemSaferCritSect);
131 Assert(!g_pMemSaferTree);
132 }
133}
134
135
136
137DECLINLINE(void *) rtMemSaferScramblePointer(void *pvUser)
138{
139 uintptr_t uPtr = (uintptr_t)pvUser;
140 uPtr ^= g_uMemSaferPtrScramblerXor;
141#if ARCH_BITS == 64
142 uPtr = ASMRotateRightU64(uPtr, g_cMemSaferPtrScramblerRotate);
143#elif ARCH_BITS == 32
144 uPtr = ASMRotateRightU32(uPtr, g_cMemSaferPtrScramblerRotate);
145#else
146# error "Unsupported/missing ARCH_BITS."
147#endif
148 return (void *)uPtr;
149}
150
151
152/**
153 * Inserts a tracking node into the tree.
154 *
155 * @param pThis The allocation tracking node to insert.
156 */
157static void rtMemSaferNodeInsert(PRTMEMSAFERNODE pThis)
158{
159 RTCritSectRwEnterExcl(&g_MemSaferCritSect);
160 pThis->Core.Key = rtMemSaferScramblePointer(pThis->Core.Key);
161 bool fRc = RTAvlPVInsert(&g_pMemSaferTree, &pThis->Core);
162 RTCritSectRwLeaveExcl(&g_MemSaferCritSect);
163 Assert(fRc);
164}
165
166
167/**
168 * Finds a tracking node into the tree.
169 *
170 * @returns The allocation tracking node for @a pvUser. NULL if not found.
171 * @param pvUser The user pointer to the allocation.
172 */
173static PRTMEMSAFERNODE rtMemSaferNodeLookup(void *pvUser)
174{
175 void *pvKey = rtMemSaferScramblePointer(pvUser);
176 RTCritSectRwEnterShared(&g_MemSaferCritSect);
177 PRTMEMSAFERNODE pThis = (PRTMEMSAFERNODE)RTAvlPVGet(&g_pMemSaferTree, pvKey);
178 RTCritSectRwLeaveShared(&g_MemSaferCritSect);
179 return pThis;
180}
181
182
183/**
184 * Removes a tracking node from the tree.
185 *
186 * @returns The allocation tracking node for @a pvUser. NULL if not found.
187 * @param pvUser The user pointer to the allocation.
188 */
189static PRTMEMSAFERNODE rtMemSaferNodeRemove(void *pvUser)
190{
191 void *pvKey = rtMemSaferScramblePointer(pvUser);
192 RTCritSectRwEnterExcl(&g_MemSaferCritSect);
193 PRTMEMSAFERNODE pThis = (PRTMEMSAFERNODE)RTAvlPVRemove(&g_pMemSaferTree, pvKey);
194 RTCritSectRwLeaveExcl(&g_MemSaferCritSect);
195 return pThis;
196}
197
198
199RTDECL(int) RTMemSaferScramble(void *pv, size_t cb)
200{
201#ifdef RT_STRICT
202 PRTMEMSAFERNODE pThis = rtMemSaferNodeLookup(pv);
203 AssertReturn(pThis, VERR_INVALID_POINTER);
204 AssertMsgReturn(cb == pThis->cbUser, ("cb=%#zx != %#zx\n", cb, pThis->cbUser), VERR_INVALID_PARAMETER);
205#endif
206
207 /* Note! This isn't supposed to be safe, just less obvious. */
208 uintptr_t *pu = (uintptr_t *)pv;
209 cb = RT_ALIGN_Z(cb, RTMEMSAFER_ALIGN);
210 while (cb > 0)
211 {
212 *pu ^= g_uMemSaferScramblerXor;
213 pu++;
214 cb -= sizeof(*pu);
215 }
216
217 return VINF_SUCCESS;
218}
219RT_EXPORT_SYMBOL(RTMemSaferScramble);
220
221
222RTDECL(int) RTMemSaferUnscramble(void *pv, size_t cb)
223{
224#ifdef RT_STRICT
225 PRTMEMSAFERNODE pThis = rtMemSaferNodeLookup(pv);
226 AssertReturn(pThis, VERR_INVALID_POINTER);
227 AssertMsgReturn(cb == pThis->cbUser, ("cb=%#zx != %#zx\n", cb, pThis->cbUser), VERR_INVALID_PARAMETER);
228#endif
229
230 /* Note! This isn't supposed to be safe, just less obvious. */
231 uintptr_t *pu = (uintptr_t *)pv;
232 cb = RT_ALIGN_Z(cb, RTMEMSAFER_ALIGN);
233 while (cb > 0)
234 {
235 *pu ^= g_uMemSaferScramblerXor;
236 pu++;
237 cb -= sizeof(*pu);
238 }
239
240 return VINF_SUCCESS;
241}
242RT_EXPORT_SYMBOL(RTMemSaferUnscramble);
243
244
245/**
246 * Initializes the pages.
247 *
248 * Fills the memory with random bytes in order to make it less obvious where the
249 * secret data starts and ends. We also zero the user memory in case the
250 * allocator does not do this.
251 *
252 * @param pThis The allocation tracer node. The Core.Key member
253 * will be set.
254 * @param pvPages The pages to initialize.
255 */
256static void rtMemSaferInitializePages(PRTMEMSAFERNODE pThis, void *pvPages)
257{
258 RTRandBytes(pvPages, PAGE_SIZE + pThis->offUser);
259
260 uint8_t *pbUser = (uint8_t *)pvPages + PAGE_SIZE + pThis->offUser;
261 pThis->Core.Key = pbUser;
262 RT_BZERO(pbUser, pThis->cbUser); /* paranoia */
263
264 RTRandBytes(pbUser + pThis->cbUser, (size_t)pThis->cPages * PAGE_SIZE - PAGE_SIZE - pThis->offUser - pThis->cbUser);
265}
266
267
268/**
269 * Allocates and initializes pages from the support driver and initializes it.
270 *
271 * @returns VBox status code.
272 * @param pThis The allocator node. Core.Key will be set on successful
273 * return (unscrambled).
274 */
275static int rtMemSaferSupR3AllocPages(PRTMEMSAFERNODE pThis)
276{
277#ifdef IN_SUP_R3
278 /*
279 * Try allocate the memory.
280 */
281 void *pvPages;
282 int rc = SUPR3PageAllocEx(pThis->cPages, 0 /* fFlags */, &pvPages, NULL /* pR0Ptr */, NULL /* paPages */);
283 if (RT_SUCCESS(rc))
284 {
285 rtMemSaferInitializePages(pThis, pvPages);
286
287 /*
288 * Configure the guard pages.
289 * SUPR3PageProtect isn't supported on all hosts, we ignore that.
290 */
291 rc = SUPR3PageProtect(pvPages, NIL_RTR0PTR, 0, PAGE_SIZE, RTMEM_PROT_NONE);
292 if (RT_SUCCESS(rc))
293 {
294 rc = SUPR3PageProtect(pvPages, NIL_RTR0PTR, (pThis->cPages - 1) * PAGE_SIZE, PAGE_SIZE, RTMEM_PROT_NONE);
295 if (RT_SUCCESS(rc))
296 return VINF_SUCCESS;
297 SUPR3PageProtect(pvPages, NIL_RTR0PTR, 0, PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
298 }
299 else if (rc == VERR_NOT_SUPPORTED)
300 return VINF_SUCCESS;
301
302 /* failed. */
303 int rc2 = SUPR3PageFreeEx(pvPages, pThis->cPages); AssertRC(rc2);
304 }
305 return rc;
306#else /* !IN_SUP_R3 */
307 return VERR_NOT_SUPPORTED;
308#endif /* !IN_SUP_R3 */
309}
310
311
312/**
313 * Allocates and initializes pages using the IPRT page allocator API.
314 *
315 * @returns VBox status code.
316 * @param pThis The allocator node. Core.Key will be set on successful
317 * return (unscrambled).
318 */
319static int rtMemSaferMemAllocPages(PRTMEMSAFERNODE pThis)
320{
321 /*
322 * Try allocate the memory.
323 */
324 int rc = VINF_SUCCESS;
325 void *pvPages = RTMemPageAlloc((size_t)pThis->cPages * PAGE_SIZE);
326 if (pvPages)
327 {
328 rtMemSaferInitializePages(pThis, pvPages);
329
330 /*
331 * Configure the guard pages.
332 */
333 rc = RTMemProtect(pvPages, PAGE_SIZE, RTMEM_PROT_NONE);
334 if (RT_SUCCESS(rc))
335 {
336 rc = RTMemProtect((uint8_t *)pvPages + (size_t)(pThis->cPages - 1U) * PAGE_SIZE, PAGE_SIZE, RTMEM_PROT_NONE);
337 if (RT_SUCCESS(rc))
338 return VINF_SUCCESS;
339 rc = RTMemProtect(pvPages, PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
340 }
341
342 /* failed. */
343 RTMemPageFree(pvPages, (size_t)pThis->cPages * PAGE_SIZE);
344 }
345 else
346 rc = VERR_NO_PAGE_MEMORY;
347
348 return rc;
349}
350
351
352RTDECL(int) RTMemSaferAllocZExTag(void **ppvNew, size_t cb, uint32_t fFlags, const char *pszTag) RT_NO_THROW
353{
354 /*
355 * Validate input.
356 */
357 AssertPtrReturn(ppvNew, VERR_INVALID_PARAMETER);
358 *ppvNew = NULL;
359 AssertReturn(cb, VERR_INVALID_PARAMETER);
360 AssertReturn(cb <= 32U*_1M - PAGE_SIZE * 3U, VERR_ALLOCATION_TOO_BIG); /* Max 32 MB minus padding and guard pages. */
361 AssertReturn(!(fFlags & ~RTMEMSAFER_F_VALID_MASK), VERR_INVALID_FLAGS);
362
363 /*
364 * Initialize globals.
365 */
366 int rc = RTOnceEx(&g_MemSaferOnce, rtMemSaferOnceInit, rtMemSaferOnceTerm, NULL);
367 if (RT_SUCCESS(rc))
368 {
369 /*
370 * Allocate a tracker node first.
371 */
372 PRTMEMSAFERNODE pThis = (PRTMEMSAFERNODE)RTMemAllocZ(sizeof(RTMEMSAFERNODE));
373 if (pThis)
374 {
375 /*
376 * Prepare the allocation.
377 */
378 pThis->cbUser = cb;
379 pThis->offUser = (RTRandU32Ex(0, 128) * RTMEMSAFER_ALIGN) & PAGE_OFFSET_MASK;
380
381 size_t cbNeeded = pThis->offUser + pThis->cbUser;
382 cbNeeded = RT_ALIGN_Z(cbNeeded, PAGE_SIZE);
383
384 pThis->cPages = (uint32_t)(cbNeeded / PAGE_SIZE) + 2; /* +2 for guard pages */
385
386 /*
387 * Try allocate the memory, using the best allocator by default and
388 * falling back on the less safe one.
389 */
390 rc = rtMemSaferSupR3AllocPages(pThis);
391 if (RT_SUCCESS(rc))
392 pThis->enmAllocator = RTMEMSAFERALLOCATOR_SUPR3;
393 else if (!(fFlags & RTMEMSAFER_F_REQUIRE_NOT_PAGABLE))
394 {
395 rc = rtMemSaferMemAllocPages(pThis);
396 if (RT_SUCCESS(rc))
397 pThis->enmAllocator = RTMEMSAFERALLOCATOR_RTMEMPAGE;
398 }
399 if (RT_SUCCESS(rc))
400 {
401 /*
402 * Insert the node.
403 */
404 *ppvNew = pThis->Core.Key;
405 rtMemSaferNodeInsert(pThis); /* (Scrambles Core.Key) */
406 return VINF_SUCCESS;
407 }
408
409 RTMemFree(pThis);
410 }
411 else
412 rc = VERR_NO_MEMORY;
413 }
414 return rc;
415}
416RT_EXPORT_SYMBOL(RTMemSaferAllocZExTag);
417
418
419RTDECL(void) RTMemSaferFree(void *pv, size_t cb) RT_NO_THROW
420{
421 if (pv)
422 {
423 PRTMEMSAFERNODE pThis = rtMemSaferNodeRemove(pv);
424 AssertReturnVoid(pThis);
425 AssertMsg(cb == pThis->cbUser, ("cb=%#zx != %#zx\n", cb, pThis->cbUser));
426
427 /*
428 * Wipe the user memory first.
429 */
430 RTMemWipeThoroughly(pv, RT_ALIGN_Z(cb, RTMEMSAFER_ALIGN), 3);
431
432 /*
433 * Free the pages.
434 */
435 uint8_t *pbPages = (uint8_t *)pv - pThis->offUser - PAGE_SIZE;
436 size_t cbPages = (size_t)pThis->cPages * PAGE_SIZE;
437 switch (pThis->enmAllocator)
438 {
439#ifdef IN_SUP_R3
440 case RTMEMSAFERALLOCATOR_SUPR3:
441 SUPR3PageProtect(pbPages, NIL_RTR0PTR, 0, PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
442 SUPR3PageProtect(pbPages, NIL_RTR0PTR, (uint32_t)(cbPages - PAGE_SIZE), PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
443 SUPR3PageFreeEx(pbPages, cbPages);
444 break;
445#endif
446 case RTMEMSAFERALLOCATOR_RTMEMPAGE:
447 RTMemProtect(pbPages, PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
448 RTMemProtect(pbPages + cbPages - PAGE_SIZE, PAGE_SIZE, RTMEM_PROT_READ | RTMEM_PROT_WRITE);
449 RTMemPageFree(pbPages, cbPages);
450 break;
451
452 default:
453 AssertFailed();
454 }
455
456 /*
457 * Free the tracking node.
458 */
459 pThis->Core.Key = NULL;
460 pThis->offUser = 0;
461 pThis->cbUser = 0;
462 RTMemFree(pThis);
463 }
464 else
465 Assert(cb == 0);
466}
467RT_EXPORT_SYMBOL(RTMemSaferFree);
468
469
470/**
471 * The simplest reallocation method: allocate new block, copy over the data,
472 * free old block.
473 */
474static int rtMemSaferReallocSimpler(size_t cbOld, void *pvOld, size_t cbNew, void **ppvNew, uint32_t fFlags, const char *pszTag)
475{
476 void *pvNew;
477 int rc = RTMemSaferAllocZExTag(&pvNew, cbNew, fFlags, pszTag);
478 if (RT_SUCCESS(rc))
479 {
480 memcpy(pvNew, pvOld, RT_MIN(cbNew, cbOld));
481 RTMemSaferFree(pvOld, cbOld);
482 *ppvNew = pvNew;
483 }
484 return rc;
485}
486
487
488RTDECL(int) RTMemSaferReallocZExTag(size_t cbOld, void *pvOld, size_t cbNew, void **ppvNew, uint32_t fFlags, const char *pszTag) RT_NO_THROW
489{
490 int rc;
491 /* Real realloc. */
492 if (cbNew && cbOld)
493 {
494 PRTMEMSAFERNODE pThis = rtMemSaferNodeLookup(pvOld);
495 AssertReturn(pThis, VERR_INVALID_POINTER);
496 AssertMsgStmt(cbOld == pThis->cbUser, ("cbOld=%#zx != %#zx\n", cbOld, pThis->cbUser), cbOld = pThis->cbUser);
497
498 if (pThis->fFlags == fFlags)
499 {
500 if (cbNew > cbOld)
501 {
502 /*
503 * Is the enough room for us to grow?
504 */
505 size_t cbMax = (size_t)(pThis->cPages - 2) * PAGE_SIZE;
506 if (cbNew <= cbMax)
507 {
508 size_t const cbAdded = (cbNew - cbOld);
509 size_t const cbAfter = cbMax - pThis->offUser - cbOld;
510 if (cbAfter >= cbAdded)
511 {
512 /*
513 * Sufficient space after the current allocation.
514 */
515 uint8_t *pbNewSpace = (uint8_t *)pvOld + cbOld;
516 RT_BZERO(pbNewSpace, cbAdded);
517 *ppvNew = pvOld;
518 }
519 else
520 {
521 /*
522 * Have to move the allocation to make enough room at the
523 * end. In order to make it a little less predictable and
524 * maybe avoid a relocation or two in the next call, divide
525 * the page offset by four until it it fits.
526 */
527 AssertReturn(rtMemSaferNodeRemove(pvOld) == pThis, VERR_INTERNAL_ERROR_3);
528 uint32_t offNewUser = pThis->offUser;
529 do
530 offNewUser = offNewUser / 2;
531 while ((pThis->offUser - offNewUser) + cbAfter < cbAdded);
532 offNewUser &= ~(RTMEMSAFER_ALIGN - 1U);
533
534 uint32_t const cbMove = pThis->offUser - offNewUser;
535 uint8_t *pbNew = (uint8_t *)pvOld - cbMove;
536 memmove(pbNew, pvOld, cbOld);
537
538 RT_BZERO(pbNew + cbOld, cbAdded);
539 if (cbMove > cbAdded)
540 RTMemWipeThoroughly(pbNew + cbNew, cbMove - cbAdded, 3);
541
542 pThis->offUser = offNewUser;
543 pThis->Core.Key = pbNew;
544 *ppvNew = pbNew;
545
546 rtMemSaferNodeInsert(pThis);
547 }
548 Assert(((uintptr_t)*ppvNew & PAGE_OFFSET_MASK) == pThis->offUser);
549 pThis->cbUser = cbNew;
550 rc = VINF_SUCCESS;
551 }
552 else
553 {
554 /*
555 * Not enough space, allocate a new block and copy over the data.
556 */
557 rc = rtMemSaferReallocSimpler(cbOld, pvOld, cbNew, ppvNew, fFlags, pszTag);
558 }
559 }
560 else
561 {
562 /*
563 * Shrinking the allocation, just wipe the memory that is no longer
564 * being used.
565 */
566 if (cbNew != cbOld)
567 {
568 uint8_t *pbAbandond = (uint8_t *)pvOld + cbNew;
569 RTMemWipeThoroughly(pbAbandond, cbOld - cbNew, 3);
570 }
571 pThis->cbUser = cbNew;
572 *ppvNew = pvOld;
573 rc = VINF_SUCCESS;
574 }
575 }
576 else if (!pThis->fFlags)
577 {
578 /*
579 * New flags added. Allocate a new block and copy over the old one.
580 */
581 rc = rtMemSaferReallocSimpler(cbOld, pvOld, cbNew, ppvNew, fFlags, pszTag);
582 }
583 else
584 {
585 /* Compatible flags. */
586 AssertMsgFailed(("fFlags=%#x old=%#x\n", fFlags, pThis->fFlags));
587 rc = VERR_INVALID_FLAGS;
588 }
589 }
590 /*
591 * First allocation. Pass it on.
592 */
593 else if (!cbOld)
594 {
595 Assert(pvOld == NULL);
596 rc = RTMemSaferAllocZExTag(ppvNew, cbNew, fFlags, pszTag);
597 }
598 /*
599 * Free operation. Pass it on.
600 */
601 else
602 {
603 RTMemSaferFree(pvOld, cbOld);
604 *ppvNew = NULL;
605 rc = VINF_SUCCESS;
606 }
607 return rc;
608}
609RT_EXPORT_SYMBOL(RTMemSaferReallocZExTag);
610
611
612RTDECL(void *) RTMemSaferAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW
613{
614 void *pvNew = NULL;
615 int rc = RTMemSaferAllocZExTag(&pvNew, cb, 0 /*fFlags*/, pszTag);
616 if (RT_SUCCESS(rc))
617 return pvNew;
618 return NULL;
619}
620RT_EXPORT_SYMBOL(RTMemSaferAllocZTag);
621
622
623RTDECL(void *) RTMemSaferReallocZTag(size_t cbOld, void *pvOld, size_t cbNew, const char *pszTag) RT_NO_THROW
624{
625 void *pvNew = NULL;
626 int rc = RTMemSaferReallocZExTag(cbOld, pvOld, cbNew, &pvNew, 0 /*fFlags*/, pszTag);
627 if (RT_SUCCESS(rc))
628 return pvNew;
629 return NULL;
630}
631RT_EXPORT_SYMBOL(RTMemSaferReallocZTag);
632
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