mirror of
https://github.com/cookiengineer/audacity
synced 2025-06-16 16:10:06 +02:00
273 lines
6.9 KiB
C
273 lines
6.9 KiB
C
/*
|
|
* falloc.c
|
|
* data for Nyquist memory allocation.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <assert.h>
|
|
#include "xlisp.h"
|
|
#include "sound.h"
|
|
#include "falloc.h"
|
|
|
|
/* special free lists */
|
|
CQUE *sample_block_free = NULL; /* really a sample_block_type */
|
|
|
|
/* special counts */
|
|
int sample_block_used = 0;
|
|
int sample_block_low_water = 0;
|
|
int sample_block_total = 0;
|
|
int snd_list_used = 0;
|
|
int sound_used = 0;
|
|
|
|
/* generic free lists */
|
|
CQUE *generic_free[MAXLISTS];
|
|
|
|
|
|
void falloc_init(void)
|
|
{
|
|
int i;
|
|
for (i = 0; i < MAXLISTS; i++) generic_free[i] = NULL;
|
|
}
|
|
|
|
|
|
/* memory pool */
|
|
char *poolp = NULL;
|
|
char *poolend = NULL;
|
|
|
|
/* sample block memory pool */
|
|
char *spoolp = NULL;
|
|
char *spoolend = NULL;
|
|
|
|
int npools = 0;
|
|
|
|
#if defined(TRACK_POOLS) && TRACK_POOLS
|
|
#define POOL_HEAD_SIZE (round_size(sizeof(CQUE)))
|
|
CQUE *pools = NULL;
|
|
#endif
|
|
|
|
void sound_already_free_test(s)
|
|
sound_type s;
|
|
{
|
|
sound_type sp;
|
|
for (sp = (sound_type) sound_free; sp; sp = (sound_type) ((CQUE *) sp)->qnext) {
|
|
if (s == sp) {
|
|
stdputstr("SOUND ALREADY FREE!!!");
|
|
fflush(stdout);
|
|
sp = 0; sp->list = 0; /* trap to debugger */
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* new_pool -- allocate a new pool from which mem is allocated */
|
|
/**/
|
|
void new_pool(void)
|
|
{
|
|
poolp = (char *) malloc(MAXPOOLSIZE);
|
|
|
|
if (poolp == NULL) {
|
|
fprintf(STDERR, "Nyquist: out of memory!\n");
|
|
EXIT(1);
|
|
}
|
|
|
|
poolend = poolp + MAXPOOLSIZE;
|
|
npools++;
|
|
/* stick to double word boundaries */
|
|
poolp = (char *) round_size(((long) poolp));
|
|
}
|
|
|
|
/* new_spool -- allocate a new spool from which sample blocks are allocated */
|
|
/**/
|
|
void new_spool(void)
|
|
{
|
|
#if defined(TRACK_POOLS) && TRACK_POOLS
|
|
spoolp = (char *) malloc(MAXSPOOLSIZE + POOL_HEAD_SIZE);
|
|
#else
|
|
spoolp = (char *) malloc(MAXSPOOLSIZE);
|
|
#endif
|
|
|
|
if (spoolp == NULL) {
|
|
fprintf(STDERR, "Nyquist: out of memory!\n");
|
|
EXIT(1);
|
|
}
|
|
|
|
#if defined(TRACK_POOLS) && TRACK_POOLS
|
|
Qenter(pools, spoolp);
|
|
spoolp += POOL_HEAD_SIZE;
|
|
#endif
|
|
|
|
spoolend = spoolp + MAXSPOOLSIZE;
|
|
npools++;
|
|
/* stick to double word boundaries */
|
|
spoolp = (char *) round_size(((long) spoolp));
|
|
}
|
|
|
|
|
|
/* find_sample_block -- get sample block when freelist is empty */
|
|
/* Try these strategies in order:
|
|
1) try free list
|
|
2) use pool to get sample_blocks_low_water + BLOCKS_PER_GC blocks or until
|
|
pool runs out
|
|
3) GC and try free list again, set sample_blocks_low_water to
|
|
sample_blocks_used
|
|
4) try pool again
|
|
5) allocate new pool and use it
|
|
*/
|
|
sample_block_type find_sample_block(void)
|
|
{
|
|
sample_block_type sp;
|
|
if (sample_block_total < sample_block_low_water + BLOCKS_PER_GC &&
|
|
check_spool(round_size(sizeof(sample_block_node)))) {
|
|
if (DEBUG_MEM) spoolp += DEBUG_MEM_INFO_SIZE;
|
|
sp = (sample_block_type) spoolp;
|
|
spoolp += round_size(sizeof(sample_block_node));
|
|
sample_block_total++;
|
|
/* printf("fp%d ", sample_block_total - sample_block_low_water); */
|
|
} else {
|
|
/* printf("falloc calling gc\n"); */
|
|
gc();
|
|
sample_block_low_water = sample_block_used;
|
|
if (!Qempty(sample_block_free)) {
|
|
Qget(sample_block_free, sample_block_type, sp);
|
|
/* printf("gc, then from freelist\n"); */
|
|
} else if (check_spool(round_size(sizeof(sample_block_node)))) {
|
|
if (DEBUG_MEM) spoolp += DEBUG_MEM_INFO_SIZE;
|
|
sp = (sample_block_type) spoolp;
|
|
spoolp += round_size(sizeof(sample_block_node));
|
|
sample_block_total++;
|
|
/* printf("gc, then from spool\n"); */
|
|
} else {
|
|
new_spool();
|
|
if (DEBUG_MEM) spoolp += DEBUG_MEM_INFO_SIZE;
|
|
sp = (sample_block_type) spoolp;
|
|
spoolp += round_size(sizeof(sample_block_node));
|
|
sample_block_total++;
|
|
/* printf("gc, then new spool\n"); */
|
|
}
|
|
}
|
|
return sp;
|
|
}
|
|
|
|
|
|
|
|
/* get_from_pool -- return size bytes from pool memory */
|
|
/**/
|
|
char *get_from_pool(size_t siz)
|
|
{
|
|
if (!check_pool(siz)) {
|
|
new_pool();
|
|
}
|
|
poolp += siz;
|
|
if (DEBUG_MEM) poolp += DEBUG_MEM_INFO_SIZE; /* allow for debug info */
|
|
return poolp - siz;
|
|
}
|
|
|
|
|
|
#if defined(TRACK_POOLS) && TRACK_POOLS
|
|
|
|
/* falloc_gc -- return empty pools to the system */
|
|
/*
|
|
* Algorithm: for each pool, move all free sample blocks
|
|
* (on the sample_block_free list) to tlist. If tlist
|
|
* has ALL of the blocks in the pool (determined by
|
|
* byte counts), the pool is returned to the heap.
|
|
*/
|
|
void falloc_gc()
|
|
{
|
|
CQUE *lp = NULL;
|
|
CQUE *cp;
|
|
CQUE *np;
|
|
CQUE *tlist = NULL;
|
|
|
|
/* Scan all allocated pools */
|
|
for (cp = pools; cp; lp = cp, cp = np) {
|
|
char *str = ((char *)cp) + POOL_HEAD_SIZE;
|
|
char *end = str + MAXSPOOLSIZE;
|
|
long tsiz = end - str;
|
|
long csiz = 0;
|
|
CQUE *tsave = NULL;
|
|
CQUE *ln = NULL;
|
|
CQUE *cn;
|
|
CQUE *nn;
|
|
|
|
/* Save pointer to next pool */
|
|
np = cp->qnext;
|
|
|
|
/* Remember head of temp free list */
|
|
tsave = tlist;
|
|
|
|
/* Scan all nodes on the free list */
|
|
for (cn = sample_block_free; cn; ln = cn, cn = nn) {
|
|
|
|
/* Get next node */
|
|
nn = cn->qnext;
|
|
|
|
/* Count it if the node belongs to this pool */
|
|
if (cn >= (CQUE *) str && cn <= (CQUE *) end) {
|
|
csiz += round_size(sizeof(sample_block_node));
|
|
|
|
Qenter(tlist, cn);
|
|
|
|
/* Unlink the node */
|
|
if (cn == sample_block_free) {
|
|
sample_block_free = nn;
|
|
cn = NULL;
|
|
}
|
|
else {
|
|
ln->qnext = nn;
|
|
cn = ln;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* The pool had inuse nodes */
|
|
if (csiz != tsiz) {
|
|
continue;
|
|
}
|
|
|
|
/* Remove the nodes from the temp free list */
|
|
tlist = tsave;
|
|
|
|
/* Maintain stats */
|
|
sample_block_total -= (tsiz / round_size(sizeof(sample_block_node)));
|
|
npools--;
|
|
|
|
/* If this is the active pool, then reset current pointers */
|
|
if (spoolp >= str && spoolp <= end) {
|
|
spoolp = NULL;
|
|
spoolend = NULL;
|
|
}
|
|
|
|
/* Release the pool to the system */
|
|
free(cp);
|
|
|
|
/* Unlink this pool from the list */
|
|
if (cp == pools) {
|
|
pools = np;
|
|
cp = NULL;
|
|
}
|
|
else {
|
|
/* lp cannot be null here: On 1st iteration, lp == NULL, but
|
|
* cp == pools, so code above is executed. Before the for-loop
|
|
* iterates, pools == np (assigned above), and cp == NULL. The
|
|
* for-loop update (lp=cp,cp=np) produces lp == NULL, cp == pools.
|
|
* Since cp == pools, this else branch will not be taken.
|
|
* The other path to this code is via the "continue" above. In that
|
|
* case, the update (lp=cp,cp=np) makes lp a valid pointer or else
|
|
* the loop exits.
|
|
* The assert(lp) is here to possibly make static analyzers happy.
|
|
*/
|
|
assert(lp);
|
|
lp->qnext = np;
|
|
cp = lp;
|
|
}
|
|
}
|
|
|
|
/* Resave list of free nodes */
|
|
sample_block_free = tlist;
|
|
}
|
|
|
|
#endif
|
|
|
|
|