minix3/net/inet/buf.c

1254 lines
25 KiB
C

/*
This file contains routines for buffer management.
Copyright 1995 Philip Homburg
*/
#include "inet.h"
#include <stdlib.h>
#include <string.h>
#include "generic/assert.h"
#include "generic/buf.h"
THIS_FILE
#ifndef BUF_USEMALLOC
#define BUF_USEMALLOC 0
#endif
#ifndef BUF512_NR
#define BUF512_NR 512
#endif
#ifndef BUF2K_NR
#define BUF2K_NR 0
#endif
#ifndef BUF32K_NR
#define BUF32K_NR 0
#endif
#define ACC_NR ((BUF512_NR+BUF2K_NR+BUF32K_NR)*3)
#define CLIENT_NR 10
#define DECLARE_TYPE(Tag, Type, Size) \
typedef struct Tag \
{ \
buf_t buf_header; \
char buf_data[Size]; \
} Type
#if BUF_USEMALLOC
#define DECLARE_STORAGE(Type, Ident, Nitems) \
PRIVATE Type *Ident
#define ALLOC_STORAGE(Ident, Nitems, Label) \
do \
{ \
printf("buf.c: malloc %d %s\n", Nitems, Label); \
Ident= malloc(sizeof(*Ident) * Nitems); \
if (!Ident) \
ip_panic(( "unable to alloc %s", Label )); \
} while(0)
#else
#define DECLARE_STORAGE(Type, Ident, Nitems) \
PRIVATE Type Ident[Nitems]
#define ALLOC_STORAGE(Ident, Nitems, Label) \
(void)0
#endif
#if BUF512_NR
DECLARE_TYPE(buf512, buf512_t, 512);
static acc_t *buf512_freelist;
DECLARE_STORAGE(buf512_t, buffers512, BUF512_NR);
static void bf_512free ARGS(( acc_t *acc ));
#endif
#if BUF2K_NR
DECLARE_TYPE(buf2K, buf2K_t, (2*1024));
static acc_t *buf2K_freelist;
DECLARE_STORAGE(buf2K_t, buffers2K, BUF2K_NR);
static void bf_2Kfree ARGS(( acc_t *acc ));
#endif
#if BUF32K_NR
DECLARE_TYPE(buf32K, buf32K_t, (32*1024));
static acc_t *buf32K_freelist;
DECLARE_STORAGE(buf32K_t, buffers32K, BUF32K_NR);
static void bf_32Kfree ARGS(( acc_t *acc ));
#endif
static acc_t *acc_freelist;
DECLARE_STORAGE(acc_t, accessors, ACC_NR);
static bf_freereq_t freereq[CLIENT_NR];
static size_t bf_buf_gran;
size_t bf_free_bufsize;
acc_t *bf_temporary_acc;
acc_t *bf_linkcheck_acc;
#ifdef BUF_CONSISTENCY_CHECK
int inet_buf_debug;
unsigned buf_generation;
static bf_checkreq_t checkreq[CLIENT_NR];
#endif
#ifndef BUF_TRACK_ALLOC_FREE
static acc_t *bf_small_memreq ARGS(( size_t size ));
#else
static acc_t *_bf_small_memreq ARGS(( char *clnt_file, int clnt_line,
size_t size ));
#define bf_small_memreq(a) _bf_small_memreq(clnt_file, clnt_line, a)
#endif
static void free_accs ARGS(( void ));
#ifdef BUF_CONSISTENCY_CHECK
static void count_free_bufs ARGS(( acc_t *list ));
static int report_buffer ARGS(( buf_t *buf, char *label, int i ));
#endif
void bf_init()
{
int i;
size_t buf_s;
acc_t *acc;
bf_buf_gran= BUF_S;
buf_s= 0;
for (i=0;i<CLIENT_NR;i++)
freereq[i]=0;
#ifdef BUF_CONSISTENCY_CHECK
for (i=0;i<CLIENT_NR;i++)
checkreq[i]=0;
#endif
#if BUF512_NR
ALLOC_STORAGE(buffers512, BUF512_NR, "512B-buffers");
#endif
#if BUF2K_NR
ALLOC_STORAGE(buffers2K, BUF2K_NR, "2K-buffers");
#endif
#if BUF32K_NR
ALLOC_STORAGE(buffers32K, BUF32K_NR, "32K-buffers");
#endif
ALLOC_STORAGE(accessors, ACC_NR, "accs");
acc_freelist= NULL;
for (i=0;i<ACC_NR;i++)
{
memset(&accessors[i], '\0', sizeof(accessors[i]));
accessors[i].acc_linkC= 0;
accessors[i].acc_next= acc_freelist;
acc_freelist= &accessors[i];
}
#define INIT_BUFFERS(Ident, Nitems, Freelist, Freefunc) \
do \
{ \
Freelist= NULL; \
for (i=0;i<Nitems;i++) \
{ \
acc= acc_freelist; \
if (!acc) \
ip_panic(( "fewer accessors than buffers")); \
acc_freelist= acc->acc_next; \
acc->acc_linkC= 0; \
\
memset(&Ident[i], '\0', sizeof(Ident[i])); \
Ident[i].buf_header.buf_linkC= 0; \
Ident[i].buf_header.buf_free= Freefunc; \
Ident[i].buf_header.buf_size= \
sizeof(Ident[i].buf_data); \
Ident[i].buf_header.buf_data_p= \
Ident[i].buf_data; \
\
acc->acc_buffer= &Ident[i].buf_header; \
acc->acc_next= Freelist; \
Freelist= acc; \
} \
if (sizeof(Ident[0].buf_data) < bf_buf_gran) \
bf_buf_gran= sizeof(Ident[0].buf_data); \
if (sizeof(Ident[0].buf_data) > buf_s) \
buf_s= sizeof(Ident[0].buf_data); \
} while(0)
#if BUF512_NR
INIT_BUFFERS(buffers512, BUF512_NR, buf512_freelist, bf_512free);
#endif
#if BUF2K_NR
INIT_BUFFERS(buffers2K, BUF2K_NR, buf2K_freelist, bf_2Kfree);
#endif
#if BUF32K_NR
INIT_BUFFERS(buffers32K, BUF32K_NR, buf32K_freelist, bf_32Kfree);
#endif
#undef INIT_BUFFERS
assert (buf_s == BUF_S);
}
#ifndef BUF_CONSISTENCY_CHECK
void bf_logon(func)
bf_freereq_t func;
#else
void bf_logon(func, checkfunc)
bf_freereq_t func;
bf_checkreq_t checkfunc;
#endif
{
int i;
for (i=0;i<CLIENT_NR;i++)
if (!freereq[i])
{
freereq[i]=func;
#ifdef BUF_CONSISTENCY_CHECK
checkreq[i]= checkfunc;
#endif
return;
}
ip_panic(( "buf.c: too many clients" ));
}
/*
bf_memreq
*/
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_memreq(size)
#else
acc_t *_bf_memreq(clnt_file, clnt_line, size)
char *clnt_file;
int clnt_line;
#endif
size_t size;
{
acc_t *head, *tail, *new_acc;
buf_t *buf;
int i,j;
size_t count;
assert (size>0);
head= NULL;
tail= NULL;
while (size)
{
new_acc= NULL;
/* Note the tricky dangling else... */
#define ALLOC_BUF(Freelist, Bufsize) \
if (Freelist && (Bufsize == BUF_S || size <= Bufsize)) \
{ \
new_acc= Freelist; \
Freelist= new_acc->acc_next; \
\
assert(new_acc->acc_linkC == 0); \
new_acc->acc_linkC= 1; \
buf= new_acc->acc_buffer; \
assert(buf->buf_linkC == 0); \
buf->buf_linkC= 1; \
} \
else
/* Sort attempts by buffer size */
#if BUF512_NR
ALLOC_BUF(buf512_freelist, 512)
#endif
#if BUF2K_NR
ALLOC_BUF(buf2K_freelist, 2*1024)
#endif
#if BUF32K_NR
ALLOC_BUF(buf32K_freelist, 32*1024)
#endif
#undef ALLOC_BUF
{
DBLOCK(2, printf("freeing buffers\n"));
bf_free_bufsize= 0;
for (i=0; bf_free_bufsize<size && i<MAX_BUFREQ_PRI;
i++)
{
for (j=0; j<CLIENT_NR; j++)
{
if (freereq[j])
(*freereq[j])(i);
}
#if DEBUG && 0
{ acc_t *acc;
j= 0; for(acc= buf512_freelist; acc; acc= acc->acc_next) j++;
printf("# of free 512-bytes buffer is now %d\n", j); }
#endif
}
#if DEBUG && 0
{ printf("last level was level %d\n", i-1); }
#endif
if (bf_free_bufsize<size)
ip_panic(( "not enough buffers freed" ));
continue;
}
#ifdef BUF_TRACK_ALLOC_FREE
new_acc->acc_alloc_file= clnt_file;
new_acc->acc_alloc_line= clnt_line;
buf->buf_alloc_file= clnt_file;
buf->buf_alloc_line= clnt_line;
#endif
if (!head)
head= new_acc;
else
tail->acc_next= new_acc;
tail= new_acc;
count= tail->acc_buffer->buf_size;
if (count > size)
count= size;
tail->acc_offset= 0;
tail->acc_length= count;
size -= count;
}
tail->acc_next= NULL;
return head;
}
/*
bf_small_memreq
*/
#ifndef BUF_TRACK_ALLOC_FREE
static acc_t *bf_small_memreq(size)
#else
static acc_t *_bf_small_memreq(clnt_file, clnt_line, size)
char *clnt_file;
int clnt_line;
#endif
size_t size;
{
return bf_memreq(size);
}
#ifndef BUF_TRACK_ALLOC_FREE
void bf_afree(acc)
#else
void _bf_afree(clnt_file, clnt_line, acc)
char *clnt_file;
int clnt_line;
#endif
acc_t *acc;
{
acc_t *next_acc;
buf_t *buf;
while (acc)
{
#if defined(bf_afree)
DIFBLOCK(1, (acc->acc_linkC <= 0),
printf("clnt_file= %s, clnt_line= %d\n",
clnt_file, clnt_line));
#endif
assert (acc->acc_linkC>0);
if (--acc->acc_linkC > 0)
break;
#ifdef BUF_TRACK_ALLOC_FREE
acc->acc_free_file= clnt_file;
acc->acc_free_line= clnt_line;
#endif
buf= acc->acc_buffer;
assert (buf);
#if defined(bf_afree)
DIFBLOCK(1, (buf->buf_linkC == 0),
printf("clnt_file= %s, clnt_line= %d\n",
clnt_file, clnt_line));
#endif
assert (buf->buf_linkC>0);
if (--buf->buf_linkC > 0)
{
acc->acc_buffer= NULL;
next_acc= acc->acc_next;
acc->acc_next= acc_freelist;
acc_freelist= acc;
#ifdef BUF_CONSISTENCY_CHECK
if (inet_buf_debug)
{
acc->acc_offset= 0xdeadbeaf;
acc->acc_length= 0xdeadbeaf;
acc->acc_buffer= (buf_t *)0xdeadbeaf;
acc->acc_ext_link= (acc_t *)0xdeadbeaf;
}
#endif
acc= next_acc;
continue;
}
bf_free_bufsize += buf->buf_size;
#ifdef BUF_TRACK_ALLOC_FREE
buf->buf_free_file= clnt_file;
buf->buf_free_line= clnt_line;
#endif
next_acc= acc->acc_next;
buf->buf_free(acc);
acc= next_acc;
continue;
}
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_dupacc(acc_ptr)
#else
acc_t *_bf_dupacc(clnt_file, clnt_line, acc_ptr)
char *clnt_file;
int clnt_line;
#endif
register acc_t *acc_ptr;
{
register acc_t *new_acc;
if (!acc_freelist)
{
free_accs();
if (!acc_freelist)
ip_panic(( "buf.c: out of accessors" ));
}
new_acc= acc_freelist;
acc_freelist= new_acc->acc_next;
*new_acc= *acc_ptr;
if (acc_ptr->acc_next)
acc_ptr->acc_next->acc_linkC++;
if (acc_ptr->acc_buffer)
acc_ptr->acc_buffer->buf_linkC++;
new_acc->acc_linkC= 1;
#ifdef BUF_TRACK_ALLOC_FREE
new_acc->acc_alloc_file= clnt_file;
new_acc->acc_alloc_line= clnt_line;
#endif
return new_acc;
}
size_t bf_bufsize(acc_ptr)
register acc_t *acc_ptr;
{
register size_t size;
assert(acc_ptr);
size=0;
while (acc_ptr)
{
assert(acc_ptr >= accessors && acc_ptr <= &accessors[ACC_NR-1]);
size += acc_ptr->acc_length;
acc_ptr= acc_ptr->acc_next;
}
return size;
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_packIffLess(pack, min_len)
#else
acc_t *_bf_packIffLess(clnt_file, clnt_line, pack, min_len)
char *clnt_file;
int clnt_line;
#endif
acc_t *pack;
int min_len;
{
if (!pack || pack->acc_length >= min_len)
return pack;
#if DEBUG
#ifdef bf_packIffLess
{ where(); printf("calling bf_pack because of %s %d: %d\n", bf_pack_file,
bf_pack_line, min_len); }
#endif
#endif
return bf_pack(pack);
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_pack(old_acc)
#else
acc_t *_bf_pack(clnt_file, clnt_line, old_acc)
char *clnt_file;
int clnt_line;
#endif
acc_t *old_acc;
{
acc_t *new_acc, *acc_ptr_old, *acc_ptr_new;
size_t size, offset_old, offset_new, block_size, block_size_old;
/* Check if old acc is good enough. */
if (!old_acc || (!old_acc->acc_next && old_acc->acc_linkC == 1 &&
old_acc->acc_buffer->buf_linkC == 1))
{
return old_acc;
}
size= bf_bufsize(old_acc);
assert(size > 0);
new_acc= bf_memreq(size);
acc_ptr_old= old_acc;
acc_ptr_new= new_acc;
offset_old= 0;
offset_new= 0;
while (size)
{
assert (acc_ptr_old);
if (offset_old == acc_ptr_old->acc_length)
{
offset_old= 0;
acc_ptr_old= acc_ptr_old->acc_next;
continue;
}
assert (offset_old < acc_ptr_old->acc_length);
block_size_old= acc_ptr_old->acc_length - offset_old;
assert (acc_ptr_new);
if (offset_new == acc_ptr_new->acc_length)
{
offset_new= 0;
acc_ptr_new= acc_ptr_new->acc_next;
continue;
}
assert (offset_new < acc_ptr_new->acc_length);
block_size= acc_ptr_new->acc_length - offset_new;
if (block_size > block_size_old)
block_size= block_size_old;
memcpy(ptr2acc_data(acc_ptr_new)+offset_new,
ptr2acc_data(acc_ptr_old)+offset_old, block_size);
offset_new += block_size;
offset_old += block_size;
size -= block_size;
}
bf_afree(old_acc);
return new_acc;
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_cut (data, offset, length)
#else
acc_t *_bf_cut (clnt_file, clnt_line, data, offset, length)
char *clnt_file;
int clnt_line;
#endif
register acc_t *data;
register unsigned offset;
register unsigned length;
{
register acc_t *head, *tail;
if (!data && !offset && !length)
return NULL;
#ifdef BUF_TRACK_ALLOC_FREE
assert(data ||
(printf("from %s, %d: %u, %u\n",
clnt_file, clnt_line, offset, length), 0));
#else
assert(data);
#endif
assert(data);
if (!length)
{
head= bf_dupacc(data);
bf_afree(head->acc_next);
head->acc_next= NULL;
head->acc_length= 0;
return head;
}
while (data && offset>=data->acc_length)
{
offset -= data->acc_length;
data= data->acc_next;
}
assert (data);
head= bf_dupacc(data);
bf_afree(head->acc_next);
head->acc_next= NULL;
head->acc_offset += offset;
head->acc_length -= offset;
if (length >= head->acc_length)
length -= head->acc_length;
else
{
head->acc_length= length;
length= 0;
}
tail= head;
data= data->acc_next;
while (data && length && length>=data->acc_length)
{
tail->acc_next= bf_dupacc(data);
tail= tail->acc_next;
bf_afree(tail->acc_next);
tail->acc_next= NULL;
data= data->acc_next;
length -= tail->acc_length;
}
if (length)
{
#ifdef bf_cut
assert (data ||
(printf("bf_cut called from %s:%d\n",
clnt_file, clnt_line), 0));
#else
assert (data);
#endif
tail->acc_next= bf_dupacc(data);
tail= tail->acc_next;
bf_afree(tail->acc_next);
tail->acc_next= NULL;
tail->acc_length= length;
}
return head;
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_delhead (data, offset)
#else
acc_t *_bf_delhead (clnt_file, clnt_line, data, offset)
char *clnt_file;
int clnt_line;
#endif
register acc_t *data;
register unsigned offset;
{
acc_t *new_acc;
assert(data);
/* Find the acc we need to modify. */
new_acc= data;
while(offset >= new_acc->acc_length)
{
offset -= new_acc->acc_length;
new_acc= new_acc->acc_next;
#ifdef BUF_TRACK_ALLOC_FREE
assert(new_acc || (printf("called from %s, %d\n",
clnt_file, clnt_line),0));
#else
assert(new_acc);
#endif
}
/* Discard the old acc(s) */
if (new_acc != data)
{
new_acc->acc_linkC++;
bf_afree(data);
data= new_acc;
}
/* Make sure that acc_linkC == 1 */
if (data->acc_linkC != 1)
{
new_acc= bf_dupacc(data);
bf_afree(data);
data= new_acc;
}
/* Delete the last bit by modifying acc_offset and acc_length */
data->acc_offset += offset;
data->acc_length -= offset;
return data;
}
/*
bf_append
*/
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_append(data_first, data_second)
#else
acc_t *_bf_append(clnt_file, clnt_line, data_first, data_second)
char *clnt_file;
int clnt_line;
#endif
acc_t *data_first;
acc_t *data_second;
{
acc_t *head, *tail, *new_acc, *acc_ptr_new, tmp_acc, *curr;
char *src_ptr, *dst_ptr;
size_t size, offset_old, offset_new, block_size_old, block_size;
if (!data_first)
return data_second;
if (!data_second)
return data_first;
head= NULL;
tail= NULL;
while (data_first)
{
if (data_first->acc_linkC == 1)
curr= data_first;
else
{
curr= bf_dupacc(data_first);
assert (curr->acc_linkC == 1);
bf_afree(data_first);
}
data_first= curr->acc_next;
if (!curr->acc_length)
{
curr->acc_next= NULL;
bf_afree(curr);
continue;
}
if (!head)
head= curr;
else
tail->acc_next= curr;
tail= curr;
}
if (!head)
return data_second;
tail->acc_next= NULL;
while (data_second && !data_second->acc_length)
{
curr= data_second;
data_second= data_second->acc_next;
if (data_second)
data_second->acc_linkC++;
bf_afree(curr);
}
if (!data_second)
return head;
if (tail->acc_length + data_second->acc_length >
tail->acc_buffer->buf_size)
{
tail->acc_next= data_second;
return head;
}
if (tail->acc_buffer->buf_size == bf_buf_gran &&
tail->acc_buffer->buf_linkC == 1)
{
if (tail->acc_offset)
{
memmove(tail->acc_buffer->buf_data_p,
ptr2acc_data(tail), tail->acc_length);
tail->acc_offset= 0;
}
dst_ptr= ptr2acc_data(tail) + tail->acc_length;
src_ptr= ptr2acc_data(data_second);
memcpy(dst_ptr, src_ptr, data_second->acc_length);
tail->acc_length += data_second->acc_length;
tail->acc_next= data_second->acc_next;
if (data_second->acc_next)
data_second->acc_next->acc_linkC++;
bf_afree(data_second);
return head;
}
new_acc= bf_small_memreq(tail->acc_length+data_second->acc_length);
acc_ptr_new= new_acc;
offset_old= 0;
offset_new= 0;
size= tail->acc_length;
while (size)
{
assert (acc_ptr_new);
if (offset_new == acc_ptr_new->acc_length)
{
offset_new= 0;
acc_ptr_new= acc_ptr_new->acc_next;
continue;
}
assert (offset_new < acc_ptr_new->acc_length);
assert (offset_old < tail->acc_length);
block_size_old= tail->acc_length - offset_old;
block_size= acc_ptr_new->acc_length - offset_new;
if (block_size > block_size_old)
block_size= block_size_old;
memcpy(ptr2acc_data(acc_ptr_new)+offset_new,
ptr2acc_data(tail)+offset_old, block_size);
offset_new += block_size;
offset_old += block_size;
size -= block_size;
}
offset_old= 0;
size= data_second->acc_length;
while (size)
{
assert (acc_ptr_new);
if (offset_new == acc_ptr_new->acc_length)
{
offset_new= 0;
acc_ptr_new= acc_ptr_new->acc_next;
continue;
}
assert (offset_new < acc_ptr_new->acc_length);
assert (offset_old < data_second->acc_length);
block_size_old= data_second->acc_length - offset_old;
block_size= acc_ptr_new->acc_length - offset_new;
if (block_size > block_size_old)
block_size= block_size_old;
memcpy(ptr2acc_data(acc_ptr_new)+offset_new,
ptr2acc_data(data_second)+offset_old, block_size);
offset_new += block_size;
offset_old += block_size;
size -= block_size;
}
tmp_acc= *tail;
*tail= *new_acc;
*new_acc= tmp_acc;
bf_afree(new_acc);
while (tail->acc_next)
tail= tail->acc_next;
tail->acc_next= data_second->acc_next;
if (data_second->acc_next)
data_second->acc_next->acc_linkC++;
bf_afree(data_second);
return head;
}
#if BUF512_NR
static void bf_512free(acc)
acc_t *acc;
{
#ifdef BUF_CONSISTENCY_CHECK
if (inet_buf_debug)
memset(acc->acc_buffer->buf_data_p, 0xa5, 512);
#endif
acc->acc_next= buf512_freelist;
buf512_freelist= acc;
}
#endif
#if BUF2K_NR
static void bf_2Kfree(acc)
acc_t *acc;
{
#ifdef BUF_CONSISTENCY_CHECK
if (inet_buf_debug)
memset(acc->acc_buffer->buf_data_p, 0xa5, 2*1024);
#endif
acc->acc_next= buf2K_freelist;
buf2K_freelist= acc;
}
#endif
#if BUF32K_NR
static void bf_32Kfree(acc)
acc_t *acc;
{
#ifdef BUF_CONSISTENCY_CHECK
if (inet_buf_debug)
memset(acc->acc_buffer->buf_data_p, 0xa5, 32*1024);
#endif
acc->acc_next= buf32K_freelist;
buf32K_freelist= acc;
}
#endif
#ifdef BUF_CONSISTENCY_CHECK
int bf_consistency_check()
{
acc_t *acc;
int silent;
int error;
int i;
buf_generation++;
for (i=0; i<CLIENT_NR; i++)
{
if (checkreq[i])
(*checkreq[i])();
}
/* Add information about free accessors */
for(acc= acc_freelist; acc; acc= acc->acc_next)
{
if (acc->acc_generation == buf_generation-1)
{
acc->acc_generation= buf_generation;
acc->acc_check_linkC= 0;
}
else
{
assert(acc->acc_generation == buf_generation &&
acc->acc_check_linkC > 0);
acc->acc_check_linkC= -acc->acc_check_linkC;
}
}
#if BUF512_NR
count_free_bufs(buf512_freelist);
#endif
#if BUF2K_NR
count_free_bufs(buf2K_freelist);
#endif
#if BUF32K_NR
count_free_bufs(buf32K_freelist);
#endif
error= 0;
/* Report about accessors */
silent= 0;
for (i=0, acc= accessors; i<ACC_NR; i++, acc++)
{
if (acc->acc_generation != buf_generation)
{
error++;
assert(acc->acc_generation == buf_generation-1);
acc->acc_generation= buf_generation;
if (!silent)
{
printf(
"acc[%d] (%p) has been lost with count %d, last allocated at %s, %d\n",
i, acc, acc->acc_linkC, acc->acc_alloc_file, acc->acc_alloc_line);
#if 0
silent= 1;
#endif
}
continue;
}
if (acc->acc_check_linkC == acc->acc_linkC)
continue;
error++;
if (acc->acc_check_linkC < 0)
{
if (!silent)
{
printf(
"acc[%d] is freed but still in use, allocated at %s, %d, freed at %s, %d\n",
i, acc->acc_alloc_file, acc->acc_alloc_line,
acc->acc_free_file, acc->acc_free_line);
}
acc->acc_check_linkC= -acc->acc_check_linkC;
if (acc->acc_check_linkC == acc->acc_linkC)
{
silent= 1;
continue;
}
}
if (!silent)
{
printf(
"# of tracked links (%d) for acc[%d] don't match with stored link count %d\n",
acc->acc_check_linkC, i, acc->acc_linkC);
printf("acc[%d] was allocated at %s, %d\n",
i, acc->acc_alloc_file, acc->acc_alloc_line);
silent=1;
}
}
/* Report about buffers */
#if BUF512_NR
{
for (i= 0; i<BUF512_NR; i++)
{
error |= report_buffer(&buffers512[i].buf_header,
"512-buffer", i);
}
}
#endif
#if BUF2K_NR
{
for (i= 0; i<BUF2K_NR; i++)
{
error |= report_buffer(&buffers2K[i].buf_header,
"2K-buffer", i);
}
}
#endif
#if BUF32K_NR
{
for (i= 0; i<BUF32K_NR; i++)
{
error |= report_buffer(&buffers32K[i].buf_header,
"32K-buffer", i);
}
}
#endif
return !error;
}
static void count_free_bufs(list)
acc_t *list;
{
acc_t *acc;
buf_t *buf;
for(acc= list; acc; acc= acc->acc_next)
{
if (acc->acc_generation != buf_generation-1)
{
assert(acc->acc_generation == buf_generation &&
acc->acc_check_linkC > 0);
acc->acc_check_linkC= -acc->acc_check_linkC;
continue;
}
acc->acc_generation= buf_generation;
acc->acc_check_linkC= 0;
buf= acc->acc_buffer;
if (buf->buf_generation == buf_generation-1)
{
buf->buf_generation= buf_generation;
buf->buf_check_linkC= 0;
continue;
}
assert(buf->buf_generation == buf_generation &&
buf->buf_check_linkC > 0);
buf->buf_check_linkC= -buf->buf_check_linkC;
}
}
static int report_buffer(buf, label, i)
buf_t *buf;
char *label;
int i;
{
if (buf->buf_generation != buf_generation)
{
assert(buf->buf_generation == buf_generation-1);
buf->buf_generation= buf_generation;
printf(
"%s[%d] (%p) has been lost with count %d, last allocated at %s, %d\n",
label, i, buf,
buf->buf_linkC, buf->buf_alloc_file,
buf->buf_alloc_line);
return 1;
}
if (buf->buf_check_linkC == buf->buf_linkC)
return 0;
if (buf->buf_check_linkC < 0)
{
printf(
"%s[%d] is freed but still in use, allocated at %s, %d, freed at %s, %d\n",
label, i, buf->buf_alloc_file, buf->buf_alloc_line,
buf->buf_free_file, buf->buf_free_line);
buf->buf_check_linkC= -buf->buf_check_linkC;
if (buf->buf_check_linkC == buf->buf_linkC)
return 1;
}
printf(
"# of tracked links (%d) for %s[%d] don't match with stored link count %d\n",
buf->buf_check_linkC, label, i, buf->buf_linkC);
printf("%s[%d] was allocated at %s, %d\n",
label, i, buf->buf_alloc_file, buf->buf_alloc_line);
return 1;
}
void bf_check_acc(acc)
acc_t *acc;
{
buf_t *buf;
while(acc != NULL)
{
if (acc->acc_generation == buf_generation)
{
assert(acc->acc_check_linkC > 0);
acc->acc_check_linkC++;
return;
}
assert(acc->acc_generation == buf_generation-1);
acc->acc_generation= buf_generation;
acc->acc_check_linkC= 1;
buf= acc->acc_buffer;
if (buf->buf_generation == buf_generation)
{
assert(buf->buf_check_linkC > 0);
buf->buf_check_linkC++;
}
else
{
assert(buf->buf_generation == buf_generation-1);
buf->buf_generation= buf_generation;
buf->buf_check_linkC= 1;
}
acc= acc->acc_next;
}
}
void _bf_mark_1acc(clnt_file, clnt_line, acc)
char *clnt_file;
int clnt_line;
acc_t *acc;
{
acc->acc_alloc_file= clnt_file;
acc->acc_alloc_line= clnt_line;
}
void _bf_mark_acc(clnt_file, clnt_line, acc)
char *clnt_file;
int clnt_line;
acc_t *acc;
{
buf_t *buf;
for (; acc; acc= acc->acc_next)
{
acc->acc_alloc_file= clnt_file;
acc->acc_alloc_line= clnt_line;
buf= acc->acc_buffer;
buf->buf_alloc_file= clnt_file;
buf->buf_alloc_line= clnt_line;
}
}
#endif
int bf_linkcheck(acc)
acc_t *acc;
{
int i;
buf_t *buffer;
for (i= 0; i<ACC_NR && acc; i++, acc= acc->acc_next)
{
if (acc->acc_linkC <= 0)
{
printf("wrong acc_linkC (%d) for acc %p\n",
acc->acc_linkC, acc);
return 0;
}
if (acc->acc_offset < 0)
{
printf("wrong acc_offset (%d) for acc %p\n",
acc->acc_offset, acc);
return 0;
}
if (acc->acc_length < 0)
{
printf("wrong acc_length (%d) for acc %p\n",
acc->acc_length, acc);
return 0;
}
buffer= acc->acc_buffer;
if (buffer == NULL)
{
printf("no buffer for acc %p\n", acc);
return 0;
}
if (buffer->buf_linkC <= 0)
{
printf(
"wrong buf_linkC (%d) for buffer %p, from acc %p\n",
buffer->buf_linkC, buffer, acc);
return 0;
}
if (acc->acc_offset + acc->acc_length > buffer->buf_size)
{
printf("%d + %d > %d for buffer %p, and acc %p\n",
acc->acc_offset, acc->acc_length,
buffer->buf_size, buffer, acc);
return 0;
}
}
if (acc != NULL)
{
printf("loop\n");
return 0;
}
return 1;
}
static void free_accs()
{
int i, j;
DBLOCK(1, printf("free_accs\n"));
assert(bf_linkcheck(bf_linkcheck_acc));
for (i=0; !acc_freelist && i<MAX_BUFREQ_PRI; i++)
{
for (j=0; j<CLIENT_NR; j++)
{
bf_free_bufsize= 0;
if (freereq[j])
{
(*freereq[j])(i);
assert(bf_linkcheck(bf_linkcheck_acc) ||
(printf("just called %p\n",
freereq[i]),0));
}
}
}
#if DEBUG
printf("last level was level %d\n", i-1);
#endif
}
#ifndef BUF_TRACK_ALLOC_FREE
acc_t *bf_align(acc, size, alignment)
#else
acc_t *_bf_align(clnt_file, clnt_line, acc, size, alignment)
char *clnt_file;
int clnt_line;
#endif
acc_t *acc;
size_t size;
size_t alignment;
{
char *ptr;
size_t buf_size;
acc_t *head, *tail;
/* Fast check if the buffer is aligned already. */
if (acc->acc_length >= size)
{
ptr= ptr2acc_data(acc);
if (((unsigned)ptr & (alignment-1)) == 0)
return acc;
}
buf_size= bf_bufsize(acc);
#ifdef bf_align
assert((size != 0 && buf_size != 0) ||
(printf("bf_align(..., %d, %d) from %s, %d\n",
size, alignment, clnt_file, clnt_line),0));
#else
assert(size != 0 && buf_size != 0);
#endif
if (buf_size <= size)
{
acc= bf_pack(acc);
return acc;
}
head= bf_cut(acc, 0, size);
tail= bf_cut(acc, size, buf_size-size);
bf_afree(acc);
head= bf_pack(head);
assert(head->acc_next == NULL);
head->acc_next= tail;
return head;
}
#if 0
int chk_acc(acc)
acc_t *acc;
{
int acc_nr;
if (!acc)
return 1;
if (acc < accessors || acc >= &accessors[ACC_NR])
return 0;
acc_nr= acc-accessors;
return acc == &accessors[acc_nr];
}
#endif
/*
* $PchId: buf.c,v 1.19 2003/09/10 08:54:23 philip Exp $
*/