aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/linden/indra/llcommon/llmemory.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'linden/indra/llcommon/llmemory.cpp')
-rw-r--r--linden/indra/llcommon/llmemory.cpp312
1 files changed, 312 insertions, 0 deletions
diff --git a/linden/indra/llcommon/llmemory.cpp b/linden/indra/llcommon/llmemory.cpp
new file mode 100644
index 0000000..9bfbf88
--- /dev/null
+++ b/linden/indra/llcommon/llmemory.cpp
@@ -0,0 +1,312 @@
1/**
2 * @file llmemory.cpp
3 * @brief Very special memory allocation/deallocation stuff here
4 *
5 * Copyright (c) 2002-2007, Linden Research, Inc.
6 *
7 * The source code in this file ("Source Code") is provided by Linden Lab
8 * to you under the terms of the GNU General Public License, version 2.0
9 * ("GPL"), unless you have obtained a separate licensing agreement
10 * ("Other License"), formally executed by you and Linden Lab. Terms of
11 * the GPL can be found in doc/GPL-license.txt in this distribution, or
12 * online at http://secondlife.com/developers/opensource/gplv2
13 *
14 * There are special exceptions to the terms and conditions of the GPL as
15 * it is applied to this Source Code. View the full text of the exception
16 * in the file doc/FLOSS-exception.txt in this software distribution, or
17 * online at http://secondlife.com/developers/opensource/flossexception
18 *
19 * By copying, modifying or distributing this software, you acknowledge
20 * that you have read and understood your obligations described above,
21 * and agree to abide by those obligations.
22 *
23 * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
24 * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
25 * COMPLETENESS OR PERFORMANCE.
26 */
27
28#include "linden_common.h"
29
30#include "llmemory.h"
31
32//----------------------------------------------------------------------------
33
34//static
35char* LLMemory::reserveMem = 0;
36
37//static
38void LLMemory::initClass()
39{
40 if (!reserveMem)
41 {
42 reserveMem = new char[16*1024]; // reserve 16K for out of memory error handling
43 }
44}
45
46//static
47void LLMemory::cleanupClass()
48{
49 delete [] reserveMem;
50 reserveMem = NULL;
51}
52
53//static
54void LLMemory::freeReserve()
55{
56 delete [] reserveMem;
57 reserveMem = NULL;
58}
59
60
61//----------------------------------------------------------------------------
62
63//static
64#if MEM_TRACK_TYPE
65S32 LLMemType::sCurDepth = 0;
66S32 LLMemType::sCurType = LLMemType::MTYPE_INIT;
67S32 LLMemType::sType[LLMemType::MTYPE_MAX_DEPTH];
68S32 LLMemType::sMemCount[LLMemType::MTYPE_NUM_TYPES] = { 0 };
69S32 LLMemType::sMaxMemCount[LLMemType::MTYPE_NUM_TYPES] = { 0 };
70S32 LLMemType::sOverheadMem = 0;
71
72const char* LLMemType::sTypeDesc[LLMemType::MTYPE_NUM_TYPES] =
73{
74 "INIT",
75 "STARTUP",
76 "MAIN",
77
78 "IMAGEBASE",
79 "IMAGERAW",
80 "IMAGEFORMATTED",
81
82 "APPFMTIMAGE",
83 "APPRAWIMAGE",
84 "APPAUXRAWIMAGE",
85
86 "DRAWABLE",
87 "OBJECT",
88 "PIPELINE",
89 "AVATAR",
90 "PARTICLES",
91 "REGIONS",
92 "INVENTORY",
93 "ANIMATION",
94 "NETWORK",
95 "PHYSICS",
96 "INTERESTLIST",
97
98 "SCRIPT",
99 "SCRIPT_RUN",
100 "SCRIPT_BYTECODE",
101
102 "IO_PUMP",
103 "IO_TCP",
104 "IO_BUFFER",
105 "IO_HTTP_SERVER"
106 "IO_SD_SERVER",
107 "IO_SD_CLIENT",
108 "IO_URL_REQUEST",
109
110 "TEMP1",
111 "TEMP2",
112 "TEMP3",
113 "TEMP4",
114 "TEMP5",
115 "TEMP6",
116 "TEMP7",
117 "TEMP8",
118 "TEMP9"
119};
120
121#endif
122S32 LLMemType::sTotalMem = 0;
123S32 LLMemType::sMaxTotalMem = 0;
124
125//static
126void LLMemType::printMem()
127{
128 S32 misc_mem = sTotalMem;
129#if MEM_TRACK_TYPE
130 for (S32 i=0; i<MTYPE_NUM_TYPES; i++)
131 {
132 if (sMemCount[i])
133 {
134 llinfos << llformat("MEM: % 20s %03d MB (%03d MB)",sTypeDesc[i],sMemCount[i]>>20,sMaxMemCount[i]>>20) << llendl;
135 }
136 misc_mem -= sMemCount[i];
137 }
138#endif
139 llinfos << llformat("MEM: % 20s %03d MB","MISC",misc_mem>>20) << llendl;
140 llinfos << llformat("MEM: % 20s %03d MB (Max=%d MB)","TOTAL",sTotalMem>>20,sMaxTotalMem>>20) << llendl;
141}
142
143#if MEM_TRACK_MEM
144
145void* ll_allocate (size_t size)
146{
147 if (size == 0)
148 {
149 llwarns << "Null allocation" << llendl;
150 }
151
152 size = (size+3)&~3;
153 S32 alloc_size = size + 4;
154#if MEM_TRACK_TYPE
155 alloc_size += 4;
156#endif
157 char* p = (char*)malloc(alloc_size);
158 if (p == NULL)
159 {
160 LLMemory::freeReserve();
161 llerrs << "Out of memory Error" << llendl;
162 }
163 LLMemType::sTotalMem += size;
164 LLMemType::sMaxTotalMem = llmax(LLMemType::sTotalMem, LLMemType::sMaxTotalMem);
165 LLMemType::sOverheadMem += 4;
166 *(size_t*)p = size;
167 p += 4;
168#if MEM_TRACK_TYPE
169 if (LLMemType::sCurType < 0 || LLMemType::sCurType >= LLMemType::MTYPE_NUM_TYPES)
170 {
171 llerrs << "Memory Type Error: new" << llendl;
172 }
173 LLMemType::sOverheadMem += 4;
174 *(S32*)p = LLMemType::sCurType;
175 p += 4;
176 LLMemType::sMemCount[LLMemType::sCurType] += size;
177 if (LLMemType::sMemCount[LLMemType::sCurType] > LLMemType::sMaxMemCount[LLMemType::sCurType])
178 {
179 LLMemType::sMaxMemCount[LLMemType::sCurType] = LLMemType::sMemCount[LLMemType::sCurType];
180 }
181#endif
182 return (void*)p;
183}
184
185void ll_release (void *pin)
186{
187 if (!pin)
188 {
189 return;
190 }
191 char* p = (char*)pin;
192#if MEM_TRACK_TYPE
193 p -= 4;
194 S32 type = *(S32*)p;
195 if (type < 0 || type >= LLMemType::MTYPE_NUM_TYPES)
196 {
197 llerrs << "Memory Type Error: delete" << llendl;
198 }
199#endif
200 p -= 4;
201 S32 size = *(size_t*)p;
202 LLMemType::sOverheadMem -= 4;
203#if MEM_TRACK_TYPE
204 LLMemType::sMemCount[type] -= size;
205 LLMemType::sOverheadMem -= 4;
206#endif
207 LLMemType::sTotalMem -= size;
208 free(p);
209}
210
211#else
212
213void* ll_allocate (size_t size)
214{
215 if (size == 0)
216 {
217 llwarns << "Null allocation" << llendl;
218 }
219 void *p = malloc(size);
220 if (p == NULL)
221 {
222 LLMemory::freeReserve();
223 llerrs << "Out of memory Error" << llendl;
224 }
225 return p;
226}
227
228void ll_release (void *p)
229{
230 free(p);
231}
232
233#endif
234
235#if MEM_TRACK_MEM
236
237void* operator new (size_t size)
238{
239 return ll_allocate(size);
240}
241
242void* operator new[] (size_t size)
243{
244 return ll_allocate(size);
245}
246
247void operator delete (void *p)
248{
249 ll_release(p);
250}
251
252void operator delete[] (void *p)
253{
254 ll_release(p);
255}
256
257#endif
258
259//----------------------------------------------------------------------------
260
261//static
262LLMutex* LLThreadSafeRefCount::sMutex = 0;
263
264//static
265void LLThreadSafeRefCount::initClass()
266{
267 if (!sMutex)
268 {
269 sMutex = new LLMutex(0);
270 }
271}
272
273//static
274void LLThreadSafeRefCount::cleanupClass()
275{
276 delete sMutex;
277 sMutex = NULL;
278}
279
280
281//----------------------------------------------------------------------------
282
283LLThreadSafeRefCount::LLThreadSafeRefCount() :
284 mRef(0)
285{
286}
287
288LLThreadSafeRefCount::~LLThreadSafeRefCount()
289{
290 if (mRef != 0)
291 {
292 llerrs << "deleting non-zero reference" << llendl;
293 }
294}
295
296//----------------------------------------------------------------------------
297
298LLRefCount::LLRefCount() :
299 mRef(0)
300{
301}
302
303LLRefCount::~LLRefCount()
304{
305 if (mRef != 0)
306 {
307 llerrs << "deleting non-zero reference" << llendl;
308 }
309}
310
311//----------------------------------------------------------------------------
312