/* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */ #include #include #include #include #include #include #include #include #include #include #include #include T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true)); static bool debug; static int sock_open_common(int pf, int type) { int s; s = socket(pf, type, 0); T_QUIET; T_ASSERT_POSIX_SUCCESS(s, "socket(%d, %d, 0)", pf, type); return s; } static int sock_open(int type) { return sock_open_common(PF_INET, type); } static int sock_bind(int s, int port) { struct sockaddr_in sin = { .sin_len = sizeof(sin), .sin_family = AF_INET, }; sin.sin_port = htons(port); return bind(s, (const struct sockaddr *)&sin, sizeof(sin)); } static int sockv6_open(int type) { return sock_open_common(PF_INET6, type); } static int sockv6_bind(int s, int port) { struct sockaddr_in6 sin6 = { .sin6_len = sizeof(sin6), .sin6_family = AF_INET6, }; sin6.sin6_port = htons(port); return bind(s, (const struct sockaddr *)&sin6, sizeof(sin6)); } static uint16_t sock_get_port(int sockfd) { int error; uint16_t p; union sockaddr_in_4_6 sin; socklen_t sin_len; sin_len = sizeof(sin); bzero(&sin, sin_len); error = getsockname(sockfd, (struct sockaddr *)&sin, &sin_len); T_QUIET; T_EXPECT_POSIX_ZERO(error, "getsockname(%d)", sockfd); if (error != 0) { return 0; } switch (sin.sa.sa_family) { case AF_INET: p = sin.sin.sin_port; break; case AF_INET6: p = sin.sin6.sin6_port; break; default: T_ASSERT_FAIL("unknown address family %d\n", sin.sa.sa_family); p = 0; break; } return p; } typedef struct { bool v6; int socket_count; int * socket_list; } SocketInfo, * SocketInfoRef; static void bind_sockets(SocketInfoRef info, const char * msg) { for (int i = 0; i < info->socket_count; i++) { int error; uint16_t port; if (info->v6) { error = sockv6_bind(info->socket_list[i], 0); } else { error = sock_bind(info->socket_list[i], 0); } port = sock_get_port(info->socket_list[i]); if (debug) { T_LOG( "%s: fd %d port is %d error %d", msg, info->socket_list[i], ntohs(port), error); } } return; } static void * second_thread(void * arg) { SocketInfoRef info = (SocketInfoRef)arg; bind_sockets(info, "second"); return NULL; } static void multithreaded_bind_test(bool v6, int socket_count) { int error; SocketInfo info; int socket_list[socket_count]; pthread_t thread; info.v6 = v6; for (int i = 0; i < socket_count; i++) { if (v6) { socket_list[i] = sockv6_open(SOCK_STREAM); } else { socket_list[i] = sock_open(SOCK_STREAM); } } info.socket_count = socket_count; info.socket_list = socket_list; error = pthread_create(&thread, NULL, second_thread, &info); T_QUIET; T_ASSERT_POSIX_ZERO(error, "pthread_create"); /* compete with second thread */ bind_sockets(&info, "main"); error = pthread_join(thread, NULL); T_QUIET; T_ASSERT_POSIX_ZERO(error, "pthread_join"); for (int i = 0; i < socket_count; i++) { error = close(socket_list[i]); T_QUIET; T_ASSERT_POSIX_ZERO(error, "close socket %d", socket_list[i]); } } static void run_multithreaded_bind_test(int number_of_runs, bool v6, int socket_count) { #if TARGET_OS_BRIDGE T_SKIP("Not enough memory to handle this test"); #else /* TARGET_OS_BRIDGE */ for (int i = 0; i < number_of_runs; i++) { multithreaded_bind_test(v6, socket_count); } T_PASS("multithreaded_bind_test %s", v6 ? "IPv6" : "IPv4"); #endif /* TARGET_OS_BRIDGE */ } T_DECL(socket_bind_35685803, "multithreaded bind IPv4 socket as root", T_META_ASROOT(false), T_META_CHECK_LEAKS(false), T_META_TAG_VM_PREFERRED) { run_multithreaded_bind_test(100, false, 100); } T_DECL(socket_bind_35685803_root, "multithreaded bind IPv4 socket", T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) { run_multithreaded_bind_test(100, false, 100); } T_DECL(socket_bind_35685803_v6, "multithreaded bind IPv6 socket as root", T_META_ASROOT(false), T_META_CHECK_LEAKS(false), T_META_TAG_VM_PREFERRED) { run_multithreaded_bind_test(100, true, 100); } T_DECL(socket_bind_35685803_v6_root, "multithreaded bind IPv6 socket", T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) { run_multithreaded_bind_test(100, true, 100); }