plan9port/src/libventi/server.c

207 lines
3 KiB
C
Raw Normal View History

2003-11-23 18:19:58 +00:00
#include <u.h>
#include <libc.h>
#include <venti.h>
#include <thread.h>
#include "queue.h"
enum
{
STACK = 8192,
};
typedef struct VtSconn VtSconn;
struct VtSconn
{
int ctl;
int ref;
QLock lk;
2003-11-23 18:19:58 +00:00
char dir[NETPATHLEN];
VtSrv *srv;
VtConn *c;
};
struct VtSrv
{
int afd;
int dead;
char adir[NETPATHLEN];
Queue *q; /* Queue(VtReq*) */
};
static void listenproc(void*);
static void connproc(void*);
static void
scincref(VtSconn *sc)
{
qlock(&sc->lk);
sc->ref++;
qunlock(&sc->lk);
}
static void
scdecref(VtSconn *sc)
{
qlock(&sc->lk);
if(--sc->ref > 0){
qunlock(&sc->lk);
return;
}
if(sc->c)
vtfreeconn(sc->c);
vtfree(sc);
}
2003-11-23 18:19:58 +00:00
VtSrv*
vtlisten(char *addr)
{
VtSrv *s;
s = vtmallocz(sizeof(VtSrv));
s->afd = announce(addr, s->adir);
if(s->afd < 0){
free(s);
return nil;
}
s->q = _vtqalloc();
proccreate(listenproc, s, STACK);
return s;
}
static void
listenproc(void *v)
{
int ctl;
char dir[NETPATHLEN];
VtSrv *srv;
VtSconn *sc;
srv = v;
for(;;){
ctl = listen(srv->adir, dir);
if(ctl < 0){
srv->dead = 1;
break;
}
sc = vtmallocz(sizeof(VtSconn));
sc->ref = 1;
2003-11-23 18:19:58 +00:00
sc->ctl = ctl;
sc->srv = srv;
strcpy(sc->dir, dir);
proccreate(connproc, sc, STACK);
}
// hangup
}
static void
connproc(void *v)
{
VtSconn *sc;
VtConn *c;
Packet *p;
VtReq *r;
int fd;
2004-05-23 00:59:17 +00:00
static int first=1;
2003-11-23 18:19:58 +00:00
2004-05-23 00:59:17 +00:00
if(first){
first=0;
fmtinstall('F', vtfcallfmt);
}
2003-11-23 18:19:58 +00:00
r = nil;
sc = v;
sc->c = nil;
2003-11-23 18:19:58 +00:00
fprint(2, "new call %s on %d\n", sc->dir, sc->ctl);
fd = accept(sc->ctl, sc->dir);
close(sc->ctl);
if(fd < 0){
fprint(2, "accept %s: %r\n", sc->dir);
goto out;
}
c = vtconn(fd, fd);
sc->c = c;
if(vtversion(c) < 0){
fprint(2, "vtversion %s: %r\n", sc->dir);
goto out;
}
if(vtsrvhello(c) < 0){
fprint(2, "vtsrvhello %s: %r\n", sc->dir);
goto out;
}
fprint(2, "new proc %s\n", sc->dir);
proccreate(vtsendproc, c, STACK);
qlock(&c->lk);
while(!c->writeq)
rsleep(&c->rpcfork);
qunlock(&c->lk);
while((p = vtrecv(c)) != nil){
r = vtmallocz(sizeof(VtReq));
if(vtfcallunpack(&r->tx, p) < 0){
packetfree(p);
fprint(2, "bad packet on %s: %r\n", sc->dir);
continue;
}
2004-05-23 00:59:17 +00:00
if(chattyventi)
fprint(2, "%s <- %F\n", argv0, &r->tx);
2003-11-23 18:19:58 +00:00
packetfree(p);
if(r->tx.type == VtTgoodbye)
break;
r->rx.tag = r->tx.tag;
r->sc = sc;
scincref(sc);
2003-11-23 18:19:58 +00:00
if(_vtqsend(sc->srv->q, r) < 0){
scdecref(sc);
2003-11-23 18:19:58 +00:00
fprint(2, "hungup queue\n");
break;
}
r = nil;
}
fprint(2, "eof on %s\n", sc->dir);
out:
if(r){
vtfcallclear(&r->tx);
vtfree(r);
}
fprint(2, "freed %s\n", sc->dir);
scdecref(sc);
2003-11-23 18:19:58 +00:00
return;
}
VtReq*
vtgetreq(VtSrv *srv)
{
return _vtqrecv(srv->q);
}
void
vtrespond(VtReq *r)
{
Packet *p;
VtSconn *sc;
sc = r->sc;
if(r->rx.tag != r->tx.tag)
abort();
if(r->rx.type != r->tx.type+1 && r->rx.type != VtRerror)
abort();
2004-05-23 00:59:17 +00:00
if(chattyventi)
fprint(2, "%s -> %F\n", argv0, &r->rx);
2003-11-23 18:19:58 +00:00
if((p = vtfcallpack(&r->rx)) == nil){
fprint(2, "fcallpack on %s: %r\n", sc->dir);
packetfree(p);
vtfcallclear(&r->rx);
return;
}
vtsend(sc->c, p);
scdecref(sc);
2003-11-23 18:19:58 +00:00
vtfcallclear(&r->tx);
vtfcallclear(&r->rx);
vtfree(r);
}