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