VirtualBox

source: vbox/trunk/include/iprt/cpp/mem.h@ 39654

Last change on this file since 39654 was 36529, checked in by vboxsync, 14 years ago

iprt::non_copyable -> RTCNonCopyable (now in utils.h), moved and renamed RTMemAutoPtr et al out of iprt/mem.h.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.4 KB
Line 
1/** @file
2 * IPRT - C++ Memory Resource Management.
3 */
4
5/*
6 * Copyright (C) 2006-2011 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_cpp_mem_h
27#define ___iprt_cpp_mem_h
28
29#include <iprt/cpp/autores.h>
30#include <iprt/assert.h>
31#include <iprt/mem.h>
32
33/** @defgroup grp_rt_cpp_autores_mem C++ Memory Resource Management
34 * @ingroup grp_rt_cpp_autores
35 * @{
36 */
37
38/**
39 * Template function wrapping RTMemFree to get the correct a_fnDestruct
40 * signature for RTCAutoRes.
41 *
42 * We can't use a more complex template here, because the g++ on RHEL 3
43 * chokes on it with an internal compiler error.
44 *
45 * @tparam T The data type that's being managed.
46 * @param aMem Pointer to the memory that should be free.
47 */
48template <class T>
49inline void RTCMemAutoDestructor(T *aMem) RT_NO_THROW
50{
51 RTMemFree(aMem);
52}
53
54
55/**
56 * RTCMemAutoPtr allocator which uses RTMemTmpAlloc().
57 *
58 * @returns Allocated memory on success, NULL on failure.
59 * @param pvOld What to reallocate, shall always be NULL.
60 * @param cbNew The amount of memory to allocate (in bytes).
61 */
62inline void *RTCMemTmpAutoAllocator(void *pvOld, size_t cbNew) RT_NO_THROW
63{
64 AssertReturn(!pvOld, NULL);
65 return RTMemTmpAlloc(cbNew);
66}
67
68
69/**
70 * Template function wrapping RTMemTmpFree to get the correct a_fnDestruct
71 * signature for RTCAutoRes.
72 *
73 * We can't use a more complex template here, because the g++ on RHEL 3
74 * chokes on it with an internal compiler error.
75 *
76 * @tparam T The data type that's being managed.
77 * @param aMem Pointer to the memory that should be free.
78 */
79template <class T>
80inline void RTCMemTmpAutoDestructor(T *aMem) RT_NO_THROW
81{
82 RTMemTmpFree(aMem);
83}
84
85
86/**
87 * Template function wrapping RTMemEfFree to get the correct a_fnDestruct
88 * signature for RTCAutoRes.
89 *
90 * We can't use a more complex template here, because the g++ on RHEL 3
91 * chokes on it with an internal compiler error.
92 *
93 * @tparam T The data type that's being managed.
94 * @param aMem Pointer to the memory that should be free.
95 */
96template <class T>
97inline void RTCMemEfAutoFree(T *aMem) RT_NO_THROW
98{
99 RTMemEfFreeNP(aMem);
100}
101
102
103/**
104 * Template function wrapping NULL to get the correct NilRes signature
105 * for RTCAutoRes.
106 *
107 * @tparam T The data type that's being managed.
108 * @returns NULL with the right type.
109 */
110template <class T>
111inline T *RTCMemAutoNil(void) RT_NO_THROW
112{
113 return (T *)(NULL);
114}
115
116
117/**
118 * An auto pointer-type template class for managing memory allocating
119 * via C APIs like RTMem (the default).
120 *
121 * The main purpose of this class is to automatically free memory that
122 * isn't explicitly used (release()'ed) when the object goes out of scope.
123 *
124 * As an additional service it can also make the allocations and
125 * reallocations for you if you like, but it can also take of memory
126 * you hand it.
127 *
128 * @tparam T The data type to manage allocations for.
129 * @tparam a_fnDestruct The function to be used to free the resource.
130 * This will default to RTMemFree.
131 * @tparam a_fnAllocator The function to be used to allocate or reallocate
132 * the managed memory.
133 * This is standard realloc() like stuff, so it's
134 * possible to support simple allocation without
135 * actually having to support reallocating memory if
136 * that's a problem. This will default to
137 * RTMemRealloc.
138 */
139template <class T,
140 void a_fnDestruct(T *) = RTCMemAutoDestructor<T>,
141# if defined(RTMEM_WRAP_TO_EF_APIS) && !defined(RTMEM_NO_WRAP_TO_EF_APIS)
142 void *a_fnAllocator(void *, size_t, const char *) = RTMemEfReallocNP
143# else
144 void *a_fnAllocator(void *, size_t, const char *) = RTMemReallocTag
145# endif
146 >
147class RTCMemAutoPtr
148 : public RTCAutoRes<T *, a_fnDestruct, RTCMemAutoNil<T> >
149{
150public:
151 /**
152 * Constructor.
153 *
154 * @param aPtr Memory pointer to manage. Defaults to NULL.
155 */
156 RTCMemAutoPtr(T *aPtr = NULL)
157 : RTCAutoRes<T *, a_fnDestruct, RTCMemAutoNil<T> >(aPtr)
158 {
159 }
160
161 /**
162 * Constructor that allocates memory.
163 *
164 * @param a_cElements The number of elements (of the data type) to allocate.
165 * @param a_fZeroed Whether the memory should be memset with zeros after
166 * the allocation. Defaults to false.
167 */
168 RTCMemAutoPtr(size_t a_cElements, bool a_fZeroed = false)
169 : RTCAutoRes<T *, a_fnDestruct, RTCMemAutoNil<T> >((T *)a_fnAllocator(NULL, a_cElements * sizeof(T), RTMEM_TAG))
170 {
171 if (a_fZeroed && RT_LIKELY(this->get() != NULL))
172 memset(this->get(), '\0', a_cElements * sizeof(T));
173 }
174
175 /**
176 * Free current memory and start managing aPtr.
177 *
178 * @param aPtr Memory pointer to manage.
179 */
180 RTCMemAutoPtr &operator=(T *aPtr)
181 {
182 this->RTCAutoRes<T *, a_fnDestruct, RTCMemAutoNil<T> >::operator=(aPtr);
183 return *this;
184 }
185
186 /**
187 * Dereference with * operator.
188 */
189 T &operator*()
190 {
191 return *this->get();
192 }
193
194 /**
195 * Dereference with -> operator.
196 */
197 T *operator->()
198 {
199 return this->get();
200 }
201
202 /**
203 * Accessed with the subscript operator ([]).
204 *
205 * @returns Reference to the element.
206 * @param a_i The element to access.
207 */
208 T &operator[](size_t a_i)
209 {
210 return this->get()[a_i];
211 }
212
213 /**
214 * Allocates memory and start manage it.
215 *
216 * Any previously managed memory will be freed before making
217 * the new allocation.
218 *
219 * @returns Success indicator.
220 * @retval true if the new allocation succeeds.
221 * @retval false on failure, no memory is associated with the object.
222 *
223 * @param a_cElements The number of elements (of the data type) to allocate.
224 * This defaults to 1.
225 * @param a_fZeroed Whether the memory should be memset with zeros after
226 * the allocation. Defaults to false.
227 */
228 bool alloc(size_t a_cElements = 1, bool a_fZeroed = false)
229 {
230 this->reset(NULL);
231 T *pNewMem = (T *)a_fnAllocator(NULL, a_cElements * sizeof(T), RTMEM_TAG);
232 if (a_fZeroed && RT_LIKELY(pNewMem != NULL))
233 memset(pNewMem, '\0', a_cElements * sizeof(T));
234 this->reset(pNewMem);
235 return pNewMem != NULL;
236 }
237
238 /**
239 * Reallocate or allocates the memory resource.
240 *
241 * Free the old value if allocation fails.
242 *
243 * The content of any additional memory that was allocated is
244 * undefined when using the default allocator.
245 *
246 * @returns Success indicator.
247 * @retval true if the new allocation succeeds.
248 * @retval false on failure, no memory is associated with the object.
249 *
250 * @param a_cElements The new number of elements (of the data type) to
251 * allocate. The size of the allocation is the number of
252 * elements times the size of the data type - this is
253 * currently what's passed down to the a_fnAllocator.
254 * This defaults to 1.
255 */
256 bool realloc(size_t a_cElements = 1)
257 {
258 T *aNewValue = (T *)a_fnAllocator(this->get(), a_cElements * sizeof(T), RTMEM_TAG);
259 if (RT_LIKELY(aNewValue != NULL))
260 this->release();
261 /* We want this both if aNewValue is non-NULL and if it is NULL. */
262 this->reset(aNewValue);
263 return aNewValue != NULL;
264 }
265};
266
267/** @} */
268
269#endif
270
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