shithub: neoventi

Download patch

ref: a734ef4f09cf5c4b7851b17f88f2970607ee25b2
parent: 8a603a071dc58deb4ab66b76572c5a82767fa8db
author: Noam Preil <noam@pixelhero.dev>
date: Sun Dec 24 11:37:59 EST 2023

error handling skeleton

--- a/main.c
+++ b/main.c
@@ -25,7 +25,6 @@
 		return 0;
 	if(*nmap > MaxAMap)
 		return 0;
-	print("reading %d map entries\n", *nmap);
 	*map = realloc(*map, *nmap * sizeof(MapEntry));
 	for(i = 0; i < *nmap; i += 1){
 		s = Brdline(b, '\n');
@@ -37,7 +36,6 @@
 			sysfatal("corrupt index map: %s", fields[1]);
 		if(stru64int(fields[2], &(*map)[i].stop) < 0)
 			sysfatal("corrupt index map: %s", fields[2]);
-	//	print("amap entry %d: [%llud, %llud)\n", i, (*map)[i].start, (*map)[i].stop);
 	}
 	return 1;
 }
@@ -116,7 +114,6 @@
 	if(Bseek(&bio, tabbase, 0) != tabbase)
 		sysfatal("seek failed: %r");
 	parsemap(&bio, &map, &nmap);
-	print("found %d arenas\n", nmap);
 	arenas = realloc(arenas, sizeof(VtArena) * (nmap + numarenas));
 	if(!arenas)
 		sysfatal("oom");
@@ -232,7 +229,6 @@
 		sysfatal("invalid magic found in index header");
 	if(!Brdu32(&bio, &version) || version != 1)
 		sysfatal("failed to read version or version unsupported");
-	print("Parsing index v1...\n");
 	line = Brdline(&bio, '\n');
 	if(Blinelen(&bio) >= NameSize)
 		sysfatal("invalid or corrupt index: name too big");
@@ -462,8 +458,9 @@
 main(int argc, char **argv)
 {
 	parseargs(argc, argv);
-	print("Initializing neoventi build 2...\n");
+	print("Initializing neoventi build 3... ");
 	init();
 	validate();
+	print("initialized, launching server.\n");
 	serve("tcp!127.1!14011");
 }
--- a/neoventi.h
+++ b/neoventi.h
@@ -30,7 +30,7 @@
 	 * or for when you accidentally {mount /dev/.../arenas /mnt/venti} */
 	PartBlank = 256 * 1024,
 	NameSize = 64,
-	/* FIXME make arena part head a full block? breaking change for new arena partitions, removes
+	/* FIXME(design break) make arena part head a full block? breaking change for new arena partitions, removes
 	 * need for special casing and magic around it. Just read the damn block. */
 	HeadSize = 512,
 	ArenaPartMagic = 0xa9e4a5e7U,
