blob: 352b3cacc792c8d088bac7a7466b8e8286158708 [file] [log] [blame]
#include <u.h>
#include <libc.h>
#include <bio.h>
#include <venti.h>
#include <libsec.h>
#include <thread.h>
enum
{
// XXX What to do here?
VtMaxLumpSize = 65535,
};
VtConn *z;
char *host;
void
usage(void)
{
fprint(2, "usage: venti/dump [-h host] score\n");
threadexitsall("usage");
}
Biobuf bout;
char spaces[256];
void
dump(int indent, uchar *score, int type)
{
int i, n;
uchar *buf;
VtEntry e;
VtRoot root;
if(spaces[0] == 0)
memset(spaces, ' ', sizeof spaces-1);
buf = vtmallocz(VtMaxLumpSize);
if(memcmp(score, vtzeroscore, VtScoreSize) == 0)
n = 0;
else
n = vtread(z, score, type, buf, VtMaxLumpSize);
if(n < 0){
Bprint(&bout, "%.*serror reading %V: %r\n", indent*4, spaces, score);
goto out;
}
switch(type){
case VtRootType:
if(vtrootunpack(&root, buf) < 0){
Bprint(&bout, "%.*serror unpacking root %V: %r\n", indent*4, spaces, score);
goto out;
}
Bprint(&bout, "%.*s%V root name=%s type=%s prev=%V bsize=%ld\n",
indent*4, spaces, score, root.name, root.type, root.prev, root.blocksize);
dump(indent+1, root.score, VtDirType);
break;
case VtDirType:
Bprint(&bout, "%.*s%V dir n=%d\n", indent*4, spaces, score, n);
for(i=0; i*VtEntrySize<n; i++){
if(vtentryunpack(&e, buf, i) < 0){
Bprint(&bout, "%.*s%d: cannot unpack\n", indent+1, spaces, i);
continue;
}
Bprint(&bout, "%.*s%d: gen=%#lux psize=%ld dsize=%ld type=%d flags=%#x size=%llud score=%V\n",
(indent+1)*4, spaces, i, e.gen, e.psize, e.dsize, e.type, e.flags, e.size, e.score);
dump(indent+2, e.score, e.type);
}
break;
case VtDataType:
Bprint(&bout, "%.*s%V data n=%d", indent*4, spaces, score, n);
for(i=0; i<n; i++){
if(i%16 == 0)
Bprint(&bout, "\n%.*s", (indent+1)*4, spaces);
Bprint(&bout, " %02x", buf[i]);
}
Bprint(&bout, "\n");
break;
default:
if(type >= VtDirType)
Bprint(&bout, "%.*s%V dir+%d\n", indent*4, spaces, score, type-VtDirType);
else
Bprint(&bout, "%.*s%V data+%d\n", indent*4, spaces, score, type-VtDirType);
for(i=0; i<n; i+=VtScoreSize)
dump(indent+1, buf+i, type-1);
break;
}
out:
free(buf);
}
void
threadmain(int argc, char *argv[])
{
int type, n;
uchar score[VtScoreSize];
uchar *buf;
char *prefix;
fmtinstall('F', vtfcallfmt);
fmtinstall('V', vtscorefmt);
ARGBEGIN{
case 'h':
host = EARGF(usage());
break;
default:
usage();
}ARGEND
if(argc != 1)
usage();
if(vtparsescore(argv[0], &prefix, score) < 0)
sysfatal("could not parse score: %r");
buf = vtmallocz(VtMaxLumpSize);
z = vtdial(host);
if(z == nil)
sysfatal("dialing venti: %r");
if(vtconnect(z) < 0)
sysfatal("vtconnect src: %r");
for(type=0; type<VtMaxType; type++){
n = vtread(z, score, type, buf, VtMaxLumpSize);
if(n >= 0)
goto havetype;
}
sysfatal("cannot find block %V", score);
havetype:
Binit(&bout, 1, OWRITE);
dump(0, score, type);
Bflush(&bout);
threadexitsall(nil);
}