// Copyright 2008 Google Inc. Released under the GPL v2. // // This test performs numerous connects (with auto-binding), to a server // listening on all local addresses using an IPv6 socket, by connecting to // 127.0.0.1, ::ffff:127.0.0.1 and ::1. // // The code is really three tests: // // - RunWithOneServer, using CreateServer and ConnectAndAccept, // uses one server socket and repeatedly connects to it. // // - RunWithOneShotServers, using CreateServerConnectAndAccept, // creates servers, connects to them and then discards them. // // - RunMultiThreaded, using ThreadedCreateServerConnectAndAccept, // ThreadedStartServer and ThreadedGetServerFD, is equivalent to // RunWithOneShotServers but uses multiple threads, one for the // server and one for the client. // // Each of these tests triggers error conditions on different kernels // to a different extent. #include <arpa/inet.h> #include <netinet/in.h> #include <pthread.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <sys/time.h> #include <time.h> #include <unistd.h> // Which loopback address to connect to. enum LoopbackAddr { V4_LOOPBACK, V6_LOOPBACK, V6_MAPPED_V4_LOOPBACK }; // Connect to a listening TCP socket, and accept the connection. static void ConnectAndAccept(enum LoopbackAddr addr, int server_fd, int port) { struct sockaddr_in6 sa; socklen_t addr_len; int client_fd, accepted_fd; if (addr == V6_LOOPBACK || addr == V6_MAPPED_V4_LOOPBACK) { char buf[INET6_ADDRSTRLEN]; memset(&sa, 0, sizeof(sa)); if ((client_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } if (addr == V6_LOOPBACK) { inet_pton(AF_INET6, "::1", &sa.sin6_addr); } else if (addr == V6_MAPPED_V4_LOOPBACK) { inet_pton(AF_INET6, "::ffff:127.0.0.1", &sa.sin6_addr); } if (!inet_ntop(AF_INET6, &sa.sin6_addr, buf, INET6_ADDRSTRLEN)) { perror("inet_ntop"); exit(1); } addr_len = sizeof(sa); sa.sin6_family = AF_INET6; sa.sin6_port = port; if (connect(client_fd, (struct sockaddr*)(&sa), sizeof(struct sockaddr_in6)) == -1) { perror("connect"); exit(1); } write(2, (addr == V6_LOOPBACK) ? "+" : "-", 1); } else { struct sockaddr_in sa4; if ((client_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } memset(&sa4, 0, sizeof(sa4)); sa4.sin_family = AF_INET; inet_pton(AF_INET, "127.0.0.1", &sa4.sin_addr); sa4.sin_port = port; if (connect(client_fd, (struct sockaddr*)(&sa4), sizeof(struct sockaddr_in)) == -1) { perror("connect"); exit(1); } write(2, ".", 1); } addr_len = sizeof(sa); if ((accepted_fd = accept(server_fd, (struct sockaddr*)(&sa), &addr_len)) == -1) { perror("accept"); exit(1); } close(client_fd); close(accepted_fd); } // Create a listening TCP socket. static void CreateServer(int* server_fd, int* port) { struct sockaddr_in6 sa; socklen_t addr_len; memset(&sa, 0, sizeof(sa)); if ((*server_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } addr_len = sizeof(sa); sa.sin6_family = AF_INET6; sa.sin6_addr = in6addr_any; sa.sin6_port = 0; if (bind(*server_fd, (struct sockaddr*)(&sa), sizeof(sa)) == -1) { perror("bind"); exit(1); } if (getsockname(*server_fd, (struct sockaddr*)(&sa), &addr_len) == -1) { perror("getsockname"); exit(1); } if (listen(*server_fd, 10) == -1) { perror("listen"); exit(1); } *port = sa.sin6_port; } // Create a socket, connect to it, accept, and discard both. static void CreateServerConnectAndAccept(enum LoopbackAddr addr) { struct sockaddr_in6 sa; socklen_t addr_len; int server_fd, client_fd, accepted_fd, connect_rc; if ((server_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } addr_len = sizeof(sa); memset(&sa, 0, sizeof(sa)); sa.sin6_family = AF_INET6; sa.sin6_addr = in6addr_any; sa.sin6_port = 0; if (bind(server_fd, (struct sockaddr*)(&sa), sizeof(sa)) == -1) { perror("bind"); exit(1); } if (getsockname(server_fd, (struct sockaddr*)(&sa), &addr_len) == -1) { perror("getsockname"); exit(1); } if (listen(server_fd, 10) == -1) { perror("listen"); exit(1); } if (addr == V6_LOOPBACK || addr == V6_MAPPED_V4_LOOPBACK) { char buf[INET6_ADDRSTRLEN]; if ((client_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } if (addr == V6_LOOPBACK) { inet_pton(AF_INET6, "::1", &sa.sin6_addr); } else if (addr == V6_MAPPED_V4_LOOPBACK) { inet_pton(AF_INET6, "::ffff:127.0.0.1", &sa.sin6_addr); } if (!inet_ntop(AF_INET6, &sa.sin6_addr, buf, INET6_ADDRSTRLEN)) { perror("inet_ntop"); exit(1); } connect_rc = connect(client_fd, (struct sockaddr*)(&sa), sizeof(struct sockaddr_in6)); write(2, (addr == V6_MAPPED_V4_LOOPBACK) ? "-" : "+", 1); } else { struct sockaddr_in sa4; if ((client_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } memset(&sa4, 0, sizeof(sa4)); sa4.sin_family = AF_INET; inet_pton(AF_INET, "127.0.0.1", &sa4.sin_addr); sa4.sin_port = sa.sin6_port; connect_rc = connect(client_fd, (struct sockaddr*)(&sa4), sizeof(struct sockaddr_in)); write(2, ".", 1); } if (connect_rc == -1) { perror("connect"); exit(1); } addr_len = sizeof(sa); if ((accepted_fd = accept(server_fd, (struct sockaddr*)(&sa), &addr_len)) == -1) { perror("accept"); exit(1); } close(accepted_fd); close(client_fd); close(server_fd); } // Globals for threaded version. static volatile int threaded_listening = 0; static int threaded_server_fd; static pthread_mutex_t threaded_mutex = PTHREAD_MUTEX_INITIALIZER; static pthread_cond_t threaded_cond = PTHREAD_COND_INITIALIZER; // Block until listening, then return server address. static int ThreadedGetServerFD() { pthread_mutex_lock(&threaded_mutex); while (!threaded_listening) { pthread_cond_wait(&threaded_cond, &threaded_mutex); } pthread_mutex_unlock(&threaded_mutex); return threaded_server_fd; } // Start a server which accepts one connection. static void* ThreadedStartServer(void* unused) { struct sockaddr_in6 sa; socklen_t addr_len = sizeof(sa); int accept_fd; if ((threaded_server_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } // Any IP, unused port. memset(&sa, 0, sizeof(sa)); sa.sin6_family = AF_INET6; sa.sin6_addr = in6addr_any; sa.sin6_port = 0; // Bind. if (bind(threaded_server_fd, (struct sockaddr*)(&sa), sizeof(sa)) == -1) { perror("bind"); exit(1); } // Listen. if (listen(threaded_server_fd, 10) == -1) { perror("listen"); exit(1); } pthread_mutex_lock(&threaded_mutex); threaded_listening = 1; pthread_cond_signal(&threaded_cond); pthread_mutex_unlock(&threaded_mutex); // Try to accept. if ((accept_fd = accept(threaded_server_fd, (struct sockaddr*)(&sa), &addr_len)) == -1) { perror("accept"); exit(1); } // All done. close(threaded_server_fd); close(accept_fd); threaded_listening = 0; return NULL; } // Start a server thread, then connect to it via TCP. static void ThreadedCreateServerConnectAndAccept(enum LoopbackAddr addr) { pthread_t pthread; int server_fd, client_fd; struct sockaddr_in6 sa; socklen_t addr_len = sizeof(sa); pthread_create(&pthread, NULL, ThreadedStartServer, NULL); // Get the server address information -- this call will block until // the server is listening. server_fd = ThreadedGetServerFD(); memset(&sa, 0, sizeof(sa)); if (getsockname(server_fd, (struct sockaddr*)(&sa), &addr_len) == -1) { perror("getsockname"); exit(1); } if (addr == V6_LOOPBACK || addr == V6_MAPPED_V4_LOOPBACK) { char buf[INET6_ADDRSTRLEN]; if ((client_fd = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } // Check that we are listening on :: if (!inet_ntop(AF_INET6, &sa.sin6_addr, buf, INET6_ADDRSTRLEN)) { fprintf(stderr, "inet_ntop failed\n"); exit(1); } if (strlen(buf) != 2) { fprintf(stderr, "Expected to listen on ::, instead listening on %s", buf); exit(1); } if (addr == V6_LOOPBACK) { inet_pton(AF_INET6, "::1", &sa.sin6_addr); } else if (addr == V6_MAPPED_V4_LOOPBACK) { inet_pton(AF_INET6, "::ffff:127.0.0.1", &sa.sin6_addr); } if (connect(client_fd, (struct sockaddr*)(&sa), sizeof(struct sockaddr_in6)) == -1) { perror("connect"); exit(1); } } else { struct sockaddr_in sa4; if ((client_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1) { perror("socket"); exit(1); } memset(&sa4, 0, sizeof(sa4)); sa4.sin_family = AF_INET; inet_aton("127.0.0.1", &sa4.sin_addr); sa4.sin_port = sa.sin6_port; if (connect(client_fd, (struct sockaddr*)(&sa4), sizeof(struct sockaddr_in)) == -1) { perror("connect"); exit(1); } } // Update progress. switch (addr) { case V4_LOOPBACK: write(2, ".", 1); break; case V6_MAPPED_V4_LOOPBACK: write(2, "-", 1); break; case V6_LOOPBACK: write(2, "+", 1); break; } // Close our connection and wait for the server thread to shutdown. close(client_fd); pthread_join(pthread, NULL); } static void RunWithOneServer(int outer, int inner) { int i, j, server_fd, port; fprintf(stderr, "Starting test with one server port for all connects\n"); for (i = 0; i < outer; ++i) { CreateServer(&server_fd, &port); for (j = 0; j < inner; ++j) { ConnectAndAccept(V4_LOOPBACK, server_fd, port); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { ConnectAndAccept(V6_MAPPED_V4_LOOPBACK, server_fd, port); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { ConnectAndAccept(V6_LOOPBACK, server_fd, port); } write(2, "\n", 1); close(server_fd); } } static void RunWithOneShotServers(int outer, int inner) { int i, j; fprintf(stderr, "Starting test with one server port per connect\n"); for (i = 0; i < outer; ++i) { for (j = 0; j < inner; ++j) { CreateServerConnectAndAccept(V4_LOOPBACK); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { CreateServerConnectAndAccept(V6_MAPPED_V4_LOOPBACK); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { CreateServerConnectAndAccept(V6_LOOPBACK); } write(2, "\n", 1); } } static void RunMultiThreaded(int outer, int inner) { int i, j; fprintf(stderr, "Starting multi-threaded test\n"); for (i = 0; i < outer; ++i) { for (j = 0; j < inner; ++j) { ThreadedCreateServerConnectAndAccept(V4_LOOPBACK); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { ThreadedCreateServerConnectAndAccept(V6_MAPPED_V4_LOOPBACK); } write(2, "\n", 1); for (j = 0; j < inner; ++j) { ThreadedCreateServerConnectAndAccept(V6_LOOPBACK); } write(2, "\n", 1); } } static const char* usage = "Usage: %s [types [outer [inner]]]\n" "Arguments:\n" "\ttypes: String consisting of [OMT], for the test types to run\n" "\t O: One server, multiple connects\n" "\t M: One server per connect (multiple server ports)\n" "\t T: Multi-threaded version of \'M\'\n" "\touter: Number of passes through the outer loops, default 10\n" "\tinner: Number of passes through the inner loops, default 75\n"; static void Usage(char *argv0) { fprintf(stderr, usage, argv0); exit(2); } int main(int argc, char** argv) { char *types = "OMT"; int i, inner = 75, outer = 10, timediff; struct timeval tv0, tv1; // Parse the options. if (argc == 4) { inner = atoi(argv[3]); if (inner <= 0) { Usage(argv[0]); } argc--; } if (argc == 3) { outer = atoi(argv[2]); if (outer <= 0) { Usage(argv[0]); } argc--; } if (argc == 2) { types = argv[1]; if (strspn(types, "OMT") != strlen(types)) { Usage(argv[0]); } argc--; } if (argc != 1) { Usage(argv[0]); } // Run the tests. gettimeofday(&tv0, NULL); for (i = 0; i < strlen(types); ++i) { switch (types[i]) { case 'O': RunWithOneServer(outer, inner); break; case 'M': RunWithOneShotServers(outer, inner); break; case 'T': RunMultiThreaded(outer, inner); break; } } gettimeofday(&tv1, NULL); timediff = (tv1.tv_sec - tv0.tv_sec) * 1000000 + tv1.tv_usec - tv0.tv_usec; fprintf(stderr, "Total time = %d.%06ds\n", timediff / 1000000, timediff % 1000000); exit(0); }