xref: /openbsd-src/regress/lib/libpthread/socket/3/socket3.c (revision d0774bb2f04aa46ef0bdd42b0257c67ce679eb9b)
1*d0774bb2Sfgsch /* $OpenBSD: socket3.c,v 1.4 2005/10/30 23:59:43 fgsch Exp $ */
219108fceSmarc /* PUBLIC DOMAIN Oct 2002 <marc@snafu.org> */
319108fceSmarc 
419108fceSmarc /* Test blocking/non-blocking mode inheritance on accept */
519108fceSmarc 
619108fceSmarc #include <sys/types.h>
719108fceSmarc #include <sys/socket.h>
819108fceSmarc 
919108fceSmarc #include <netinet/in.h>
1019108fceSmarc 
1119108fceSmarc #include <fcntl.h>
1219108fceSmarc #include <poll.h>
1319108fceSmarc #include <pthread.h>
1419108fceSmarc #include <string.h>
1519108fceSmarc #include <unistd.h>
1619108fceSmarc 
1719108fceSmarc #include "test.h"
1819108fceSmarc 
1919108fceSmarc /*
2019108fceSmarc  * connect to the test port passed in arg, then close the connection
2119108fceSmarc  * and return.
2219108fceSmarc  */
23db3296cfSderaadt static void *
sock_connect(void * arg)2419108fceSmarc sock_connect(void *arg)
2519108fceSmarc {
2619108fceSmarc 	struct sockaddr_in sin;
2719108fceSmarc 	int port;
2819108fceSmarc 	int sock;
2919108fceSmarc 
3019108fceSmarc 	SET_NAME("connect");
31*d0774bb2Sfgsch 	port = *(int *)arg;
3219108fceSmarc 	CHECKe(sock = socket(AF_INET, SOCK_STREAM, 0));
3319108fceSmarc 	sin.sin_family = AF_INET;
3419108fceSmarc 	sin.sin_port = htons(port);
3519108fceSmarc 	sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
3619108fceSmarc 	CHECKe(connect(sock, (struct sockaddr *)&sin, sizeof sin));
3719108fceSmarc 	CHECKe(close(sock));
3819108fceSmarc 	return NULL;
3919108fceSmarc }
4019108fceSmarc 
4119108fceSmarc /*
4219108fceSmarc  * listen for a connection, accept it using a non-blocking socket, and
4319108fceSmarc  * verify that the blocking mode of the socket returned from accept is
4419108fceSmarc  * also non-blocking
4519108fceSmarc  */
46db3296cfSderaadt static void *
sock_accept(void * arg)4719108fceSmarc sock_accept(void *arg)
4819108fceSmarc {
4919108fceSmarc 	pthread_t connect_thread;
5019108fceSmarc 	struct pollfd fds;
5119108fceSmarc 	struct sockaddr_in sa;
5219108fceSmarc 	struct sockaddr accept_sa;
5319108fceSmarc 	int accept_fd;
5419108fceSmarc 	int accept_sa_size;
5519108fceSmarc 	int flags;
5619108fceSmarc 	int listen_fd;
5719108fceSmarc 	int port;
5819108fceSmarc 
5919108fceSmarc 	SET_NAME("accept");
6019108fceSmarc 
6119108fceSmarc 	/* listen for a connection */
6219108fceSmarc 
6319108fceSmarc 	port = 6543;
6419108fceSmarc 	memset(&sa, 0, sizeof sa);
6519108fceSmarc 	sa.sin_family = AF_INET;
6619108fceSmarc 	sa.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
6719108fceSmarc 	sa.sin_port = htons(port);
6819108fceSmarc 	CHECKe(listen_fd = socket(AF_INET, SOCK_STREAM, 0));
6919108fceSmarc 	printf("listen_fd = %d\n", listen_fd);
7019108fceSmarc 	while (1) {
7119108fceSmarc 		if (bind(listen_fd, (struct sockaddr *)&sa, sizeof(sa)) == 0)
7219108fceSmarc 			break;
7319108fceSmarc 		if (errno == EADDRINUSE) {
7419108fceSmarc 			sa.sin_port = htons(++port);
7519108fceSmarc 			continue;
7619108fceSmarc 		}
7719108fceSmarc 		DIE(errno, "bind");
7819108fceSmarc 	}
7919108fceSmarc 	CHECKe(listen(listen_fd, 2));
8019108fceSmarc 
8119108fceSmarc 	/* Create another thread to connect to the listening socket. */
8219108fceSmarc 	CHECKr(pthread_create(&connect_thread, NULL, sock_connect,
83*d0774bb2Sfgsch 	    (void *)&port));
8419108fceSmarc 
8519108fceSmarc 	/*
8619108fceSmarc 	 * Use poll to check for a pending connection as the socket
8719108fceSmarc 	 * passed to accept will be in non-blocking mode.
8819108fceSmarc 	 */
8919108fceSmarc 	fds.fd = listen_fd;
9019108fceSmarc 	fds.events = POLLIN;
9119108fceSmarc 	CHECKe(poll(&fds, 1, INFTIM));
9219108fceSmarc 
9319108fceSmarc 	/*
9419108fceSmarc 	 * set non blocking mode on the listening socket and close stdin
9519108fceSmarc 	 * (fd 0) so the accept will use fd 0 (needed to test boundary
9619108fceSmarc 	 * condition in the pthread accept code).
9719108fceSmarc 	 */
9819108fceSmarc 	flags = fcntl(listen_fd, F_GETFL);
9919108fceSmarc         CHECKr(fcntl(listen_fd, F_SETFL, flags |= O_NONBLOCK));
10019108fceSmarc 	CHECKe(close(STDIN_FILENO));
10119108fceSmarc 	accept_sa_size = sizeof accept_sa;
10219108fceSmarc 	CHECKe(accept_fd = accept(listen_fd, &accept_sa, &accept_sa_size));
103ef686a3dSmarc 	/* verify O_NONBLOCK on the accepted fd */
10419108fceSmarc 	flags = fcntl(accept_fd, F_GETFL);
10519108fceSmarc 	printf("accept_fd = %d, flags = %x\n", accept_fd, flags);
106ef686a3dSmarc 	ASSERT(flags & O_NONBLOCK);
10719108fceSmarc 	CHECKe(close(listen_fd));
10819108fceSmarc 	CHECKe(close(accept_fd));
10919108fceSmarc 	CHECKr(pthread_join(connect_thread, NULL));
11019108fceSmarc 	return NULL;
11119108fceSmarc }
11219108fceSmarc 
11319108fceSmarc int
main(int argc,char * argv[])11419108fceSmarc main(int argc, char * argv[])
11519108fceSmarc {
11619108fceSmarc 	pthread_t accept_thread;
11719108fceSmarc 
11819108fceSmarc 	CHECKr(pthread_create(&accept_thread, NULL, sock_accept, NULL));
11919108fceSmarc 	CHECKr(pthread_join(accept_thread, NULL));
12019108fceSmarc 	SUCCEED;
12119108fceSmarc }
122