Update contrib.
4 ** The author disclaims copyright to this source code. In place of
5 ** a legal notice, here is a blessing:
7 ** May you do good and not evil.
8 ** May you find forgiveness for yourself and forgive others.
9 ** May you share freely, never taking more than you give.
11 *************************************************************************
13 ** This file contains an alternative memory allocation system for SQLite.
14 ** This system is implemented as a wrapper around the system provided
15 ** by the operating system - vanilla malloc(), realloc() and free().
17 ** This system differentiates between requests for "small" allocations
18 ** (by default those of 128 bytes or less) and "large" allocations (all
19 ** others). The 256 byte threshhold is configurable at runtime.
21 ** All requests for large allocations are passed through to the
24 ** Requests for small allocations are met by allocating space within
25 ** one or more larger "chunks" of memory obtained from the default
26 ** memory allocation system. Chunks of memory are usually 64KB or
27 ** larger. The algorithm used to manage space within each chunk is
28 ** the same as that used by mem5.c.
30 ** This strategy is designed to prevent the default memory allocation
31 ** system (usually the system malloc) from suffering from heap
32 ** fragmentation. On some systems, heap fragmentation can cause a
33 ** significant real-time slowdown.
35 ** $Id: mem6.c,v 1.10 2008/09/02 17:52:52 danielk1977 Exp $
38 #ifdef SQLITE_ENABLE_MEMSYS6
40 #include "sqliteInt.h"
43 ** Maximum size of any "small" allocation is ((1<<LOGMAX)*Mem6Chunk.nAtom).
44 ** Mem6Chunk.nAtom is always at least 8, so this is not a practical
50 ** Default value for the "small" allocation size threshold.
52 #define SMALL_MALLOC_DEFAULT_THRESHOLD 256
55 ** Minimum size for a memory chunk.
57 #define MIN_CHUNKSIZE (1<<16)
59 #define LOG2_MINALLOC 4
62 typedef struct Mem6Chunk Mem6Chunk;
63 typedef struct Mem6Link Mem6Link;
66 ** A minimum allocation is an instance of the following structure.
67 ** Larger allocations are an array of these structures where the
68 ** size of the array is a power of 2.
71 int next; /* Index of next free chunk */
72 int prev; /* Index of previous free chunk */
76 ** Masks used for mem5.aCtrl[] elements.
78 #define CTRL_LOGSIZE 0x1f /* Log2 Size of this block relative to POW2_MIN */
79 #define CTRL_FREE 0x20 /* True if not checked out */
85 ** Lists of free blocks of various sizes.
87 int aiFreelist[LOGMAX+1];
89 int nCheckedOut; /* Number of currently outstanding allocations */
92 ** Space for tracking which blocks are checked out and the size
93 ** of each block. One byte per block.
98 ** Memory available for allocation
100 int nAtom; /* Smallest possible allocation in bytes */
101 int nBlock; /* Number of nAtom sized blocks in zPool */
102 u8 *zPool; /* Pointer to memory chunk from which allocations are made */
105 #define MEM6LINK(idx) ((Mem6Link *)(&pChunk->zPool[(idx)*pChunk->nAtom]))
107 static SQLITE_WSD struct Mem6Global {
108 int nMinAlloc; /* Minimum allowed allocation size */
109 int nThreshold; /* Allocs larger than this go to malloc() */
110 int nLogThreshold; /* log2 of (nThreshold/nMinAlloc) */
111 sqlite3_mutex *mutex;
112 Mem6Chunk *pChunk; /* Singly linked list of all memory chunks */
113 } mem6 = { 48642791 };
115 #define mem6 GLOBAL(struct Mem6Global, mem6)
118 ** Unlink the chunk at pChunk->aPool[i] from list it is currently
119 ** on. It should be found on pChunk->aiFreelist[iLogsize].
121 static void memsys6Unlink(Mem6Chunk *pChunk, int i, int iLogsize){
123 assert( i>=0 && i<pChunk->nBlock );
124 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
125 assert( (pChunk->aCtrl[i] & CTRL_LOGSIZE)==iLogsize );
127 next = MEM6LINK(i)->next;
128 prev = MEM6LINK(i)->prev;
130 pChunk->aiFreelist[iLogsize] = next;
132 MEM6LINK(prev)->next = next;
135 MEM6LINK(next)->prev = prev;
140 ** Link the chunk at mem5.aPool[i] so that is on the iLogsize
143 static void memsys6Link(Mem6Chunk *pChunk, int i, int iLogsize){
145 assert( i>=0 && i<pChunk->nBlock );
146 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
147 assert( (pChunk->aCtrl[i] & CTRL_LOGSIZE)==iLogsize );
149 x = MEM6LINK(i)->next = pChunk->aiFreelist[iLogsize];
150 MEM6LINK(i)->prev = -1;
152 assert( x<pChunk->nBlock );
153 MEM6LINK(x)->prev = i;
155 pChunk->aiFreelist[iLogsize] = i;
160 ** Find the first entry on the freelist iLogsize. Unlink that
161 ** entry and return its index.
163 static int memsys6UnlinkFirst(Mem6Chunk *pChunk, int iLogsize){
167 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
168 i = iFirst = pChunk->aiFreelist[iLogsize];
170 memsys6Unlink(pChunk, iFirst, iLogsize);
174 static int roundupLog2(int n){
175 static const char LogTable256[256] = {
179 3, 3, 3, 3, /* 5..8 */
180 4, 4, 4, 4, 4, 4, 4, 4, /* 9..16 */
181 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, /* 17..32 */
182 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
183 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, /* 33..64 */
184 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
185 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
186 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
187 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, /* 65..128 */
188 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
189 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
190 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
191 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
192 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
193 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
194 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
195 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, /* 129..256 */
198 assert(n<=(1<<16) && n>0);
199 if( n<=256 ) return LogTable256[n-1];
200 return LogTable256[(n>>8) - ((n&0xFF)?0:1)] + 8;
204 ** Allocate and return a block of (pChunk->nAtom << iLogsize) bytes from chunk
205 ** pChunk. If the allocation request cannot be satisfied, return 0.
207 static void *chunkMalloc(Mem6Chunk *pChunk, int iLogsize){
208 int i; /* Index of a mem5.aPool[] slot */
209 int iBin; /* Index into mem5.aiFreelist[] */
211 /* Make sure mem5.aiFreelist[iLogsize] contains at least one free
212 ** block. If not, then split a block of the next larger power of
213 ** two in order to create a new free block of size iLogsize.
215 for(iBin=iLogsize; pChunk->aiFreelist[iBin]<0 && iBin<=mem6.nLogThreshold; iBin++){}
216 if( iBin>mem6.nLogThreshold ) return 0;
217 i = memsys6UnlinkFirst(pChunk, iBin);
218 while( iBin>iLogsize ){
222 pChunk->aCtrl[i+newSize] = CTRL_FREE | iBin;
223 memsys6Link(pChunk, i+newSize, iBin);
225 pChunk->aCtrl[i] = iLogsize;
227 /* Return a pointer to the allocated memory. */
228 pChunk->nCheckedOut++;
229 return (void*)&pChunk->zPool[i*pChunk->nAtom];
233 ** Free the allocation pointed to by p, which is guaranteed to be non-zero
234 ** and a part of chunk object pChunk.
236 static void chunkFree(Mem6Chunk *pChunk, void *pOld){
240 /* Set iBlock to the index of the block pointed to by pOld in
241 ** the array of pChunk->nAtom byte blocks pointed to by pChunk->zPool.
243 iBlock = ((u8 *)pOld-pChunk->zPool)/pChunk->nAtom;
245 /* Check that the pointer pOld points to a valid, non-free block. */
246 assert( iBlock>=0 && iBlock<pChunk->nBlock );
247 assert( ((u8 *)pOld-pChunk->zPool)%pChunk->nAtom==0 );
248 assert( (pChunk->aCtrl[iBlock] & CTRL_FREE)==0 );
250 iLogsize = pChunk->aCtrl[iBlock] & CTRL_LOGSIZE;
252 assert( iBlock+size-1<pChunk->nBlock );
254 pChunk->aCtrl[iBlock] |= CTRL_FREE;
255 pChunk->aCtrl[iBlock+size-1] |= CTRL_FREE;
257 pChunk->aCtrl[iBlock] = CTRL_FREE | iLogsize;
258 while( iLogsize<mem6.nLogThreshold ){
260 if( (iBlock>>iLogsize) & 1 ){
261 iBuddy = iBlock - size;
263 iBuddy = iBlock + size;
266 if( (iBuddy+(1<<iLogsize))>pChunk->nBlock ) break;
267 if( pChunk->aCtrl[iBuddy]!=(CTRL_FREE | iLogsize) ) break;
268 memsys6Unlink(pChunk, iBuddy, iLogsize);
271 pChunk->aCtrl[iBuddy] = CTRL_FREE | iLogsize;
272 pChunk->aCtrl[iBlock] = 0;
275 pChunk->aCtrl[iBlock] = CTRL_FREE | iLogsize;
276 pChunk->aCtrl[iBuddy] = 0;
280 pChunk->nCheckedOut--;
281 memsys6Link(pChunk, iBlock, iLogsize);
285 ** Return the actual size of the block pointed to by p, which is guaranteed
286 ** to have been allocated from chunk pChunk.
288 static int chunkSize(Mem6Chunk *pChunk, void *p){
291 int i = ((u8 *)p-pChunk->zPool)/pChunk->nAtom;
292 assert( i>=0 && i<pChunk->nBlock );
293 iSize = pChunk->nAtom * (1 << (pChunk->aCtrl[i]&CTRL_LOGSIZE));
299 ** Return true if there are currently no outstanding allocations.
301 static int chunkIsEmpty(Mem6Chunk *pChunk){
302 return (pChunk->nCheckedOut==0);
306 ** Initialize the buffer zChunk, which is nChunk bytes in size, as
307 ** an Mem6Chunk object. Return a copy of the zChunk pointer.
309 static Mem6Chunk *chunkInit(u8 *zChunk, int nChunk, int nMinAlloc){
312 Mem6Chunk *pChunk = (Mem6Chunk *)zChunk;
314 assert( nChunk>sizeof(Mem6Chunk) );
315 assert( nMinAlloc>sizeof(Mem6Link) );
317 memset(pChunk, 0, sizeof(Mem6Chunk));
318 pChunk->nAtom = nMinAlloc;
319 pChunk->nBlock = ((nChunk-sizeof(Mem6Chunk)) / (pChunk->nAtom+sizeof(u8)));
321 pChunk->zPool = (u8 *)&pChunk[1];
322 pChunk->aCtrl = &pChunk->zPool[pChunk->nBlock*pChunk->nAtom];
324 for(ii=0; ii<=mem6.nLogThreshold; ii++){
325 pChunk->aiFreelist[ii] = -1;
329 for(ii=mem6.nLogThreshold; ii>=0; ii--){
330 int nAlloc = (1<<ii);
331 while( (iOffset+nAlloc)<=pChunk->nBlock ){
332 pChunk->aCtrl[iOffset] = ii | CTRL_FREE;
333 memsys6Link(pChunk, iOffset, ii);
342 static void mem6Enter(void){
343 sqlite3_mutex_enter(mem6.mutex);
346 static void mem6Leave(void){
347 sqlite3_mutex_leave(mem6.mutex);
351 ** Based on the number and size of the currently allocated chunks, return
352 ** the size of the next chunk to allocate, in bytes.
354 static int nextChunkSize(void){
355 int iTotal = MIN_CHUNKSIZE;
357 for(p=mem6.pChunk; p; p=p->pNext){
363 static void freeChunk(Mem6Chunk *pChunk){
364 Mem6Chunk **pp = &mem6.pChunk;
365 for( pp=&mem6.pChunk; *pp!=pChunk; pp = &(*pp)->pNext );
370 static void *memsys6Malloc(int nByte){
373 int nTotal = nByte+8;
376 if( nTotal>mem6.nThreshold ){
380 if( nTotal>(1<<LOG2_MINALLOC) ){
381 iLogsize = roundupLog2(nTotal) - LOG2_MINALLOC;
384 for(pChunk=mem6.pChunk; pChunk; pChunk=pChunk->pNext){
385 p = chunkMalloc(pChunk, iLogsize);
391 int iSize = nextChunkSize();
394 pChunk = chunkInit((u8 *)p, iSize, mem6.nMinAlloc);
395 pChunk->pNext = mem6.pChunk;
396 mem6.pChunk = pChunk;
397 p = chunkMalloc(pChunk, iLogsize);
401 iOffset = ((u8*)p - (u8*)pChunk);
408 ((u32 *)p)[0] = iOffset;
409 ((u32 *)p)[1] = nByte;
410 return &((u32 *)p)[2];
413 static int memsys6Size(void *pPrior){
414 if( pPrior==0 ) return 0;
415 return ((u32*)pPrior)[-1];
418 static void memsys6Free(void *pPrior){
420 void *p = &((u32 *)pPrior)[-2];
421 iSlot = ((u32 *)p)[0];
425 pChunk = (Mem6Chunk *)(&((u8 *)p)[-1 * iSlot]);
426 chunkFree(pChunk, p);
427 if( chunkIsEmpty(pChunk) ){
436 static void *memsys6Realloc(void *p, int nByte){
439 if( p && nByte<=memsys6Size(p) ){
442 p2 = memsys6Malloc(nByte);
444 memcpy(p2, p, memsys6Size(p));
452 static int memsys6Roundup(int n){
453 if( n>mem6.nThreshold ){
456 return (1<<roundupLog2(n));
460 static int memsys6Init(void *pCtx){
461 u8 bMemstat = sqlite3GlobalConfig.bMemstat;
462 mem6.nMinAlloc = (1 << LOG2_MINALLOC);
464 mem6.nThreshold = sqlite3GlobalConfig.nSmall;
465 if( mem6.nThreshold<=0 ){
466 mem6.nThreshold = SMALL_MALLOC_DEFAULT_THRESHOLD;
468 mem6.nLogThreshold = roundupLog2(mem6.nThreshold) - LOG2_MINALLOC;
470 mem6.mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MEM);
475 static void memsys6Shutdown(void *pCtx){
476 memset(&mem6, 0, sizeof(mem6));
480 ** This routine is the only routine in this file with external
481 ** linkage. It returns a pointer to a static sqlite3_mem_methods
482 ** struct populated with the memsys6 methods.
484 const sqlite3_mem_methods *sqlite3MemGetMemsys6(void){
485 static const sqlite3_mem_methods memsys6Methods = {
495 return &memsys6Methods;