VirtualBox

source: vbox/trunk/include/iprt/mem.h@ 32036

Last change on this file since 32036 was 31252, checked in by vboxsync, 14 years ago

Made 'DEFS += RTMEM_WRAP_TO_EF_APIS' work better.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.6 KB
Line 
1/** @file
2 * IPRT - Memory Management and Manipulation.
3 */
4
5/*
6 * Copyright (C) 2006-2010 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_mem_h
27#define ___iprt_mem_h
28
29
30#include <iprt/cdefs.h>
31#include <iprt/types.h>
32#ifdef __cplusplus /** @todo remove when spitting. */
33# include <iprt/cpp/autores.h>
34#endif
35
36
37#ifdef IN_RC
38# error "There are no RTMem APIs available Guest Context!"
39#endif
40
41
42/** @defgroup grp_rt_mem RTMem - Memory Management and Manipulation
43 * @ingroup grp_rt
44 * @{
45 */
46
47RT_C_DECLS_BEGIN
48
49/** @def RTMEM_ALIGNMENT
50 * The alignment of the memory blocks returned by RTMemAlloc(), RTMemAllocZ(),
51 * RTMemRealloc(), RTMemTmpAlloc() and RTMemTmpAllocZ() for allocations greater
52 * than RTMEM_ALIGNMENT.
53 *
54 * @note This alignment is not forced if the electric fence is active!
55 */
56#define RTMEM_ALIGNMENT 8
57
58/** @def RTMEM_TAG
59 * The default allocation tag used by the RTMem allocation APIs.
60 *
61 * When not defined before the inclusion of iprt/mem.h or iprt/memobj.h, this
62 * will default to the pointer to the current file name. The memory API will
63 * make of use of this as pointer to a volatile but read-only string.
64 */
65#ifndef RTMEM_TAG
66# define RTMEM_TAG (__FILE__)
67#endif
68
69
70/** @name Allocate temporary memory.
71 * @{ */
72/**
73 * Allocates temporary memory with default tag.
74 *
75 * Temporary memory blocks are used for not too large memory blocks which
76 * are believed not to stick around for too long. Using this API instead
77 * of RTMemAlloc() not only gives the heap manager room for optimization
78 * but makes the code easier to read.
79 *
80 * @returns Pointer to the allocated memory.
81 * @returns NULL on failure, assertion raised in strict builds.
82 * @param cb Size in bytes of the memory block to allocated.
83 */
84#define RTMemTmpAlloc(cb) RTMemTmpAllocTag((cb), RTMEM_TAG)
85
86/**
87 * Allocates temporary memory with custom tag.
88 *
89 * Temporary memory blocks are used for not too large memory blocks which
90 * are believed not to stick around for too long. Using this API instead
91 * of RTMemAlloc() not only gives the heap manager room for optimization
92 * but makes the code easier to read.
93 *
94 * @returns Pointer to the allocated memory.
95 * @returns NULL on failure, assertion raised in strict builds.
96 * @param cb Size in bytes of the memory block to allocated.
97 * @param pszTag Allocation tag used for statistics and such.
98 */
99RTDECL(void *) RTMemTmpAllocTag(size_t cb, const char *pszTag) RT_NO_THROW;
100
101/**
102 * Allocates zero'ed temporary memory with default tag.
103 *
104 * Same as RTMemTmpAlloc() but the memory will be zero'ed.
105 *
106 * @returns Pointer to the allocated memory.
107 * @returns NULL on failure, assertion raised in strict builds.
108 * @param cb Size in bytes of the memory block to allocated.
109 */
110#define RTMemTmpAllocZ(cb) RTMemTmpAllocZTag((cb), RTMEM_TAG)
111
112/**
113 * Allocates zero'ed temporary memory with custom tag.
114 *
115 * Same as RTMemTmpAlloc() but the memory will be zero'ed.
116 *
117 * @returns Pointer to the allocated memory.
118 * @returns NULL on failure, assertion raised in strict builds.
119 * @param cb Size in bytes of the memory block to allocated.
120 * @param pszTag Allocation tag used for statistics and such.
121 */
122RTDECL(void *) RTMemTmpAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW;
123
124/**
125 * Free temporary memory.
126 *
127 * @param pv Pointer to memory block.
128 */
129RTDECL(void) RTMemTmpFree(void *pv) RT_NO_THROW;
130
131/** @} */
132
133
134/**
135 * Allocates memory with default tag.
136 *
137 * @returns Pointer to the allocated memory.
138 * @returns NULL on failure, assertion raised in strict builds.
139 * @param cb Size in bytes of the memory block to allocated.
140 * @param pszTag Allocation tag used for statistics and such.
141 */
142#define RTMemAlloc(cb) RTMemAllocTag((cb), RTMEM_TAG)
143
144/**
145 * Allocates memory with custom tag.
146 *
147 * @returns Pointer to the allocated memory.
148 * @returns NULL on failure, assertion raised in strict builds.
149 * @param cb Size in bytes of the memory block to allocated.
150 * @param pszTag Allocation tag used for statistics and such.
151 */
152RTDECL(void *) RTMemAllocTag(size_t cb, const char *pszTag) RT_NO_THROW;
153
154/**
155 * Allocates zero'ed memory with default tag.
156 *
157 * Instead of memset(pv, 0, sizeof()) use this when you want zero'ed
158 * memory. This keeps the code smaller and the heap can skip the memset
159 * in about 0.42% of calls :-).
160 *
161 * @returns Pointer to the allocated memory.
162 * @returns NULL on failure.
163 * @param cb Size in bytes of the memory block to allocated.
164 */
165#define RTMemAllocZ(cb) RTMemAllocZTag((cb), RTMEM_TAG)
166
167/**
168 * Allocates zero'ed memory with custom tag.
169 *
170 * Instead of memset(pv, 0, sizeof()) use this when you want zero'ed
171 * memory. This keeps the code smaller and the heap can skip the memset
172 * in about 0.42% of calls :-).
173 *
174 * @returns Pointer to the allocated memory.
175 * @returns NULL on failure.
176 * @param cb Size in bytes of the memory block to allocated.
177 * @param pszTag Allocation tag used for statistics and such.
178 */
179RTDECL(void *) RTMemAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW;
180
181/**
182 * Wrapper around RTMemAlloc for automatically aligning variable sized
183 * allocations so that the various electric fence heaps works correctly.
184 *
185 * @returns See RTMemAlloc.
186 * @param cbUnaligned The unaligned size.
187 */
188#define RTMemAllocVar(cbUnaligned) RTMemAllocVarTag((cbUnaligned), RTMEM_TAG)
189
190/**
191 * Wrapper around RTMemAllocTag for automatically aligning variable sized
192 * allocations so that the various electric fence heaps works correctly.
193 *
194 * @returns See RTMemAlloc.
195 * @param cbUnaligned The unaligned size.
196 * @param pszTag Allocation tag used for statistics and such.
197 */
198RTDECL(void *) RTMemAllocVarTag(size_t cbUnaligned, const char *pszTag) RT_NO_THROW;
199
200/**
201 * Wrapper around RTMemAllocZ for automatically aligning variable sized
202 * allocations so that the various electric fence heaps works correctly.
203 *
204 * @returns See RTMemAllocZ.
205 * @param cbUnaligned The unaligned size.
206 */
207#define RTMemAllocZVar(cbUnaligned) RTMemAllocZVarTag((cbUnaligned), RTMEM_TAG)
208
209/**
210 * Wrapper around RTMemAllocZTag for automatically aligning variable sized
211 * allocations so that the various electric fence heaps works correctly.
212 *
213 * @returns See RTMemAllocZ.
214 * @param cbUnaligned The unaligned size.
215 * @param pszTag Allocation tag used for statistics and such.
216 */
217RTDECL(void *) RTMemAllocZVarTag(size_t cbUnaligned, const char *pszTag) RT_NO_THROW;
218
219/**
220 * Duplicates a chunk of memory into a new heap block (default tag).
221 *
222 * @returns New heap block with the duplicate data.
223 * @returns NULL if we're out of memory.
224 * @param pvSrc The memory to duplicate.
225 * @param cb The amount of memory to duplicate.
226 */
227#define RTMemDup(pvSrc, cb) RTMemDupTag((pvSrc), (cb), RTMEM_TAG)
228
229/**
230 * Duplicates a chunk of memory into a new heap block (custom tag).
231 *
232 * @returns New heap block with the duplicate data.
233 * @returns NULL if we're out of memory.
234 * @param pvSrc The memory to duplicate.
235 * @param cb The amount of memory to duplicate.
236 * @param pszTag Allocation tag used for statistics and such.
237 */
238RTDECL(void *) RTMemDupTag(const void *pvSrc, size_t cb, const char *pszTag) RT_NO_THROW;
239
240/**
241 * Duplicates a chunk of memory into a new heap block with some additional
242 * zeroed memory (default tag).
243 *
244 * @returns New heap block with the duplicate data.
245 * @returns NULL if we're out of memory.
246 * @param pvSrc The memory to duplicate.
247 * @param cbSrc The amount of memory to duplicate.
248 * @param cbExtra The amount of extra memory to allocate and zero.
249 */
250#define RTMemDupEx(pvSrc, cbSrc, cbExtra) RTMemDupExTag((pvSrc), (cbSrc), (cbExtra), RTMEM_TAG)
251
252/**
253 * Duplicates a chunk of memory into a new heap block with some additional
254 * zeroed memory (default tag).
255 *
256 * @returns New heap block with the duplicate data.
257 * @returns NULL if we're out of memory.
258 * @param pvSrc The memory to duplicate.
259 * @param cbSrc The amount of memory to duplicate.
260 * @param cbExtra The amount of extra memory to allocate and zero.
261 * @param pszTag Allocation tag used for statistics and such.
262 */
263RTDECL(void *) RTMemDupExTag(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag) RT_NO_THROW;
264
265/**
266 * Reallocates memory with default tag.
267 *
268 * @returns Pointer to the allocated memory.
269 * @returns NULL on failure.
270 * @param pvOld The memory block to reallocate.
271 * @param cbNew The new block size (in bytes).
272 */
273#define RTMemRealloc(pvOld, cbNew) RTMemReallocTag((pvOld), (cbNew), RTMEM_TAG)
274
275/**
276 * Reallocates memory with custom tag.
277 *
278 * @returns Pointer to the allocated memory.
279 * @returns NULL on failure.
280 * @param pvOld The memory block to reallocate.
281 * @param cbNew The new block size (in bytes).
282 * @param pszTag Allocation tag used for statistics and such.
283 */
284RTDECL(void *) RTMemReallocTag(void *pvOld, size_t cbNew, const char *pszTag) RT_NO_THROW;
285
286/**
287 * Frees memory.
288 *
289 * @param pv Pointer to memory block.
290 */
291RTDECL(void) RTMemFree(void *pv) RT_NO_THROW;
292
293
294
295/**
296 * Allocates memory which may contain code (default tag).
297 *
298 * @returns Pointer to the allocated memory.
299 * @returns NULL on failure.
300 * @param cb Size in bytes of the memory block to allocate.
301 */
302#define RTMemExecAlloc(cb) RTMemExecAllocTag((cb), RTMEM_TAG)
303
304/**
305 * Allocates memory which may contain code (custom tag).
306 *
307 * @returns Pointer to the allocated memory.
308 * @returns NULL on failure.
309 * @param cb Size in bytes of the memory block to allocate.
310 * @param pszTag Allocation tag used for statistics and such.
311 */
312RTDECL(void *) RTMemExecAllocTag(size_t cb, const char *pszTag) RT_NO_THROW;
313
314/**
315 * Free executable/read/write memory allocated by RTMemExecAlloc().
316 *
317 * @param pv Pointer to memory block.
318 */
319RTDECL(void) RTMemExecFree(void *pv) RT_NO_THROW;
320
321#if defined(IN_RING0) && defined(RT_ARCH_AMD64) && defined(RT_OS_LINUX)
322/**
323 * Donate read+write+execute memory to the exec heap.
324 *
325 * This API is specific to AMD64 and Linux/GNU. A kernel module that desires to
326 * use RTMemExecAlloc on AMD64 Linux/GNU will have to donate some statically
327 * allocated memory in the module if it wishes for GCC generated code to work.
328 * GCC can only generate modules that work in the address range ~2GB to ~0
329 * currently.
330 *
331 * The API only accept one single donation.
332 *
333 * @returns IPRT status code.
334 * @param pvMemory Pointer to the memory block.
335 * @param cb The size of the memory block.
336 */
337RTR0DECL(int) RTR0MemExecDonate(void *pvMemory, size_t cb) RT_NO_THROW;
338#endif /* R0+AMD64+LINUX */
339
340/**
341 * Allocate page aligned memory with default tag.
342 *
343 * @returns Pointer to the allocated memory.
344 * @returns NULL if we're out of memory.
345 * @param cb Size of the memory block. Will be rounded up to page size.
346 */
347#define RTMemPageAlloc(cb) RTMemPageAllocTag((cb), RTMEM_TAG)
348
349/**
350 * Allocate page aligned memory with custom tag.
351 *
352 * @returns Pointer to the allocated memory.
353 * @returns NULL if we're out of memory.
354 * @param cb Size of the memory block. Will be rounded up to page size.
355 * @param pszTag Allocation tag used for statistics and such.
356 */
357RTDECL(void *) RTMemPageAllocTag(size_t cb, const char *pszTag) RT_NO_THROW;
358
359/**
360 * Allocate zero'ed page aligned memory with default tag.
361 *
362 * @returns Pointer to the allocated memory.
363 * @returns NULL if we're out of memory.
364 * @param cb Size of the memory block. Will be rounded up to page size.
365 */
366#define RTMemPageAllocZ(cb) RTMemPageAllocZTag((cb), RTMEM_TAG)
367
368/**
369 * Allocate zero'ed page aligned memory with custom tag.
370 *
371 * @returns Pointer to the allocated memory.
372 * @returns NULL if we're out of memory.
373 * @param cb Size of the memory block. Will be rounded up to page size.
374 * @param pszTag Allocation tag used for statistics and such.
375 */
376RTDECL(void *) RTMemPageAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW;
377
378/**
379 * Free a memory block allocated with RTMemPageAlloc() or RTMemPageAllocZ().
380 *
381 * @param pv Pointer to the block as it was returned by the allocation function.
382 * NULL will be ignored.
383 * @param cb The allocation size. Will be rounded up to page size.
384 * Ignored if @a pv is NULL.
385 */
386RTDECL(void) RTMemPageFree(void *pv, size_t cb) RT_NO_THROW;
387
388/** Page level protection flags for RTMemProtect().
389 * @{
390 */
391/** No access at all. */
392#define RTMEM_PROT_NONE 0
393/** Read access. */
394#define RTMEM_PROT_READ 1
395/** Write access. */
396#define RTMEM_PROT_WRITE 2
397/** Execute access. */
398#define RTMEM_PROT_EXEC 4
399/** @} */
400
401/**
402 * Change the page level protection of a memory region.
403 *
404 * @returns iprt status code.
405 * @param pv Start of the region. Will be rounded down to nearest page boundary.
406 * @param cb Size of the region. Will be rounded up to the nearest page boundary.
407 * @param fProtect The new protection, a combination of the RTMEM_PROT_* defines.
408 */
409RTDECL(int) RTMemProtect(void *pv, size_t cb, unsigned fProtect) RT_NO_THROW;
410
411/**
412 * Goes thru some pains to make sure the specified memory block is thoroughly
413 * scrambled.
414 *
415 * @param pv The start of the memory block.
416 * @param cb The size of the memory block.
417 * @param cMinPasses The minimum number of passes to make.
418 */
419RTDECL(void) RTMemWipeThoroughly(void *pv, size_t cb, size_t cMinPasses) RT_NO_THROW;
420
421#ifdef IN_RING0
422
423/**
424 * Allocates physical contiguous memory (below 4GB).
425 * The allocation is page aligned and the content is undefined.
426 *
427 * @returns Pointer to the memory block. This is page aligned.
428 * @param pPhys Where to store the physical address.
429 * @param cb The allocation size in bytes. This is always
430 * rounded up to PAGE_SIZE.
431 */
432RTR0DECL(void *) RTMemContAlloc(PRTCCPHYS pPhys, size_t cb) RT_NO_THROW;
433
434/**
435 * Frees memory allocated ysing RTMemContAlloc().
436 *
437 * @param pv Pointer to return from RTMemContAlloc().
438 * @param cb The cb parameter passed to RTMemContAlloc().
439 */
440RTR0DECL(void) RTMemContFree(void *pv, size_t cb) RT_NO_THROW;
441
442/**
443 * Copy memory from an user mode buffer into a kernel buffer.
444 *
445 * @retval VINF_SUCCESS on success.
446 * @retval VERR_ACCESS_DENIED on error.
447 *
448 * @param pvDst The kernel mode destination address.
449 * @param R3PtrSrc The user mode source address.
450 * @param cb The number of bytes to copy.
451 */
452RTR0DECL(int) RTR0MemUserCopyFrom(void *pvDst, RTR3PTR R3PtrSrc, size_t cb);
453
454/**
455 * Copy memory from a kernel buffer into a user mode one.
456 *
457 * @retval VINF_SUCCESS on success.
458 * @retval VERR_ACCESS_DENIED on error.
459 *
460 * @param R3PtrDst The user mode destination address.
461 * @param pvSrc The kernel mode source address.
462 * @param cb The number of bytes to copy.
463 */
464RTR0DECL(int) RTR0MemUserCopyTo(RTR3PTR R3PtrDst, void const *pvSrc, size_t cb);
465
466/**
467 * Tests if the specified address is in the user addressable range.
468 *
469 * This function does not check whether the memory at that address is accessible
470 * or anything of that sort, only if the address it self is in the user mode
471 * range.
472 *
473 * @returns true if it's in the user addressable range. false if not.
474 * @param R3Ptr The user mode pointer to test.
475 *
476 * @remarks Some systems may have overlapping kernel and user address ranges.
477 * One prominent example of this is the x86 version of Mac OS X. Use
478 * RTR0MemAreKrnlAndUsrDifferent() to check.
479 */
480RTR0DECL(bool) RTR0MemUserIsValidAddr(RTR3PTR R3Ptr);
481
482/**
483 * Tests if the specified address is in the kernel mode range.
484 *
485 * This function does not check whether the memory at that address is accessible
486 * or anything of that sort, only if the address it self is in the kernel mode
487 * range.
488 *
489 * @returns true if it's in the kernel range. false if not.
490 * @param pv The alleged kernel mode pointer.
491 *
492 * @remarks Some systems may have overlapping kernel and user address ranges.
493 * One prominent example of this is the x86 version of Mac OS X. Use
494 * RTR0MemAreKrnlAndUsrDifferent() to check.
495 */
496RTR0DECL(bool) RTR0MemKernelIsValidAddr(void *pv);
497
498/**
499 * Are user mode and kernel mode address ranges distinctly different.
500 *
501 * This determins whether RTR0MemKernelIsValidAddr and RTR0MemUserIsValidAddr
502 * can be used for deciding whether some arbitrary address is a user mode or a
503 * kernel mode one.
504 *
505 * @returns true if they are, false if not.
506 */
507RTR0DECL(bool) RTR0MemAreKrnlAndUsrDifferent(void);
508
509#endif /* IN_RING0 */
510
511
512/** @name Electrical Fence Version of some APIs.
513 * @{
514 */
515
516/**
517 * Same as RTMemTmpAllocTag() except that it's fenced.
518 *
519 * @returns Pointer to the allocated memory.
520 * @returns NULL on failure.
521 * @param cb Size in bytes of the memory block to allocate.
522 * @param pszTag Allocation tag used for statistics and such.
523 */
524RTDECL(void *) RTMemEfTmpAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
525
526/**
527 * Same as RTMemTmpAllocZTag() except that it's fenced.
528 *
529 * @returns Pointer to the allocated memory.
530 * @returns NULL on failure.
531 * @param cb Size in bytes of the memory block to allocate.
532 * @param pszTag Allocation tag used for statistics and such.
533 */
534RTDECL(void *) RTMemEfTmpAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
535
536/**
537 * Same as RTMemTmpFree() except that it's for fenced memory.
538 *
539 * @param pv Pointer to memory block.
540 */
541RTDECL(void) RTMemEfTmpFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW;
542
543/**
544 * Same as RTMemAllocTag() except that it's fenced.
545 *
546 * @returns Pointer to the allocated memory. Free with RTMemEfFree().
547 * @returns NULL on failure.
548 * @param cb Size in bytes of the memory block to allocate.
549 * @param pszTag Allocation tag used for statistics and such.
550 */
551RTDECL(void *) RTMemEfAlloc(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
552
553/**
554 * Same as RTMemAllocZTag() except that it's fenced.
555 *
556 * @returns Pointer to the allocated memory.
557 * @returns NULL on failure.
558 * @param cb Size in bytes of the memory block to allocate.
559 * @param pszTag Allocation tag used for statistics and such.
560 */
561RTDECL(void *) RTMemEfAllocZ(size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
562
563/**
564 * Same as RTMemAllocVarTag() except that it's fenced.
565 *
566 * @returns Pointer to the allocated memory. Free with RTMemEfFree().
567 * @returns NULL on failure.
568 * @param cbUnaligned Size in bytes of the memory block to allocate.
569 * @param pszTag Allocation tag used for statistics and such.
570 */
571RTDECL(void *) RTMemEfAllocVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
572
573/**
574 * Same as RTMemAllocZVarTag() except that it's fenced.
575 *
576 * @returns Pointer to the allocated memory.
577 * @returns NULL on failure.
578 * @param cbUnaligned Size in bytes of the memory block to allocate.
579 * @param pszTag Allocation tag used for statistics and such.
580 */
581RTDECL(void *) RTMemEfAllocZVar(size_t cbUnaligned, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
582
583/**
584 * Same as RTMemReallocTag() except that it's fenced.
585 *
586 * @returns Pointer to the allocated memory.
587 * @returns NULL on failure.
588 * @param pvOld The memory block to reallocate.
589 * @param cbNew The new block size (in bytes).
590 * @param pszTag Allocation tag used for statistics and such.
591 */
592RTDECL(void *) RTMemEfRealloc(void *pvOld, size_t cbNew, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
593
594/**
595 * Free memory allocated by any of the RTMemEf* allocators.
596 *
597 * @param pv Pointer to memory block.
598 */
599RTDECL(void) RTMemEfFree(void *pv, RT_SRC_POS_DECL) RT_NO_THROW;
600
601/**
602 * Same as RTMemDupTag() except that it's fenced.
603 *
604 * @returns New heap block with the duplicate data.
605 * @returns NULL if we're out of memory.
606 * @param pvSrc The memory to duplicate.
607 * @param cb The amount of memory to duplicate.
608 * @param pszTag Allocation tag used for statistics and such.
609 */
610RTDECL(void *) RTMemEfDup(const void *pvSrc, size_t cb, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
611
612/**
613 * Same as RTMemEfDupExTag except that it's fenced.
614 *
615 * @returns New heap block with the duplicate data.
616 * @returns NULL if we're out of memory.
617 * @param pvSrc The memory to duplicate.
618 * @param cbSrc The amount of memory to duplicate.
619 * @param cbExtra The amount of extra memory to allocate and zero.
620 * @param pszTag Allocation tag used for statistics and such.
621 */
622RTDECL(void *) RTMemEfDupEx(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag, RT_SRC_POS_DECL) RT_NO_THROW;
623
624/** @def RTMEM_WRAP_TO_EF_APIS
625 * Define RTMEM_WRAP_TO_EF_APIS to wrap RTMem APIs to RTMemEf APIs.
626 */
627#if defined(RTMEM_WRAP_TO_EF_APIS) && defined(IN_RING3) && !defined(RTMEM_NO_WRAP_TO_EF_APIS)
628# define RTMemTmpAllocTag(cb, pszTag) RTMemEfTmpAlloc((cb), (pszTag), RT_SRC_POS)
629# define RTMemTmpAllocZTag(cb, pszTag) RTMemEfTmpAllocZ((cb), (pszTag), RT_SRC_POS)
630# define RTMemTmpFree(pv) RTMemEfTmpFree((pv), RT_SRC_POS)
631# define RTMemAllocTag(cb, pszTag) RTMemEfAlloc((cb), (pszTag), RT_SRC_POS)
632# define RTMemAllocZTag(cb, pszTag) RTMemEfAllocZ((cb), (pszTag), RT_SRC_POS)
633# define RTMemAllocVarTag(cbUnaligned, pszTag) RTMemEfAllocVar((cbUnaligned), (pszTag), RT_SRC_POS)
634# define RTMemAllocZVarTag(cbUnaligned, pszTag) RTMemEfAllocZVar((cbUnaligned), (pszTag), RT_SRC_POS)
635# define RTMemReallocTag(pvOld, cbNew, pszTag) RTMemEfRealloc((pvOld), (cbNew), (pszTag), RT_SRC_POS)
636# define RTMemFree(pv) RTMemEfFree((pv), RT_SRC_POS)
637# define RTMemDupTag(pvSrc, cb, pszTag) RTMemEfDup((pvSrc), (cb), (pszTag), RT_SRC_POS)
638# define RTMemDupExTag(pvSrc, cbSrc, cbExtra, pszTag) RTMemEfDupEx((pvSrc), (cbSrc), (cbExtra), (pszTag), RT_SRC_POS)
639#endif
640#ifdef DOXYGEN_RUNNING
641# define RTMEM_WRAP_TO_EF_APIS
642#endif
643
644/**
645 * Fenced drop-in replacement for RTMemTmpAllocTag.
646 * @copydoc RTMemTmpAllocTag
647 */
648RTDECL(void *) RTMemEfTmpAllocNP(size_t cb, const char *pszTag) RT_NO_THROW;
649
650/**
651 * Fenced drop-in replacement for RTMemTmpAllocZTag.
652 * @copydoc RTMemTmpAllocZTag
653 */
654RTDECL(void *) RTMemEfTmpAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW;
655
656/**
657 * Fenced drop-in replacement for RTMemTmpFreeTag.
658 * @copydoc RTMemTmpFreeTag
659 */
660RTDECL(void) RTMemEfTmpFreeNP(void *pv) RT_NO_THROW;
661
662/**
663 * Fenced drop-in replacement for RTMemAllocTag.
664 * @copydoc RTMemAllocTag
665 */
666RTDECL(void *) RTMemEfAllocNP(size_t cb, const char *pszTag) RT_NO_THROW;
667
668/**
669 * Fenced drop-in replacement for RTMemAllocZTag.
670 * @copydoc RTMemAllocZTag
671 */
672RTDECL(void *) RTMemEfAllocZNP(size_t cb, const char *pszTag) RT_NO_THROW;
673
674/**
675 * Fenced drop-in replacement for RTMemAllocVarTag
676 * @copydoc RTMemAllocVarTag
677 */
678RTDECL(void *) RTMemEfAllocVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW;
679
680/**
681 * Fenced drop-in replacement for RTMemAllocZVarTag.
682 * @copydoc RTMemAllocZVarTag
683 */
684RTDECL(void *) RTMemEfAllocZVarNP(size_t cbUnaligned, const char *pszTag) RT_NO_THROW;
685
686/**
687 * Fenced drop-in replacement for RTMemReallocTag.
688 * @copydoc RTMemReallocTag
689 */
690RTDECL(void *) RTMemEfReallocNP(void *pvOld, size_t cbNew, const char *pszTag) RT_NO_THROW;
691
692/**
693 * Fenced drop-in replacement for RTMemFree.
694 * @copydoc RTMemFree
695 */
696RTDECL(void) RTMemEfFreeNP(void *pv) RT_NO_THROW;
697
698/**
699 * Fenced drop-in replacement for RTMemDupExTag.
700 * @copydoc RTMemDupExTag
701 */
702RTDECL(void *) RTMemEfDupNP(const void *pvSrc, size_t cb, const char *pszTag) RT_NO_THROW;
703
704/**
705 * Fenced drop-in replacement for RTMemDupExTag.
706 * @copydoc RTMemDupExTag
707 */
708RTDECL(void *) RTMemEfDupExNP(const void *pvSrc, size_t cbSrc, size_t cbExtra, const char *pszTag) RT_NO_THROW;
709
710/** @} */
711
712RT_C_DECLS_END
713
714
715#ifdef __cplusplus /** @todo Split this out into iprt/cpp/mem.h! */
716# include <iprt/assert.h>
717
718/**
719 * Template function wrapping RTMemFree to get the correct Destruct
720 * signature for RTAutoRes.
721 *
722 * We can't use a more complex template here, because the g++ on RHEL 3
723 * chokes on it with an internal compiler error.
724 *
725 * @tparam T The data type that's being managed.
726 * @param aMem Pointer to the memory that should be free.
727 */
728template <class T>
729inline void RTMemAutoDestructor(T *aMem) RT_NO_THROW
730{
731 RTMemFree(aMem);
732}
733
734
735/**
736 * RTMemAutoPtr allocator which uses RTMemTmpAlloc().
737 *
738 * @returns Allocated memory on success, NULL on failure.
739 * @param pvOld What to reallocate, shall always be NULL.
740 * @param cbNew The amount of memory to allocate (in bytes).
741 */
742inline void *RTMemTmpAutoAllocator(void *pvOld, size_t cbNew) RT_NO_THROW
743{
744 AssertReturn(!pvOld, NULL);
745 return RTMemTmpAlloc(cbNew);
746}
747
748
749/**
750 * Template function wrapping RTMemTmpFree to get the correct Destruct
751 * signature for RTAutoRes.
752 *
753 * We can't use a more complex template here, because the g++ on RHEL 3
754 * chokes on it with an internal compiler error.
755 *
756 * @tparam T The data type that's being managed.
757 * @param aMem Pointer to the memory that should be free.
758 */
759template <class T>
760inline void RTMemTmpAutoDestructor(T *aMem) RT_NO_THROW
761{
762 RTMemTmpFree(aMem);
763}
764
765
766/**
767 * Template function wrapping RTMemEfFree to get the correct Destruct
768 * signature for RTAutoRes.
769 *
770 * We can't use a more complex template here, because the g++ on RHEL 3
771 * chokes on it with an internal compiler error.
772 *
773 * @tparam T The data type that's being managed.
774 * @param aMem Pointer to the memory that should be free.
775 */
776template <class T>
777inline void RTMemEfAutoFree(T *aMem) RT_NO_THROW
778{
779 RTMemEfFreeNP(aMem);
780}
781
782
783/**
784 * Template function wrapping NULL to get the correct NilRes signature
785 * for RTAutoRes.
786 *
787 * @tparam T The data type that's being managed.
788 * @returns NULL with the right type.
789 */
790template <class T>
791inline T * RTMemAutoNil(void) RT_NO_THROW
792{
793 return (T *)(NULL);
794}
795
796
797/**
798 * An auto pointer-type template class for managing memory allocating
799 * via C APIs like RTMem (the default).
800 *
801 * The main purpose of this class is to automatically free memory that
802 * isn't explicitly used (release()'ed) when the object goes out of scope.
803 *
804 * As an additional service it can also make the allocations and
805 * reallocations for you if you like, but it can also take of memory
806 * you hand it.
807 *
808 * @tparam T The data type to manage allocations for.
809 * @tparam Destruct The function to be used to free the resource.
810 * This will default to RTMemFree.
811 * @tparam Allocator The function to be used to allocate or reallocate
812 * the managed memory.
813 * This is standard realloc() like stuff, so it's possible
814 * to support simple allocation without actually having
815 * to support reallocating memory if that's a problem.
816 * This will default to RTMemRealloc.
817 */
818template <class T,
819 void Destruct(T *) = RTMemAutoDestructor<T>,
820# if defined(RTMEM_WRAP_TO_EF_APIS) && !defined(RTMEM_NO_WRAP_TO_EF_APIS)
821 void *Allocator(void *, size_t, const char *) = RTMemEfReallocNP
822# else
823 void *Allocator(void *, size_t, const char *) = RTMemReallocTag
824# endif
825 >
826class RTMemAutoPtr
827 : public RTAutoRes<T *, Destruct, RTMemAutoNil<T> >
828{
829public:
830 /**
831 * Constructor.
832 *
833 * @param aPtr Memory pointer to manage. Defaults to NULL.
834 */
835 RTMemAutoPtr(T *aPtr = NULL)
836 : RTAutoRes<T *, Destruct, RTMemAutoNil<T> >(aPtr)
837 {
838 }
839
840 /**
841 * Constructor that allocates memory.
842 *
843 * @param a_cElements The number of elements (of the data type) to allocate.
844 * @param a_fZeroed Whether the memory should be memset with zeros after
845 * the allocation. Defaults to false.
846 */
847 RTMemAutoPtr(size_t a_cElements, bool a_fZeroed = false)
848 : RTAutoRes<T *, Destruct, RTMemAutoNil<T> >((T *)Allocator(NULL, a_cElements * sizeof(T), RTMEM_TAG))
849 {
850 if (a_fZeroed && RT_LIKELY(this->get() != NULL))
851 memset(this->get(), '\0', a_cElements * sizeof(T));
852 }
853
854 /**
855 * Free current memory and start managing aPtr.
856 *
857 * @param aPtr Memory pointer to manage.
858 */
859 RTMemAutoPtr &operator=(T *aPtr)
860 {
861 this->RTAutoRes<T *, Destruct, RTMemAutoNil<T> >::operator=(aPtr);
862 return *this;
863 }
864
865 /**
866 * Dereference with * operator.
867 */
868 T &operator*()
869 {
870 return *this->get();
871 }
872
873 /**
874 * Dereference with -> operator.
875 */
876 T *operator->()
877 {
878 return this->get();
879 }
880
881 /**
882 * Accessed with the subscript operator ([]).
883 *
884 * @returns Reference to the element.
885 * @param a_i The element to access.
886 */
887 T &operator[](size_t a_i)
888 {
889 return this->get()[a_i];
890 }
891
892 /**
893 * Allocates memory and start manage it.
894 *
895 * Any previously managed memory will be freed before making
896 * the new allocation.
897 *
898 * @returns Success indicator.
899 * @retval true if the new allocation succeeds.
900 * @retval false on failure, no memory is associated with the object.
901 *
902 * @param a_cElements The number of elements (of the data type) to allocate.
903 * This defaults to 1.
904 * @param a_fZeroed Whether the memory should be memset with zeros after
905 * the allocation. Defaults to false.
906 */
907 bool alloc(size_t a_cElements = 1, bool a_fZeroed = false)
908 {
909 this->reset(NULL);
910 T *pNewMem = (T *)Allocator(NULL, a_cElements * sizeof(T), RTMEM_TAG);
911 if (a_fZeroed && RT_LIKELY(pNewMem != NULL))
912 memset(pNewMem, '\0', a_cElements * sizeof(T));
913 this->reset(pNewMem);
914 return pNewMem != NULL;
915 }
916
917 /**
918 * Reallocate or allocates the memory resource.
919 *
920 * Free the old value if allocation fails.
921 *
922 * The content of any additional memory that was allocated is
923 * undefined when using the default allocator.
924 *
925 * @returns Success indicator.
926 * @retval true if the new allocation succeeds.
927 * @retval false on failure, no memory is associated with the object.
928 *
929 * @param a_cElements The new number of elements (of the data type) to
930 * allocate. The size of the allocation is the number of
931 * elements times the size of the data type - this is
932 * currently what's passed down to the Allocator.
933 * This defaults to 1.
934 */
935 bool realloc(size_t a_cElements = 1)
936 {
937 T *aNewValue = (T *)Allocator(this->get(), a_cElements * sizeof(T), RTMEM_TAG);
938 if (RT_LIKELY(aNewValue != NULL))
939 this->release();
940 /* We want this both if aNewValue is non-NULL and if it is NULL. */
941 this->reset(aNewValue);
942 return aNewValue != NULL;
943 }
944};
945
946
947#endif /* __cplusplus */
948
949
950/** @} */
951
952
953#endif
954
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