mirror of
https://github.com/9fans/plan9port.git
synced 2025-01-15 11:20:03 +00:00
156 lines
2.8 KiB
C
156 lines
2.8 KiB
C
|
/*
|
||
|
* Multiplexed Venti client. It would be nice if we
|
||
|
* could turn this into a generic library routine rather
|
||
|
* than keep it Venti specific. A user-level 9P client
|
||
|
* could use something like this too.
|
||
|
*
|
||
|
* This is a little more complicated than it might be
|
||
|
* because we want it to work well within and without libthread.
|
||
|
*
|
||
|
* The mux code is inspired by tra's, which is inspired by the Plan 9 kernel.
|
||
|
*/
|
||
|
|
||
|
#include <u.h>
|
||
|
#include <libc.h>
|
||
|
#include <venti.h>
|
||
|
|
||
|
typedef struct Rwait Rwait;
|
||
|
struct Rwait
|
||
|
{
|
||
|
Rendez r;
|
||
|
Packet *p;
|
||
|
int done;
|
||
|
int sleeping;
|
||
|
};
|
||
|
|
||
|
static int gettag(VtConn*, Rwait*);
|
||
|
static void puttag(VtConn*, Rwait*, int);
|
||
|
static void muxrpc(VtConn*, Packet*);
|
||
|
Packet *vtrpc(VtConn*, Packet*);
|
||
|
|
||
|
Packet*
|
||
|
vtrpc(VtConn *z, Packet *p)
|
||
|
{
|
||
|
int i;
|
||
|
uchar tag, buf[2], *top;
|
||
|
Rwait *r;
|
||
|
|
||
|
/* must malloc because stack could be private */
|
||
|
r = vtmallocz(sizeof(Rwait));
|
||
|
|
||
|
qlock(&z->lk);
|
||
|
r->r.l = &z->lk;
|
||
|
tag = gettag(z, r);
|
||
|
|
||
|
/* slam tag into packet */
|
||
|
top = packetpeek(p, buf, 0, 2);
|
||
|
if(top == nil){
|
||
|
packetfree(p);
|
||
|
return nil;
|
||
|
}
|
||
|
if(top == buf){
|
||
|
werrstr("first two bytes must be in same packet fragment");
|
||
|
packetfree(p);
|
||
|
return nil;
|
||
|
}
|
||
|
top[1] = tag;
|
||
|
qunlock(&z->lk);
|
||
|
if(vtsend(z, p) < 0)
|
||
|
return nil;
|
||
|
|
||
|
qlock(&z->lk);
|
||
|
/* wait for the muxer to give us our packet */
|
||
|
r->sleeping = 1;
|
||
|
z->nsleep++;
|
||
|
while(z->muxer && !r->done)
|
||
|
rsleep(&r->r);
|
||
|
z->nsleep--;
|
||
|
r->sleeping = 0;
|
||
|
|
||
|
/* if not done, there's no muxer: start muxing */
|
||
|
if(!r->done){
|
||
|
if(z->muxer)
|
||
|
abort();
|
||
|
z->muxer = 1;
|
||
|
while(!r->done){
|
||
|
qunlock(&z->lk);
|
||
|
if((p = vtrecv(z)) == nil){
|
||
|
z->muxer = 0;
|
||
|
return nil;
|
||
|
}
|
||
|
qlock(&z->lk);
|
||
|
muxrpc(z, p);
|
||
|
}
|
||
|
z->muxer = 0;
|
||
|
/* if there is anyone else sleeping, wake them to mux */
|
||
|
if(z->nsleep){
|
||
|
for(i=0; i<256; i++)
|
||
|
if(z->wait[i] != nil && ((Rwait*)z->wait[i])->sleeping)
|
||
|
break;
|
||
|
if(i==256)
|
||
|
fprint(2, "libventi: nsleep botch\n");
|
||
|
else
|
||
|
rwakeup(&((Rwait*)z->wait[i])->r);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
p = r->p;
|
||
|
puttag(z, r, tag);
|
||
|
vtfree(r);
|
||
|
qunlock(&z->lk);
|
||
|
return p;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
gettag(VtConn *z, Rwait *r)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
Again:
|
||
|
while(z->ntag == 256)
|
||
|
rsleep(&z->tagrend);
|
||
|
for(i=0; i<256; i++)
|
||
|
if(z->wait[i] == 0){
|
||
|
z->ntag++;
|
||
|
z->wait[i] = r;
|
||
|
return i;
|
||
|
}
|
||
|
fprint(2, "libventi: ntag botch\n");
|
||
|
goto Again;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
puttag(VtConn *z, Rwait *r, int tag)
|
||
|
{
|
||
|
assert(z->wait[tag] == r);
|
||
|
z->wait[tag] = nil;
|
||
|
z->ntag--;
|
||
|
rwakeup(&z->tagrend);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
muxrpc(VtConn *z, Packet *p)
|
||
|
{
|
||
|
uchar tag, buf[2], *top;
|
||
|
Rwait *r;
|
||
|
|
||
|
if((top = packetpeek(p, buf, 0, 2)) == nil){
|
||
|
fprint(2, "libventi: short packet in vtrpc\n");
|
||
|
packetfree(p);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
tag = top[1];
|
||
|
if((r = z->wait[tag]) == nil){
|
||
|
fprint(2, "libventi: unexpected packet tag %d in vtrpc\n", tag);
|
||
|
abort();
|
||
|
packetfree(p);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
r->p = p;
|
||
|
r->done = 1;
|
||
|
rwakeup(&r->r);
|
||
|
}
|
||
|
|