code: 9ferno

ref: 7c22230886134121893f02ee283da85f3a6807e6
dir: /liblogfs/replace.c/

View raw version
#include "logfsos.h"
#include "logfs.h"
#include "local.h"
#include "fcall.h"

static char *
copypages(LogfsServer *server, long newb, long oldb, Pageset copymask, LogfsLowLevelReadResult *llrrp, int *markedbadp)
{
	char *errmsg;
	int page;
	LogfsLowLevel *ll;
	int ppb;
	int pagesize;
	uchar *buf;

	if(copymask == 0)
		return nil;

	ll = server->ll;
	ppb = 1 << ll->l2pagesperblock;
	pagesize = 1 << ll->l2pagesize;
	*markedbadp = 0;
	*llrrp = LogfsLowLevelReadResultOk;
	errmsg = nil;

	buf = logfsrealloc(nil, 1 << ll->l2pagesize);
	if(buf == nil)
		return Enomem;

	for(page = ppb - 1; page >= 0; page--) {
		Pageset m;

		m = logfsdatapagemask(1, page);

		if(copymask & m) {
			LogfsLowLevelReadResult llrr;
			if(server->trace > 1)
				print("copypages read page %d\n", page);
			errmsg = (*ll->readpagerange)(ll, buf, oldb, page, 0, pagesize, &llrr);
			if(errmsg != nil)
				break;
			if(llrr > *llrrp)
				*llrrp = llrr;
			if(server->trace > 1)
				print("copypages write page %d\n", page);
			errmsg = (*ll->writepage)(ll, buf, newb, page);
			if(errmsg) {
				if(strcmp(errmsg, Eio) == 0) {
					(*ll->markblockbad)(ll, newb);
					*markedbadp = 1;
				}
				break;
			}
			if(server->trace > 1)
				print("copypages end page %d\n", page);
		}
	}
	logfsfreemem(buf);
	return errmsg;
}

char *
logfsservercopyactivedata(LogfsServer *server, long newb, long oldblockindex, int forcepage0, LogfsLowLevelReadResult *llrrp, int *markedbadp)
{
	LogfsLowLevel *ll = server->ll;
	ulong newpath;
	DataBlock *ob;
	char *errmsg;
	Pageset copymask;

	ob = server->datablock + oldblockindex;
	copymask = ~ob->free;
	if(forcepage0)
		copymask |= logfsdatapagemask(1, 0);
	if(server->trace > 1)
		print("copyactivedata %ld: (%ld -> %ld)\n", oldblockindex, ob->block, newb);
	newpath = mkdatapath(dataseqof(ob->path), copygensucc(copygenof(ob->path)));
	(*ll->setblocktag)(ll, newb, LogfsTdata);
	(*ll->setblockpath)(ll, newb, newpath);
	errmsg = copypages(server, newb, ob->block, copymask, llrrp, markedbadp);
	if(errmsg)
		return errmsg;
	/*
	 * anything not copied is now not dirty
	 */
	ob->dirty &= copymask;
	ob->block = newb;
	ob->path = newpath;
	return nil;
}

/*
 * unconditionally replace a datablock, and mark the old one bad
 * NB: if page 0 is apparently unused, force it to be copied, and mark
 * it free and dirty afterwards
 */
char *
logfsserverreplacedatablock(LogfsServer *server, long index)
{
	long newb;
	LogfsLowLevel *ll = server->ll;

	newb = logfsfindfreeblock(ll, AllocReasonReplace);
	/* TODO - recover space by scavenging other blocks, or recycling the log */
	while(newb >= 0) {
		char *errmsg;
		LogfsLowLevelReadResult llrr;
		long oldblock;
		int markedbad;
		DataBlock *db;

		db = server->datablock + index;
		oldblock = db->block;
		errmsg = logfsservercopyactivedata(server, newb, index, 1, &llrr, &markedbad);
		if(errmsg) {
			if(!markedbad)
				return errmsg;
			newb = logfsfindfreeblock(ll, AllocReasonReplace);
			continue;
		}
		(*ll->markblockbad)(ll, oldblock);
		return nil;
	}
	return logfsefullreplacing;
}

char *
logfsserverreplacelogblock(LogfsServer *server, LogSegment *seg, long index)
{
	ulong opath;
	LogfsLowLevel *ll = server->ll;
	long oldb = seg->blockmap[index];

	opath = (*ll->getblockpath)(ll, oldb);

	for(;;) {
		long newb;
		int pages;
		char *errmsg;
		LogfsLowLevelReadResult llrr;
		int markedbad;

		newb  = logfsfindfreeblock(ll, AllocReasonReplace);
		if(newb < 0)
			return "full replacing log block";
		/* TODO - scavenge data space for a spare block */
		(*ll->setblocktag)(ll, newb, LogfsTlog);
		(*ll->setblockpath)(ll, newb, mklogpath(seg->gen, index, copygensucc(copygenof(opath))));
		if(index == seg->curblockindex)
			pages = seg->curpage;
		else
			pages = 1 << server->ll->l2pagesperblock;
		errmsg = copypages(server, newb, oldb, logfsdatapagemask(pages, 0), &llrr, &markedbad);
		if(errmsg == nil) {
			(*ll->markblockbad)(ll, seg->blockmap[index]);
			seg->blockmap[index] = newb;
			return nil;
		}
		if(!markedbad)
			return errmsg;
	}
}

char *
logfsserverreplaceblock(LogfsServer *server, LogSegment *seg, long index)
{
	if(seg)
		return logfsserverreplacelogblock(server, seg, index);
	else
		return logfsserverreplacedatablock(server, index);
}