ref: 82a3f55c5fb7b9f7a82449e4eb943c535ec3e491
dir: /liblogfs/replace.c/
#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);
}