mirror of
https://github.com/darlinghq/darling-xnu.git
synced 2024-11-23 12:39:55 +00:00
108 lines
2.3 KiB
C
108 lines
2.3 KiB
C
/* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */
|
|
|
|
#include <darwintest.h>
|
|
#include <poll.h>
|
|
#include <sys/socket.h>
|
|
#include <unistd.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <errno.h>
|
|
|
|
static int
|
|
sockv6_open(void)
|
|
{
|
|
int s;
|
|
|
|
s = socket(AF_INET6, SOCK_DGRAM, 0);
|
|
T_QUIET;
|
|
T_ASSERT_POSIX_SUCCESS(s, "socket(AF_INET6, SOCK_DGRAM, 0)");
|
|
return s;
|
|
}
|
|
|
|
static int
|
|
sockv6_bind(int s, in_port_t port)
|
|
{
|
|
struct sockaddr_in6 sin6;
|
|
|
|
bzero(&sin6, sizeof(sin6));
|
|
sin6.sin6_len = sizeof(sin6);
|
|
sin6.sin6_family = AF_INET6;
|
|
sin6.sin6_port = port;
|
|
return bind(s, (const struct sockaddr *)&sin6, sizeof(sin6));
|
|
}
|
|
|
|
static void
|
|
sockv6_set_v6only(int s)
|
|
{
|
|
int on = 1;
|
|
int ret;
|
|
|
|
ret = setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
|
|
T_QUIET;
|
|
T_ASSERT_POSIX_SUCCESS(ret, "setsockopt(%d, IPV6_ONLY)", s);
|
|
}
|
|
|
|
static bool
|
|
alloc_and_bind_ports(in_port_t port_start, in_port_t port_end,
|
|
int bind_attempts)
|
|
{
|
|
int bound_count = 0;
|
|
bool success = true;
|
|
|
|
for (in_port_t i = port_start; success && i <= port_end; i++) {
|
|
int s6 = -1;
|
|
int s6_other = -1;
|
|
int ret;
|
|
|
|
s6 = sockv6_open();
|
|
sockv6_set_v6only(s6);
|
|
if (sockv6_bind(s6, i) != 0) {
|
|
/* find the next available port */
|
|
goto loop_done;
|
|
}
|
|
s6_other = sockv6_open();
|
|
ret = sockv6_bind(s6_other, i);
|
|
T_WITH_ERRNO;
|
|
T_QUIET;
|
|
T_ASSERT_TRUE(ret != 0, "socket %d bind %d", s6_other, i);
|
|
/*
|
|
* After bind fails, try binding to a different port.
|
|
* For non-root user, this will panic without the fix for
|
|
* <rdar://problem/35243417>.
|
|
*/
|
|
if (sockv6_bind(s6_other, i + 1) == 0) {
|
|
bound_count++;
|
|
if (bound_count >= bind_attempts) {
|
|
break;
|
|
}
|
|
}
|
|
loop_done:
|
|
if (s6 >= 0) {
|
|
close(s6);
|
|
}
|
|
if (s6_other >= 0) {
|
|
close(s6_other);
|
|
}
|
|
}
|
|
T_ASSERT_TRUE(bound_count == bind_attempts,
|
|
"number of successful binds %d (out of %d)",
|
|
bound_count, bind_attempts);
|
|
return success;
|
|
}
|
|
|
|
|
|
T_DECL(socket_bind_35243417,
|
|
"bind IPv6 only UDP socket, then bind IPv6 socket.",
|
|
T_META_ASROOT(false),
|
|
T_META_CHECK_LEAKS(false))
|
|
{
|
|
alloc_and_bind_ports(1, 65534, 10);
|
|
}
|
|
|
|
T_DECL(socket_bind_35243417_root,
|
|
"bind IPv6 only UDP socket, then bind IPv6 socket.",
|
|
T_META_ASROOT(true))
|
|
{
|
|
alloc_and_bind_ports(1, 65534, 10);
|
|
}
|