📄 buf.c
字号:
/*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 + -