/* This file contains routines for buffer management. Copyright 1995 Philip Homburg */ #define BUF_IMPLEMENTATION 1 /* Avoid some macros */ #include "inet.h" #include #include #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_NR DECLARE_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_NR DECLARE_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_NR DECLARE_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 )); #endif PRIVATE 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_CHECK int inet_buf_debug; unsigned buf_generation; PRIVATE bf_checkreq_t checkreq[CLIENT_NR]; #endif #ifndef BUF_TRACK_ALLOC_FREE FORWARD acc_t *bf_small_memreq ARGS(( size_t size )); #else FORWARD 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 FORWARD void free_accs ARGS(( void )); #ifdef BUF_CONSISTENCY_CHECK FORWARD void count_free_bufs ARGS(( acc_t *list )); FORWARD int report_buffer ARGS(( buf_t *buf, char *label, int i )); #endif PUBLIC void bf_init() { int i; size_t buf_s; acc_t *acc; bf_buf_gran= BUF_S; buf_s= 0; for (i=0;iacc_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 PUBLIC void bf_logon(func) bf_freereq_t func; #else PUBLIC void bf_logon(func, checkfunc) bf_freereq_t func; bf_checkreq_t checkfunc; #endif { int i; for (i=0;i0); 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_bufsizeacc_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_bufsizeacc_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 PRIVATE acc_t *bf_small_memreq(size) #else PRIVATE 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 PUBLIC void bf_afree(acc) #else PUBLIC 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 PUBLIC acc_t *bf_dupacc(acc_ptr) #else PUBLIC 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; } 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_FREE PUBLIC acc_t *bf_packIffLess(pack, min_len) #else PUBLIC 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 PUBLIC acc_t *bf_pack(old_acc) #else PUBLIC 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 PUBLIC acc_t *bf_cut (data, offset, length) #else PUBLIC 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 PUBLIC acc_t *bf_delhead (data, offset) #else PUBLIC 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 PUBLIC acc_t *bf_append(data_first, data_second) #else PUBLIC 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 PRIVATE 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 PRIVATE 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 PRIVATE 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 PUBLIC int bf_consistency_check() { acc_t *acc; int silent; int error; int i; buf_generation++; for (i=0; iacc_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; iacc_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; iacc_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; } } PRIVATE 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; } PUBLIC 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; } } PUBLIC 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; } PUBLIC 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 PUBLIC int bf_linkcheck(acc) acc_t *acc; { int i; buf_t *buffer; for (i= 0; iacc_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; } PRIVATE void free_accs() { int i, j; DBLOCK(1, printf("free_accs\n")); assert(bf_linkcheck(bf_linkcheck_acc)); for (i=0; !acc_freelist && iacc_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 $ */