shithub: neoventi

ref: 0a0dc432673d0c5f0902b0d053e08c8fe41a0eea
dir: /server.c/

View raw version
#include <u.h>
#include <libc.h>
#include <bio.h>
#include "neoventi.h"

// 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, ...)
{
	u16int len;
	va_list args;
	va_start(args, msg);
	msg = vsmprint(msg, args);
	werrstr(msg);
	free(msg);
	va_end(args);
	if(tbuf != nil){
		len = snprint(tbuf+6, 0x10000, "neoventi: %r");
		tbuf[2] = VtRerror;
		tbuf[4] = len >> 8;
		tbuf[5] = len & 0xFF;
		len += 4;
		tbuf[0] = len >> 8;
		tbuf[1] = len & 0xFF;
		if(write(conn.fd, tbuf, len+2) != len+2)
			fprint(2, "failed to report error: %r");
	}
	longjmp(conn.bounce, 1);
}

static void
vthangup(VtConn conn)
{
	longjmp(conn.bounce, 2);
}

// Convenience function: reads a venti packet from conn into buf
static void
vtrecv(VtConn conn, char *buf)
{
	u16int len;
	switch(read(conn.fd, buf, 2)){
	case 0:
		vthangup(conn);
	case 1:
		vterr(conn, nil, "received a single byte for message size");
	case 2:
		len = (buf[0] << 8 | buf[1]);
		if(read(conn.fd, buf + 2, len) != len)
			vterr(conn, nil, "failed to read message: %r");
	}
}

static void
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");
	vterr(conn, nil, "handshake failed");
}

static void
vtread(VtConn conn, char *buf)
{
	u8int *score;
	VtArena arena;
	u64int addr;
	u16int size;
	u8int blocks;
	score = (u8int*)buf + 4;
	if(!vtreadlookup(score, &arena, &addr, &size, &blocks))
		sysfatal("todo graceful read errors");
	// Response: VtRread, msg tag, data
	buf[0] = (size+2)>>8;
	buf[1] = (size+2) & 0xFF;
	buf[2] = VtRread;
	readclump((uchar*)buf+4, arena, addr, blocks);
	if(write(conn.fd, buf, size + 4) != size+4)
		vterr(conn, buf, "failed to write data");
}

static int
vtconnhandle(VtConn conn, char *buf)
{
	switch(buf[2]){
	case VtTread:
		vtread(conn, buf);
		return 1;
	case VtTgoodbye:
		vthangup(conn);
	case VtTsync:
		vterr(conn, buf, "TODO: sync not supported yet");
	default:
		vterr(conn, buf, "TODO safely hang up vtconns");
	}
	return 0;
}

static void
vthello(VtConn conn, char *buf)
{
	vtversion(conn);
	vtrecv(conn, buf);
	if(buf[2] != VtThello)
		vterr(conn, buf, "received message before hello: %d", buf[2]);
	if(buf[4] != 0 || buf[5] != 2 || buf[6] != '0' || buf[7] != '2')
		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)
		vterr(conn, buf, "failed to rhello: %r");
}

static void
handle(int ctl, char *dir)
{
	char buf[MaxPacketSize];
	VtConn conn;
	switch(setjmp(conn.bounce)){
	case 0:
		break;
	case 1:
		fprint(2, "abandoning client: %r\n");
	case 2:
		close(conn.fd);
		return;
	default:
		fprint(2, "internal error: unexpected bounce code\n");
		break;
	}
	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);
		if(!vtconnhandle(conn, buf)){
			close(conn.fd);
			return;
		}
	}
}

void
serve(char *addr)
{
	char adir[NETPATHLEN], dir[NETPATHLEN];
	int fd, ctl;
	fd = announce(addr, adir);
	if(fd < 0)
		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);
	}
	fprint(2, "server has died\n");
}