1254 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			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 $
 | 
						|
 */
 |