return of venti
This commit is contained in:
parent
88bb285e3d
commit
a0d146edd7
68 changed files with 14443 additions and 2 deletions
293
src/cmd/venti/srv/checkindex.c
Normal file
293
src/cmd/venti/srv/checkindex.c
Normal file
|
|
@ -0,0 +1,293 @@
|
|||
#include "stdinc.h"
|
||||
#include "dat.h"
|
||||
#include "fns.h"
|
||||
|
||||
static int extra, missing, wrong;
|
||||
|
||||
static void
|
||||
phdr(DBlock *eb)
|
||||
{
|
||||
static int did;
|
||||
|
||||
if(!did){
|
||||
did = 1;
|
||||
print("# diff actual correct\n");
|
||||
}
|
||||
print("%s block 0x%llux\n", eb->part->name, eb->addr);
|
||||
}
|
||||
|
||||
static void
|
||||
pie(IEntry *ie, char c)
|
||||
{
|
||||
print("%c %V %22lld %3d %5d %3d\n",
|
||||
c, ie->score, ie->ia.addr, ie->ia.type, ie->ia.size, ie->ia.blocks);
|
||||
}
|
||||
|
||||
static int
|
||||
checkbucket(Index *ix, u32int buck, IBucket *ib)
|
||||
{
|
||||
ISect *is;
|
||||
DBlock *eb;
|
||||
IBucket eib;
|
||||
IEntry ie, eie;
|
||||
int i, ei, ok, c, hdr;
|
||||
|
||||
is = ix->sects[indexsect0(ix, buck)];
|
||||
if(buck < is->start || buck >= is->stop){
|
||||
seterr(EAdmin, "cannot find index section for bucket %lud\n", (ulong)buck);
|
||||
return -1;
|
||||
}
|
||||
buck -= is->start;
|
||||
eb = getdblock(is->part, is->blockbase + ((u64int)buck << is->blocklog), OREAD);
|
||||
if(eb == nil)
|
||||
return -1;
|
||||
unpackibucket(&eib, eb->data, is->bucketmagic);
|
||||
|
||||
ok = 0;
|
||||
ei = 0;
|
||||
hdr = 0;
|
||||
for(i = 0; i < ib->n; i++){
|
||||
while(ei < eib.n){
|
||||
c = ientrycmp(&ib->data[i * IEntrySize], &eib.data[ei * IEntrySize]);
|
||||
if(c == 0){
|
||||
unpackientry(&ie, &ib->data[i * IEntrySize]);
|
||||
unpackientry(&eie, &eib.data[ei * IEntrySize]);
|
||||
if(iaddrcmp(&ie.ia, &eie.ia) != 0){
|
||||
if(!hdr){
|
||||
phdr(eb);
|
||||
hdr = 1;
|
||||
}
|
||||
wrong++;
|
||||
pie(&eie, '<');
|
||||
pie(&ie, '>');
|
||||
}
|
||||
ei++;
|
||||
goto cont;
|
||||
}
|
||||
if(c < 0)
|
||||
break;
|
||||
if(!hdr){
|
||||
phdr(eb);
|
||||
hdr = 1;
|
||||
}
|
||||
unpackientry(&eie, &eib.data[ei*IEntrySize]);
|
||||
extra++;
|
||||
pie(&eie, '<');
|
||||
ei++;
|
||||
ok = -1;
|
||||
}
|
||||
if(!hdr){
|
||||
phdr(eb);
|
||||
hdr = 1;
|
||||
}
|
||||
unpackientry(&ie, &ib->data[i*IEntrySize]);
|
||||
missing++;
|
||||
pie(&ie, '>');
|
||||
ok = -1;
|
||||
cont:;
|
||||
}
|
||||
for(; ei < eib.n; ei++){
|
||||
if(!hdr){
|
||||
phdr(eb);
|
||||
hdr = 1;
|
||||
}
|
||||
unpackientry(&eie, &eib.data[ei*IEntrySize]);
|
||||
pie(&eie, '<');
|
||||
ok = -1;
|
||||
}
|
||||
putdblock(eb);
|
||||
return ok;
|
||||
}
|
||||
|
||||
int
|
||||
checkindex(Index *ix, Part *part, u64int off, u64int clumps, int zero)
|
||||
{
|
||||
IEStream *ies;
|
||||
IBucket ib, zib;
|
||||
ZBlock *z, *b;
|
||||
u32int next, buck;
|
||||
int ok, bok;
|
||||
u64int found = 0;
|
||||
|
||||
//ZZZ make buffer size configurable
|
||||
b = alloczblock(ix->blocksize, 0, ix->blocksize);
|
||||
z = alloczblock(ix->blocksize, 1, ix->blocksize);
|
||||
ies = initiestream(part, off, clumps, 64*1024);
|
||||
if(b == nil || z == nil || ies == nil){
|
||||
werrstr("allocating: %r");
|
||||
ok = -1;
|
||||
goto breakout;
|
||||
return -1;
|
||||
}
|
||||
ok = 0;
|
||||
next = 0;
|
||||
memset(&ib, 0, sizeof ib);
|
||||
ib.data = b->data;
|
||||
zib.data = z->data;
|
||||
zib.n = 0;
|
||||
zib.buck = 0;
|
||||
for(;;){
|
||||
buck = buildbucket(ix, ies, &ib, ix->blocksize-IBucketSize);
|
||||
found += ib.n;
|
||||
if(zero){
|
||||
for(; next != buck; next++){
|
||||
if(next == ix->buckets){
|
||||
if(buck != TWID32){
|
||||
ok = -1;
|
||||
werrstr("internal error: bucket out of range");
|
||||
}
|
||||
if(ok < 0)
|
||||
werrstr("%d spurious entries, %d missing, %d wrong", extra, missing, wrong);
|
||||
goto breakout;
|
||||
}
|
||||
bok = checkbucket(ix, next, &zib);
|
||||
if(bok < 0)
|
||||
ok = -1;
|
||||
}
|
||||
}
|
||||
if(buck >= ix->buckets){
|
||||
if(buck == TWID32)
|
||||
break;
|
||||
werrstr("internal error: bucket out of range");
|
||||
ok = -1;
|
||||
goto breakout;
|
||||
}
|
||||
bok = checkbucket(ix, buck, &ib);
|
||||
if(bok < 0)
|
||||
ok = -1;
|
||||
next = buck + 1;
|
||||
}
|
||||
breakout:
|
||||
freeiestream(ies);
|
||||
freezblock(z);
|
||||
freezblock(b);
|
||||
return ok;
|
||||
}
|
||||
|
||||
int
|
||||
checkbloom(Bloom *b1, Bloom *b2, int fix)
|
||||
{
|
||||
u32int *a1, *a2;
|
||||
int i, n, extra, missing;
|
||||
|
||||
if(b1==nil && b2==nil)
|
||||
return 0;
|
||||
if(b1==nil || b2==nil){
|
||||
werrstr("nil/non-nil");
|
||||
return -1;
|
||||
}
|
||||
wbbloomhead(b1);
|
||||
wbbloomhead(b2);
|
||||
if(memcmp(b1->data, b2->data, BloomHeadSize) != 0){
|
||||
werrstr("bloom header mismatch");
|
||||
return -1;
|
||||
}
|
||||
a1 = (u32int*)b1->data;
|
||||
a2 = (u32int*)b2->data;
|
||||
n = b1->size/4;
|
||||
extra = 0;
|
||||
missing = 0;
|
||||
for(i=BloomHeadSize/4; i<n; i++){
|
||||
if(a1[i] != a2[i]){
|
||||
print("%.8ux/%.8ux.", a1[i], a2[i]);
|
||||
extra += countbits(a1[i] & ~a2[i]);
|
||||
missing += countbits(a2[i] & ~a1[i]);
|
||||
}
|
||||
}
|
||||
if(extra || missing)
|
||||
fprint(2, "bloom filter: %d spurious bits, %d missing bits\n", extra, missing);
|
||||
else
|
||||
fprint(2, "bloom filter: correct\n");
|
||||
if(!fix && missing){
|
||||
werrstr("missing bits");
|
||||
return -1;
|
||||
}
|
||||
if(fix && (missing || extra)){
|
||||
memmove(b1->data, b2->data, b1->size);
|
||||
return writebloom(b1);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
usage(void)
|
||||
{
|
||||
fprint(2, "usage: checkindex [-f] [-B blockcachesize] config tmp\n");
|
||||
threadexitsall(0);
|
||||
}
|
||||
|
||||
Config conf;
|
||||
|
||||
void
|
||||
threadmain(int argc, char *argv[])
|
||||
{
|
||||
Bloom *oldbloom, *newbloom;
|
||||
Part *part;
|
||||
u64int clumps, base;
|
||||
u32int bcmem;
|
||||
int fix, skipz, ok;
|
||||
|
||||
fix = 0;
|
||||
bcmem = 0;
|
||||
skipz = 0;
|
||||
ARGBEGIN{
|
||||
case 'B':
|
||||
bcmem = unittoull(ARGF());
|
||||
break;
|
||||
case 'f':
|
||||
fix++;
|
||||
break;
|
||||
case 'Z':
|
||||
skipz = 1;
|
||||
break;
|
||||
default:
|
||||
usage();
|
||||
break;
|
||||
}ARGEND
|
||||
|
||||
if(argc != 2)
|
||||
usage();
|
||||
|
||||
ventifmtinstall();
|
||||
|
||||
part = initpart(argv[1], ORDWR|ODIRECT);
|
||||
if(part == nil)
|
||||
sysfatal("can't initialize temporary partition: %r");
|
||||
|
||||
if(!fix)
|
||||
readonly = 1;
|
||||
|
||||
if(initventi(argv[0], &conf) < 0)
|
||||
sysfatal("can't init venti: %r");
|
||||
oldbloom = mainindex->bloom;
|
||||
newbloom = nil;
|
||||
if(oldbloom){
|
||||
newbloom = vtmallocz(sizeof *newbloom);
|
||||
bloominit(newbloom, oldbloom->size, nil);
|
||||
newbloom->data = vtmallocz(oldbloom->size);
|
||||
}
|
||||
if(bcmem < maxblocksize * (mainindex->narenas + mainindex->nsects * 4 + 16))
|
||||
bcmem = maxblocksize * (mainindex->narenas + mainindex->nsects * 4 + 16);
|
||||
if(0) fprint(2, "initialize %d bytes of disk block cache\n", bcmem);
|
||||
initdcache(bcmem);
|
||||
|
||||
fprint(2, "checkindex: building entry list\n");
|
||||
clumps = sortrawientries(mainindex, part, &base, newbloom);
|
||||
if(clumps == TWID64)
|
||||
sysfatal("can't build sorted index: %r");
|
||||
fprint(2, "checkindex: checking %lld entries at %lld\n", clumps, base);
|
||||
ok = 0;
|
||||
if(checkindex(mainindex, part, base, clumps, !skipz) < 0){
|
||||
fprint(2, "checkindex: %r\n");
|
||||
ok = -1;
|
||||
}
|
||||
if(checkbloom(oldbloom, newbloom, fix) < 0){
|
||||
fprint(2, "checkbloom: %r\n");
|
||||
ok = -1;
|
||||
}
|
||||
if(ok < 0)
|
||||
sysfatal("errors found");
|
||||
fprint(2, "checkindex: index is correct\n");
|
||||
threadexitsall(0);
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue