First public contribution.
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.7 2008/07/28 19:34:53 drh 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]))
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 */
116 ** Unlink the chunk at pChunk->aPool[i] from list it is currently
117 ** on. It should be found on pChunk->aiFreelist[iLogsize].
119 static void memsys6Unlink(Mem6Chunk *pChunk, int i, int iLogsize){
121 assert( i>=0 && i<pChunk->nBlock );
122 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
123 assert( (pChunk->aCtrl[i] & CTRL_LOGSIZE)==iLogsize );
125 next = MEM6LINK(i)->next;
126 prev = MEM6LINK(i)->prev;
128 pChunk->aiFreelist[iLogsize] = next;
130 MEM6LINK(prev)->next = next;
133 MEM6LINK(next)->prev = prev;
138 ** Link the chunk at mem5.aPool[i] so that is on the iLogsize
141 static void memsys6Link(Mem6Chunk *pChunk, int i, int iLogsize){
143 assert( i>=0 && i<pChunk->nBlock );
144 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
145 assert( (pChunk->aCtrl[i] & CTRL_LOGSIZE)==iLogsize );
147 x = MEM6LINK(i)->next = pChunk->aiFreelist[iLogsize];
148 MEM6LINK(i)->prev = -1;
150 assert( x<pChunk->nBlock );
151 MEM6LINK(x)->prev = i;
153 pChunk->aiFreelist[iLogsize] = i;
158 ** Find the first entry on the freelist iLogsize. Unlink that
159 ** entry and return its index.
161 static int memsys6UnlinkFirst(Mem6Chunk *pChunk, int iLogsize){
165 assert( iLogsize>=0 && iLogsize<=mem6.nLogThreshold );
166 i = iFirst = pChunk->aiFreelist[iLogsize];
168 memsys6Unlink(pChunk, iFirst, iLogsize);
172 static int roundupLog2(int n){
173 static const char LogTable256[256] = {
177 3, 3, 3, 3, /* 5..8 */
178 4, 4, 4, 4, 4, 4, 4, 4, /* 9..16 */
179 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, /* 17..32 */
180 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
181 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, /* 33..64 */
182 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
183 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
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, /* 65..128 */
186 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
187 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
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, /* 129..256 */
196 assert(n<=(1<<16) && n>0);
197 if( n<=256 ) return LogTable256[n-1];
198 return LogTable256[(n>>8) - ((n&0xFF)?0:1)] + 8;
202 ** Allocate and return a block of (pChunk->nAtom << iLogsize) bytes from chunk
203 ** pChunk. If the allocation request cannot be satisfied, return 0.
205 static void *chunkMalloc(Mem6Chunk *pChunk, int iLogsize){
206 int i; /* Index of a mem5.aPool[] slot */
207 int iBin; /* Index into mem5.aiFreelist[] */
209 /* Make sure mem5.aiFreelist[iLogsize] contains at least one free
210 ** block. If not, then split a block of the next larger power of
211 ** two in order to create a new free block of size iLogsize.
213 for(iBin=iLogsize; pChunk->aiFreelist[iBin]<0 && iBin<=mem6.nLogThreshold; iBin++){}
214 if( iBin>mem6.nLogThreshold ) return 0;
215 i = memsys6UnlinkFirst(pChunk, iBin);
216 while( iBin>iLogsize ){
220 pChunk->aCtrl[i+newSize] = CTRL_FREE | iBin;
221 memsys6Link(pChunk, i+newSize, iBin);
223 pChunk->aCtrl[i] = iLogsize;
225 /* Return a pointer to the allocated memory. */
226 pChunk->nCheckedOut++;
227 return (void*)&pChunk->zPool[i*pChunk->nAtom];
231 ** Free the allocation pointed to by p, which is guaranteed to be non-zero
232 ** and a part of chunk object pChunk.
234 static void chunkFree(Mem6Chunk *pChunk, void *pOld){
238 /* Set iBlock to the index of the block pointed to by pOld in
239 ** the array of pChunk->nAtom byte blocks pointed to by pChunk->zPool.
241 iBlock = ((u8 *)pOld-pChunk->zPool)/pChunk->nAtom;
243 /* Check that the pointer pOld points to a valid, non-free block. */
244 assert( iBlock>=0 && iBlock<pChunk->nBlock );
245 assert( ((u8 *)pOld-pChunk->zPool)%pChunk->nAtom==0 );
246 assert( (pChunk->aCtrl[iBlock] & CTRL_FREE)==0 );
248 iLogsize = pChunk->aCtrl[iBlock] & CTRL_LOGSIZE;
250 assert( iBlock+size-1<pChunk->nBlock );
252 pChunk->aCtrl[iBlock] |= CTRL_FREE;
253 pChunk->aCtrl[iBlock+size-1] |= CTRL_FREE;
255 pChunk->aCtrl[iBlock] = CTRL_FREE | iLogsize;
256 while( iLogsize<mem6.nLogThreshold ){
258 if( (iBlock>>iLogsize) & 1 ){
259 iBuddy = iBlock - size;
261 iBuddy = iBlock + size;
264 if( (iBuddy+(1<<iLogsize))>pChunk->nBlock ) break;
265 if( pChunk->aCtrl[iBuddy]!=(CTRL_FREE | iLogsize) ) break;
266 memsys6Unlink(pChunk, iBuddy, iLogsize);
269 pChunk->aCtrl[iBuddy] = CTRL_FREE | iLogsize;
270 pChunk->aCtrl[iBlock] = 0;
273 pChunk->aCtrl[iBlock] = CTRL_FREE | iLogsize;
274 pChunk->aCtrl[iBuddy] = 0;
278 pChunk->nCheckedOut--;
279 memsys6Link(pChunk, iBlock, iLogsize);
283 ** Return the actual size of the block pointed to by p, which is guaranteed
284 ** to have been allocated from chunk pChunk.
286 static int chunkSize(Mem6Chunk *pChunk, void *p){
289 int i = ((u8 *)p-pChunk->zPool)/pChunk->nAtom;
290 assert( i>=0 && i<pChunk->nBlock );
291 iSize = pChunk->nAtom * (1 << (pChunk->aCtrl[i]&CTRL_LOGSIZE));
297 ** Return true if there are currently no outstanding allocations.
299 static int chunkIsEmpty(Mem6Chunk *pChunk){
300 return (pChunk->nCheckedOut==0);
304 ** Initialize the buffer zChunk, which is nChunk bytes in size, as
305 ** an Mem6Chunk object. Return a copy of the zChunk pointer.
307 static Mem6Chunk *chunkInit(u8 *zChunk, int nChunk, int nMinAlloc){
310 Mem6Chunk *pChunk = (Mem6Chunk *)zChunk;
312 assert( nChunk>sizeof(Mem6Chunk) );
313 assert( nMinAlloc>sizeof(Mem6Link) );
315 memset(pChunk, 0, sizeof(Mem6Chunk));
316 pChunk->nAtom = nMinAlloc;
317 pChunk->nBlock = ((nChunk-sizeof(Mem6Chunk)) / (pChunk->nAtom+sizeof(u8)));
319 pChunk->zPool = (u8 *)&pChunk[1];
320 pChunk->aCtrl = &pChunk->zPool[pChunk->nBlock*pChunk->nAtom];
322 for(ii=0; ii<=mem6.nLogThreshold; ii++){
323 pChunk->aiFreelist[ii] = -1;
327 for(ii=mem6.nLogThreshold; ii>=0; ii--){
328 int nAlloc = (1<<ii);
329 while( (iOffset+nAlloc)<=pChunk->nBlock ){
330 pChunk->aCtrl[iOffset] = ii | CTRL_FREE;
331 memsys6Link(pChunk, iOffset, ii);
340 static void mem6Enter(void){
341 sqlite3_mutex_enter(mem6.mutex);
344 static void mem6Leave(void){
345 sqlite3_mutex_leave(mem6.mutex);
349 ** Based on the number and size of the currently allocated chunks, return
350 ** the size of the next chunk to allocate, in bytes.
352 static int nextChunkSize(void){
353 int iTotal = MIN_CHUNKSIZE;
355 for(p=mem6.pChunk; p; p=p->pNext){
361 static void freeChunk(Mem6Chunk *pChunk){
362 Mem6Chunk **pp = &mem6.pChunk;
363 for( pp=&mem6.pChunk; *pp!=pChunk; pp = &(*pp)->pNext );
368 static void *memsys6Malloc(int nByte){
371 int nTotal = nByte+8;
374 if( nTotal>mem6.nThreshold ){
378 if( nTotal>(1<<LOG2_MINALLOC) ){
379 iLogsize = roundupLog2(nTotal) - LOG2_MINALLOC;
382 for(pChunk=mem6.pChunk; pChunk; pChunk=pChunk->pNext){
383 p = chunkMalloc(pChunk, iLogsize);
389 int iSize = nextChunkSize();
392 pChunk = chunkInit((u8 *)p, iSize, mem6.nMinAlloc);
393 pChunk->pNext = mem6.pChunk;
394 mem6.pChunk = pChunk;
395 p = chunkMalloc(pChunk, iLogsize);
399 iOffset = ((u8*)p - (u8*)pChunk);
406 ((u32 *)p)[0] = iOffset;
407 ((u32 *)p)[1] = nByte;
408 return &((u32 *)p)[2];
411 static int memsys6Size(void *pPrior){
412 if( pPrior==0 ) return 0;
413 return ((u32*)pPrior)[-1];
416 static void memsys6Free(void *pPrior){
418 void *p = &((u32 *)pPrior)[-2];
419 iSlot = ((u32 *)p)[0];
423 pChunk = (Mem6Chunk *)(&((u8 *)p)[-1 * iSlot]);
424 chunkFree(pChunk, p);
425 if( chunkIsEmpty(pChunk) ){
434 static void *memsys6Realloc(void *p, int nByte){
437 if( p && nByte<=memsys6Size(p) ){
440 p2 = memsys6Malloc(nByte);
442 memcpy(p2, p, memsys6Size(p));
450 static int memsys6Roundup(int n){
451 if( n>mem6.nThreshold ){
454 return (1<<roundupLog2(n));
458 static int memsys6Init(void *pCtx){
459 u8 bMemstat = sqlite3Config.bMemstat;
460 mem6.nMinAlloc = (1 << LOG2_MINALLOC);
462 mem6.nThreshold = sqlite3Config.nSmall;
463 if( mem6.nThreshold<=0 ){
464 mem6.nThreshold = SMALL_MALLOC_DEFAULT_THRESHOLD;
466 mem6.nLogThreshold = roundupLog2(mem6.nThreshold) - LOG2_MINALLOC;
468 mem6.mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MEM);
473 static void memsys6Shutdown(void *pCtx){
474 memset(&mem6, 0, sizeof(mem6));
478 ** This routine is the only routine in this file with external
479 ** linkage. It returns a pointer to a static sqlite3_mem_methods
480 ** struct populated with the memsys6 methods.
482 const sqlite3_mem_methods *sqlite3MemGetMemsys6(void){
483 static const sqlite3_mem_methods memsys6Methods = {
493 return &memsys6Methods;