@@ -44,6 +44,7 @@
 
 typedef struct {
 	int fd;
+	jmp_buf bounce;
 } VtConn;
 
 typedef struct {
--- a/server.c
+++ b/server.c
@@ -3,22 +3,34 @@
 #include <bio.h>
 #include "neoventi.h"
 
-// Convenience function: reads a venti packed from conn into buf
+// Handles an error on `conn` handling client request `tbuf`
+// Only the tag must be preserved in the buffer
+static void
+vterr(VtConn conn, char *tbuf, char *msg, ...)
+{
+	va_list args;
+	va_start(args, msg);
+	msg = vsmprint(msg, args);
+	werrstr(msg);
+	free(msg);
+	va_end(args);
+	if(tbuf != nil){
+		// TODO: send Rerror message
+	}
+	longjmp(conn.bounce, 1);
+}
+
+// Convenience function: reads a venti packet from conn into buf
 // Invariant: buf MUST have 0x10002 bytes available! see venti(6)
-static int
+static void
 vtrecv(VtConn conn, char *buf)
 {
 	u16int len;
-	if(read(conn.fd, buf, 2) != 2){
-		werrstr("Failed to read message size: %r");
-		return 0;
-	}
+	if(read(conn.fd, buf, 2) != 2)
+		vterr(conn, nil, "failed to read message size: %r");
 	len = (buf[0] << 8 | buf[1]);
-	if(read(conn.fd, buf + 2, len) != len){
-		werrstr("Failed to read message: %r");
-		return 0;
-	}
-	return 1;
+	if(read(conn.fd, buf + 2, len) != len)
+		vterr(conn, nil, "failed to read message: %r");
 }
 
 static void
@@ -25,13 +37,14 @@
 vtversion(VtConn conn)
 {
 	char c;
+	// Response is one line of unknown size; discard bytes until EOL
 	if(fprint(conn.fd, "venti-02-neoventi\n") == 18)
 		while(read(conn.fd, &c, 1) == 1)
 			if(c == '\n')
 				return;
+	// If the handshake fails, make it clear there's a problem and give up
 	fprint(conn.fd, "FUCK OFF\n");
-	close(conn.fd);
-	sysfatal("venti handshake failed: %r");
+	vterr(conn, nil, "handshake failed");
 }
 
 static void
@@ -41,7 +54,6 @@
 	VtArena arena;
 	u64int addr;
 	u16int size;
-	u32int off;
 	u8int blocks;
 	uchar *dbuf;
 	score = (u8int*)buf + 4;
@@ -68,41 +80,48 @@
 	case VtTgoodbye:
 		return 0;
 	case VtTsync:
-		print("we don't support vtsync yet. Hanging up!\n");
-		return 0;
+		vterr(conn, buf, "TODO: sync not supported yet");
 	default:
-		sysfatal("TODO safely hang up vtconns");
+		vterr(conn, buf, "TODO safely hang up vtconns");
 	}
 	return 0;
 }
 
 static void
-handle(int ctl, char *dir)
+vthello(VtConn conn, char *buf)
 {
-	char buf[0x10002];
-	VtConn conn;
-	conn.fd = accept(ctl, dir);
-	if(conn.fd < 0)
-		sysfatal("failed to accept connection: %r");
-	print("received a connection at %s, fd %d\n", dir, conn.fd);
 	vtversion(conn);
-	if(!vtrecv(conn, buf))
-		sysfatal("msg recv failed: %r");
+	vtrecv(conn, buf);
 	if(buf[2] != VtThello)
-		sysfatal("received message before hello: %d", buf[2]);
+		vterr(conn, buf, "received message before hello: %d", buf[2]);
 	if(buf[4] != 0 || buf[5] != 2 || buf[6] != '0' || buf[7] != '2')
-		sysfatal("unsupported protocol version requested in Thello: %d %d %d %d", buf[4], buf[5], buf[6], buf[7]);
+		vterr(conn, buf, "unsupported protocol version requested in Thello: %d %d %d %d", buf[4], buf[5], buf[6], buf[7]);
 	buf[2] = VtRhello;
 	buf[6] = 'n';
 	buf[7] = 'o';
 	buf[1] = 8;
 	if(write(conn.fd, buf, 10) != 10)
-		sysfatal("failed to rhello: %r");
-	while(vtrecv(conn, buf)){
-		if(!vtconnhandle(conn, buf))
-			break;
+		vterr(conn, buf, "failed to rhello: %r");
+}
+
+static void
+handle(int ctl, char *dir)
+{
+	char buf[0x10002];
+	VtConn conn;
+	if(setjmp(conn.bounce) != 0){
+		fprint(2, "abandoning client: %r\n");
+		close(conn.fd);
+		return;
 	}
-	close(conn.fd);
+	conn.fd = accept(ctl, dir);
+	if(conn.fd < 0)
+		vterr(conn, nil, "failed to accept connection: %r");
+	vthello(conn, buf);
+	while(1){
+		vtrecv(conn, buf);
+		vtconnhandle(conn, buf);
+	}
 }
 
 void
@@ -115,6 +134,7 @@
 		sysfatal("%r");
 	procsetname("neoventi/server");
 	for(ctl = listen(adir, dir); ctl >= 0; ctl = listen(adir, dir)){
+		// TODO: handle the client in a worker, allow multiple clients
 		handle(ctl, dir);
 		close(ctl);
 	}