X-Git-Url: http://dolda2000.com/gitweb/?a=blobdiff_plain;f=daemon%2Fnet.c;h=121050cf4eba6bfe11d89a1393adfff3c84e1388;hb=8d7a1e880011bbb660895f022a9f250f3cfda391;hp=7a68dae5c5dca11856a4824edda73fa7fd78ca6c;hpb=363c7059ebeac9d54a372a26737e2235d849eae5;p=doldaconnect.git diff --git a/daemon/net.c b/daemon/net.c index 7a68dae..121050c 100644 --- a/daemon/net.c +++ b/daemon/net.c @@ -211,6 +211,7 @@ static struct socket *newsock1(int dgram) new->refcount = 1; new->state = -1; new->dgram = dgram; + new->maxbuf = 65536; numsocks++; return(new); } @@ -233,6 +234,15 @@ static void sksetstate(struct socket *sk, int state) sk->back->state = state; } +struct socket *netsockpipe(void) +{ + struct socket *sk; + + sk = sockpair(0); + sksetstate(sk, SOCK_EST); + return(sk); +} + static void closeufd(struct ufd *ufd) { if(ufd->fd != -1) @@ -348,10 +358,41 @@ void getsock(struct socket *sk) sk->refcount++; } +static void sockdebug(int level, struct socket *sk, char *format, ...) +{ + va_list args; + char *tb; + + if((sk->dbgnm == NULL) || (level > sk->dbglvl)) + return; + va_start(args, format); + tb = vsprintf2(format, args); + va_end(args); + fprintf(stderr, "%s: %s\n", sk->dbgnm, tb); + free(tb); +} + +void socksetdebug(struct socket *sk, int level, char *nm, ...) +{ + va_list args; + char *tb; + + va_start(args, nm); + tb = vsprintf2(nm, args); + va_end(args); + sk->dbgnm = sprintf2("%s (f)", tb); + sk->back->dbgnm = sprintf2("%s (b)", tb); + free(tb); + sk->dbglvl = level; + sk->back->dbglvl = level; + sockdebug(1, sk, "enabled debugging"); +} + static void freesock(struct socket *sk) { struct dgrambuf *buf; + sockdebug(1, sk, "freeing socket"); if(sk->dgram) { while((buf = sk->buf.d.f) != NULL) { sk->buf.d.f = buf->next; @@ -361,6 +402,8 @@ static void freesock(struct socket *sk) if(sk->buf.s.buf != NULL) free(sk->buf.s.buf); } + if(sk->dbgnm != NULL) + free(sk->dbgnm); free(sk); numsocks--; } @@ -380,6 +423,14 @@ void putsock(struct socket *sk) } } +void quitsock(struct socket *sk) +{ + sk->readcb = NULL; + sk->writecb = NULL; + sk->errcb = NULL; + putsock(sk); +} + static void linksock(struct scons **list, struct socket *sk) { struct scons *sc; @@ -458,6 +509,7 @@ void *sockgetinbuf(struct socket *sk, size_t *size) if((sk->buf.s.buf == NULL) || (sk->buf.s.datasize == 0)) { *size = 0; + sockdebug(2, sk, "read 0 bytes", *size); return(NULL); } buf = sk->buf.s.buf; @@ -466,6 +518,7 @@ void *sockgetinbuf(struct socket *sk, size_t *size) sk->buf.s.bufsize = sk->buf.s.datasize = 0; sockread(sk); } + sockdebug(2, sk, "read %zi bytes", *size); return(buf); } @@ -475,6 +528,7 @@ void sockqueue(struct socket *sk, void *data, size_t size) struct sockaddr *remote; socklen_t remotelen; + sockdebug(2, sk, "queued %zi bytes", size); if(size == 0) return; if(sk->state == SOCK_STL) @@ -735,6 +789,7 @@ static int sockflush(struct ufd *ufd) void closesock(struct socket *sk) { + sockdebug(1, sk, "closed"); sksetstate(sk, SOCK_STL); if(sk->back->eos == 0) sk->back->eos = 1; @@ -756,9 +811,26 @@ size_t sockgetdatalen(struct socket *sk) return(ret); } -size_t sockqueuesize(struct socket *sk) +/* size_t sockqueuesize(struct socket *sk) */ +/* { */ +/* return(sockgetdatalen(sk->back)); */ +/* } */ + +size_t socktqueuesize(struct socket *sk) { - return(sockgetdatalen(sk->back)); + size_t ret; + + ret = 0; + while(1) { + ret += sockgetdatalen(sk->back); + if((sk = sk->back->pnext) == NULL) + return(ret); + } +} + +ssize_t sockqueueleft(struct socket *sk) +{ + return(sk->back->maxbuf - sockgetdatalen(sk->back)); } /* @@ -931,6 +1003,8 @@ struct socket *netcsconn(struct sockaddr *addr, socklen_t addrlen, void (*func)( memcpy(sk->ufd->d.s.remote = smalloc(addrlen), addr, sk->ufd->d.s.remotelen = addrlen); sk->back->conncb = func; sk->back->data = data; + getsock(sk->back); + putsock(sk); if(!connect(sk->ufd->fd, addr, addrlen)) { sksetstate(sk, SOCK_EST); @@ -1002,10 +1076,18 @@ static void runbatches(void) static void cleansocks(void) { struct ufd *ufd, *next; + int dead; for(ufd = ufds; ufd != NULL; ufd = next) { next = ufd->next; - if(ufd->sk && ((ufd->fd < 0) || (sockgetdatalen(ufd->sk) == 0))) { + if(ufd->sk) { + dead = (ufd->fd < 0); + if(ufd->sk->state == SOCK_STL) + dead = 1; + if((ufd->sk->state == SOCK_EST) && (sockgetdatalen(ufd->sk) == 0)) + dead = 1; + if(!dead) + continue; if(ufd->sk->eos == 1) { ufd->sk->eos = 2; closeufd(ufd); @@ -1038,7 +1120,7 @@ int pollsocks(int timeout) for(maxfd = 0, ufd = ufds; ufd != NULL; ufd = ufd->next) { if(ufd->fd < 0) continue; - if(!ufd->ignread) + if(!ufd->ignread && ((ufd->sk == NULL) || (sockqueueleft(ufd->sk) > 0))) FD_SET(ufd->fd, &rfds); if(ufd->sk != NULL) { if(sockgetdatalen(ufd->sk) > 0) @@ -1529,13 +1611,13 @@ int getucred(struct socket *sk, uid_t *uid, gid_t *gid) return(0); } -void sockblock(struct socket *sk, int block) -{ - struct ufd *ufd; +/* void sockblock(struct socket *sk, int block) */ +/* { */ +/* struct ufd *ufd; */ - ufd = getskufd(sk); - ufd->ignread = block; -} +/* ufd = getskufd(sk); */ +/* ufd->ignread = block; */ +/* } */ int sockfamily(struct socket *sk) {