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;
|
2004-04-29 17:13:24 +00:00
|
|
|
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*);
|
|
|
|
|
2004-04-29 17:13:24 +00:00
|
|
|
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));
|
2004-04-29 17:13:24 +00:00
|
|
|
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-06-09 14:10:32 +00:00
|
|
|
if(first && chattyventi){
|
2004-05-23 00:59:17 +00:00
|
|
|
first=0;
|
|
|
|
fmtinstall('F', vtfcallfmt);
|
|
|
|
}
|
2003-11-23 18:19:58 +00:00
|
|
|
r = nil;
|
|
|
|
sc = v;
|
2004-04-29 17:13:24 +00:00
|
|
|
sc->c = nil;
|
2004-06-09 14:10:32 +00:00
|
|
|
if(0) fprint(2, "new call %s on %d\n", sc->dir, sc->ctl);
|
2003-11-23 18:19:58 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2004-06-09 14:10:32 +00:00
|
|
|
if(0) fprint(2, "new proc %s\n", sc->dir);
|
2003-11-23 18:19:58 +00:00
|
|
|
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;
|
2004-04-29 17:13:24 +00:00
|
|
|
scincref(sc);
|
2003-11-23 18:19:58 +00:00
|
|
|
if(_vtqsend(sc->srv->q, r) < 0){
|
2004-04-29 17:13:24 +00:00
|
|
|
scdecref(sc);
|
2003-11-23 18:19:58 +00:00
|
|
|
fprint(2, "hungup queue\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
r = nil;
|
|
|
|
}
|
|
|
|
|
2004-06-09 14:10:32 +00:00
|
|
|
if(0) fprint(2, "eof on %s\n", sc->dir);
|
2003-11-23 18:19:58 +00:00
|
|
|
|
|
|
|
out:
|
|
|
|
if(r){
|
|
|
|
vtfcallclear(&r->tx);
|
|
|
|
vtfree(r);
|
|
|
|
}
|
2004-06-09 14:10:32 +00:00
|
|
|
if(0) fprint(2, "freed %s\n", sc->dir);
|
2004-04-29 17:13:24 +00:00
|
|
|
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);
|
2004-04-29 17:13:24 +00:00
|
|
|
scdecref(sc);
|
2003-11-23 18:19:58 +00:00
|
|
|
vtfcallclear(&r->tx);
|
|
|
|
vtfcallclear(&r->rx);
|
|
|
|
vtfree(r);
|
|
|
|
}
|
|
|
|
|