📄 recindex.c
字号:
/* $Id: recindex.c,v 1.34 2002/08/02 19:26:55 adam Exp $ Copyright (C) 1995,1996,1997,1998,1999,2000,2001,2002 Index Data ApsThis file is part of the Zebra server.Zebra is free software; you can redistribute it and/or modify it underthe terms of the GNU General Public License as published by the FreeSoftware Foundation; either version 2, or (at your option) any laterversion.Zebra is distributed in the hope that it will be useful, but WITHOUT ANYWARRANTY; without even the implied warranty of MERCHANTABILITY orFITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public Licensefor more details.You should have received a copy of the GNU General Public Licensealong with Zebra; see the file LICENSE.zebra. If not, write to theFree Software Foundation, 59 Temple Place - Suite 330, Boston, MA02111-1307, USA.*//* * Format of first block * next (4 bytes) * ref_count (4 bytes) * block (504 bytes) * * Format of subsequent blocks * next (4 bytes) * block (508 bytes) * * Format of each record * sysno * (length, data) - pairs * length = 0 if same as previous */#include <stdio.h>#include <assert.h>#include <string.h>#include "recindxp.h"#if HAVE_BZLIB_H#include <bzlib.h>#endifstatic void rec_write_head (Records p){ int r; assert (p); assert (p->index_BFile); r = bf_write (p->index_BFile, 0, 0, sizeof(p->head), &p->head); if (r) { logf (LOG_FATAL|LOG_ERRNO, "write head of %s", p->index_fname); exit (1); }}static void rec_tmp_expand (Records p, int size){ if (p->tmp_size < size + 2048 || p->tmp_size < p->head.block_size[REC_BLOCK_TYPES-1]*2) { xfree (p->tmp_buf); p->tmp_size = size + p->head.block_size[REC_BLOCK_TYPES-1]*2 + 2048; p->tmp_buf = (char *) xmalloc (p->tmp_size); }}static int read_indx (Records p, int sysno, void *buf, int itemsize, int ignoreError){ int r; int pos = (sysno-1)*itemsize; r = bf_read (p->index_BFile, 1+pos/128, pos%128, itemsize, buf); if (r != 1 && !ignoreError) { logf (LOG_FATAL|LOG_ERRNO, "read in %s at pos %ld", p->index_fname, (long) pos); exit (1); } return r;}static void write_indx (Records p, int sysno, void *buf, int itemsize){ int pos = (sysno-1)*itemsize; bf_write (p->index_BFile, 1+pos/128, pos%128, itemsize, buf);}static void rec_release_blocks (Records p, int sysno){ struct record_index_entry entry; int freeblock; char block_and_ref[sizeof(short) + sizeof(int)]; int dst_type; int first = 1; if (read_indx (p, sysno, &entry, sizeof(entry), 1) != 1) return ; freeblock = entry.next; assert (freeblock > 0); dst_type = freeblock & 7; assert (dst_type < REC_BLOCK_TYPES); freeblock = freeblock / 8; while (freeblock) { if (bf_read (p->data_BFile[dst_type], freeblock, 0, sizeof(block_and_ref), block_and_ref) != 1) { logf (LOG_FATAL|LOG_ERRNO, "read in rec_del_single"); exit (1); } if (first) { short ref; memcpy (&ref, block_and_ref + sizeof(int), sizeof(ref)); --ref; memcpy (block_and_ref + sizeof(int), &ref, sizeof(ref)); if (ref) { if (bf_write (p->data_BFile[dst_type], freeblock, 0, sizeof(block_and_ref), block_and_ref)) { logf (LOG_FATAL|LOG_ERRNO, "write in rec_del_single"); exit (1); } return; } first = 0; } if (bf_write (p->data_BFile[dst_type], freeblock, 0, sizeof(freeblock), &p->head.block_free[dst_type])) { logf (LOG_FATAL|LOG_ERRNO, "write in rec_del_single"); exit (1); } p->head.block_free[dst_type] = freeblock; memcpy (&freeblock, block_and_ref, sizeof(int)); p->head.block_used[dst_type]--; } p->head.total_bytes -= entry.size;}static void rec_delete_single (Records p, Record rec){ struct record_index_entry entry; rec_release_blocks (p, rec->sysno); entry.next = p->head.index_free; entry.size = 0; p->head.index_free = rec->sysno; write_indx (p, rec->sysno, &entry, sizeof(entry));}static void rec_write_tmp_buf (Records p, int size, int *sysnos){ struct record_index_entry entry; int no_written = 0; char *cptr = p->tmp_buf; int block_prev = -1, block_free; int dst_type = 0; int i; for (i = 1; i<REC_BLOCK_TYPES; i++) if (size >= p->head.block_move[i]) dst_type = i; while (no_written < size) { block_free = p->head.block_free[dst_type]; if (block_free) { if (bf_read (p->data_BFile[dst_type], block_free, 0, sizeof(*p->head.block_free), &p->head.block_free[dst_type]) != 1) { logf (LOG_FATAL|LOG_ERRNO, "read in %s at free block %d", p->data_fname[dst_type], block_free); exit (1); } } else block_free = p->head.block_last[dst_type]++; if (block_prev == -1) { entry.next = block_free*8 + dst_type; entry.size = size; p->head.total_bytes += size; while (*sysnos > 0) { write_indx (p, *sysnos, &entry, sizeof(entry)); sysnos++; } } else { memcpy (cptr, &block_free, sizeof(int)); bf_write (p->data_BFile[dst_type], block_prev, 0, 0, cptr); cptr = p->tmp_buf + no_written; } block_prev = block_free; no_written += p->head.block_size[dst_type] - sizeof(int); p->head.block_used[dst_type]++; } assert (block_prev != -1); block_free = 0; memcpy (cptr, &block_free, sizeof(int)); bf_write (p->data_BFile[dst_type], block_prev, 0, sizeof(int) + (p->tmp_buf+size) - cptr, cptr);}Records rec_open (BFiles bfs, int rw, int compression_method){ Records p; int i, r; int version; p = (Records) xmalloc (sizeof(*p)); p->compression_method = compression_method; p->rw = rw; p->tmp_size = 1024; p->tmp_buf = (char *) xmalloc (p->tmp_size); p->index_fname = "reci"; p->index_BFile = bf_open (bfs, p->index_fname, 128, rw); if (p->index_BFile == NULL) { logf (LOG_FATAL|LOG_ERRNO, "open %s", p->index_fname); exit (1); } r = bf_read (p->index_BFile, 0, 0, 0, p->tmp_buf); switch (r) { case 0: memcpy (p->head.magic, REC_HEAD_MAGIC, sizeof(p->head.magic)); sprintf (p->head.version, "%3d", REC_VERSION); p->head.index_free = 0; p->head.index_last = 1; p->head.no_records = 0; p->head.total_bytes = 0; for (i = 0; i<REC_BLOCK_TYPES; i++) { p->head.block_free[i] = 0; p->head.block_last[i] = 1; p->head.block_used[i] = 0; } p->head.block_size[0] = 128; p->head.block_move[0] = 0; for (i = 1; i<REC_BLOCK_TYPES; i++) { p->head.block_size[i] = p->head.block_size[i-1] * 4; p->head.block_move[i] = p->head.block_size[i] * 24; } if (rw) rec_write_head (p); break; case 1: memcpy (&p->head, p->tmp_buf, sizeof(p->head)); if (memcmp (p->head.magic, REC_HEAD_MAGIC, sizeof(p->head.magic))) { logf (LOG_FATAL, "file %s has bad format", p->index_fname); exit (1); } version = atoi (p->head.version); if (version != REC_VERSION) { logf (LOG_FATAL, "file %s is version %d, but version" " %d is required", p->index_fname, version, REC_VERSION); exit (1); } break; } for (i = 0; i<REC_BLOCK_TYPES; i++) { char str[80]; sprintf (str, "recd%c", i + 'A'); p->data_fname[i] = (char *) xmalloc (strlen(str)+1); strcpy (p->data_fname[i], str); p->data_BFile[i] = NULL; } for (i = 0; i<REC_BLOCK_TYPES; i++) { if (!(p->data_BFile[i] = bf_open (bfs, p->data_fname[i], p->head.block_size[i], rw))) { logf (LOG_FATAL|LOG_ERRNO, "bf_open %s", p->data_fname[i]); exit (1); } } p->cache_max = 400; p->cache_cur = 0; p->record_cache = (struct record_cache_entry *) xmalloc (sizeof(*p->record_cache)*p->cache_max); zebra_mutex_init (&p->mutex); return p;}static void rec_encode_unsigned (unsigned n, unsigned char *buf, int *len){ (*len) = 0; while (n > 127) { buf[*len] = 128 + (n & 127); n = n >> 7; (*len)++; } buf[*len] = n; (*len)++;}static void rec_decode_unsigned(unsigned *np, unsigned char *buf, int *len){ unsigned n = 0; unsigned w = 1; (*len) = 0; while (buf[*len] > 127) { n += w*(buf[*len] & 127); w = w << 7; (*len)++; } n += w * buf[*len]; (*len)++; *np = n;}static void rec_cache_flush_block1 (Records p, Record rec, Record last_rec, char **out_buf, int *out_size, int *out_offset){ int i; int len; for (i = 0; i<REC_NO_INFO; i++) { if (*out_offset + (int) rec->size[i] + 20 > *out_size) { int new_size = *out_offset + rec->size[i] + 65536; char *np = (char *) xmalloc (new_size); if (*out_offset) memcpy (np, *out_buf, *out_offset); xfree (*out_buf); *out_size = new_size; *out_buf = np; } if (i == 0) { rec_encode_unsigned (rec->sysno, *out_buf + *out_offset, &len); (*out_offset) += len; } if (rec->size[i] == 0) { rec_encode_unsigned (1, *out_buf + *out_offset, &len); (*out_offset) += len; } else if (last_rec && rec->size[i] == last_rec->size[i] && !memcmp (rec->info[i], last_rec->info[i], rec->size[i])) { rec_encode_unsigned (0, *out_buf + *out_offset, &len); (*out_offset) += len; } else { rec_encode_unsigned (rec->size[i]+1, *out_buf + *out_offset, &len); (*out_offset) += len; memcpy (*out_buf + *out_offset, rec->info[i], rec->size[i]); (*out_offset) += rec->size[i]; } }}static void rec_write_multiple (Records p, int saveCount){ int i; short ref_count = 0; char compression_method; Record last_rec = 0; int out_size = 1000; int out_offset = 0; char *out_buf = (char *) xmalloc (out_size); int *sysnos = (int *) xmalloc (sizeof(*sysnos) * (p->cache_cur + 1)); int *sysnop = sysnos; for (i = 0; i<p->cache_cur - saveCount; i++) { struct record_cache_entry *e = p->record_cache + i; switch (e->flag) { case recordFlagNew: rec_cache_flush_block1 (p, e->rec, last_rec, &out_buf, &out_size, &out_offset); *sysnop++ = e->rec->sysno; ref_count++; e->flag = recordFlagNop; last_rec = e->rec; break; case recordFlagWrite: rec_release_blocks (p, e->rec->sysno); rec_cache_flush_block1 (p, e->rec, last_rec, &out_buf, &out_size, &out_offset); *sysnop++ = e->rec->sysno; ref_count++; e->flag = recordFlagNop; last_rec = e->rec; break; case recordFlagDelete: rec_delete_single (p, e->rec); e->flag = recordFlagNop; break; default: break; } }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -