diff options
author | Guenter Geiger <ggeiger@users.sourceforge.net> | 2002-07-29 17:06:19 +0000 |
---|---|---|
committer | Guenter Geiger <ggeiger@users.sourceforge.net> | 2002-07-29 17:06:19 +0000 |
commit | 57045df5fe3ec557e57dc7434ac1a07b5521bffc (patch) | |
tree | 7174058b41b73c808107c7090d9a4e93ee202341 /pd/src/u_pdreceive.c | |
parent | da38b3424229e59f956252c3d89895e43e84e278 (diff) |
This commit was generated by cvs2svn to compensate for changes in r58,
which included commits to RCS files with non-trunk default branches.
svn path=/trunk/; revision=59
Diffstat (limited to 'pd/src/u_pdreceive.c')
-rw-r--r-- | pd/src/u_pdreceive.c | 305 |
1 files changed, 305 insertions, 0 deletions
diff --git a/pd/src/u_pdreceive.c b/pd/src/u_pdreceive.c new file mode 100644 index 00000000..8d3f83e9 --- /dev/null +++ b/pd/src/u_pdreceive.c @@ -0,0 +1,305 @@ +/* Copyright (c) 2000 Miller Puckette. +* For information on usage and redistribution, and for a DISCLAIMER OF ALL +* WARRANTIES, see the file, "LICENSE.txt," in the Pd distribution. */ + +/* the "pdreceive" command. This is a standalone program that receives messages +from Pd via the netsend/netreceive ("FUDI") protocol, and copies them to +standard output. */ + +#include <sys/time.h> +#include <sys/types.h> +#include <string.h> +#include <stdio.h> +#include <errno.h> +#include <unistd.h> +#include <stdlib.h> +#ifdef UNIX +#include <sys/socket.h> +#include <netinet/in.h> +#include <netdb.h> +#define SOCKET_ERROR -1 +#else +#include <winsock.h> +#endif + +typedef struct _fdpoll +{ + int fdp_fd; + char *fdp_inbuf; + int fdp_inhead; + int fdp_intail; + int fdp_udp; +} t_fdpoll; + +static int nfdpoll; +static t_fdpoll *fdpoll; +static int maxfd; +static int sockfd; +static int protocol; + +static void sockerror(char *s); +static void closesocket(int fd); +static void dopoll(void); +#define BUFSIZE 4096 + +int main(int argc, char **argv) +{ + int portno; + struct sockaddr_in server; + int nretry = 10; + if (argc < 2 || sscanf(argv[1], "%d", &portno) < 1 || portno <= 0) + goto usage; + if (argc >= 3) + { + if (!strcmp(argv[2], "tcp")) + protocol = SOCK_STREAM; + else if (!strcmp(argv[2], "udp")) + protocol = SOCK_DGRAM; + else goto usage; + } + else protocol = SOCK_STREAM; + sockfd = socket(AF_INET, protocol, 0); + if (sockfd < 0) + { + sockerror("socket()"); + exit(1); + } + maxfd = sockfd + 1; + server.sin_family = AF_INET; + server.sin_addr.s_addr = INADDR_ANY; + +#ifdef IRIX + /* this seems to work only in IRIX but is unnecessary in + Linux. Not sure what NT needs in place of this. */ + if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, 0, 0) < 0) + post("setsockopt failed\n"); +#endif + + /* assign client port number */ + server.sin_port = htons((unsigned short)portno); + + /* name the socket */ + if (bind(sockfd, (struct sockaddr *)&server, sizeof(server)) < 0) + { + sockerror("bind"); + closesocket(sockfd); + return (0); + } + if (protocol == SOCK_STREAM) + { + if (listen(sockfd, 5) < 0) + { + sockerror("listen"); + closesocket(sockfd); + exit(1); + } + } + /* now loop forever selecting on sockets */ + while (1) + dopoll(); + +usage: + fprintf(stderr, "usage: pdreceive <portnumber> [udp|tcp]\n"); + fprintf(stderr, "(default is tcp)\n"); + exit(1); +} + +static void addport(int fd) +{ + int nfd = nfdpoll; + t_fdpoll *fp; + fdpoll = (t_fdpoll *)realloc(fdpoll, + (nfdpoll+1) * sizeof(t_fdpoll)); + fp = fdpoll + nfdpoll; + fp->fdp_fd = fd; + nfdpoll++; + if (fd >= maxfd) maxfd = fd + 1; + fp->fdp_inhead = fp->fdp_intail = 0; + if (!(fp->fdp_inbuf = malloc(BUFSIZE))) + { + fprintf(stderr, "out of memory"); + exit(1); + } + printf("number_connected %d;\n", nfdpoll); +} + +static void rmport(t_fdpoll *x) +{ + int nfd = nfdpoll; + int i, size = nfdpoll * sizeof(t_fdpoll); + t_fdpoll *fp; + for (i = nfdpoll, fp = fdpoll; i--; fp++) + { + if (fp == x) + { + closesocket(fp->fdp_fd); + free(fp->fdp_inbuf); + while (i--) + { + fp[0] = fp[1]; + fp++; + } + fdpoll = (t_fdpoll *)realloc(fdpoll, + (nfdpoll-1) * sizeof(t_fdpoll)); + nfdpoll--; + printf("number_connected %d;\n", nfdpoll); + return; + } + } + fprintf(stderr, "warning: item removed from poll list but not found"); +} + +static void doconnect(void) +{ + int fd = accept(sockfd, 0, 0); + if (fd < 0) + perror("accept"); + else addport(fd); +} + +static void udpread(void) +{ + char buf[BUFSIZE]; + int ret = recv(sockfd, buf, BUFSIZE, 0); + if (ret < 0) + { + sockerror("recv (udp)"); + close(sockfd); + exit(1); + } + else if (ret > 0) + { + if (write(1, buf, ret) < ret) + { + perror("write"); + exit(1); + } + } +} + +static int tcpmakeoutput(t_fdpoll *x) +{ + char messbuf[BUFSIZE+1], *bp = messbuf; + int indx; + int inhead = x->fdp_inhead; + int intail = x->fdp_intail; + char *inbuf = x->fdp_inbuf; + if (intail == inhead) + return (0); + for (indx = intail; indx != inhead; indx = (indx+1)&(BUFSIZE-1)) + { + /* search for a semicolon. */ + char c = *bp++ = inbuf[indx]; + if (c == ';') + { + intail = (indx+1)&(BUFSIZE-1); + if (inbuf[intail] == '\n') + intail = (intail+1)&(BUFSIZE-1); + *bp++ = '\n'; + write(1, messbuf, bp - messbuf); + x->fdp_inhead = inhead; + x->fdp_intail = intail; + return (1); + } + } + return (0); +} + +static void tcpread(t_fdpoll *x) +{ + char *semi; + int readto = + (x->fdp_inhead >= x->fdp_intail ? BUFSIZE : x->fdp_intail-1); + int ret; + + /* the input buffer might be full. If so, drop the whole thing */ + if (readto == x->fdp_inhead) + { + fprintf(stderr, "pd: dropped message from gui\n"); + x->fdp_inhead = x->fdp_intail = 0; + readto = BUFSIZE; + } + else + { + ret = recv(x->fdp_fd, x->fdp_inbuf + x->fdp_inhead, + readto - x->fdp_inhead, 0); + if (ret < 0) + { + sockerror("recv (tcp)"); + rmport(x); + } + else if (ret == 0) + rmport(x); + else + { + x->fdp_inhead += ret; + if (x->fdp_inhead >= BUFSIZE) + x->fdp_inhead = 0; + while (tcpmakeoutput(x)) + ; + } + } +} + +static void dopoll(void) +{ + int i; + t_fdpoll *fp; + fd_set readset, writeset, exceptset; + FD_ZERO(&writeset); + FD_ZERO(&readset); + FD_ZERO(&exceptset); + + FD_SET(sockfd, &readset); + if (protocol == SOCK_STREAM) + { + for (fp = fdpoll, i = nfdpoll; i--; fp++) + FD_SET(fp->fdp_fd, &readset); + } + if (select(maxfd+1, &readset, &writeset, &exceptset, 0) < 0) + { + perror("select"); + exit(1); + } + if (protocol == SOCK_STREAM) + { + for (i = 0; i < nfdpoll; i++) + if (FD_ISSET(fdpoll[i].fdp_fd, &readset)) + tcpread(&fdpoll[i]); + if (FD_ISSET(sockfd, &readset)) + doconnect(); + } + else + { + if (FD_ISSET(sockfd, &readset)) + udpread(); + } +} + + +static void sockerror(char *s) +{ +#ifdef NT + int err = WSAGetLastError(); + if (err == 10054) return; + else if (err == 10044) + { + fprintf(stderr, + "Warning: you might not have TCP/IP \"networking\" turned on\n"); + } +#endif +#ifdef UNIX + int err = errno; +#endif + fprintf(stderr, "%s: %s (%d)\n", s, strerror(err), err); +} + +static void closesocket(int fd) +{ +#ifdef UNIX + close(fd); +#endif +#ifdef NT + closesocket(fd); +#endif +} |