| #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); | 
 | } |