xref: /freebsd/tools/test/stress2/testcases/tcp/tcp.c (revision 9a6201ac2ff2e6a15816c7d6efbccaa803eb122b)
18a272653SPeter Holm /*-
28a272653SPeter Holm  * Copyright (c) 2008 Peter Holm <pho@FreeBSD.org>
38a272653SPeter Holm  * All rights reserved.
48a272653SPeter Holm  *
58a272653SPeter Holm  * Redistribution and use in source and binary forms, with or without
68a272653SPeter Holm  * modification, are permitted provided that the following conditions
78a272653SPeter Holm  * are met:
88a272653SPeter Holm  * 1. Redistributions of source code must retain the above copyright
98a272653SPeter Holm  *    notice, this list of conditions and the following disclaimer.
108a272653SPeter Holm  * 2. Redistributions in binary form must reproduce the above copyright
118a272653SPeter Holm  *    notice, this list of conditions and the following disclaimer in the
128a272653SPeter Holm  *    documentation and/or other materials provided with the distribution.
138a272653SPeter Holm  *
148a272653SPeter Holm  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
158a272653SPeter Holm  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
168a272653SPeter Holm  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
178a272653SPeter Holm  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
188a272653SPeter Holm  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
198a272653SPeter Holm  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
208a272653SPeter Holm  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
218a272653SPeter Holm  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
228a272653SPeter Holm  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
238a272653SPeter Holm  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
248a272653SPeter Holm  * SUCH DAMAGE.
258a272653SPeter Holm  *
268a272653SPeter Holm  */
278a272653SPeter Holm 
288a272653SPeter Holm #include <sys/param.h>
298a272653SPeter Holm #include <sys/socket.h>
308a272653SPeter Holm #include <sys/wait.h>
318a272653SPeter Holm 
328a272653SPeter Holm #include <netinet/in.h>
338a272653SPeter Holm 
348a272653SPeter Holm #include <err.h>
358a272653SPeter Holm #include <errno.h>
368a272653SPeter Holm #include <netdb.h>
378a272653SPeter Holm #include <signal.h>
388a272653SPeter Holm #include <stdio.h>
398a272653SPeter Holm #include <stdlib.h>
408a272653SPeter Holm #include <string.h>
418a272653SPeter Holm #include <unistd.h>
428a272653SPeter Holm 
438a272653SPeter Holm #include "stress.h"
448a272653SPeter Holm 
458a272653SPeter Holm #define NB (400 * 1024 * 1024)
468a272653SPeter Holm 
478a272653SPeter Holm static int port;
488a272653SPeter Holm static int bufsize;
498a272653SPeter Holm 
508a272653SPeter Holm static void
reader(void)518a272653SPeter Holm reader(void) {
528a272653SPeter Holm 	struct sockaddr_in inetaddr, inetpeer;
538a272653SPeter Holm 	socklen_t len;
548a272653SPeter Holm 	int on;
55*9a6201acSPeter Holm 	int n, *buf;
568a272653SPeter Holm 	int tcpsock, msgsock;
578a272653SPeter Holm 
588a272653SPeter Holm 	alarm(op->run_time + 30);
598a272653SPeter Holm 	on = 1;
608a272653SPeter Holm 	if ((tcpsock = socket(AF_INET, SOCK_STREAM, 0)) < 0)
618a272653SPeter Holm 		err(1, "socket(), %s:%d", __FILE__, __LINE__);
628a272653SPeter Holm 
638a272653SPeter Holm 	if (setsockopt(tcpsock,
648a272653SPeter Holm 	    SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on)) < 0)
658a272653SPeter Holm 		err(1, "setsockopt(), %s:%d", __FILE__, __LINE__);
668a272653SPeter Holm 
678a272653SPeter Holm 	inetaddr.sin_family = AF_INET;
688a272653SPeter Holm 	inetaddr.sin_addr.s_addr = INADDR_ANY;
698a272653SPeter Holm 	inetaddr.sin_port = htons(port);
708a272653SPeter Holm 	inetaddr.sin_len = sizeof(inetaddr);
718a272653SPeter Holm 
728a272653SPeter Holm 	if (bind(tcpsock,
738a272653SPeter Holm 	    (struct sockaddr *)&inetaddr, sizeof (inetaddr)) < 0)
748a272653SPeter Holm 		err(1, "bind(), %s:%d", __FILE__, __LINE__);
758a272653SPeter Holm 
768a272653SPeter Holm 	if (listen(tcpsock, 5) < 0)
778a272653SPeter Holm 		err(1, "listen(), %s:%d", __FILE__, __LINE__);
788a272653SPeter Holm 
798a272653SPeter Holm 	if ((random_int(1,100) > 60) || (op->hog == 1)) {
808a272653SPeter Holm 		usleep(random_int(1000000,1000000) * 60);
818a272653SPeter Holm 	}
828a272653SPeter Holm 
838a272653SPeter Holm 	len = sizeof(inetpeer);
848a272653SPeter Holm 	if ((msgsock = accept(tcpsock,
858a272653SPeter Holm 	    (struct sockaddr *)&inetpeer, &len)) < 0)
868a272653SPeter Holm 		err(1, "accept(), %s:%d", __FILE__, __LINE__);
878a272653SPeter Holm 
888a272653SPeter Holm 	if ((buf = malloc(bufsize)) == NULL)
898a272653SPeter Holm 			err(1, "malloc(%d), %s:%d", bufsize, __FILE__, __LINE__);
908a272653SPeter Holm 	while (done_testing == 0) {
918a272653SPeter Holm 		if ((n = read(msgsock, buf, bufsize)) < 0)
928a272653SPeter Holm 			err(1, "read(), %s:%d", __FILE__, __LINE__);
938a272653SPeter Holm 		if (n == 0) break;
948a272653SPeter Holm 	}
958a272653SPeter Holm 	close(msgsock);
968a272653SPeter Holm 	return;
978a272653SPeter Holm }
988a272653SPeter Holm 
998a272653SPeter Holm static void
writer(void)1008a272653SPeter Holm writer(void) {
1018a272653SPeter Holm 	struct sockaddr_in inetaddr;
1028a272653SPeter Holm 	struct hostent *hostent;
1038a272653SPeter Holm 	int i, *buf, r;
1048a272653SPeter Holm 	int tcpsock, on;
1058a272653SPeter Holm 
1068a272653SPeter Holm 	alarm(op->run_time + 30);
1078a272653SPeter Holm 	on = 1;
1088a272653SPeter Holm 	for (i = 1; i < 5; i++) {
1098a272653SPeter Holm 		if ((tcpsock = socket(AF_INET, SOCK_STREAM, 0)) < 0)
1108a272653SPeter Holm 			err(1, "socket(), %s:%d", __FILE__, __LINE__);
1118a272653SPeter Holm 
1128a272653SPeter Holm 		if (setsockopt(tcpsock,
1138a272653SPeter Holm 		    SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on)) < 0)
1148a272653SPeter Holm 			err(1, "setsockopt(), %s:%d", __FILE__, __LINE__);
1158a272653SPeter Holm 
1168a272653SPeter Holm 		hostent = gethostbyname ("localhost");
1178a272653SPeter Holm 		bzero((char *) &inetaddr, sizeof(inetaddr));
1188a272653SPeter Holm 		memcpy (&inetaddr.sin_addr.s_addr, hostent->h_addr,
1198a272653SPeter Holm 			sizeof (struct in_addr));
1208a272653SPeter Holm 
1218a272653SPeter Holm 		inetaddr.sin_family = AF_INET;
1228a272653SPeter Holm 		inetaddr.sin_port = htons(port);
1238a272653SPeter Holm 		inetaddr.sin_len = sizeof(inetaddr);
1248a272653SPeter Holm 
1258a272653SPeter Holm 		r = connect(tcpsock, (struct sockaddr *) &inetaddr,
1268a272653SPeter Holm 			sizeof(inetaddr));
1278a272653SPeter Holm 		if (r == 0)
1288a272653SPeter Holm 			break;
1298a272653SPeter Holm 		sleep(1);
1308a272653SPeter Holm 		close(tcpsock);
1318a272653SPeter Holm 	}
1328a272653SPeter Holm 	if (r < 0)
1338a272653SPeter Holm 		err(1, "connect(), %s:%d", __FILE__, __LINE__);
1348a272653SPeter Holm 
1358a272653SPeter Holm 	if ((buf = malloc(bufsize)) == NULL)
1368a272653SPeter Holm 			err(1, "malloc(%d), %s:%d", bufsize, __FILE__, __LINE__);
1378a272653SPeter Holm 	for (i = 0; i < bufsize / (int)sizeof(int); i++)
1388a272653SPeter Holm 		buf[i] = i;
1398a272653SPeter Holm 
1408a272653SPeter Holm 	for (;;) {
1418a272653SPeter Holm 		for (i = 0; i < NB; i+= bufsize) {
1428a272653SPeter Holm 			if (write(tcpsock, buf, bufsize) < 0) {
1438a272653SPeter Holm 				if (errno == EPIPE)
1448a272653SPeter Holm 					return;
1458a272653SPeter Holm 				if (errno != ECONNRESET)
1468a272653SPeter Holm 					err(1, "write(%d), %s:%d", tcpsock,
1478a272653SPeter Holm 						__FILE__, __LINE__);
1488a272653SPeter Holm 				_exit(EXIT_SUCCESS);
1498a272653SPeter Holm 			}
1508a272653SPeter Holm 		}
1518a272653SPeter Holm 	}
1528a272653SPeter Holm 	return;
1538a272653SPeter Holm }
1548a272653SPeter Holm 
1558a272653SPeter Holm int
setup(int nb)1568a272653SPeter Holm setup(int nb)
1578a272653SPeter Holm {
1588a272653SPeter Holm 	port = 12340 + nb;
1598a272653SPeter Holm 	bufsize = 2 << random_int(1, 12);
1608a272653SPeter Holm 	return (0);
1618a272653SPeter Holm }
1628a272653SPeter Holm 
1638a272653SPeter Holm void
cleanup(void)1648a272653SPeter Holm cleanup(void)
1658a272653SPeter Holm {
1668a272653SPeter Holm }
1678a272653SPeter Holm 
1688a272653SPeter Holm int
test(void)1698a272653SPeter Holm test(void)
1708a272653SPeter Holm {
1718a272653SPeter Holm 	pid_t pid;
1728a272653SPeter Holm 
1738a272653SPeter Holm 	if ((pid = fork()) == 0) {
1748a272653SPeter Holm 		writer();
1758a272653SPeter Holm 		_exit(EXIT_SUCCESS);
1768a272653SPeter Holm 
1778a272653SPeter Holm 	} else if (pid > 0) {
1788a272653SPeter Holm 		reader();
1798a272653SPeter Holm 		kill(pid, SIGINT);
1808a272653SPeter Holm 		if (waitpid(pid, NULL, 0) != pid)
1818a272653SPeter Holm 			err(1, "waitpid(%d)", pid);
1828a272653SPeter Holm 	} else
1838a272653SPeter Holm 		err(1, "fork(), %s:%d",  __FILE__, __LINE__);
1848a272653SPeter Holm 
1858a272653SPeter Holm 	return (0);
1868a272653SPeter Holm }
187