Ice Sheet System Model  4.18
Code documentation
MemOps.h
Go to the documentation of this file.
1 /* \file MemOps.h
2  * \brief: header file for memory allocations as well as templated new/delete checking for non-null pointers
3  */
4 
5 #ifndef _MEM_OPS_H_
6 #define _MEM_OPS_H_
7 
8 #include <cassert>
9 
10 /* memory management of types T with non-trivial constructors require C++ style memory management*/
11 #define USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
12 /* but for speed one may alternatively use C memory management but can do so safely only for T that are at most
13  * plain old data structures (POD)*/
14 #ifndef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
15 #include <cstdlib>
16 #endif
17 
18 static char const DEFCONTIG = 'f';
19 
20 /* AD (mostly ADOLC) is sensitive to calls to ensurecontiguous. These changes limit its use.*/
21 #ifdef _HAVE_AD_
22 template <class T> T* xNew(unsigned int size, const char* const contig = &DEFCONTIG){
23 #else
24 template <class T> T* xNew(unsigned int size){
25 #endif
26 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
27  T* aT_p=new T[size];
28  assert(aT_p);
29  return aT_p;
30 #else
31  T* aT_p=(T*)malloc(size*sizeof(T));
32  assert(aT_p);
33  return aT_p;
34 #endif
35 }
36 template <class T> T** xNew(unsigned int dim1, unsigned int dim2) { /*{{{*/
37 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
38  T* buf=xNew<T>(dim1*dim2);
39  T** aT_pp =new T*[dim1];
40  assert(aT_pp );
41  for (unsigned int i=0;i<dim1;++i) {
42  aT_pp [i]=buf;
43  buf+=dim2;
44  }
45  return aT_pp ;
46 #else
47  T* buf=(T*)malloc(dim1*dim2*sizeof(T));
48  assert(buf );
49  T** aT_pp =(T**)malloc(dim1*sizeof(T*));
50  assert(aT_pp );
51  for (unsigned int i=0;i<dim1;++i) {
52  aT_pp [i]=buf;
53  buf+=dim2;
54  }
55  return aT_pp ;
56 #endif
57 }/*}}}*/
58 // AD (mostly ADOLC) is sensitive to calls to ensurecontiguous. These changes limit its use.
59 #ifdef _HAVE_AD_
60 template <class T> T* xNewZeroInit(unsigned int size,const char* const contig = &DEFCONTIG){
61 #else
62 template <class T> T* xNewZeroInit(unsigned int size){
63 #endif
64 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
65 #ifdef _HAVE_AD_
66  T* aT_p=xNew<T>(size,contig);
67 #else
68  T* aT_p=xNew<T>(size);
69 #endif
70  for (unsigned int i=0; i<size;++i)
71  aT_p[i]=(T)0;
72  return aT_p;
73 #else
74  T* aT_p=(T*)calloc(size,sizeof(T));
75  assert(aT_p);
76  return aT_p;
77 #endif
78 }
79 template <class T> T** xNewZeroInit(unsigned int dim1, unsigned int dim2) {/*{{{*/
80 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
81  T** aT_pp=xNew<T>(dim1,dim2);
82  for (unsigned int i=0; i<dim1*dim2;++i)
83  (*aT_pp)[i]=(T)0;
84  return aT_pp;
85 #else
86  T* buf=(T*)calloc(dim1*dim2*sizeof(T));
87  assert(buf );
88  T** aT_pp =(T**)malloc(dim1*sizeof(T*));
89  assert(aT_pp );
90  for (unsigned int i=0;i<dim1;++i) {
91  aT_pp [i]=buf;
92  buf+=dim2;
93  }
94  return aT_pp ;
95 #endif
96 }/*}}}*/
97 template <class T> void xDelete(T**& aT_pp) {/*{{{*/
98  if (aT_pp) {
99 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
100  delete [](*aT_pp);
101  delete [](aT_pp);
102 #else
103  free((void*)*aT_pp)
104  free((void**)aT_pp);
105 #endif
106  }
107  aT_pp=0;
108 }/*}}}*/
109 template <class T> void xDelete(T*& aT_p) {/*{{{*/
110  if (aT_p)
111 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
112  delete []aT_p;
113 #else
114  free((void*)aT_p);
115 #endif
116  aT_p=0;
117 }/*}}}*/
118 template <class T> T* xReNew(T* old, unsigned int old_size, unsigned int size) {/*{{{*/
119 #ifdef USE_CXX_MEMORY_MANAGMENT_FOR_NON_POD_TYPES
120  T* aT_p=0;
121  if (!old) { // no old memory
122  if (size)
123  aT_p=xNew<T>(size); // according to realloc behavior in manual page
124  }
125  else { // have old memory
126  if (!size) // but 0 size
127  xDelete<T>(old); // according to realloc behavior in manual page
128  else { // non-zero size
129  assert(old_size); // have old memory - need to have old_size set or this call is bad
130  // allocate new, delete old; ; even for the case when size is
131  // less than old_size we can't just keep the memory unchanged
132  // because otherwise classes that have ctors/dtors with side-effects
133  // may misbehave, for example classes with static instance/operations counters.
134  aT_p=xNew<T>(size);
135  unsigned int iMax=(old_size<size)?old_size:size;
136  for (unsigned int i=0; i<iMax;++i) {
137  // we need to copy the items by explicit assignments
138  aT_p[i]=old[i];
139  }
140  xDelete<T>(old);
141  }
142  }
143  return aT_p;
144 #else
145  T* aT_p=0;
146  aT_p=(T*)realloc((void*)old,size*sizeof(T));
147  if (size)
148  assert(aT_p); // according to realloc behavior in manual page
149  return aT_p;
150 #endif
151 }/*}}}*/
152 template <class T> T* xMemCpy(T* dest, const T* src, unsigned int size) {/*{{{*/
153  assert(dest); assert(src);
154  for (int i=0; i<size;++i)
155  dest[i]=src[i];
156  return dest;
157 };
158 /*}}}*/
159 
160 #if defined(_HAVE_ADOLC_) && !defined(_WRAPPERS_)
161 #include "../Numerics/types.h"
162 template <> adouble* xNew(unsigned int size, const char* const contig);
163 #endif
164 
165 #endif
xReNew
T * xReNew(T *old, unsigned int old_size, unsigned int size)
Definition: MemOps.h:118
xDelete
void xDelete(T **&aT_pp)
Definition: MemOps.h:97
DEFCONTIG
static char const DEFCONTIG
Definition: MemOps.h:18
xNewZeroInit
T * xNewZeroInit(unsigned int size)
Definition: MemOps.h:62
xNew
T * xNew(unsigned int size)
Definition: MemOps.h:24
xMemCpy
T * xMemCpy(T *dest, const T *src, unsigned int size)
Definition: MemOps.h:152