379 lines
8.6 KiB
C
379 lines
8.6 KiB
C
/* $Header$ */
|
|
/*
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
*/
|
|
#include <limits.h>
|
|
#include <stdlib.h>
|
|
#include "param.h"
|
|
#include "impl.h"
|
|
#include "check.h"
|
|
#include "log.h"
|
|
#include "phys.h"
|
|
|
|
/* Malloc space is traversed by N doubly-linked lists of chunks, each
|
|
containing a couple of house-keeping data addressed as a
|
|
'mallink' and a piece of useful space, called the block.
|
|
The N lists are accessed through their starting pointers in
|
|
free_list[]. Free_list[n] points to a list of chunks between
|
|
2**(n+LOG_MIN_SIZE) and 2**(n+LOG_MIN_SIZE+1)-1, which means
|
|
that the smallest chunk is 2**LOG_MIN_SIZE (== MIN_SIZE).
|
|
*/
|
|
|
|
#ifdef SYSTEM
|
|
#include <system.h>
|
|
#define SBRK sys_break
|
|
#else
|
|
#define SBRK _sbrk
|
|
#define ILL_BREAK (void *)(-1) /* funny failure value */
|
|
#endif
|
|
extern void *SBRK(int incr);
|
|
#ifdef STORE
|
|
#define MAX_STORE 32
|
|
private do_free(mallink *ml), sell_out(void);
|
|
privatedata mallink *store[MAX_STORE];
|
|
#endif /* STORE */
|
|
|
|
void *
|
|
malloc(register size_t n)
|
|
{check_mallinks("malloc entry");{
|
|
register mallink *ml;
|
|
register int min_class;
|
|
|
|
if (n == 0) {
|
|
return NULL;
|
|
}
|
|
if (n < MIN_SIZE) n = align(MIN_SIZE); else n = align(n);
|
|
#ifdef STORE
|
|
if (n <= MAX_STORE*MIN_SIZE) {
|
|
/* look in the store first */
|
|
register mallink **stp = &store[(n >> LOG_MIN_SIZE) - 1];
|
|
|
|
if (ml = *stp) {
|
|
*stp = log_next_of(ml);
|
|
set_store(ml, 0);
|
|
check_mallinks("malloc fast exit");
|
|
assert(! in_store(ml));
|
|
return block_of_mallink(ml);
|
|
}
|
|
}
|
|
#endif /* STORE */
|
|
|
|
check_work_empty("malloc, entry");
|
|
|
|
/* Acquire a chunk of at least size n if at all possible;
|
|
Try everything.
|
|
*/
|
|
{
|
|
/* Inline substitution of "smallest".
|
|
*/
|
|
register size_t n1 = n;
|
|
|
|
assert(n1 < (1L << LOG_MAX_SIZE));
|
|
min_class = 0;
|
|
|
|
do {
|
|
n1 >>= 1;
|
|
min_class++;
|
|
} while (n1 >= MIN_SIZE);
|
|
}
|
|
|
|
if (min_class >= MAX_FLIST)
|
|
return NULL; /* we don't deal in blocks that big */
|
|
ml = first_present(min_class);
|
|
if (ml == MAL_NULL) {
|
|
/* Try and extend */
|
|
register void *p;
|
|
#define GRABSIZE 4096 /* Power of 2 */
|
|
register size_t req =
|
|
((MIN_SIZE<<min_class)+ mallink_size() + GRABSIZE - 1) &
|
|
~(GRABSIZE-1);
|
|
|
|
if (!ml_last) {
|
|
/* first align SBRK() */
|
|
|
|
p = SBRK(0);
|
|
SBRK((int) (align((size_type) p) - (size_type) p));
|
|
}
|
|
|
|
p = SBRK((int)req);
|
|
assert((size_type)p == align((size_type)p));
|
|
if (p == ILL_BREAK) {
|
|
req = n + mallink_size();
|
|
p = SBRK((int)req);
|
|
}
|
|
if (p == ILL_BREAK) {
|
|
/* Now this is bad. The system will not give us
|
|
more memory. We can only liquidate our store
|
|
and hope it helps.
|
|
*/
|
|
#ifdef STORE
|
|
sell_out();
|
|
ml = first_present(min_class);
|
|
if (ml == MAL_NULL) {
|
|
#endif /* STORE */
|
|
/* In this emergency we try to locate a suitable
|
|
chunk in the free_list just below the safe
|
|
one; some of these chunks may fit the job.
|
|
*/
|
|
ml = search_free_list(min_class - 1, n);
|
|
if (!ml) /* really out of space */
|
|
return NULL;
|
|
started_working_on(ml);
|
|
unlink_free_chunk(ml);
|
|
check_mallinks("suitable_chunk, forced");
|
|
#ifdef STORE
|
|
}
|
|
else started_working_on(ml);
|
|
#endif /* STORE */
|
|
}
|
|
else {
|
|
ml = create_chunk(p, req);
|
|
}
|
|
check_mallinks("suitable_chunk, extended");
|
|
}
|
|
else started_working_on(ml);
|
|
|
|
/* we have a chunk */
|
|
set_free(ml, 0);
|
|
calc_checksum(ml);
|
|
check_mallinks("suitable_chunk, removed");
|
|
n += mallink_size();
|
|
if (n + MIN_SIZE <= size_of(ml)) {
|
|
truncate(ml, n);
|
|
}
|
|
stopped_working_on(ml);
|
|
check_mallinks("malloc exit");
|
|
check_work_empty("malloc exit");
|
|
#ifdef STORE
|
|
assert(! in_store(ml));
|
|
#endif
|
|
return block_of_mallink(ml);
|
|
}}
|
|
|
|
void
|
|
free(void *addr)
|
|
{check_mallinks("free entry");{
|
|
register mallink *ml;
|
|
|
|
if (addr == NULL) {
|
|
check_mallinks("free(0) very fast exit");
|
|
return;
|
|
}
|
|
|
|
ml = mallink_of_block(addr);
|
|
#ifdef STORE
|
|
|
|
if (free_of(ml) || in_store(ml))
|
|
return; /* user frees free block */
|
|
if (size_of(ml) <= MAX_STORE*MIN_SIZE) {
|
|
/* return to store */
|
|
mallink **stp = &store[(size_of(ml) >> LOG_MIN_SIZE) - 1];
|
|
|
|
set_log_next(ml, *stp);
|
|
*stp = ml;
|
|
set_store(ml, 1);
|
|
calc_checksum(ml);
|
|
check_mallinks("free fast exit");
|
|
}
|
|
else {
|
|
do_free(ml);
|
|
check_mallinks("free exit");
|
|
}
|
|
}}
|
|
|
|
private
|
|
do_free(register mallink *ml)
|
|
{{
|
|
#endif
|
|
|
|
#ifndef STORE
|
|
if (free_of(ml)) return;
|
|
#endif /* STORE */
|
|
started_working_on(ml);
|
|
set_free(ml, 1);
|
|
calc_checksum(ml);
|
|
if (! last_mallink(ml)) {
|
|
register mallink *next = phys_next_of(ml);
|
|
|
|
if (free_of(next)) coalesce_forw(ml, next);
|
|
}
|
|
|
|
if (! first_mallink(ml)) {
|
|
register mallink *prev = phys_prev_of(ml);
|
|
|
|
if (free_of(prev)) {
|
|
coalesce_backw(ml, prev);
|
|
ml = prev;
|
|
}
|
|
}
|
|
link_free_chunk(ml);
|
|
stopped_working_on(ml);
|
|
check_work_empty("free");
|
|
|
|
/* Compile-time checks on param.h */
|
|
switch (0) {
|
|
case MIN_SIZE < OFF_SET * sizeof(mallink): break;
|
|
case 1: break;
|
|
/* If this statement does not compile due to duplicate case
|
|
entry, the minimum size block cannot hold the links for
|
|
the free blocks. Either raise LOG_MIN_SIZE or switch
|
|
off NON_STANDARD.
|
|
*/
|
|
}
|
|
switch(0) {
|
|
case sizeof(void *) != sizeof(size_type): break;
|
|
case 1: break;
|
|
/* If this statement does not compile due to duplicate
|
|
case entry, size_type is not defined correctly.
|
|
Redefine and compile again.
|
|
*/
|
|
}
|
|
}}
|
|
|
|
void *
|
|
realloc(void *addr, register size_t n)
|
|
{check_mallinks("realloc entry");{
|
|
register mallink *ml, *ph_next;
|
|
register size_t size;
|
|
|
|
if (addr == NULL) {
|
|
/* Behave like most Unix realloc's when handed a
|
|
null-pointer
|
|
*/
|
|
return malloc(n);
|
|
}
|
|
if (n == 0) {
|
|
free(addr);
|
|
return NULL;
|
|
}
|
|
ml = mallink_of_block(addr);
|
|
if (n < MIN_SIZE) n = align(MIN_SIZE); else n = align(n);
|
|
#ifdef STORE
|
|
if (in_store(ml)) {
|
|
register mallink *stp = store[(size_of(ml) >> LOG_MIN_SIZE) - 1];
|
|
mallink *stp1 = NULL;
|
|
while (ml != stp) {
|
|
stp1 = stp;
|
|
stp = log_next_of(stp);
|
|
}
|
|
stp = log_next_of(stp);
|
|
if (! stp1) store[(size_of(ml) >> LOG_MIN_SIZE) - 1] = stp;
|
|
else set_log_next(stp1, stp);
|
|
set_store(ml, 0);
|
|
calc_checksum(ml);
|
|
}
|
|
#endif
|
|
if (free_of(ml)) {
|
|
unlink_free_chunk(ml);
|
|
set_free(ml, 0); /* user reallocs free block */
|
|
}
|
|
started_working_on(ml);
|
|
size = size_of(ml);
|
|
if ( /* we can simplify the problem by adding the next chunk: */
|
|
n > size &&
|
|
!last_mallink(ml) &&
|
|
(ph_next = phys_next_of(ml), free_of(ph_next)) &&
|
|
n <= size + mallink_size() + size_of(ph_next)
|
|
) {
|
|
/* add in the physically next chunk */
|
|
unlink_free_chunk(ph_next);
|
|
combine_chunks(ml, ph_next);
|
|
size = size_of(ml);
|
|
check_mallinks("realloc, combining");
|
|
}
|
|
if (n > size) { /* this didn't help */
|
|
void *new;
|
|
register char *l1, *l2 = addr;
|
|
|
|
stopped_working_on(ml);
|
|
if (!(new = l1 = malloc(n))) return NULL; /* no way */
|
|
while (size--) *l1++ = *l2++;
|
|
free(addr);
|
|
check_work_empty("mv_realloc");
|
|
#ifdef STORE
|
|
assert(! in_store(mallink_of_block(new)));
|
|
#endif
|
|
return new;
|
|
}
|
|
/* it helped, but maybe too well */
|
|
n += mallink_size();
|
|
if (n + MIN_SIZE <= size_of(ml)) {
|
|
truncate(ml, n);
|
|
}
|
|
stopped_working_on(ml);
|
|
check_mallinks("realloc exit");
|
|
check_work_empty("realloc");
|
|
#ifdef STORE
|
|
assert(! in_store(ml));
|
|
#endif
|
|
return addr;
|
|
}}
|
|
|
|
void *
|
|
calloc(size_t nmemb, size_t size)
|
|
{check_mallinks("calloc entry");{
|
|
long *l1, *l2;
|
|
size_t n;
|
|
|
|
if (size == 0) return NULL;
|
|
if (nmemb == 0) return NULL;
|
|
|
|
/* Check for overflow on the multiplication. The peephole-optimizer
|
|
* will eliminate all but one of the possibilities.
|
|
*/
|
|
if (sizeof(size_t) == sizeof(int)) {
|
|
if (UINT_MAX / size < nmemb) return NULL;
|
|
} else if (sizeof(size_t) == sizeof(long)) {
|
|
if (ULONG_MAX / size < nmemb) return NULL;
|
|
} else return NULL; /* can't happen, can it ? */
|
|
|
|
n = size * nmemb;
|
|
if (n < MIN_SIZE) n = align(MIN_SIZE); else n = align(n);
|
|
if (n >= (1L << LOG_MAX_SIZE)) return NULL;
|
|
l1 = (long *) malloc(n);
|
|
l2 = l1 + (n / sizeof(long)); /* n is at least long aligned */
|
|
while ( l2 != l1 ) *--l2 = 0;
|
|
check_mallinks("calloc exit");
|
|
check_work_empty("calloc exit");
|
|
return (void *)l1;
|
|
}}
|
|
/* Auxiliary routines */
|
|
|
|
#ifdef STORE
|
|
private
|
|
sell_out(void) {
|
|
/* Frees all block in store.
|
|
*/
|
|
register mallink **stp;
|
|
|
|
for (stp = &store[0]; stp < &store[MAX_STORE]; stp++) {
|
|
register mallink *ml = *stp;
|
|
|
|
while (ml) {
|
|
*stp = log_next_of(ml);
|
|
set_store(ml, 0);
|
|
do_free(ml);
|
|
ml = *stp;
|
|
}
|
|
}
|
|
|
|
}
|
|
#endif /* STORE */
|
|
|
|
#ifdef ASSERT
|
|
public
|
|
m_assert(const char *fn, int ln)
|
|
{
|
|
char ch;
|
|
|
|
while (*fn)
|
|
write(2, fn++, 1);
|
|
write(2, ": malloc assert failed in line ", 31);
|
|
ch = (ln / 100) + '0'; write(2, &ch, 1); ln %= 100;
|
|
ch = (ln / 10) + '0'; write(2, &ch, 1); ln %= 10;
|
|
ch = (ln / 1) + '0'; write(2, &ch, 1);
|
|
write(2, "\n", 1);
|
|
maldump(1);
|
|
}
|
|
#endif /* ASSERT */
|