Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * bump.c
4 : * Bump allocator definitions.
5 : *
6 : * Bump is a MemoryContext implementation designed for memory usages which
7 : * require allocating a large number of chunks, none of which ever need to be
8 : * pfree'd or realloc'd. Chunks allocated by this context have no chunk header
9 : * and operations which ordinarily require looking at the chunk header cannot
10 : * be performed. For example, pfree, realloc, GetMemoryChunkSpace and
11 : * GetMemoryChunkContext are all not possible with bump allocated chunks. The
12 : * only way to release memory allocated by this context type is to reset or
13 : * delete the context.
14 : *
15 : * Portions Copyright (c) 2024-2025, PostgreSQL Global Development Group
16 : *
17 : * IDENTIFICATION
18 : * src/backend/utils/mmgr/bump.c
19 : *
20 : *
21 : * Bump is best suited to cases which require a large number of short-lived
22 : * chunks where performance matters. Because bump allocated chunks don't
23 : * have a chunk header, it can fit more chunks on each block. This means we
24 : * can do more with less memory and fewer cache lines. The reason it's best
25 : * suited for short-lived usages of memory is that ideally, pointers to bump
26 : * allocated chunks won't be visible to a large amount of code. The more
27 : * code that operates on memory allocated by this allocator, the more chances
28 : * that some code will try to perform a pfree or one of the other operations
29 : * which are made impossible due to the lack of chunk header. In order to
30 : * detect accidental usage of the various disallowed operations, we do add a
31 : * MemoryChunk chunk header in MEMORY_CONTEXT_CHECKING builds and have the
32 : * various disallowed functions raise an ERROR.
33 : *
34 : * Allocations are MAXALIGNed.
35 : *
36 : *-------------------------------------------------------------------------
37 : */
38 :
39 : #include "postgres.h"
40 :
41 : #include "lib/ilist.h"
42 : #include "port/pg_bitutils.h"
43 : #include "utils/memdebug.h"
44 : #include "utils/memutils.h"
45 : #include "utils/memutils_memorychunk.h"
46 : #include "utils/memutils_internal.h"
47 :
48 : #define Bump_BLOCKHDRSZ MAXALIGN(sizeof(BumpBlock))
49 :
50 : /* No chunk header unless built with MEMORY_CONTEXT_CHECKING */
51 : #ifdef MEMORY_CONTEXT_CHECKING
52 : #define Bump_CHUNKHDRSZ sizeof(MemoryChunk)
53 : #else
54 : #define Bump_CHUNKHDRSZ 0
55 : #endif
56 :
57 : #define Bump_CHUNK_FRACTION 8
58 :
59 : /* The keeper block is allocated in the same allocation as the set */
60 : #define KeeperBlock(set) ((BumpBlock *) ((char *) (set) + \
61 : MAXALIGN(sizeof(BumpContext))))
62 : #define IsKeeperBlock(set, blk) (KeeperBlock(set) == (blk))
63 :
64 : typedef struct BumpBlock BumpBlock; /* forward reference */
65 :
66 : typedef struct BumpContext
67 : {
68 : MemoryContextData header; /* Standard memory-context fields */
69 :
70 : /* Bump context parameters */
71 : uint32 initBlockSize; /* initial block size */
72 : uint32 maxBlockSize; /* maximum block size */
73 : uint32 nextBlockSize; /* next block size to allocate */
74 : uint32 allocChunkLimit; /* effective chunk size limit */
75 :
76 : dlist_head blocks; /* list of blocks with the block currently
77 : * being filled at the head */
78 : } BumpContext;
79 :
80 : /*
81 : * BumpBlock
82 : * BumpBlock is the unit of memory that is obtained by bump.c from
83 : * malloc(). It contains zero or more allocations, which are the
84 : * units requested by palloc().
85 : */
86 : struct BumpBlock
87 : {
88 : dlist_node node; /* doubly-linked list of blocks */
89 : #ifdef MEMORY_CONTEXT_CHECKING
90 : BumpContext *context; /* pointer back to the owning context */
91 : #endif
92 : char *freeptr; /* start of free space in this block */
93 : char *endptr; /* end of space in this block */
94 : };
95 :
96 : /*
97 : * BumpIsValid
98 : * True iff set is valid bump context.
99 : */
100 : #define BumpIsValid(set) \
101 : (PointerIsValid(set) && IsA(set, BumpContext))
102 :
103 : /*
104 : * We always store external chunks on a dedicated block. This makes fetching
105 : * the block from an external chunk easy since it's always the first and only
106 : * chunk on the block.
107 : */
108 : #define ExternalChunkGetBlock(chunk) \
109 : (BumpBlock *) ((char *) chunk - Bump_BLOCKHDRSZ)
110 :
111 : /* Inlined helper functions */
112 : static inline void BumpBlockInit(BumpContext *context, BumpBlock *block,
113 : Size blksize);
114 : static inline bool BumpBlockIsEmpty(BumpBlock *block);
115 : static inline void BumpBlockMarkEmpty(BumpBlock *block);
116 : static inline Size BumpBlockFreeBytes(BumpBlock *block);
117 : static inline void BumpBlockFree(BumpContext *set, BumpBlock *block);
118 :
119 :
120 : /*
121 : * BumpContextCreate
122 : * Create a new Bump context.
123 : *
124 : * parent: parent context, or NULL if top-level context
125 : * name: name of context (must be statically allocated)
126 : * minContextSize: minimum context size
127 : * initBlockSize: initial allocation block size
128 : * maxBlockSize: maximum allocation block size
129 : */
130 : MemoryContext
131 370316 : BumpContextCreate(MemoryContext parent, const char *name, Size minContextSize,
132 : Size initBlockSize, Size maxBlockSize)
133 : {
134 : Size firstBlockSize;
135 : Size allocSize;
136 : BumpContext *set;
137 : BumpBlock *block;
138 :
139 : /* ensure MemoryChunk's size is properly maxaligned */
140 : StaticAssertDecl(Bump_CHUNKHDRSZ == MAXALIGN(Bump_CHUNKHDRSZ),
141 : "sizeof(MemoryChunk) is not maxaligned");
142 :
143 : /*
144 : * First, validate allocation parameters. Asserts seem sufficient because
145 : * nobody varies their parameters at runtime. We somewhat arbitrarily
146 : * enforce a minimum 1K block size. We restrict the maximum block size to
147 : * MEMORYCHUNK_MAX_BLOCKOFFSET as MemoryChunks are limited to this in
148 : * regards to addressing the offset between the chunk and the block that
149 : * the chunk is stored on. We would be unable to store the offset between
150 : * the chunk and block for any chunks that were beyond
151 : * MEMORYCHUNK_MAX_BLOCKOFFSET bytes into the block if the block was to be
152 : * larger than this.
153 : */
154 : Assert(initBlockSize == MAXALIGN(initBlockSize) &&
155 : initBlockSize >= 1024);
156 : Assert(maxBlockSize == MAXALIGN(maxBlockSize) &&
157 : maxBlockSize >= initBlockSize &&
158 : AllocHugeSizeIsValid(maxBlockSize)); /* must be safe to double */
159 : Assert(minContextSize == 0 ||
160 : (minContextSize == MAXALIGN(minContextSize) &&
161 : minContextSize >= 1024 &&
162 : minContextSize <= maxBlockSize));
163 : Assert(maxBlockSize <= MEMORYCHUNK_MAX_BLOCKOFFSET);
164 :
165 : /* Determine size of initial block */
166 370316 : allocSize = MAXALIGN(sizeof(BumpContext)) + Bump_BLOCKHDRSZ +
167 : Bump_CHUNKHDRSZ;
168 370316 : if (minContextSize != 0)
169 0 : allocSize = Max(allocSize, minContextSize);
170 : else
171 370316 : allocSize = Max(allocSize, initBlockSize);
172 :
173 : /*
174 : * Allocate the initial block. Unlike other bump.c blocks, it starts with
175 : * the context header and its block header follows that.
176 : */
177 370316 : set = (BumpContext *) malloc(allocSize);
178 370316 : if (set == NULL)
179 : {
180 0 : MemoryContextStats(TopMemoryContext);
181 0 : ereport(ERROR,
182 : (errcode(ERRCODE_OUT_OF_MEMORY),
183 : errmsg("out of memory"),
184 : errdetail("Failed while creating memory context \"%s\".",
185 : name)));
186 : }
187 :
188 : /*
189 : * Avoid writing code that can fail between here and MemoryContextCreate;
190 : * we'd leak the header and initial block if we ereport in this stretch.
191 : */
192 370316 : dlist_init(&set->blocks);
193 :
194 : /* Fill in the initial block's block header */
195 370316 : block = KeeperBlock(set);
196 : /* determine the block size and initialize it */
197 370316 : firstBlockSize = allocSize - MAXALIGN(sizeof(BumpContext));
198 370316 : BumpBlockInit(set, block, firstBlockSize);
199 :
200 : /* add it to the doubly-linked list of blocks */
201 370316 : dlist_push_head(&set->blocks, &block->node);
202 :
203 : /*
204 : * Fill in BumpContext-specific header fields. The Asserts above should
205 : * ensure that these all fit inside a uint32.
206 : */
207 370316 : set->initBlockSize = (uint32) initBlockSize;
208 370316 : set->maxBlockSize = (uint32) maxBlockSize;
209 370316 : set->nextBlockSize = (uint32) initBlockSize;
210 :
211 : /*
212 : * Compute the allocation chunk size limit for this context.
213 : *
214 : * Limit the maximum size a non-dedicated chunk can be so that we can fit
215 : * at least Bump_CHUNK_FRACTION of chunks this big onto the maximum sized
216 : * block. We must further limit this value so that it's no more than
217 : * MEMORYCHUNK_MAX_VALUE. We're unable to have non-external chunks larger
218 : * than that value as we store the chunk size in the MemoryChunk 'value'
219 : * field in the call to MemoryChunkSetHdrMask().
220 : */
221 370316 : set->allocChunkLimit = Min(maxBlockSize, MEMORYCHUNK_MAX_VALUE);
222 370316 : while ((Size) (set->allocChunkLimit + Bump_CHUNKHDRSZ) >
223 1851580 : (Size) ((Size) (maxBlockSize - Bump_BLOCKHDRSZ) / Bump_CHUNK_FRACTION))
224 1481264 : set->allocChunkLimit >>= 1;
225 :
226 : /* Finally, do the type-independent part of context creation */
227 370316 : MemoryContextCreate((MemoryContext) set, T_BumpContext, MCTX_BUMP_ID,
228 : parent, name);
229 :
230 370316 : ((MemoryContext) set)->mem_allocated = allocSize;
231 :
232 370316 : return (MemoryContext) set;
233 : }
234 :
235 : /*
236 : * BumpReset
237 : * Frees all memory which is allocated in the given set.
238 : *
239 : * The code simply frees all the blocks in the context apart from the keeper
240 : * block.
241 : */
242 : void
243 372166 : BumpReset(MemoryContext context)
244 : {
245 372166 : BumpContext *set = (BumpContext *) context;
246 : dlist_mutable_iter miter;
247 :
248 : Assert(BumpIsValid(set));
249 :
250 : #ifdef MEMORY_CONTEXT_CHECKING
251 : /* Check for corruption and leaks before freeing */
252 : BumpCheck(context);
253 : #endif
254 :
255 764386 : dlist_foreach_modify(miter, &set->blocks)
256 : {
257 392220 : BumpBlock *block = dlist_container(BumpBlock, node, miter.cur);
258 :
259 392220 : if (IsKeeperBlock(set, block))
260 372166 : BumpBlockMarkEmpty(block);
261 : else
262 20054 : BumpBlockFree(set, block);
263 : }
264 :
265 : /* Reset block size allocation sequence, too */
266 372166 : set->nextBlockSize = set->initBlockSize;
267 :
268 : /* Ensure there is only 1 item in the dlist */
269 : Assert(!dlist_is_empty(&set->blocks));
270 : Assert(!dlist_has_next(&set->blocks, dlist_head_node(&set->blocks)));
271 372166 : }
272 :
273 : /*
274 : * BumpDelete
275 : * Free all memory which is allocated in the given context.
276 : */
277 : void
278 370316 : BumpDelete(MemoryContext context)
279 : {
280 : /* Reset to release all releasable BumpBlocks */
281 370316 : BumpReset(context);
282 : /* And free the context header and keeper block */
283 370316 : free(context);
284 370316 : }
285 :
286 : /*
287 : * Helper for BumpAlloc() that allocates an entire block for the chunk.
288 : *
289 : * BumpAlloc()'s comment explains why this is separate.
290 : */
291 : pg_noinline
292 : static void *
293 18 : BumpAllocLarge(MemoryContext context, Size size, int flags)
294 : {
295 18 : BumpContext *set = (BumpContext *) context;
296 : BumpBlock *block;
297 : #ifdef MEMORY_CONTEXT_CHECKING
298 : MemoryChunk *chunk;
299 : #endif
300 : Size chunk_size;
301 : Size required_size;
302 : Size blksize;
303 :
304 : /* validate 'size' is within the limits for the given 'flags' */
305 18 : MemoryContextCheckSize(context, size, flags);
306 :
307 : #ifdef MEMORY_CONTEXT_CHECKING
308 : /* ensure there's always space for the sentinel byte */
309 : chunk_size = MAXALIGN(size + 1);
310 : #else
311 18 : chunk_size = MAXALIGN(size);
312 : #endif
313 :
314 18 : required_size = chunk_size + Bump_CHUNKHDRSZ;
315 18 : blksize = required_size + Bump_BLOCKHDRSZ;
316 :
317 18 : block = (BumpBlock *) malloc(blksize);
318 18 : if (block == NULL)
319 0 : return NULL;
320 :
321 18 : context->mem_allocated += blksize;
322 :
323 : /* the block is completely full */
324 18 : block->freeptr = block->endptr = ((char *) block) + blksize;
325 :
326 : #ifdef MEMORY_CONTEXT_CHECKING
327 : /* block with a single (used) chunk */
328 : block->context = set;
329 :
330 : chunk = (MemoryChunk *) (((char *) block) + Bump_BLOCKHDRSZ);
331 :
332 : /* mark the MemoryChunk as externally managed */
333 : MemoryChunkSetHdrMaskExternal(chunk, MCTX_BUMP_ID);
334 :
335 : chunk->requested_size = size;
336 : /* set mark to catch clobber of "unused" space */
337 : Assert(size < chunk_size);
338 : set_sentinel(MemoryChunkGetPointer(chunk), size);
339 : #endif
340 : #ifdef RANDOMIZE_ALLOCATED_MEMORY
341 : /* fill the allocated space with junk */
342 : randomize_mem((char *) MemoryChunkGetPointer(chunk), size);
343 : #endif
344 :
345 : /*
346 : * Add the block to the tail of allocated blocks list. The current block
347 : * is left at the head of the list as it may still have space for
348 : * non-large allocations.
349 : */
350 18 : dlist_push_tail(&set->blocks, &block->node);
351 :
352 : #ifdef MEMORY_CONTEXT_CHECKING
353 : /* Ensure any padding bytes are marked NOACCESS. */
354 : VALGRIND_MAKE_MEM_NOACCESS((char *) MemoryChunkGetPointer(chunk) + size,
355 : chunk_size - size);
356 :
357 : /* Disallow access to the chunk header. */
358 : VALGRIND_MAKE_MEM_NOACCESS(chunk, Bump_CHUNKHDRSZ);
359 :
360 : return MemoryChunkGetPointer(chunk);
361 : #else
362 18 : return (void *) (((char *) block) + Bump_BLOCKHDRSZ);
363 : #endif
364 : }
365 :
366 : /*
367 : * Small helper for allocating a new chunk from a chunk, to avoid duplicating
368 : * the code between BumpAlloc() and BumpAllocFromNewBlock().
369 : */
370 : static inline void *
371 22950234 : BumpAllocChunkFromBlock(MemoryContext context, BumpBlock *block, Size size,
372 : Size chunk_size)
373 : {
374 : #ifdef MEMORY_CONTEXT_CHECKING
375 : MemoryChunk *chunk;
376 : #else
377 : void *ptr;
378 : #endif
379 :
380 : /* validate we've been given a block with enough free space */
381 : Assert(block != NULL);
382 : Assert((block->endptr - block->freeptr) >= Bump_CHUNKHDRSZ + chunk_size);
383 :
384 : #ifdef MEMORY_CONTEXT_CHECKING
385 : chunk = (MemoryChunk *) block->freeptr;
386 : #else
387 22950234 : ptr = (void *) block->freeptr;
388 : #endif
389 :
390 : /* point the freeptr beyond this chunk */
391 22950234 : block->freeptr += (Bump_CHUNKHDRSZ + chunk_size);
392 : Assert(block->freeptr <= block->endptr);
393 :
394 : #ifdef MEMORY_CONTEXT_CHECKING
395 : /* Prepare to initialize the chunk header. */
396 : VALGRIND_MAKE_MEM_UNDEFINED(chunk, Bump_CHUNKHDRSZ);
397 :
398 : MemoryChunkSetHdrMask(chunk, block, chunk_size, MCTX_BUMP_ID);
399 : chunk->requested_size = size;
400 : /* set mark to catch clobber of "unused" space */
401 : Assert(size < chunk_size);
402 : set_sentinel(MemoryChunkGetPointer(chunk), size);
403 :
404 : #ifdef RANDOMIZE_ALLOCATED_MEMORY
405 : /* fill the allocated space with junk */
406 : randomize_mem((char *) MemoryChunkGetPointer(chunk), size);
407 : #endif
408 :
409 : /* Ensure any padding bytes are marked NOACCESS. */
410 : VALGRIND_MAKE_MEM_NOACCESS((char *) MemoryChunkGetPointer(chunk) + size,
411 : chunk_size - size);
412 :
413 : /* Disallow access to the chunk header. */
414 : VALGRIND_MAKE_MEM_NOACCESS(chunk, Bump_CHUNKHDRSZ);
415 :
416 : return MemoryChunkGetPointer(chunk);
417 : #else
418 22950234 : return ptr;
419 : #endif /* MEMORY_CONTEXT_CHECKING */
420 : }
421 :
422 : /*
423 : * Helper for BumpAlloc() that allocates a new block and returns a chunk
424 : * allocated from it.
425 : *
426 : * BumpAlloc()'s comment explains why this is separate.
427 : */
428 : pg_noinline
429 : static void *
430 20036 : BumpAllocFromNewBlock(MemoryContext context, Size size, int flags,
431 : Size chunk_size)
432 : {
433 20036 : BumpContext *set = (BumpContext *) context;
434 : BumpBlock *block;
435 : Size blksize;
436 : Size required_size;
437 :
438 : /*
439 : * The first such block has size initBlockSize, and we double the space in
440 : * each succeeding block, but not more than maxBlockSize.
441 : */
442 20036 : blksize = set->nextBlockSize;
443 20036 : set->nextBlockSize <<= 1;
444 20036 : if (set->nextBlockSize > set->maxBlockSize)
445 2 : set->nextBlockSize = set->maxBlockSize;
446 :
447 : /* we'll need space for the chunk, chunk hdr and block hdr */
448 20036 : required_size = chunk_size + Bump_CHUNKHDRSZ + Bump_BLOCKHDRSZ;
449 : /* round the size up to the next power of 2 */
450 20036 : if (blksize < required_size)
451 0 : blksize = pg_nextpower2_size_t(required_size);
452 :
453 20036 : block = (BumpBlock *) malloc(blksize);
454 :
455 20036 : if (block == NULL)
456 0 : return MemoryContextAllocationFailure(context, size, flags);
457 :
458 20036 : context->mem_allocated += blksize;
459 :
460 : /* initialize the new block */
461 20036 : BumpBlockInit(set, block, blksize);
462 :
463 : /* add it to the doubly-linked list of blocks */
464 20036 : dlist_push_head(&set->blocks, &block->node);
465 :
466 20036 : return BumpAllocChunkFromBlock(context, block, size, chunk_size);
467 : }
468 :
469 : /*
470 : * BumpAlloc
471 : * Returns a pointer to allocated memory of given size or raises an ERROR
472 : * on allocation failure, or returns NULL when flags contains
473 : * MCXT_ALLOC_NO_OOM.
474 : *
475 : * No request may exceed:
476 : * MAXALIGN_DOWN(SIZE_MAX) - Bump_BLOCKHDRSZ - Bump_CHUNKHDRSZ
477 : * All callers use a much-lower limit.
478 : *
479 : *
480 : * Note: when using valgrind, it doesn't matter how the returned allocation
481 : * is marked, as mcxt.c will set it to UNDEFINED.
482 : * This function should only contain the most common code paths. Everything
483 : * else should be in pg_noinline helper functions, thus avoiding the overhead
484 : * of creating a stack frame for the common cases. Allocating memory is often
485 : * a bottleneck in many workloads, so avoiding stack frame setup is
486 : * worthwhile. Helper functions should always directly return the newly
487 : * allocated memory so that we can just return that address directly as a tail
488 : * call.
489 : */
490 : void *
491 22950252 : BumpAlloc(MemoryContext context, Size size, int flags)
492 : {
493 22950252 : BumpContext *set = (BumpContext *) context;
494 : BumpBlock *block;
495 : Size chunk_size;
496 : Size required_size;
497 :
498 : Assert(BumpIsValid(set));
499 :
500 : #ifdef MEMORY_CONTEXT_CHECKING
501 : /* ensure there's always space for the sentinel byte */
502 : chunk_size = MAXALIGN(size + 1);
503 : #else
504 22950252 : chunk_size = MAXALIGN(size);
505 : #endif
506 :
507 : /*
508 : * If requested size exceeds maximum for chunks we hand the request off to
509 : * BumpAllocLarge().
510 : */
511 22950252 : if (chunk_size > set->allocChunkLimit)
512 18 : return BumpAllocLarge(context, size, flags);
513 :
514 22950234 : required_size = chunk_size + Bump_CHUNKHDRSZ;
515 :
516 : /*
517 : * Not an oversized chunk. We try to first make use of the latest block,
518 : * but if there's not enough space in it we must allocate a new block.
519 : */
520 22950234 : block = dlist_container(BumpBlock, node, dlist_head_node(&set->blocks));
521 :
522 22950234 : if (BumpBlockFreeBytes(block) < required_size)
523 20036 : return BumpAllocFromNewBlock(context, size, flags, chunk_size);
524 :
525 : /* The current block has space, so just allocate chunk there. */
526 22930198 : return BumpAllocChunkFromBlock(context, block, size, chunk_size);
527 : }
528 :
529 : /*
530 : * BumpBlockInit
531 : * Initializes 'block' assuming 'blksize'. Does not update the context's
532 : * mem_allocated field.
533 : */
534 : static inline void
535 390352 : BumpBlockInit(BumpContext *context, BumpBlock *block, Size blksize)
536 : {
537 : #ifdef MEMORY_CONTEXT_CHECKING
538 : block->context = context;
539 : #endif
540 390352 : block->freeptr = ((char *) block) + Bump_BLOCKHDRSZ;
541 390352 : block->endptr = ((char *) block) + blksize;
542 :
543 : /* Mark unallocated space NOACCESS. */
544 : VALGRIND_MAKE_MEM_NOACCESS(block->freeptr, blksize - Bump_BLOCKHDRSZ);
545 390352 : }
546 :
547 : /*
548 : * BumpBlockIsEmpty
549 : * Returns true iff 'block' contains no chunks
550 : */
551 : static inline bool
552 0 : BumpBlockIsEmpty(BumpBlock *block)
553 : {
554 : /* it's empty if the freeptr has not moved */
555 0 : return (block->freeptr == ((char *) block + Bump_BLOCKHDRSZ));
556 : }
557 :
558 : /*
559 : * BumpBlockMarkEmpty
560 : * Set a block as empty. Does not free the block.
561 : */
562 : static inline void
563 372166 : BumpBlockMarkEmpty(BumpBlock *block)
564 : {
565 : #if defined(USE_VALGRIND) || defined(CLOBBER_FREED_MEMORY)
566 : char *datastart = ((char *) block) + Bump_BLOCKHDRSZ;
567 : #endif
568 :
569 : #ifdef CLOBBER_FREED_MEMORY
570 : wipe_mem(datastart, block->freeptr - datastart);
571 : #else
572 : /* wipe_mem() would have done this */
573 : VALGRIND_MAKE_MEM_NOACCESS(datastart, block->freeptr - datastart);
574 : #endif
575 :
576 : /* Reset the block, but don't return it to malloc */
577 372166 : block->freeptr = ((char *) block) + Bump_BLOCKHDRSZ;
578 372166 : }
579 :
580 : /*
581 : * BumpBlockFreeBytes
582 : * Returns the number of bytes free in 'block'
583 : */
584 : static inline Size
585 22950234 : BumpBlockFreeBytes(BumpBlock *block)
586 : {
587 22950234 : return (block->endptr - block->freeptr);
588 : }
589 :
590 : /*
591 : * BumpBlockFree
592 : * Remove 'block' from 'set' and release the memory consumed by it.
593 : */
594 : static inline void
595 20054 : BumpBlockFree(BumpContext *set, BumpBlock *block)
596 : {
597 : /* Make sure nobody tries to free the keeper block */
598 : Assert(!IsKeeperBlock(set, block));
599 :
600 : /* release the block from the list of blocks */
601 20054 : dlist_delete(&block->node);
602 :
603 20054 : ((MemoryContext) set)->mem_allocated -= ((char *) block->endptr - (char *) block);
604 :
605 : #ifdef CLOBBER_FREED_MEMORY
606 : wipe_mem(block, ((char *) block->endptr - (char *) block));
607 : #endif
608 :
609 20054 : free(block);
610 20054 : }
611 :
612 : /*
613 : * BumpFree
614 : * Unsupported.
615 : */
616 : void
617 0 : BumpFree(void *pointer)
618 : {
619 0 : elog(ERROR, "%s is not supported by the bump memory allocator", "pfree");
620 : }
621 :
622 : /*
623 : * BumpRealloc
624 : * Unsupported.
625 : */
626 : void *
627 0 : BumpRealloc(void *pointer, Size size, int flags)
628 : {
629 0 : elog(ERROR, "%s is not supported by the bump memory allocator", "realloc");
630 : return NULL; /* keep compiler quiet */
631 : }
632 :
633 : /*
634 : * BumpGetChunkContext
635 : * Unsupported.
636 : */
637 : MemoryContext
638 0 : BumpGetChunkContext(void *pointer)
639 : {
640 0 : elog(ERROR, "%s is not supported by the bump memory allocator", "GetMemoryChunkContext");
641 : return NULL; /* keep compiler quiet */
642 : }
643 :
644 : /*
645 : * BumpGetChunkSpace
646 : * Unsupported.
647 : */
648 : Size
649 0 : BumpGetChunkSpace(void *pointer)
650 : {
651 0 : elog(ERROR, "%s is not supported by the bump memory allocator", "GetMemoryChunkSpace");
652 : return 0; /* keep compiler quiet */
653 : }
654 :
655 : /*
656 : * BumpIsEmpty
657 : * Is a BumpContext empty of any allocated space?
658 : */
659 : bool
660 0 : BumpIsEmpty(MemoryContext context)
661 : {
662 0 : BumpContext *set = (BumpContext *) context;
663 : dlist_iter iter;
664 :
665 : Assert(BumpIsValid(set));
666 :
667 0 : dlist_foreach(iter, &set->blocks)
668 : {
669 0 : BumpBlock *block = dlist_container(BumpBlock, node, iter.cur);
670 :
671 0 : if (!BumpBlockIsEmpty(block))
672 0 : return false;
673 : }
674 :
675 0 : return true;
676 : }
677 :
678 : /*
679 : * BumpStats
680 : * Compute stats about memory consumption of a Bump context.
681 : *
682 : * printfunc: if not NULL, pass a human-readable stats string to this.
683 : * passthru: pass this pointer through to printfunc.
684 : * totals: if not NULL, add stats about this context into *totals.
685 : * print_to_stderr: print stats to stderr if true, elog otherwise.
686 : */
687 : void
688 6 : BumpStats(MemoryContext context, MemoryStatsPrintFunc printfunc,
689 : void *passthru, MemoryContextCounters *totals, bool print_to_stderr)
690 : {
691 6 : BumpContext *set = (BumpContext *) context;
692 6 : Size nblocks = 0;
693 6 : Size totalspace = 0;
694 6 : Size freespace = 0;
695 : dlist_iter iter;
696 :
697 : Assert(BumpIsValid(set));
698 :
699 18 : dlist_foreach(iter, &set->blocks)
700 : {
701 12 : BumpBlock *block = dlist_container(BumpBlock, node, iter.cur);
702 :
703 12 : nblocks++;
704 12 : totalspace += (block->endptr - (char *) block);
705 12 : freespace += (block->endptr - block->freeptr);
706 : }
707 :
708 6 : if (printfunc)
709 : {
710 : char stats_string[200];
711 :
712 0 : snprintf(stats_string, sizeof(stats_string),
713 : "%zu total in %zu blocks; %zu free; %zu used",
714 : totalspace, nblocks, freespace, totalspace - freespace);
715 0 : printfunc(context, passthru, stats_string, print_to_stderr);
716 : }
717 :
718 6 : if (totals)
719 : {
720 6 : totals->nblocks += nblocks;
721 6 : totals->totalspace += totalspace;
722 6 : totals->freespace += freespace;
723 : }
724 6 : }
725 :
726 :
727 : #ifdef MEMORY_CONTEXT_CHECKING
728 :
729 : /*
730 : * BumpCheck
731 : * Walk through chunks and check consistency of memory.
732 : *
733 : * NOTE: report errors as WARNING, *not* ERROR or FATAL. Otherwise you'll
734 : * find yourself in an infinite loop when trouble occurs, because this
735 : * routine will be entered again when elog cleanup tries to release memory!
736 : */
737 : void
738 : BumpCheck(MemoryContext context)
739 : {
740 : BumpContext *bump = (BumpContext *) context;
741 : const char *name = context->name;
742 : dlist_iter iter;
743 : Size total_allocated = 0;
744 :
745 : /* walk all blocks in this context */
746 : dlist_foreach(iter, &bump->blocks)
747 : {
748 : BumpBlock *block = dlist_container(BumpBlock, node, iter.cur);
749 : int nchunks;
750 : char *ptr;
751 : bool has_external_chunk = false;
752 :
753 : if (IsKeeperBlock(bump, block))
754 : total_allocated += block->endptr - (char *) bump;
755 : else
756 : total_allocated += block->endptr - (char *) block;
757 :
758 : /* check block belongs to the correct context */
759 : if (block->context != bump)
760 : elog(WARNING, "problem in Bump %s: bogus context link in block %p",
761 : name, block);
762 :
763 : /* now walk through the chunks and count them */
764 : nchunks = 0;
765 : ptr = ((char *) block) + Bump_BLOCKHDRSZ;
766 :
767 : while (ptr < block->freeptr)
768 : {
769 : MemoryChunk *chunk = (MemoryChunk *) ptr;
770 : BumpBlock *chunkblock;
771 : Size chunksize;
772 :
773 : /* allow access to the chunk header */
774 : VALGRIND_MAKE_MEM_DEFINED(chunk, Bump_CHUNKHDRSZ);
775 :
776 : if (MemoryChunkIsExternal(chunk))
777 : {
778 : chunkblock = ExternalChunkGetBlock(chunk);
779 : chunksize = block->endptr - (char *) MemoryChunkGetPointer(chunk);
780 : has_external_chunk = true;
781 : }
782 : else
783 : {
784 : chunkblock = MemoryChunkGetBlock(chunk);
785 : chunksize = MemoryChunkGetValue(chunk);
786 : }
787 :
788 : /* move to the next chunk */
789 : ptr += (chunksize + Bump_CHUNKHDRSZ);
790 :
791 : nchunks += 1;
792 :
793 : /* chunks have both block and context pointers, so check both */
794 : if (chunkblock != block)
795 : elog(WARNING, "problem in Bump %s: bogus block link in block %p, chunk %p",
796 : name, block, chunk);
797 : }
798 :
799 : if (has_external_chunk && nchunks > 1)
800 : elog(WARNING, "problem in Bump %s: external chunk on non-dedicated block %p",
801 : name, block);
802 :
803 : }
804 :
805 : Assert(total_allocated == context->mem_allocated);
806 : }
807 :
808 : #endif /* MEMORY_CONTEXT_CHECKING */
|