xref: /csrg-svn/sys/tests/netiso/tisink.c (revision 58241)
149352Sbostic /*-
249352Sbostic  * Copyright (c) 1988, 1990 The Regents of the University of California.
344323Ssklower  * All rights reserved.
444323Ssklower  *
544323Ssklower  * %sccs.include.redist.c%
644323Ssklower  */
749352Sbostic 
844323Ssklower #ifndef lint
949352Sbostic char copyright[] =
1049352Sbostic "@(#) Copyright (c) 1988, 1990 The Regents of the University of California.\n\
1149352Sbostic  All rights reserved.\n";
1244323Ssklower #endif /* not lint */
1344323Ssklower 
1449352Sbostic #ifndef lint
15*58241Ssklower static char sccsid[] = "@(#)tisink.c	7.11 (Berkeley) 02/25/93";
1649352Sbostic #endif /* not lint */
1749352Sbostic 
1844323Ssklower /*
1948712Ssklower  * This is a test program to be a sink for ISO packets.
2044323Ssklower  */
2155605Ssklower #include <unistd.h>
2244323Ssklower #include <sys/param.h>
2344323Ssklower #include <sys/uio.h>
2444323Ssklower #include <sys/socket.h>
2544323Ssklower #include <sys/ioctl.h>
2655605Ssklower #include <sys/syscall.h>
2744323Ssklower #include <net/route.h>
2844323Ssklower #include <net/if.h>
2944323Ssklower #define  TCPT_NTIMERS 4
3044323Ssklower #include <netiso/iso.h>
3144323Ssklower #include <netiso/tp_param.h>
3244323Ssklower #include <netiso/tp_user.h>
3344323Ssklower 
3444323Ssklower #include <stdio.h>
3544323Ssklower #include <errno.h>
3644323Ssklower #include <ctype.h>
3744323Ssklower #include <netdb.h>
3844323Ssklower 
3944323Ssklower 
4044323Ssklower #define dbprintf if(verbose)printf
4150890Ssklower #ifdef __STDC__
4250890Ssklower #define try(a,b,c) {x = (a b); dbprintf("%s%s returns %d\n",c,#a,x);\
4350890Ssklower 		if (x<0) {perror(#a); myexit(0);}}
4450890Ssklower #else
4544323Ssklower #define try(a,b,c) {x = (a b); dbprintf("%s%s returns %d\n",c,"a",x);\
4650890Ssklower 		if (x<0) {perror("a"); myexit(0);}}
4750890Ssklower #endif
4844323Ssklower 
4944323Ssklower 
5044323Ssklower struct  ifreq ifr;
5144323Ssklower short port = 3000;
5244323Ssklower struct  sockaddr_iso faddr, laddr = { sizeof(laddr), AF_ISO };
5344323Ssklower struct  sockaddr_iso *siso = &laddr;
5448711Ssklower char **xenvp;
5544323Ssklower 
5655605Ssklower long size, forkp = 0, confp = 0, mynamep, verbose = 1, echop = 0;
5750515Ssklower long records, intercept = 0, isode_mode = 0, dgramp = 0, tp0mode = 0;
58*58241Ssklower long dumpnodata = 0, playtag = 0, select_mode = 0, tuba = 0;
5950515Ssklower void savedata();
6044323Ssklower 
6144323Ssklower char buf[2048];
6244323Ssklower char your_it[] = "You're it!";
6355605Ssklower fd_set readfds, exceptfds;
6444323Ssklower 
6544323Ssklower char *Servername;
6644323Ssklower 
main(argc,argv,envp)6748711Ssklower main(argc, argv, envp)
6844323Ssklower int argc;
6944323Ssklower char *argv[];
7048711Ssklower char *envp[];
7144323Ssklower {
7244323Ssklower 	register char **av = argv;
7344323Ssklower 	register char *cp;
7448712Ssklower 	struct iso_addr *iso_addr();
7544323Ssklower 
7648711Ssklower 	xenvp = envp;
7744323Ssklower 	while(--argc > 0) {
7844323Ssklower 		av++;
7944323Ssklower 		if(strcmp(*av,"Servername")==0) {
8044323Ssklower 			av++;
8144323Ssklower 			Servername = *av;
8244323Ssklower 			argc--;
8344323Ssklower 		} else if (strcmp(*av,"host")==0) {
8444323Ssklower 			av++;
8548712Ssklower 			laddr.siso_addr = *iso_addr(*av);
8644323Ssklower 			argc--;
8744323Ssklower 		} else if (strcmp(*av,"port")==0) {
8844323Ssklower 			av++;
8944323Ssklower 			sscanf(*av,"%hd",&port);
9044323Ssklower 			argc--;
9144323Ssklower 		} else if (strcmp(*av,"size")==0) {
9244323Ssklower 			av++;
9344323Ssklower 			sscanf(*av,"%ld",&size);
9444323Ssklower 			argc--;
9550890Ssklower 		} else if (strcmp(*av, "echo")==0) {
9650890Ssklower 			echop++;
9744600Ssklower 		} else if (strcmp(*av, "intercept")==0) {
9844600Ssklower 			intercept++;
9944323Ssklower 		}
10044323Ssklower 	}
10144323Ssklower 	if (Servername) {
10244323Ssklower 		int tlen = laddr.siso_tlen = strlen(Servername);
10344323Ssklower 		int len =  TSEL(siso) + tlen - (caddr_t) &siso;
10444323Ssklower 		if (len > sizeof(*siso)) {
10544323Ssklower 			siso = (struct sockaddr_iso *)malloc(len);
10644323Ssklower 			*siso = laddr;
10744323Ssklower 			siso->siso_len = len;
10844323Ssklower 		}
10944323Ssklower 		bcopy(Servername, TSEL(siso), tlen);
11044323Ssklower 	} else {
11144323Ssklower 		port = htons(port);
11244323Ssklower 		laddr.siso_tlen = sizeof(port);
11344323Ssklower 		bcopy((char *)&port, TSEL(siso), sizeof(port));
11444323Ssklower 	}
11544323Ssklower 	tisink();
11644323Ssklower }
11744323Ssklower #define BIG 2048
11844323Ssklower #define MIDLIN 512
11944323Ssklower char readbuf[BIG];
12044323Ssklower struct iovec iov[1] = {
12144323Ssklower 	readbuf,
12244323Ssklower 	sizeof readbuf,
12344323Ssklower };
12444323Ssklower char name[MIDLIN];
12544323Ssklower union {
12644323Ssklower     struct {
12744323Ssklower 	    struct cmsghdr	cmhdr;
12844323Ssklower 	    char		cmdata[128 - sizeof(struct cmsghdr)];
12944323Ssklower     } cm;
13044323Ssklower     char data[128];
13144323Ssklower } cbuf;
13244323Ssklower #define control cbuf.data
13344323Ssklower struct msghdr msghdr = {
13444323Ssklower 	name, sizeof(name),
13544323Ssklower 	iov, sizeof(iov)/sizeof(iov[1]),
13644323Ssklower 	control, sizeof control,
13744323Ssklower 	0 /* flags */
13844323Ssklower };
13944323Ssklower 
tisink()14044323Ssklower tisink()
14144323Ssklower {
14248712Ssklower 	int x, s, pid, on = 1, loop = 0, n, ns;
14344323Ssklower 	extern int errno;
144*58241Ssklower 	int socktype = (dgramp ? SOCK_DGRAM :
145*58241Ssklower 			(tuba ? SOCK_STREAM :SOCK_SEQPACKET));
146*58241Ssklower 	int proto = (tp0mode ? ISOPROTO_TP0 : (tuba ? ISOPROTO_TCP : 0 ));
14748712Ssklower 	int addrlen = sizeof(faddr);
14844323Ssklower 
14950515Ssklower 	try(socket, (AF_ISO, socktype, proto),"");
15044323Ssklower 
15144323Ssklower 	s = x;
15244323Ssklower 
15344323Ssklower 	try(bind, (s, (struct sockaddr *) siso, siso->siso_len), "");
15444323Ssklower 
15544323Ssklower 	/*try(setsockopt, (s, SOL_SOCKET, SO_DEBUG, &on, sizeof (on)), ""); */
15648712Ssklower 	if (dgramp) {
15748712Ssklower 		ns  =  s;
15848712Ssklower 		goto dgram1;
15948712Ssklower 	}
16044323Ssklower 
16144323Ssklower 	try(listen, (s, 5), "");
16244424Ssklower 	if (intercept) {
16344424Ssklower 	    try(setsockopt,
16444424Ssklower 		(s, SOL_TRANSPORT, TPOPT_INTERCEPT, &on, sizeof(on)), "");
16544424Ssklower 	}
16644323Ssklower 	for(;;) {
16748712Ssklower 		int child;
16844323Ssklower 		char childname[50];
16944323Ssklower 
17050515Ssklower 		try (accept, (s, (struct sockaddr *)&faddr, &addrlen), "");
17144323Ssklower 		ns = x;
17244323Ssklower 		dumpit("connection from:", &faddr, sizeof faddr);
17344600Ssklower 		if (mynamep || intercept) {
17444323Ssklower 			addrlen = sizeof(faddr);
17550515Ssklower 			try (getsockname,
17650515Ssklower 			      (ns, (struct sockaddr *)&faddr, &addrlen), "");
17744323Ssklower 			dumpit("connected as:", &faddr, addrlen);
17844323Ssklower 		}
17944323Ssklower 		loop++;
18048712Ssklower 		if(loop > 3) myexit(0);
18144323Ssklower 		if (forkp) {
18244323Ssklower 			try(fork, (), "");
18344323Ssklower 		} else
18444323Ssklower 			x = 0;
18544323Ssklower 		if (x == 0)  {
18644323Ssklower 		    long n, count = 0, cn, flags;
18744323Ssklower 		    records = 0;
18844323Ssklower 		    if (confp) {
18944323Ssklower 			msghdr.msg_iovlen = 0;
19044323Ssklower 			msghdr.msg_namelen = 0;
19144323Ssklower 			msghdr.msg_controllen =
19244323Ssklower 			    cbuf.cm.cmhdr.cmsg_len = sizeof (cbuf.cm.cmhdr);
19344323Ssklower 			cbuf.cm.cmhdr.cmsg_level = SOL_TRANSPORT;
19444323Ssklower 			cbuf.cm.cmhdr.cmsg_type = TPOPT_CFRM_DATA;
19544323Ssklower 			n = sendmsg(ns, &msghdr, 0);
19648712Ssklower 			if (n < 0) {
19744323Ssklower 				printf("confirm: errno is %d\n", errno);
19844323Ssklower 				fflush(stdout);
19944323Ssklower 				perror("Confirm error");
20044323Ssklower 			} else {
20144323Ssklower 				dbprintf("confim ok\n");
20244323Ssklower 			}
20355605Ssklower 			sleep(3);
20444323Ssklower 		    }
20548711Ssklower #ifdef ISODE_MODE
20648711Ssklower 		    if (isode_mode) {
20748711Ssklower 			static char fdbuf[10];
20848711Ssklower 			static char *nargv[4] =
20948711Ssklower 			    {"/usr/sbin/isod.tsap", fdbuf, "", 0};
21048711Ssklower 			sprintf(fdbuf, "Z%d", ns);
21148711Ssklower 			old_isod_main(3, nargv, xenvp);
21255605Ssklower 			myexit(0);
21355605Ssklower 		    }
21448711Ssklower #endif
21544323Ssklower 		    for (;;) {
21648712Ssklower 		    dgram1:
21744323Ssklower 			msghdr.msg_iovlen = 1;
21844323Ssklower 			msghdr.msg_controllen = sizeof(control);
21948712Ssklower 			msghdr.msg_namelen = (dgramp ? (sizeof name) : 0);
22044323Ssklower 			iov->iov_len = sizeof(readbuf);
22155605Ssklower 			if (select_mode)
22255605Ssklower 			    sel_recvwait(ns);
22344323Ssklower 			n = recvmsg(ns, &msghdr, 0);
22444323Ssklower 			flags = msghdr.msg_flags;
22544323Ssklower 			count++;
22644323Ssklower 			dbprintf("recvmsg from child %d got %d ctl %d flags %x\n",
22748712Ssklower 				getpid(), n, (cn = msghdr.msg_controllen), flags);
22844323Ssklower 			fflush(stdout);
22948712Ssklower 			if (dgramp && msghdr.msg_namelen && verbose)
23048712Ssklower 				dumpit("from:\n", name, msghdr.msg_namelen);
23144323Ssklower 			if (cn && verbose)
23244323Ssklower 				dumpit("control data:\n", control, cn);
23344323Ssklower 			if (n < 0) {
23444323Ssklower 				fprintf(stderr, "errno is %d\n", errno);
23544323Ssklower 				perror("recvmsg");
23644323Ssklower 				/*sleep (10);*/
23744323Ssklower 				break;
23844323Ssklower 			} else {
23944323Ssklower 				if (verbose)
24044323Ssklower 					dumpit("data:\n", readbuf, n);
24144323Ssklower 			}
24250515Ssklower 			if (echop)
24350515Ssklower 				savedata(n, flags);
24450515Ssklower 			if (flags & MSG_EOR) {
24544323Ssklower 				records++;
24650515Ssklower 				if (echop)
24750515Ssklower 					answerback(ns);
24844323Ssklower 			}
24944323Ssklower 			errno = 0;
25044323Ssklower 		    }
25150890Ssklower 		    myexit(0);
25244323Ssklower 		}
25344323Ssklower 	}
25444323Ssklower }
25550515Ssklower struct savebuf {
25650515Ssklower 	struct savebuf *s_next;
25750515Ssklower 	struct savebuf *s_prev;
25850515Ssklower 	int	s_n;
25950515Ssklower 	int	s_flags;
26050515Ssklower } savebuf = {&savebuf, &savebuf};
26150515Ssklower 
26250515Ssklower void
savedata(n,flags)26350515Ssklower savedata(n, flags)
26450515Ssklower int n, flags;
26544323Ssklower {
26650515Ssklower 	register struct savebuf *s = (struct savebuf *)malloc(n + sizeof *s);
26750515Ssklower 	if (s == 0)
26850515Ssklower 		return;
26950515Ssklower 	insque(s, savebuf.s_prev);
27050515Ssklower 	s->s_n = n;
27150515Ssklower 	s->s_flags = flags;
27250515Ssklower 	bcopy(readbuf, (char *)(s + 1), n);
27344323Ssklower }
27444323Ssklower 
answerback(ns)27550515Ssklower answerback(ns)
27650515Ssklower {
27750515Ssklower 	int n;
27850515Ssklower 	register struct savebuf *s = savebuf.s_next, *t;
27950515Ssklower 	static struct iovec iov[1];
28050515Ssklower 	static struct msghdr msghdr = { 0, 0, iov, 1, 0, 0, 0};
28150515Ssklower 	while (s != &savebuf) {
28250515Ssklower 		iov->iov_len = s->s_n;
28350515Ssklower 		iov->iov_base = (char *)(s + 1);
28450515Ssklower 		n = sendmsg(ns, &msghdr, s->s_flags);
28550515Ssklower 		dbprintf("echoed %d\n", n);
28650515Ssklower 		t = s; s = s->s_next; remque(t); free((char *)t);
28750515Ssklower 	}
28850515Ssklower }
28950515Ssklower 
dumpit(what,where,n)29044323Ssklower dumpit(what, where, n)
29144323Ssklower char *what; unsigned short *where; int n;
29244323Ssklower {
29344323Ssklower 	unsigned short *s = where;
29444323Ssklower 	unsigned short *z = where + (n+1)/2;
29544323Ssklower 	int count = 0;
29650890Ssklower 	if (dumpnodata)
29750890Ssklower 		return;
29844323Ssklower 	printf(what);
29944323Ssklower 	while(s < z) {
30044323Ssklower 		count++;
30144323Ssklower 		printf("%x ",*s++);
30244323Ssklower 		if ((count & 15) == 0)
30344323Ssklower 			putchar('\n');
30444323Ssklower 	}
30544323Ssklower 	if (count & 15)
30644323Ssklower 		putchar('\n');
30744323Ssklower 	fflush(stdout);
30844323Ssklower }
myexit(n)30944323Ssklower myexit(n)
31044323Ssklower {
31144323Ssklower 	fflush(stderr);
31244323Ssklower 	printf("got %d records\n", records);
31344323Ssklower 	fflush(stdout);
31444323Ssklower 	exit(n);
31544323Ssklower }
31655605Ssklower 
sel_recvwait(fd)31755605Ssklower sel_recvwait(fd)
31855605Ssklower int fd;
31955605Ssklower {
32055605Ssklower 	int x;
32155605Ssklower 	do {
32255605Ssklower 		FD_ZERO(&readfds);
32355605Ssklower 		FD_ZERO(&exceptfds);
32455605Ssklower 		FD_SET(fd, &readfds);
32555605Ssklower 		FD_SET(fd, &exceptfds);
32655605Ssklower 		x = select(fd+1, &readfds, (fd_set *)0, &exceptfds, (void *)0);
32755605Ssklower 		dbprintf("select returns %d\n", x);
32855605Ssklower 	} while (x <= 0 ||
32955605Ssklower 		 (FD_ISSET(fd,&readfds) == 0 && FD_ISSET(fd,&exceptfds) == 0));
33055605Ssklower }
33155605Ssklower 
33255605Ssklower #include <sys/syscall.h>
33355605Ssklower /* Here for gdb trapping */
setsockopt(s,level,optname,optval,optlen)33455605Ssklower setsockopt(s, level, optname, optval, optlen)
33555605Ssklower int s, level, optname, optlen;
33655605Ssklower const void *optval;
33755605Ssklower {
33855605Ssklower 
33955605Ssklower 	dbprintf("setsocket called s %d, level 0x%x, optname %d, optlen %d\n",
34055605Ssklower 			s, level, optname, optlen);
34155605Ssklower 	dumpit("", optval, optlen);
34255605Ssklower 	return syscall(SYS_setsockopt, s, level, optname, optval, optlen);
34355605Ssklower }
344