⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 buf.c

📁 minix3.1.1源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/*This file contains routines for buffer management.Copyright 1995 Philip Homburg*/#define BUF_IMPLEMENTATION	1	/* Avoid some macros */#include "inet.h"#include <stdlib.h>#include <string.h>#include "generic/assert.h"#include "generic/buf.h"#include "generic/type.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	7#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_NRDECLARE_TYPE(buf512, buf512_t, 512);PRIVATE acc_t *buf512_freelist;DECLARE_STORAGE(buf512_t, buffers512, BUF512_NR);FORWARD void bf_512free ARGS(( acc_t *acc ));#endif#if BUF2K_NRDECLARE_TYPE(buf2K, buf2K_t, (2*1024));PRIVATE acc_t *buf2K_freelist;DECLARE_STORAGE(buf2K_t, buffers2K, BUF2K_NR);FORWARD void bf_2Kfree ARGS(( acc_t *acc ));#endif#if BUF32K_NRDECLARE_TYPE(buf32K, buf32K_t, (32*1024));PRIVATE acc_t *buf32K_freelist;DECLARE_STORAGE(buf32K_t, buffers32K, BUF32K_NR);FORWARD void bf_32Kfree ARGS(( acc_t *acc ));#endifPRIVATE acc_t *acc_freelist;DECLARE_STORAGE(acc_t, accessors, ACC_NR);PRIVATE bf_freereq_t freereq[CLIENT_NR];PRIVATE size_t bf_buf_gran;PUBLIC size_t bf_free_bufsize;PUBLIC acc_t *bf_temporary_acc;PUBLIC acc_t *bf_linkcheck_acc;#ifdef BUF_CONSISTENCY_CHECKint inet_buf_debug;unsigned buf_generation; PRIVATE bf_checkreq_t checkreq[CLIENT_NR];#endif#ifndef BUF_TRACK_ALLOC_FREEFORWARD acc_t *bf_small_memreq ARGS(( size_t size ));#elseFORWARD 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)#endifFORWARD void free_accs ARGS(( void ));#ifdef BUF_CONSISTENCY_CHECKFORWARD void count_free_bufs ARGS(( acc_t *list ));FORWARD int report_buffer ARGS(( buf_t *buf, char *label, int i ));#endifPUBLIC 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_CHECKPUBLIC void bf_logon(func)bf_freereq_t func;#elsePUBLIC 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_FREEPUBLIC acc_t *bf_memreq(size)#elsePUBLIC acc_t *_bf_memreq(clnt_file, clnt_line, size)char *clnt_file;int clnt_line;#endifsize_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_FREEPRIVATE acc_t *bf_small_memreq(size)#elsePRIVATE acc_t *_bf_small_memreq(clnt_file, clnt_line, size)char *clnt_file;int clnt_line;#endifsize_t size;{	return bf_memreq(size);}#ifndef BUF_TRACK_ALLOC_FREEPUBLIC void bf_afree(acc)#elsePUBLIC void _bf_afree(clnt_file, clnt_line, acc)char *clnt_file;int clnt_line;#endifacc_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_FREEPUBLIC acc_t *bf_dupacc(acc_ptr)#elsePUBLIC acc_t *_bf_dupacc(clnt_file, clnt_line, acc_ptr)char *clnt_file;int clnt_line;#endifregister 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;}PUBLIC 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_FREEPUBLIC acc_t *bf_packIffLess(pack, min_len)#elsePUBLIC acc_t *_bf_packIffLess(clnt_file, clnt_line, pack, min_len)char *clnt_file;int clnt_line;#endifacc_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_FREEPUBLIC acc_t *bf_pack(old_acc)#elsePUBLIC acc_t *_bf_pack(clnt_file, clnt_line, old_acc)char *clnt_file;int clnt_line;#endifacc_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_FREEPUBLIC acc_t *bf_cut (data, offset, length)#elsePUBLIC acc_t *_bf_cut (clnt_file, clnt_line, data, offset, length)char *clnt_file;int clnt_line;#endifregister 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_FREEPUBLIC acc_t *bf_delhead (data, offset)#elsePUBLIC acc_t *_bf_delhead (clnt_file, clnt_line, data, offset)char *clnt_file;int clnt_line;#endifregister acc_t *data;register unsigned offset;{	acc_t *new_acc;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -