📄 falloc.h
字号:
/* * falloc.h * nyquist memory allocation data structures and macros * * there is an falloc and ffree for each major type of data structure * there is an falloc and ffree for generic (not so common) structures * there is an frelease for some structures. this reduces the * reference count for the particular structure by 1; it * does not continue recursively. *//* Debugging support: * When DEBUG_MEM is set, each piece of allocated storage will contain * a pointer to a string naming the caller or other allocation info, * and a sequence number. (8 extra bytes are allocated for this info). * * When storage is freed, the ID is set to NULL, and the routine * dbg_mem_check(ptr) will abort if ID is NULL. Call this routine to * avoid following a pointer to data that was previously freed. * * The goal of this support is to allow you to "go back" to the point * where memory is corrupted; specifically where a memory block is freed * too early. * * When a memory-related bug is crashing the system: * (1) Recompile with DEBUG_MEM on. * (2) Develop some Nyquist code that will predictably crash the system. * (3) When Nyquist crashes, use a debugger to find where the bad * pointer came from. See if the source of the pointer was freed. * (4) If the source of the pointer was freed, then notice the sequence * number. * (5) Rerun with dbg_mem_seq_num set to the number noted in (4). * (6) Nyquist will print when the storage in question was allocated and * freed. Use the debugger to find out why the storage is * freed too early and who did it. * (7) If the source of the pointer in (3) was not freed, you're on your * own. * * The DEBUG_MEM related routines are: * dbg_mem_allocated: called when memory is allocated * dbg_mem_freed: called when memory is freed * dbg_mem_released: called when memory is released * dbg_mem_check: called to check memory * * see also xldmem.c: * by setting xldmem_trace to a pointer, you can trace when the * pointer is referenced by anything in the heap *//* to get size_t on pmax: */#ifdef pmax#include "sys/types.h"#endif#include "cque.h"#include "debug.h"#define DEBUG_MEM 1#define DEBUG_MEM_INFO_SIZE (sizeof(long) + sizeof(char *))/* special free lists */extern CQUE *sample_block_free; /* really a sample_block_type *//* special counts */extern int sample_block_total;extern int sample_block_used;extern int snd_list_used;extern int sound_used;extern long table_memory;/* generic free lists */#define MAXLISTS 128extern CQUE *generic_free[MAXLISTS];/* memory pool */#define MAXPOOLSIZE 1000000extern char *poolp;extern char *poolend;extern int npools;extern int sample_blocks_since_gc;void falloc_init(void);void new_pool(void);sample_block_type find_sample_block(void);char *get_from_pool(size_t siz);#define round_size(n) (((n) + 7) & ~7)/* check_pool -- returns true if enough bytes are available */#if DEBUG_MEM#define check_pool(size) (poolp + (size) + DEBUG_MEM_INFO_SIZE <= poolend)#else#define check_pool(size) (poolp + (size) <= poolend)#endif#define BLOCKS_PER_GC 100#define falloc_sample_block(sp, who) { \ if (!Qempty(sample_block_free)) \ Qget(sample_block_free, sample_block_type, sp) \ else sp = find_sample_block(); \ /* sample_block_test(sp, "falloc_sample_block"); */ \ /* printf("[%x] ", sp); */ \ if (DEBUG_MEM) dbg_mem_allocated(sp, who); \ sp->refcnt = 1; \ sample_block_used++; \}#define ffree_sample_block(sp, who) { \ /* printf("freeing sample_block@%x\n", sp); */ \ if (DEBUG_MEM) dbg_mem_freed(sp, who); \ Qenter(sample_block_free, sp); \ sample_block_used--; \}#define frelease_sample_block(sp, who) { \ sp->refcnt--; \ if (DEBUG_MEM) dbg_mem_released(sp, who); \ if (sp->refcnt <= 0) { \ ffree_sample_block(sp); \ } \}/* NOTE: This must not cause garbage collection. * LVAL parameters to snd_make_? functions are not * protected and falloc_sound is invoked there. */#define snd_list_free (generic_free[round_size(sizeof(snd_list_node)) >> 3])#define falloc_snd_list(sp, who) { \ if (!Qempty(snd_list_free)) \ Qget(snd_list_free, snd_list_type, sp) \ else \ sp = (snd_list_type)get_from_pool(round_size(sizeof(snd_list_node)));\ snd_list_used++; \ if (DEBUG_MEM) dbg_mem_allocated(sp, who); \}#define ffree_snd_list(sp, who) { \ if (DEBUG_MEM) dbg_mem_freed(sp, who); \ Qenter(snd_list_free, sp); \ snd_list_used--; \}#define frelease_snd_list(sp, who) { \ sp->refcnt--; \ if (DEBUG_MEM) dbg_mem_released(sp, who); \ if (sp->refcnt <= 0) { \ ffree_snd_list(sp, who); \ } \}#define sound_free (generic_free[round_size(sizeof(sound_node)) >> 3])#define NORMALSOUNDALLOC#ifdef NORMALSOUNDALLOC#define falloc_sound(sp, who) { \ if (!Qempty(sound_free)) { \ Qget(sound_free, sound_type, sp); \ } else { \ sp = (sound_type) get_from_pool(round_size(sizeof(sound_node))); \ } \ sound_used++; \ if (DEBUG_MEM) dbg_mem_allocated(sp, who); \}#else#define falloc_sound(sp) \ sp =(sound_type) \ get_from_pool(round_size(sizeof(sound_node)))#endif/* note: usually you call sound_unref, not this macro */#define ffree_sound(sp, who) { \/* sound_already_free_test(); */ \ if (DEBUG_MEM) dbg_mem_freed(sp, who); \ Qenter(sound_free, sp); \ sound_used--; \}/* falloc_generic -- sp gets new node of type sptype *//**/#define falloc_generic(sp, sptype, who) { \ int size = round_size(sizeof(sptype)); \ falloc_generic_bytes(sp, sptype, size, who) }/* falloc_generic_n -- sp gets new array of n sptype's *//**/#define falloc_generic_n(sp, sptype, n, who) { \ int min_size = sizeof(sptype) * (n); \ int size = round_size(min_size); \ falloc_generic_bytes(sp, sptype, size, who) }#define falloc_generic_bytes(sp, sptype, size, who) \ if ((size >> 3) >= MAXLISTS) { \ stdputstr("falloc_generic problem\n"); \ sp = (sptype *) malloc(size); \ } else if (!Qempty(generic_free[size >> 3])) { \ Qget(generic_free[size >> 3], sptype *, sp); \ } else { \ sp = (sptype *) get_from_pool(size); \ } \ if (DEBUG_MEM) dbg_mem_allocated(sp, who); \/* printf("GENERIC ALLOC %x\n", sp); *//* ffree_generic puts an item back on proper freelist *//* NOTE: sIzE is capitalized funny so that it will not * match an actual parameter, e.g. if the caller writes * ffree_generic(ptr, size), we don't want the expanded * code to include: "int size = round_size(size) >> 3"! */#define ffree_generic(sp, nn, who) { \ int sIzE = round_size(nn) >> 3; \ if (DEBUG_MEM) dbg_mem_freed(sp, who); \ /* printf("GENERIC FREE %x SIZE %d\n", sp, nnn); */ \ if ((sIzE) >= MAXLISTS) { \ free(sp); \ } else { \ Qenter(generic_free[sIzE], sp); \ } \}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -