shithub: gefs

Download patch

ref: c7bc6cca77498f87fa9230d9628b82c46c12dba5
parent: e808b613b7b70a5b23b25bef22ad0e7d5ab846dc
author: Ori Bernstein <[email protected]>
date: Fri May 26 11:15:31 EDT 2023

fs: pull out superblock from arena, duplicate arenas to allow for better crash resilience

--- a/blk.c
+++ b/blk.c
@@ -102,13 +102,16 @@
 
 	switch(b->type){
 	default:
-		fprint(2, "invalid block @%llx\n", bp);
+		fprint(2, "invalid block type %d @%llx\n", b->type, bp);
 		abort();
 		break;
 	case Tdat:
-	case Tarena:
+	case Tsuper:
 		b->data = b->buf;
 		break;
+	case Tarena:
+		b->data = b->buf + 2;
+		break;
 	case Tdlist:
 		b->deadsz = UNPACK16(b->buf+2);
 		b->deadp = unpackbp(b->buf+4, Blksz-4);
@@ -285,6 +288,8 @@
 	assert(off % Blksz == 0);
 	assert(op == LogAlloc || op == LogFree);
 	assert(lb == nil || lb->type == Tlog);
+	assert(off >= a->hd->bp.addr + Blksz);
+	assert(off < a->tl->bp.addr);
 	dprint("logop %llx+%llx@%x: %s\n", off, len, lb?lb->logsz:-1, (op == LogAlloc) ? "Alloc" : "Free");
 	/*
 	 * move to the next block when we have
@@ -354,7 +359,7 @@
 	bh = UNPACK64(b->data);
 	/* the hash covers the log and offset */
 	if(bh != bufhash(b->data+Loghashsz, Logspc-Loghashsz)){
-		werrstr("corrupt log");
+		werrstr("corrupt log block %B [%llx]", bp, blkhash(b));
 		return -1;
 	}
 	for(i = Loghashsz; i < Logspc; i += n){
@@ -785,6 +790,8 @@
 
 	switch(b->type){
 	default:
+		abort();
+		break;
 	case Tpivot:
 		PACK16(b->buf+2, b->nval);
 		PACK16(b->buf+4, b->valsz);
@@ -806,6 +813,7 @@
 	case Tdat:
 	case Tmagic:
 	case Tarena:
+	case Tsuper:
 		break;
 	}
 
@@ -829,6 +837,7 @@
 	}
 	if((b = readblk(bp.addr, flg)) == nil){
 		qunlock(&fs->blklk[i]);
+		abort();
 		return nil;
 	}
 	b->alloced = getcallerpc(&bp);
@@ -836,6 +845,7 @@
 	if((flg&GBnochk) == 0 && h != bp.hash){
 		fprint(2, "corrupt block %p %B: %.16llux != %.16llux\n", b, bp, h, bp.hash);
 		qunlock(&fs->blklk[i]);
+		abort();
 		return nil;
 	}
 	b->bp.hash = h;
@@ -851,8 +861,6 @@
 holdblk(Blk *b)
 {
 	ainc(&b->ref);
-	b->lasthold1 = b->lasthold0;
-	b->lasthold0 = b->lasthold;
 	b->lasthold = getcallerpc(&b);
 	return b;
 }
@@ -863,8 +871,6 @@
 	assert(b == nil || b->ref > 0);
 	if(b == nil || adec(&b->ref) != 0)
 		return;
-	b->lastdrop1 = b->lastdrop0;
-	b->lastdrop0 = b->lastdrop;
 	b->lastdrop = getcallerpc(&b);
 	/*
 	 * While a freed block can get resurrected
@@ -990,6 +996,7 @@
 	Arena *a;
 
 	b->qgen = agetv(&fs->qgen);
+	b->enqueued = getcallerpc(&b);
 	a = getarena(b->bp.addr);
 	assert(checkflag(b, Bdirty));
 	assert(b->bp.addr >= 0);
@@ -1115,6 +1122,7 @@
 	}
 	while(fs->syncing != 0)
 		rsleep(&fs->syncrz);
+	/* pass 0: sync arena contents */
 	for(i = 0; i < fs->narena; i++){
 		a = &fs->arenas[i];
 		lock(a);
@@ -1125,14 +1133,49 @@
 			a->deferhd = (Bptr){-1, -1, -1};
 			a->defertl = nil;
 		}
-		packarena(a->b->data, Blksz, a, fs);
-		finalize(a->b);
 		finalize(a->logtl);
 		if(syncblk(a->logtl) == -1)
 			sysfatal("sync arena: %r");
-		if(syncblk(a->b) == -1)
+		unlock(a);
+	}
+	/*
+	 * pass 1: sync block headers; if we crash here,
+	 *  the block footers are consistent, and we can
+	 *  use them.
+         */
+	for(i = 0; i < fs->narena; i++){
+		a = &fs->arenas[i];
+		lock(a);
+		packarena(a->hd->data, Blksz, a);
+		finalize(a->hd);
+		if(syncblk(a->hd) == -1)
 			sysfatal("sync arena: %r");
 		unlock(a);
 	}
+	/*
+	 * pass 2: sync block footers; if we crash here,
+	 *  the block headers are consistent, and we can
+	 *  use them.
+         */
+	for(i = 0; i < fs->narena; i++){
+		a = &fs->arenas[i];
+		lock(a);
+		packarena(a->tl->data, Blksz, a);
+		finalize(a->tl);
+		if(syncblk(a->tl) == -1)
+			sysfatal("sync arena: %r");
+		unlock(a);
+	}
+	for(i = 0; i < fs->narena; i++)
+		fs->arenabp[i] = fs->arenas[i].hd->bp;
+
+	packsb(fs->sb0->buf, Blksz, fs);
+	packsb(fs->sb1->buf, Blksz, fs);
+	finalize(fs->sb0);
+	finalize(fs->sb1);
+	if(syncblk(fs->sb0) == -1)
+		sysfatal("sync sb: %r");
+	if(syncblk(fs->sb1) == -1)
+		sysfatal("sync sb: %r");
 	qunlock(&fs->synclk);
 }
--- a/dat.h
+++ b/dat.h
@@ -67,19 +67,20 @@
 	Kvmax	= Keymax + Inlmax,	/* Key and value */
 	Kpmax	= Keymax + Ptrsz,	/* Key and pointer */
 	Wstatmax = 4+8+8+8,		/* mode, size, atime, mtime */
+	Arenasz	= 8+8+8+8,		/* loghd, loghash, size, used */
 	
 	Pivhdsz		= 10,
 	Leafhdsz	= 6,
 	Loghdsz		= 2,
 	Loghashsz	= 8,
-	Dlhdsz		= 2+2+Ptrsz,		/* type, size, chain */
+	Dlhdsz		= 2+2+Ptrsz,			/* type, size, chain */
 	Dlspc		= Blksz - Dlhdsz,
-	Rootsz		= 4+Ptrsz,		/* root pointer */
+	Rootsz		= 4+Ptrsz,			/* root pointer */
 	Pivsz		= Blksz - Pivhdsz,
-	Bufspc		= (Blksz - Pivhdsz)/2,	/* pivot room */
+	Bufspc		= (Blksz - Pivhdsz)/2,		/* pivot room */
 	Pivspc		= Blksz - Pivhdsz - Bufspc,
 	Logspc		= Blksz - Loghdsz,
-	Logslop		= 16+16+8,	/* val, nextb, chain */
+	Logslop		= 16+16+8,			/* val, nextb, chain */
 	Leafspc 	= Blksz - Leafhdsz,
 	Msgmax  	= 1 + (Kvmax > Kpmax ? Kvmax : Kpmax)
 };
@@ -244,7 +245,8 @@
 	Tlog,
 	Tdlist,
 	Tmagic,
-	Tarena = 0x6765,	/* 'ge' bigendian */
+	Tarena,
+	Tsuper = 0x6765,	/* 'ge' bigendian */
 };
 
 enum {
@@ -438,6 +440,7 @@
 	vlong	arenasz;
 	vlong	nextqid;
 	vlong	nextgen;
+	Bptr	*arenabp;
 };
 
 /*
@@ -446,6 +449,9 @@
  */
 struct Gefs {
 	Fshdr;
+	/* superblocks */
+	Blk	*sb0;	/* primary */
+	Blk	*sb1;	/* backup */
 	/* arena allocation */
 	Arena	*arenas;
 	long	roundrobin;
@@ -518,7 +524,8 @@
 	Avltree *free;
 	Blk	**queue;
 	int	nqueue;
-	Blk	*b;			/* arena block */
+	Blk	*hd;			/* arena header */
+	Blk	*tl;			/* arena footer */
 	Blk	**q;			/* write queue */
 	vlong	nq;
 	vlong	size;
@@ -666,13 +673,8 @@
 
 	/* debug */
 	uintptr lasthold;
-	uintptr lasthold0;
-	uintptr lasthold1;
-
 	uintptr lastdrop;
-	uintptr lastdrop0;
-	uintptr lastdrop1;
-
+	uintptr	enqueued;
 	uintptr cached;
 	uintptr uncached;
 	uintptr	alloced;
--- a/fns.h
+++ b/fns.h
@@ -72,7 +72,7 @@
 void	closesnap(Tree*);
 void	reamfs(char*);
 void	growfs(char*);
-int	loadarena(Arena*, Fshdr *fi, vlong);
+int	loadarena(Arena*, Bptr, vlong);
 void	loadfs(char*);
 void	sync(void);
 int	loadlog(Arena*, Bptr);
@@ -134,36 +134,41 @@
 char*	pack64(int*, char*, char*, uvlong);
 char*	packstr(int*, char*, char*, char*);
 
-/* void* is a bit hacky, but we want both signed and unsigned to work */
-char*	unpack8(int*, char*, char*, void*);
-char*	unpack16(int*, char*, char*, void*);
-char*	unpack32(int*, char*, char*, void*);
-char*	unpack64(int*, char*, char*, void*);
-char*	unpackstr(int*, char*, char*, char**);
 int	dir2kv(vlong, Xdir*, Kvp*, char*, int);
-int	kv2statbuf(Kvp*, char*, int);
 int	dir2statbuf(Xdir*, char*, int);
-int	kv2dir(Kvp*, Xdir*);
-void	kv2qid(Kvp*, Qid*);
-void	kv2dlist(Kvp*, Dlist*);
 void	dlist2kv(Dlist*, Kvp*, char*, int);
-void	tree2kv(Tree*, Kvp*, char*, int);
 void	lbl2kv(char*, vlong, Kvp*, char*, int);
 void	link2kv(vlong, vlong, Kvp*, char*, int);
+void	tree2kv(Tree*, Kvp*, char*, int);
+
+int	kv2dir(Kvp*, Xdir*);
+void	kv2dlist(Kvp*, Dlist*);
 void	kv2link(Kvp*, vlong*, vlong*);
+void	kv2qid(Kvp*, Qid*);
+int	kv2statbuf(Kvp*, char*, int);
 
+char*	packarena(char*, int, Arena*);
 char*	packbp(char*, int, Bptr*);
-Bptr	unpackbp(char*, int);
-char*	packtree(char*, int, Tree*);
-Tree*	unpacktree(Tree*, char*, int);
 char*	packdkey(char*, int, vlong, char*);
-char*	unpackdkey(char*, int, vlong*);
 char*	packdval(char*, int, Xdir*);
-char*	packsnap(char*, int, vlong);
 char*	packlabel(char*, int, char*);
+char*	packsnap(char*, int, vlong);
 char*	packsuper(char*, int, vlong);
-char*	packarena(char*, int, Arena*, Fshdr*);
-char*	unpackarena(Arena*, Fshdr*, char*, int);
+char*	packtree(char*, int, Tree*);
+char*	packsb(char*, int, Fshdr*);
+
+char*	unpackarena(Arena*, char*, int);
+Bptr	unpackbp(char*, int);
+char*	unpackdkey(char*, int, vlong*);
+Tree*	unpacktree(Tree*, char*, int);
+char*	unpacksb(Fshdr*, char*, int);
+
+/* void* is a bit hacky, but we want both signed and unsigned to work */
+char*	unpack8(int*, char*, char*, void*);
+char*	unpack16(int*, char*, char*, void*);
+char*	unpack32(int*, char*, char*, void*);
+char*	unpack64(int*, char*, char*, void*);
+char*	unpackstr(int*, char*, char*, char**);
 
 /* fmt */
 int	Bconv(Fmt*);
--- a/load.c
+++ b/load.c
@@ -16,34 +16,37 @@
 	return 0;
 }
 
-static void
-mergeinfo(Gefs *fs, Fshdr *fi)
-{
-	if(fi->blksz != Blksz || fi->bufspc != Bufspc)
-		sysfatal("parameter mismatch");
-	if(fs->gotinfo && fs->narena != fi->narena)
-		sysfatal("arena count mismatch");
-	if(fs->gotinfo && fi->nextgen != fs->nextgen)
-		fprint(2, "not all arenas synced: rolling back\n");
-	fs->Fshdr = *fi;
-}
-
 int
-loadarena(Arena *a, Fshdr *fi, vlong o)
+loadarena(Arena *a, Bptr hdbp, vlong asz)
 {
-	Blk *b;
+	Blk *hd, *tl, *b;
 	Bptr bp;
 
-	bp.addr = o;
-	bp.hash = -1;
-	bp.gen = -1;
-	if((b = getblk(bp, GBnochk)) == nil)
+	/* try to load block pointers with consistency check */
+	bp = hdbp;
+	hd = getblk(bp, 0);
+	bp.addr += asz;
+	tl = getblk(bp, 0);
+
+	/* if neither head nor tail is consistent, we're hosed */
+	b = (hd != nil) ? hd : tl;
+	if(b == nil)
 		return -1;
-	if(unpackarena(a, fi, b->data, Blksz) == nil)
+
+	/* otherwise, we could have crashed mid-pass, just load the blocks */
+	bp = hdbp;
+	if(hd == nil && (hd = getblk(bp, GBnochk)) == nil)
 		return -1;
+	bp.addr += asz;
+	if(tl == nil && (tl = getblk(bp, GBnochk)) == nil)
+		return -1;
+
+	if(unpackarena(a, b->data, Arenasz) == nil)
+		return -1;
 	if((a->free = avlcreate(rangecmp)) == nil)
 		return -1;
-	a->b = b;
+	a->hd = hd;
+	a->tl = tl;
 	return 0;
 }
 
@@ -51,8 +54,8 @@
 loadfs(char *dev)
 {
 	Mount *mnt;
-	Fshdr fi;
 	Arena *a;
+	Bptr bp;
 	char *e;
 	Tree *t;
 	int i, k;
@@ -71,16 +74,25 @@
 	fs->narena = 1;
 	if((fs->fd = open(dev, ORDWR)) == -1)
 		sysfatal("open %s: %r", dev);
-	if((fs->arenas = calloc(1, sizeof(Arena))) == nil)
+	bp = (Bptr){0, -1, -1};
+	if((fs->sb0 = getblk(bp, GBnochk)) == nil)
+		sysfatal("superblock: %r\n");
+	bp = (Bptr){512*MiB, -1, -1};
+	if((fs->sb1 = getblk(bp, GBnochk)) == nil)
+		sysfatal("superblock: %r\n");
+	if(unpacksb(fs, fs->sb0->buf, Blksz) == nil)
+		sysfatal("superblock: %r");
+	if(unpacksb(fs, fs->sb0->buf, Blksz) == nil)
+		sysfatal("superblock: %r");
+	if((fs->arenas = calloc(fs->narena, sizeof(Arena))) == nil)
 		sysfatal("malloc: %r");
 	for(i = 0; i < fs->narena; i++){
 		a = &fs->arenas[i];
-		if((loadarena(a, &fi, i*fs->arenasz)) == -1)
+		if((loadarena(a, fs->arenabp[i], fs->arenasz)) == -1)
 			sysfatal("loadfs: %r");
 		a->reserve = a->size / 1024;
 		if(a->reserve < 32*MiB)
 			a->reserve = 32*MiB;
-		mergeinfo(fs, &fi);
 		if(!fs->gotinfo){
 			if((fs->arenas = realloc(fs->arenas, fs->narena*sizeof(Arena))) == nil)
 				sysfatal("malloc: %r");
@@ -92,9 +104,7 @@
 	for(i = 0; i < fs->narena; i++){
 		a = &fs->arenas[i];
 		if(loadlog(a, a->loghd) == -1)
-			sysfatal("load log: %r");
-		if(compresslog(a) == -1)
-			sysfatal("compress log: %r");
+			sysfatal("load log %B: %r", a->loghd);
 	}
 
 	fprint(2, "load %s:\n", dev);
--- a/main.c
+++ b/main.c
@@ -221,7 +221,7 @@
 	loadfs(dev);
 	fs->wrchan = mkchan(32);
 	fs->nsyncers = nproc/2;
-	fs->nreaders = 2;
+	fs->nreaders = 1;
 	if(fs->nsyncers > fs->narena)
 		fs->nsyncers = fs->narena;
 	for(i = 0; i < fs->nsyncers; i++)
--- a/pack.c
+++ b/pack.c
@@ -521,10 +521,51 @@
 }
 
 char*
-packarena(char *p, int sz, Arena *a, Fshdr *fi)
+packarena(char *p, int sz, Arena *a)
 {
+	char *e;
+
+	assert(sz >= Arenasz);
+	e = p + Arenasz;
+	PACK64(p, a->loghd.addr);	p += 8;	/* freelist addr */
+	PACK64(p, a->loghd.hash);	p += 8;	/* freelist hash */
+	PACK64(p, a->size);		p += 8;	/* arena size */
+	PACK64(p, a->used);		p += 8;	/* arena used */
+	assert(p <= e);
+	return p;
+}
+
+char*
+unpackarena(Arena *a, char *p, int sz)
+{
+	char *e;
+
+	assert(sz >= Arenasz);
+	memset(a, 0, sizeof(*a));
+
+	e = p + Arenasz;
+	a->loghd.addr = UNPACK64(p);	p += 8;
+	a->loghd.hash = UNPACK64(p);	p += 8;
+	a->loghd.gen = -1;		p += 0;
+	a->size = UNPACK64(p);		p += 8;
+	a->used = UNPACK64(p);		p += 8;
+	a->logtl = nil;
+
+	a->deferhd.addr = -1;
+	a->deferhd.hash = -1;
+	a->deferhd.gen = -1;
+	a->defertl = nil;
+	assert(p <= e);
+	return p;
+}
+
+char*
+packsb(char *p, int sz, Fshdr *fi)
+{
+	int i;
+
 	assert(sz == Blksz);
-	memcpy(p, "gefs0004", 8);	p += 8;
+	memcpy(p, "gefs0005", 8);	p += 8;
 	PACK32(p, Blksz);		p += 4;
 	PACK32(p, Bufspc);		p += 4;
 	PACK32(p, fi->snap.ht);		p += 4;
@@ -534,44 +575,39 @@
 	PACK64(p, fi->arenasz);		p += 8;
 	PACK64(p, fi->nextqid);		p += 8;
 	PACK64(p, fi->nextgen);		p += 8;
-	PACK64(p, a->loghd.addr);	p += 8;	/* freelist addr */
-	PACK64(p, a->loghd.hash);	p += 8;	/* freelist hash */
-	PACK64(p, a->size);		p += 8;	/* arena size */
-	PACK64(p, a->used);		p += 8;	/* arena used */
+	for(i = 0; i < fi->narena; i++){
+		PACK64(p, fi->arenabp[i].addr);	p += 8;
+		PACK64(p, fi->arenabp[i].hash);	p += 8;
+	}
 	return p;
 }
 
 char*
-unpackarena(Arena *a, Fshdr *fi, char *p, int sz)
+unpacksb(Fshdr *fi, char *p, int sz)
 {
+	int i;
+
 	assert(sz == Blksz);
-	memset(a, 0, sizeof(*a));
-	memset(fi, 0, sizeof(*fi));
-	if(memcmp(p, "gefs0004", 8) != 0){
+	if(memcmp(p, "gefs0005", 8) != 0){
 		werrstr("wrong block header %.8s\n", p);
 		return nil;
 	}
 	p += 8;
-	fi->blksz = UNPACK32(p);		p += 4;
-	fi->bufspc = UNPACK32(p);		p += 4;
+	fi->blksz = UNPACK32(p);	p += 4;
+	fi->bufspc = UNPACK32(p);	p += 4;
 	fi->snap.ht = UNPACK32(p);	p += 4;
 	fi->snap.bp.addr = UNPACK64(p);	p += 8;
 	fi->snap.bp.hash = UNPACK64(p);	p += 8;
 	fi->snap.bp.gen = -1;		p += 0;
-	fi->narena = UNPACK32(p);		p += 4;
+	fi->narena = UNPACK32(p);	p += 4;
 	fi->arenasz = UNPACK64(p);	p += 8;
 	fi->nextqid = UNPACK64(p);	p += 8;
 	fi->nextgen = UNPACK64(p);	p += 8;
-	a->loghd.addr = UNPACK64(p);	p += 8;
-	a->loghd.hash = UNPACK64(p);	p += 8;
-	a->loghd.gen = -1;		p += 0;
-	a->size = UNPACK64(p);		p += 8;
-	a->used = UNPACK64(p);		p += 8;
-	a->logtl = nil;
-
-	a->deferhd.addr = -1;
-	a->deferhd.hash = -1;
-	a->deferhd.gen = -1;
-	a->defertl = nil;
+	fi->arenabp = malloc(fi->narena * sizeof(Bptr));
+	for(i = 0; i < fi->narena; i++){
+		fi->arenabp[i].addr = UNPACK64(p);	p += 8;
+		fi->arenabp[i].hash = UNPACK64(p);	p += 8;
+		fi->arenabp[i].gen = -1;
+	}
 	return p;
 }
--- a/ream.c
+++ b/ream.c
@@ -158,14 +158,16 @@
 }
 
 static void
-initarena(Arena *a, Fshdr *fi, vlong start, vlong asz)
+initarena(Arena *a, vlong start, vlong asz)
 {
 	vlong addr, bo, bh;
 	char *p;
-	Blk *b;
+	Blk *b, *hd, *tl;
 
 	b = cachepluck();
-	addr = start+Blksz;	/* arena loghder */
+	if(start == 512*MiB)
+		start += Blksz;
+	addr = start+Blksz;	/* leave room for arena hdr */
 
 	a->loghd.addr = -1;
 	a->loghd.hash = -1;
@@ -174,7 +176,6 @@
 	memset(b->buf, 0, sizeof(b->buf));
 	b->type = Tlog;
 	b->bp.addr = addr;
-	b->logsz = 32;
 	b->data = b->buf + Loghdsz;
 	setflag(b, Bdirty);
 
@@ -183,7 +184,13 @@
 	PACK64(p, asz-Blksz);		p += 8;	/* len */
 	PACK64(p, b->bp.addr|LogAlloc);	p += 8;	/* addr */
 	PACK64(p, Blksz);		p += 8;	/* len */
+	/* backup sb */
+	if(start <= 512*MiB && start+asz > 512*MiB){
+		PACK64(p, (512*MiB)|LogAlloc1);
+		p += 8;
+	}
 	PACK64(p, (uvlong)LogEnd);	/* done */
+	b->logsz = p - b->data;
 	finalize(b);
 	if(syncblk(b) == -1)
 		sysfatal("ream: init log");
@@ -192,11 +199,6 @@
 	bh = b->bp.hash;
 	bo = b->bp.addr;
 
-	b = cachepluck();
-	memset(b->buf, 0, sizeof(b->buf));
-	b->type = Tarena;
-	b->bp.addr = start;
-	b->data = b->buf;
 	a->loghd.addr = bo;
 	a->loghd.hash = bh;
 	a->loghd.gen = -1;
@@ -203,17 +205,38 @@
 	a->size = asz;
 	a->used = Blksz;
 	a->logtl = nil;
-	packarena(b->data, Blksz, a, fi);
-	finalize(b);
-	if(syncblk(b) == -1)
+
+	hd = cachepluck();
+	tl = cachepluck();
+
+	memset(hd->buf, 0, sizeof(hd->buf));
+	hd->type = Tarena;
+	hd->bp.addr = start;
+	hd->data = hd->buf+2;
+	finalize(hd);
+
+	memset(tl->buf, 0, sizeof(tl->buf));
+	tl->type = Tarena;
+	tl->bp.addr = start+asz;
+	tl->data = tl->buf+2;
+	finalize(tl);
+
+	packarena(hd->data, Arenasz, a);
+	packarena(tl->data, Arenasz, a);
+	finalize(hd);
+	finalize(tl);
+	if(syncblk(hd) == -1)
 		sysfatal("ream: write arena: %r");
-	dropblk(b);
+	if(syncblk(tl) == -1)
+		sysfatal("ream: write arena: %r");
+	a->hd = hd;
+	a->tl = tl;
 }
 
 void
 reamfs(char *dev)
 {
-	Blk *sb, *mb, *ab, *ub;
+	Blk *sb0, *sb1, *tb, *mb, *ab, *ub;
 	vlong sz, asz, off;
 	Mount *mnt, *adm;
 	Arena *a;
@@ -226,18 +249,20 @@
 		sysfatal("ream: %r");
 	sz = d->length;
 	free(d);
-	if(sz < 512*MiB)
+
+	if(sz < 512*MiB+Blksz)
 		sysfatal("ream: disk too small");
 	if((mnt = mallocz(sizeof(Mount), 1)) == nil)
 		sysfatal("ream: alloc mount: %r");
 	if((mnt->root = mallocz(sizeof(Tree), 1)) == nil)
 		sysfatal("ream: alloc tree: %r");
-
 	if((adm = mallocz(sizeof(Mount), 1)) == nil)
 		sysfatal("ream: alloc mount: %r");
 	if((adm->root = mallocz(sizeof(Tree), 1)) == nil)
 		sysfatal("ream: alloc tree: %r");
 
+	sz = sz - sz%Blksz - 2*Blksz;
+
 	fs->narena = (sz + 64ULL*GiB - 1) / (64ULL*GiB);
 	if(fs->narena < 8)
 		fs->narena = 8;
@@ -246,19 +271,29 @@
 	if((fs->arenas = calloc(fs->narena, sizeof(Arena))) == nil)
 		sysfatal("malloc: %r");
 
+
+	off = Blksz;
 	asz = sz/fs->narena;
 	asz = asz - (asz % Blksz) - Blksz;
 	fs->arenasz = asz;
-	off = 0;
+
+	sb0 = cachepluck();
+	sb1 = cachepluck();
+	sb0->bp = (Bptr){0, -1, -1};
+	sb1->bp = (Bptr){512*MiB, -1, -1};
+
+	fs->arenabp = malloc(fs->narena * sizeof(Bptr));
 	for(i = 0; i < fs->narena; i++){
+		a = &fs->arenas[i];
 		print("\tarena %d: %lld blocks at %llx\n", i, asz/Blksz, off);
-		initarena(&fs->arenas[i], fs, off, asz);
-		off += asz;
+		initarena(a, off, asz);
+		fs->arenabp[i] = a->hd->bp;
+		off += asz+Blksz;
 	}
 	
 	for(i = 0; i < fs->narena; i++){
 		a = &fs->arenas[i];
-		if((loadarena(a, fs, i*asz)) == -1)
+		if((loadarena(a, a->hd->bp, asz)) == -1)
 			sysfatal("ream: loadarena: %r");
 		if(loadlog(a, a->loghd) == -1)
 			sysfatal("load log: %r");
@@ -265,6 +300,7 @@
 		if(compresslog(a) == -1)
 			sysfatal("compress log: %r");
 	}
+
 	if((mb = newblk(mnt->root, Tleaf)) == nil)
 		sysfatal("ream: allocate root: %r");
 	holdblk(mb);
@@ -303,32 +339,65 @@
 	 * a single snap block that the tree will insert
 	 * into, and take a snapshot as the initial state.
 	 */
-	if((sb = newblk(mnt->root, Tleaf)) == nil)
+	if((tb = newblk(mnt->root, Tleaf)) == nil)
 		sysfatal("ream: allocate snaps: %r");
-	holdblk(sb);
-	initsnap(sb, mb, ab);
-	finalize(sb);
-	syncblk(sb);
+	holdblk(tb);
+	initsnap(tb, mb, ab);
+	finalize(tb);
+	syncblk(tb);
 
-	fs->snap.bp = sb->bp;
+	fs->snap.bp = tb->bp;
 	fs->snap.ht = 1;
+	fs->nextqid = Nreamqid;
 
 	dropblk(mb);
 	dropblk(ab);
 	dropblk(ub);
-	dropblk(sb);
+	dropblk(tb);
 	fs->nextqid = Nreamqid;
 
+	/*
+	 * We need to write back all of the arenas
+	 * with the updated free lists
+	 */
 	for(i = 0; i < fs->narena; i++){
 		a = &fs->arenas[i];
 		finalize(a->logtl);
 		if(syncblk(a->logtl) == -1)
 			sysfatal("sync arena: %r");
-		packarena(a->b->data, Blksz, a, fs);
-		finalize(a->b);
-		if(syncblk(a->b) == -1)
+		packarena(a->hd->data, Blksz, a);
+		finalize(a->hd);
+		if(syncblk(a->hd) == -1)
 			sysfatal("sync arena: %r");
+		packarena(a->tl->data, Blksz, a);
+		finalize(a->tl);
+		if(syncblk(a->tl) == -1)
+			sysfatal("sync arena: %r");
+		fs->arenabp[i] = a->hd->bp;
+		dropblk(a->hd);
+		dropblk(a->tl);
 	}
+
+	dropblk(mb);
+	dropblk(ab);
+	dropblk(ub);
+	dropblk(tb);
+
+	/*
+	 * Finally, write back the superblock and backup
+	 * superblock.
+	 */
+	packsb(sb0->buf, Blksz, fs);
+	packsb(sb1->buf, Blksz, fs);
+	finalize(sb0);
+	finalize(sb1);
+	if(syncblk(sb0) == -1)
+		sysfatal("sync superblock: %r");
+	if(syncblk(sb1) == -1)
+		sysfatal("sync superblock: %r");
+	dropblk(sb0);
+	dropblk(sb1);
+
 	free(mnt);
 }
 
@@ -337,8 +406,8 @@
 {
 	vlong sz, off;
 	int i, narena;
+	Bptr bp;
 	Arena *a;
-	Fshdr fi;
 	Dir *d;
 
 	if((fs->fd = open(dev, ORDWR)) == -1)
@@ -348,19 +417,17 @@
 	sz = d->length;
 	free(d);
 
-	if((fs->arenas = calloc(1, sizeof(Arena))) == nil)
+	bp = (Bptr){0, -1, -1};
+	if((fs->sb0 = getblk(bp, GBnochk)) == nil)
+		sysfatal("superblock: %r\n");
+	if(unpacksb(fs, fs->sb0->buf, Blksz) == nil)
+		sysfatal("superblock: %r");
+	if((fs->arenas = calloc(fs->narena, sizeof(Arena))) == nil)
 		sysfatal("malloc: %r");
-	fs->narena = 1;
 	for(i = 0; i < fs->narena; i++){
-	Arena *a;
 		a = &fs->arenas[i];
-		if((loadarena(a, &fi, i*fs->arenasz)) == -1)
+		if((loadarena(a, fs->arenabp[i], fs->arenasz)) == -1)
 			sysfatal("growfs: %r");
-		if(fs->narena == 1){
-			fs->Fshdr = fi;
-			if((fs->arenas = realloc(fs->arenas, fs->narena*sizeof(Arena))) == nil)
-				sysfatal("malloc: %r");
-		}
 	}
 	narena = sz/fs->arenasz;
 	off = fs->arenasz * fs->narena;
@@ -368,21 +435,35 @@
 		sysfatal("disk too small for more arenas");
 	if((fs->arenas = realloc(fs->arenas, narena*sizeof(Arena))) == nil)
 		sysfatal("malloc: %r");
+	if((fs->arenabp = realloc(fs->arenas, narena*sizeof(Bptr))) == nil)
+		sysfatal("malloc: %r");
 	for(i = fs->narena; i < narena; i++){
 		a = &fs->arenas[i];
 		print("\tadding %d: %lld blocks at %llx\n", i, fs->arenasz/Blksz, off);
-		initarena(&fs->arenas[i], fs, off, fs->arenasz);
-		if((loadarena(a, &fi, i*fs->arenasz)) == -1)
+		initarena(&fs->arenas[i], off, fs->arenasz);
+		if((loadarena(a, fs->arenabp[i], fs->arenasz)) == -1)
 			sysfatal("growfs: %r");
+		fs->arenabp[i] = a->hd->bp;
 		off += fs->arenasz;
 	}
-
 	fs->narena = narena;
 	for(i = 0; i < narena; i++){
 		a = &fs->arenas[i];
-		packarena(a->b->data, Blksz, a, fs);
-		finalize(a->b);
-		if(syncblk(a->b) == -1)
+		packarena(a->hd->data, Blksz, a);
+		packarena(a->tl->data, Blksz, a);
+		finalize(a->hd);
+		finalize(a->tl);
+		if(syncblk(a->hd) == -1)
 			sysfatal("sync arena: %r");
+		if(syncblk(a->tl) == -1)
+			sysfatal("sync arena: %r");
 	}
+	packsb(fs->sb0->buf, Blksz, fs);
+	packsb(fs->sb1->buf, Blksz, fs);
+	finalize(fs->sb0);
+	finalize(fs->sb1);
+	if(syncblk(fs->sb0) == -1)
+		sysfatal("sync superblock: %r");
+	if(syncblk(fs->sb1) == -1)
+		sysfatal("sync superblock: %r");
 }