]>
git.saurik.com Git - apple/security.git/blob - OSX/libsecurity_utilities/lib/alloc.h
   2  * Copyright (c) 2000-2004,2011-2012,2014 Apple Inc. All Rights Reserved. 
   4  * @APPLE_LICENSE_HEADER_START@ 
   6  * This file contains Original Code and/or Modifications of Original Code 
   7  * as defined in and that are subject to the Apple Public Source License 
   8  * Version 2.0 (the 'License'). You may not use this file except in 
   9  * compliance with the License. Please obtain a copy of the License at 
  10  * http://www.opensource.apple.com/apsl/ and read it before using this 
  13  * The Original Code and all software distributed under the License are 
  14  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  15  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  16  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, 
  17  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  18  * Please see the License for the specific language governing rights and 
  19  * limitations under the License. 
  21  * @APPLE_LICENSE_HEADER_END@ 
  26 // alloc - abstract malloc-like allocator abstraction 
  31 #include <security_utilities/utilities.h> 
  39 // An abstract allocator superclass, based on the simple malloc/realloc/free paradigm 
  40 // that CDSA loves so much. If you have an allocation strategy and want objects 
  41 // to be allocated through it, inherit from this. 
  46         virtual void *malloc(size_t) throw(std::bad_alloc
) = 0; 
  47         virtual void free(void *) throw() = 0; 
  48         virtual void *realloc(void *, size_t) throw(std::bad_alloc
) = 0; 
  51         // Template versions for added expressiveness. 
  52         // Note that the integers are element counts, not byte sizes. 
  54         template <class T
> T 
*alloc() throw(std::bad_alloc
) 
  55         { return reinterpret_cast<T 
*>(malloc(sizeof(T
))); } 
  57         template <class T
> T 
*alloc(UInt32 count
) throw(std::bad_alloc
) 
  58         { return reinterpret_cast<T 
*>(malloc(sizeof(T
) * count
)); } 
  60         template <class T
> T 
*alloc(T 
*old
, UInt32 count
) throw(std::bad_alloc
) 
  61         { return reinterpret_cast<T 
*>(realloc(old
, sizeof(T
) * count
)); } 
  65         // Happier malloc/realloc for any type. Note that these still have 
  66         // the original (byte-sized) argument profile. 
  68         template <class T
> T 
*malloc(size_t size
) throw(std::bad_alloc
) 
  69         { return reinterpret_cast<T 
*>(malloc(size
)); } 
  71         template <class T
> T 
*realloc(void *addr
, size_t size
) throw(std::bad_alloc
) 
  72         { return reinterpret_cast<T 
*>(realloc(addr
, size
)); } 
  74         // All right, if you *really* have to have calloc... 
  75         void *calloc(size_t size
, size_t count
) throw(std::bad_alloc
) 
  77                 void *addr 
= malloc(size 
* count
); 
  78                 memset(addr
, 0, size 
* count
); 
  82         // compare Allocators for identity 
  83         virtual bool operator == (const Allocator 
&alloc
) const throw(); 
  86         // allocator chooser options 
  92         static Allocator 
&standard(UInt32 request 
= normal
); 
  97 // You'd think that this is operator delete(const T *, Allocator &), but you'd 
  98 // be wrong. Specialized operator delete is only called during constructor cleanup. 
  99 // Use this to cleanly destroy things. 
 102 inline void destroy(T 
*obj
, Allocator 
&alloc
) throw() 
 108 // untyped (release memory only, no destructor call) 
 109 inline void destroy(void *obj
, Allocator 
&alloc
) throw() 
 116 // A mixin class to automagically manage your allocator. 
 117 // To allow allocation (of your object) from any instance of Allocator, 
 118 // inherit from CssmHeap. Your users can then create heap instances of your thing by 
 119 //              new (an-allocator) YourClass(...) 
 121 //              new YourClass(...) 
 122 // for the default allocation source. The beauty is that when someone does a 
 123 //              delete pointer-to-your-instance 
 124 // then the magic fairies will find the allocator that created the object and ask it 
 125 // to free the memory (by calling its free() method). 
 126 // The price of all that glory is memory overhead - typically one pointer per object. 
 130         void *operator new (size_t size
, Allocator 
*alloc 
= NULL
) throw(std::bad_alloc
); 
 131         void operator delete (void *addr
, size_t size
) throw(); 
 132         void operator delete (void *addr
, size_t size
, Allocator 
*alloc
) throw(); 
 137 // Here is a version of auto_ptr that works with Allocators. It is designed 
 138 // to be pretty much a drop-in replacement. It requires an allocator as a constructor 
 139 // argument, of course. 
 140 // Note that CssmAutoPtr<void> is perfectly valid, unlike its auto_ptr look-alike. 
 141 // You can't dereference it, naturally. 
 146         Allocator 
&allocator
; 
 148         CssmAutoPtr(Allocator 
&alloc 
= Allocator::standard()) 
 149         : allocator(alloc
), mine(NULL
) { } 
 150         CssmAutoPtr(Allocator 
&alloc
, T 
*p
) 
 151         : allocator(alloc
), mine(p
) { } 
 153         : allocator(Allocator::standard()), mine(p
) { } 
 154         template <class T1
> CssmAutoPtr(CssmAutoPtr
<T1
> &src
) 
 155         : allocator(src
.allocator
), mine(src
.release()) { } 
 156         template <class T1
> CssmAutoPtr(Allocator 
&alloc
, CssmAutoPtr
<T1
> &src
) 
 157         : allocator(alloc
), mine(src
.release()) { assert(allocator 
== src
.allocator
); } 
 159         ~CssmAutoPtr()                          { allocator
.free(mine
); } 
 161         T 
*get() const throw()          { return mine
; } 
 162         T 
*release()                            { T 
*result 
= mine
; mine 
= NULL
; return result
; } 
 163         void reset()                            { allocator
.free(mine
); mine 
= NULL
; } 
 165         operator T 
* () const           { return mine
; } 
 166         T 
*operator -> () const         { return mine
; } 
 167         T 
&operator * () const          { assert(mine
); return *mine
; } 
 173 // specialization for void (i.e. void *), omitting the troublesome dereferencing ops. 
 175 class CssmAutoPtr
<void> { 
 177         Allocator 
&allocator
; 
 179         CssmAutoPtr(Allocator 
&alloc
) : allocator(alloc
), mine(NULL
) { } 
 180         CssmAutoPtr(Allocator 
&alloc
, void *p
) : allocator(alloc
), mine(p
) { } 
 181         template <class T1
> CssmAutoPtr(CssmAutoPtr
<T1
> &src
) 
 182         : allocator(src
.allocator
), mine(src
.release()) { } 
 183         template <class T1
> CssmAutoPtr(Allocator 
&alloc
, CssmAutoPtr
<T1
> &src
) 
 184         : allocator(alloc
), mine(src
.release()) { assert(allocator 
== src
.allocator
); } 
 186         ~CssmAutoPtr()                          { destroy(mine
, allocator
); } 
 188         void *get() throw()             { return mine
; } 
 189         void *release()                         { void *result 
= mine
; mine 
= NULL
; return result
; } 
 190         void reset()                            { allocator
.free(mine
); mine 
= NULL
; } 
 198 // Convenience forms of CssmAutoPtr that automatically make their (initial) object. 
 201 class CssmNewAutoPtr 
: public CssmAutoPtr
<T
> { 
 203         CssmNewAutoPtr(Allocator 
&alloc 
= Allocator::standard()) 
 204         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T
) { } 
 207         CssmNewAutoPtr(Allocator 
&alloc
, A1 
&arg1
) : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
)) { } 
 209         CssmNewAutoPtr(Allocator 
&alloc
, const A1 
&arg1
) 
 210         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
)) { } 
 212         template <class A1
, class A2
> 
 213         CssmNewAutoPtr(Allocator 
&alloc
, A1 
&arg1
, A2 
&arg2
) 
 214         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
, arg2
)) { } 
 215         template <class A1
, class A2
> 
 216         CssmNewAutoPtr(Allocator 
&alloc
, const A1 
&arg1
, A2 
&arg2
) 
 217         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
, arg2
)) { } 
 218         template <class A1
, class A2
> 
 219         CssmNewAutoPtr(Allocator 
&alloc
, A1 
&arg1
, const A2 
&arg2
) 
 220         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
, arg2
)) { } 
 221         template <class A1
, class A2
> 
 222         CssmNewAutoPtr(Allocator 
&alloc
, const A1 
&arg1
, const A2 
&arg2
) 
 223         : CssmAutoPtr
<T
>(alloc
, new(alloc
) T(arg1
, arg2
)) { } 
 227 } // end namespace Security 
 231 // Global C++ allocation hooks to use Allocators (global namespace) 
 233 inline void *operator new (size_t size
, Allocator 
&allocator
) throw (std::bad_alloc
) 
 234 { return allocator
.malloc(size
); } 
 236 inline void *operator new[] (size_t size
, Allocator 
&allocator
) throw (std::bad_alloc
) 
 237 { return allocator
.malloc(size
); }