diff options
Diffstat (limited to 'tests/socket')
-rw-r--r-- | tests/socket/socket_sctp_test.c | 234 | ||||
-rw-r--r-- | tests/socket/socket_sctp_test.err | 8 | ||||
-rw-r--r-- | tests/socket/socket_sctp_test.ok | 24 | ||||
-rw-r--r-- | tests/socket/socket_test.c | 347 | ||||
-rw-r--r-- | tests/socket/socket_test.err | 6 | ||||
-rw-r--r-- | tests/socket/socket_test.ok | 29 |
6 files changed, 642 insertions, 6 deletions
diff --git a/tests/socket/socket_sctp_test.c b/tests/socket/socket_sctp_test.c new file mode 100644 index 00000000..5948abc0 --- /dev/null +++ b/tests/socket/socket_sctp_test.c @@ -0,0 +1,234 @@ +/* + * (C) 2017 by Harald Welte <laforge@gnumonks.org> + * All Rights Reserved + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + */ + +#include <stdio.h> +#include <stdlib.h> +#include <unistd.h> +#include <string.h> +#include <fcntl.h> +#include <inttypes.h> + +#include <sys/socket.h> +#include <arpa/inet.h> +#include <netinet/in.h> + +#include <osmocom/core/application.h> +#include <osmocom/core/utils.h> +#include <osmocom/core/socket.h> +#include <osmocom/core/logging.h> +#include <osmocom/core/bits.h> + +#include "config.h" + +void *ctx = NULL; + +#ifdef HAVE_LIBSCTP +static uint16_t sock_get_local_port(int fd, bool is_v6) { + struct sockaddr_storage sa; + struct sockaddr_in *sin = (struct sockaddr_in *)&sa; + struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)&sa; + socklen_t len = sizeof(sa); + int local_port; + + OSMO_ASSERT(getsockname(fd, (struct sockaddr*)&sa, &len) == 0); + if(!is_v6) + local_port = osmo_load16be(&sin->sin_port); + else + local_port = osmo_load16be(&sin6->sin6_port); + //printf("Checking osmo_sock_init2_multiaddr() port: %" PRIu16 "\n", listen_port_v4); + return local_port; +} + +/* Test API osmo_sock_init2_multiaddr with 1 local/remote address */ +static int test_sockinit2_multiaddr(const char **addrv4_loc, const char **addrv6_loc, + const char **addrv4_rem, const char **addrv6_rem, + size_t addrv4_size, size_t addrv6_size) +{ + int fd, rc; + int listen_fd_v4, listen_fd_v6; + int listen_port_v4, listen_port_v6; + + printf("Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv4 port\n"); + + listen_fd_v4 = osmo_sock_init2_multiaddr(AF_INET, SOCK_STREAM, IPPROTO_SCTP, + addrv4_loc, addrv4_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND); + OSMO_ASSERT(listen_fd_v4 >= 0); + /* expect it to be blocking */ + rc = fcntl(listen_fd_v4, F_GETFL); + OSMO_ASSERT(!(rc & O_NONBLOCK)); + + listen_port_v4 = sock_get_local_port(listen_fd_v4, false); + + printf("Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv6 port\n"); + + listen_fd_v6 = osmo_sock_init2_multiaddr(AF_INET6, SOCK_STREAM, IPPROTO_SCTP, + addrv6_loc, addrv6_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND); + OSMO_ASSERT(listen_fd_v6 >= 0); + /* expect it to be blocking */ + rc = fcntl(listen_fd_v6, F_GETFL); + OSMO_ASSERT(!(rc & O_NONBLOCK)); + + listen_port_v6 = sock_get_local_port(listen_fd_v6, true); + + printf("Checking osmo_sock_init2_multiaddr() for OSMO_SOCK_F_NONBLOCK\n"); + fd = osmo_sock_init2_multiaddr(AF_INET, SOCK_STREAM, IPPROTO_SCTP, + addrv4_loc, addrv4_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND|OSMO_SOCK_F_NONBLOCK); + OSMO_ASSERT(fd >= 0); + /* expect it to be blocking */ + rc = fcntl(fd, F_GETFL); + OSMO_ASSERT(rc & O_NONBLOCK); + close(fd); + + printf("Checking osmo_sock_init2_multiaddr() for invalid flags\n"); + fd = osmo_sock_init2_multiaddr(AF_INET, SOCK_STREAM, IPPROTO_SCTP, + addrv4_loc, addrv4_size, 0, + NULL, 0, 0, 0); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init2_multiaddr() for combined BIND + CONNECT\n"); + fd = osmo_sock_init2_multiaddr(AF_INET, SOCK_STREAM, IPPROTO_SCTP, + addrv4_rem, addrv4_size, 0, + addrv4_rem, addrv4_size, listen_port_v4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + close(fd); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv4 & IPv6\n"); + fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addrv4_rem, addrv4_size, 0, + addrv6_rem, addrv6_size, listen_port_v6, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv6 & IPv4\n"); + fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addrv6_rem, addrv6_size, 0, + addrv4_rem, addrv4_size, listen_port_v4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv4\n"); + fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addrv4_rem, addrv4_size, 0, + addrv4_rem, addrv4_size, listen_port_v4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + close(fd); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv6\n"); + fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addrv6_rem, addrv6_size, 0, + addrv6_rem, addrv6_size, listen_port_v6, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + close(fd); + + close(listen_fd_v4); + close(listen_fd_v6); + printf("Done\n"); + return 0; +} + +/* Test API osmo_sock_init2_multiaddr with 1 local/remote address */ +static int test_sockinit2_multiaddr_simple(void) +{ + const char *addrv4_loc[] = { "0.0.0.0" }; + const char *addrv6_loc[] = { "::" }; + const char *addrv4_rem[] = { "127.0.0.1" }; + const char *addrv6_rem[] = { "::1" }; + + return test_sockinit2_multiaddr(addrv4_loc, addrv6_loc, + addrv4_rem, addrv6_rem, 1, 1); +} + +/* Test API osmo_sock_init2_multiaddr with several local/remote address */ +static int test_sockinit2_multiaddr_several(void) +{ + const char *addrv4_localhost[] = { "127.0.0.1", "127.0.0.2" }; + const char *addrv6_localhost[] = { "::1" }; + + return test_sockinit2_multiaddr(addrv4_localhost, addrv6_localhost, + addrv4_localhost, addrv6_localhost, 2, 1); +} + +/* Test API osmo_sock_init2_multiaddr with several local/remote address, using both ipv4+v6 */ +static int test_sockinit2_multiaddr_mixed(void) +{ + const char *addr_localhost[] = { "127.0.0.1", "127.0.0.2", "::1" }; + size_t addr_size = ARRAY_SIZE(addr_localhost); + + int listen_fd, listen_port, fd; + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_INET IPv4+v6 fails\n"); + listen_fd = osmo_sock_init2_multiaddr(AF_INET, SOCK_STREAM, IPPROTO_SCTP, + addr_localhost, addr_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND); + OSMO_ASSERT(listen_fd < 0); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_INET6 IPv4+v6 fails\n"); + listen_fd = osmo_sock_init2_multiaddr(AF_INET6, SOCK_STREAM, IPPROTO_SCTP, + addr_localhost, addr_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND); + OSMO_ASSERT(listen_fd < 0); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_UNSPEC IPv4+v6 succeeds\n"); + listen_fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addr_localhost, addr_size, 0, + NULL, 0, 0, OSMO_SOCK_F_BIND); + OSMO_ASSERT(listen_fd >= 0); + + listen_port = sock_get_local_port(listen_fd, true); + + printf("Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv4\n"); + fd = osmo_sock_init2_multiaddr(AF_UNSPEC, SOCK_STREAM, IPPROTO_SCTP, + addr_localhost, addr_size, 0, + addr_localhost, addr_size, listen_port, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + close(fd); + + close(listen_fd); + return 0; +} +#endif /* ifdef HAVE_LIBSCTP */ + +const struct log_info_cat default_categories[] = { +}; + +static struct log_info info = { + .cat = default_categories, + .num_cat = ARRAY_SIZE(default_categories), +}; + +int main(int argc, char *argv[]) +{ + ctx = talloc_named_const(NULL, 0, "socket_test_sctp"); + osmo_init_logging2(ctx, &info); + log_set_use_color(osmo_stderr_target, 0); + log_set_print_filename2(osmo_stderr_target, LOG_FILENAME_NONE); + log_set_print_category(osmo_stderr_target, 0); + log_set_print_category_hex(osmo_stderr_target, 0); +#ifdef HAVE_LIBSCTP + test_sockinit2_multiaddr_simple(); + test_sockinit2_multiaddr_several(); + test_sockinit2_multiaddr_mixed(); +#endif /* ifdef HAVE_LIBSCTP */ + + return EXIT_SUCCESS; +} diff --git a/tests/socket/socket_sctp_test.err b/tests/socket/socket_sctp_test.err new file mode 100644 index 00000000..7583a2a2 --- /dev/null +++ b/tests/socket/socket_sctp_test.err @@ -0,0 +1,8 @@ +invalid: you have to specify either BIND or CONNECT flags +Invalid v4 vs v6 in local vs remote addresses: local: v4 remote: v6 +Invalid v4 vs v6 in local vs remote addresses: local: v6 remote: v4 +invalid: you have to specify either BIND or CONNECT flags +Invalid v4 vs v6 in local vs remote addresses: local: v4 remote: v6 +Invalid v4 vs v6 in local vs remote addresses: local: v6 remote: v4 +getaddrinfo(::1, 0) failed: Address family for hostname not supported +getaddrinfo(127.0.0.1, 0) failed: Address family for hostname not supported diff --git a/tests/socket/socket_sctp_test.ok b/tests/socket/socket_sctp_test.ok new file mode 100644 index 00000000..7608e889 --- /dev/null +++ b/tests/socket/socket_sctp_test.ok @@ -0,0 +1,24 @@ +Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv4 port +Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv6 port +Checking osmo_sock_init2_multiaddr() for OSMO_SOCK_F_NONBLOCK +Checking osmo_sock_init2_multiaddr() for invalid flags +Checking osmo_sock_init2_multiaddr() for combined BIND + CONNECT +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv4 & IPv6 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv6 & IPv4 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv4 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv6 +Done +Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv4 port +Checking osmo_sock_init2_multiaddr() with bind to a random local SCTP IPv6 port +Checking osmo_sock_init2_multiaddr() for OSMO_SOCK_F_NONBLOCK +Checking osmo_sock_init2_multiaddr() for invalid flags +Checking osmo_sock_init2_multiaddr() for combined BIND + CONNECT +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv4 & IPv6 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) must fail on mixed IPv6 & IPv4 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv4 +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv6 +Done +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_INET IPv4+v6 fails +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_INET6 IPv4+v6 fails +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND on AF_UNSPEC IPv4+v6 succeeds +Checking osmo_sock_init2_multiaddr(AF_UNSPEC) BIND + CONNECT on IPv4 diff --git a/tests/socket/socket_test.c b/tests/socket/socket_test.c index 37e02819..34130b26 100644 --- a/tests/socket/socket_test.c +++ b/tests/socket/socket_test.c @@ -12,10 +12,6 @@ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * - * You should have received a copy of the GNU General Public License along - * with this program; if not, write to the Free Software Foundation, Inc., - * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. - * */ #include <stdio.h> @@ -23,6 +19,8 @@ #include <unistd.h> #include <string.h> #include <fcntl.h> +#include <inttypes.h> +#include <errno.h> #include <sys/socket.h> #include <arpa/inet.h> @@ -32,8 +30,9 @@ #include <osmocom/core/utils.h> #include <osmocom/core/socket.h> #include <osmocom/core/logging.h> +#include <osmocom/core/bits.h> -#include "../config.h" +#include "config.h" void *ctx = NULL; @@ -120,11 +119,341 @@ static int test_sockinit2(void) * FreeBSD 10 or 11 VM at home */ OSMO_ASSERT(!strncmp(name, "(r=127.0.0.1:53<->l=127.0.0.1", 29)); #endif + + printf("Checking osmo_sock_init2(AF_UNSPEC) must fail on mixed IPv4 & IPv6\n"); + fd = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP, "127.0.0.1", 0, "::1", 53, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init2(AF_UNSPEC) must fail on mixed IPv6 & IPv4\n"); + fd = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP, "::1", 0, "127.0.0.1", 53, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init2(AF_UNSPEC) BIND + CONNECT on IPv4\n"); + fd = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP, "127.0.0.1", 0, "127.0.0.1", 53, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + + printf("Checking osmo_sock_init2(AF_UNSPEC) BIND + CONNECT on IPv6\n"); + fd = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP, "::1", 0, "::1", 53, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + + printf("Checking osmo_sock_init2(AF_UNSPEC) BIND on IPv4\n"); + fd = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP, "127.0.0.1", 0, NULL, 0, + OSMO_SOCK_F_BIND); + OSMO_ASSERT(fd >= 0); + talloc_free(name); return 0; } +static int test_get_ip_and_port(void) +{ + int fd, rc; + char ip[INET6_ADDRSTRLEN] = { }; + char port[6] = { }; + + printf("Checking test_get_ip_and_port() for combined BIND + CONNECT on IPv4\n"); + fd = osmo_sock_init2(AF_INET, SOCK_DGRAM, IPPROTO_UDP, "127.0.0.1", 0, "127.0.0.1", 55, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + + OSMO_ASSERT(fd >= 0); + + /* get the remote */ + rc = osmo_sock_get_ip_and_port(fd, ip, sizeof(ip), port, sizeof(port), false); + OSMO_ASSERT(rc == 0); + OSMO_ASSERT(strncmp(ip, "127.0.0.1", INET6_ADDRSTRLEN) == 0); + OSMO_ASSERT(strncmp(port, "55", 6) == 0); + + printf("Checking test_get_ip_and_port() for combined BIND + CONNECT on IPv6\n"); + fd = osmo_sock_init2(AF_INET6, SOCK_DGRAM, IPPROTO_UDP, "::1", 0, "::1", 55, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + + /* get the remote */ + rc = osmo_sock_get_ip_and_port(fd, ip, sizeof(ip), port, sizeof(port), false); + OSMO_ASSERT(rc == 0); + OSMO_ASSERT(strncmp(ip, "::1", INET6_ADDRSTRLEN) == 0); + OSMO_ASSERT(strncmp(port, "55", 6) == 0); + + return 0; +} + +static int test_sockinit_osa(void) +{ + int fd, rc; + char *name; + + struct osmo_sockaddr localhost4 = {}; + struct osmo_sockaddr localhost6 = {}; + struct osmo_sockaddr localhost4_noport = {}; + struct osmo_sockaddr localhost6_noport = {}; + struct osmo_sockaddr any4 = {}; + struct osmo_sockaddr any6 = {}; + struct osmo_sockaddr invalid = {}; + + localhost4.u.sin = (struct sockaddr_in){ + .sin_family = AF_INET, + .sin_addr.s_addr = inet_addr("127.0.0.1"), + .sin_port = htons(42), + }; + + localhost6.u.sin6 = (struct sockaddr_in6){ + .sin6_family = AF_INET6, + .sin6_port = htons(42), + }; + inet_pton(AF_INET6, "::1", &localhost6.u.sin6.sin6_addr); + + localhost4_noport = localhost4; + localhost4_noport.u.sin.sin_port = htons(0); + localhost6_noport = localhost6; + localhost6_noport.u.sin6.sin6_port = htons(0); + + any4.u.sin = (struct sockaddr_in){ + .sin_family = AF_INET, + .sin_addr.s_addr = inet_addr("0.0.0.0"), + .sin_port = htons(0), + }; + any6.u.sin6 = (struct sockaddr_in6){ + .sin6_family = AF_INET6, + .sin6_port = htons(0), + }; + inet_pton(AF_INET6, "::", &any6.u.sin6.sin6_addr); + + invalid.u.sa.sa_family = AF_UNSPEC; + + printf("Checking osmo_sock_init_osa() with bind to a random local UDP port\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, + &any4, NULL, OSMO_SOCK_F_BIND); + OSMO_ASSERT(fd >= 0); + name = osmo_sock_get_name(ctx, fd); + /* expect it to be not connected. We cannot match on INADDR_ANY, + * as apparently that won't work on FreeBSD if there's only one + * address (e.g. 137.0.0.1) assigned to the entire system, like + * the Osmocom FreeBSD build slaves */ + OSMO_ASSERT(!strncmp(name, "(r=NULL<->", 9)); + talloc_free(name); + /* expect it to be blocking */ + rc = fcntl(fd, F_GETFL); + OSMO_ASSERT(!(rc & O_NONBLOCK)); + close(fd); + + printf("Checking osmo_sock_init_osa() IPv4 for OSMO_SOCK_F_NONBLOCK\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, + &any4, NULL, OSMO_SOCK_F_BIND|OSMO_SOCK_F_NONBLOCK); + OSMO_ASSERT(fd >= 0); + /* expect it to be blocking */ + rc = fcntl(fd, F_GETFL); + OSMO_ASSERT(rc & O_NONBLOCK); + close(fd); + + printf("Checking osmo_sock_init_osa() IPv6 for OSMO_SOCK_F_NONBLOCK\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, + &any6, NULL, OSMO_SOCK_F_BIND|OSMO_SOCK_F_NONBLOCK); + OSMO_ASSERT(fd >= 0); + /* expect it to be blocking */ + rc = fcntl(fd, F_GETFL); + OSMO_ASSERT(rc & O_NONBLOCK); + close(fd); + + printf("Checking osmo_sock_init_osa() for invalid flags\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &any4, NULL, 0); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init_osa() for combined BIND + CONNECT on IPv4\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &localhost4_noport, &localhost4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + name = osmo_sock_get_name(ctx, fd); +#ifndef __FreeBSD__ + /* For some reason, on the jenkins.osmocom.org build slave with + * FreeBSD 10 inside a jail, it fails. Works fine on laforge's + * FreeBSD 10 or 11 VM at home */ + OSMO_ASSERT(!strncmp(name, "(r=127.0.0.1:42<->l=127.0.0.1", 29)); +#endif + close(fd); + + printf("Checking osmo_sock_init_osa() for combined BIND + CONNECT on IPv6\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &localhost6_noport, &localhost6, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd >= 0); + name = osmo_sock_get_name(ctx, fd); +#ifndef __FreeBSD__ + /* For some reason, on the jenkins.osmocom.org build slave with + * FreeBSD 10 inside a jail, it fails. Works fine on laforge's + * FreeBSD 10 or 11 VM at home */ + OSMO_ASSERT(!strncmp(name, "(r=::1:42<->l=::1", 17)); +#endif + close(fd); + + printf("Checking osmo_sock_init_osa() must fail on mixed IPv4 & IPv6\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &localhost4_noport, &localhost6, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init_osa() must fail on mixed IPv6 & IPv4\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &localhost6_noport, &localhost4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + printf("Checking osmo_sock_init_osa() must fail on invalid osmo_sockaddr\n"); + fd = osmo_sock_init_osa(SOCK_DGRAM, IPPROTO_UDP, &invalid, &localhost4, + OSMO_SOCK_F_BIND|OSMO_SOCK_F_CONNECT); + OSMO_ASSERT(fd < 0); + + talloc_free(name); + + return 0; +} + +static void test_osa_str(void) +{ + char buf[256]; + const char *result; + struct osmo_sockaddr localhost4 = {}; + struct osmo_sockaddr localhost6 = {}; + struct osmo_sockaddr osa = {}; + + localhost4.u.sin = (struct sockaddr_in){ + .sin_family = AF_INET, + .sin_addr.s_addr = inet_addr("127.0.0.1"), + .sin_port = htons(42), + }; + + localhost6.u.sin6 = (struct sockaddr_in6){ + .sin6_family = AF_INET6, + .sin6_port = htons(42), + }; + inet_pton(AF_INET6, "::1", &localhost6.u.sin6.sin6_addr); + + /* test a too short str */ + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, 1, &localhost4); + printf("Checking osmo_sockaddr_to_str_buf to small IPv4\n"); + OSMO_ASSERT(result == NULL); + + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, sizeof(buf), &localhost4); + printf("Checking osmo_sockaddr_to_str_buf IPv4\n"); + OSMO_ASSERT(!strncmp("127.0.0.1:42", result, sizeof(buf))); + + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, 256, &localhost6); + printf("Checking osmo_sockaddr_to_str_buf IPv6\n"); + OSMO_ASSERT(!strncmp("[::1]:42", result, sizeof(buf))); + + memset(&buf[0], 0, sizeof(buf)); + printf("Checking osmo_sockaddr_to_str_buf too short IPv6\n"); + result = osmo_sockaddr_to_str_buf(buf, 8, &localhost6); + OSMO_ASSERT(result == NULL); + osmo_sockaddr_to_str_buf2(buf, 8, &localhost6); + OSMO_ASSERT(!strncmp("[::1]:4", buf, sizeof(buf))); + + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, 5, &localhost6); + printf("Checking osmo_sockaddr_to_str_buf too short IPv6\n"); + OSMO_ASSERT(result == NULL); + + localhost6.u.sin6.sin6_port = 0; + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, 5, &localhost6); + printf("Checking osmo_sockaddr_to_str_buf only 5 bytes IPv6\n"); + OSMO_ASSERT(result == NULL); + + inet_pton(AF_INET6, "::", &localhost6.u.sin6.sin6_addr); + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, 5, &localhost6); + printf("Checking osmo_sockaddr_to_str_buf only 5 bytes IPv6\n"); + OSMO_ASSERT(!strncmp("[::]", result, sizeof(buf))); + + inet_pton(AF_INET6, "2003:1234:5678:90ab:cdef:1234:4321:4321", &localhost6.u.sin6.sin6_addr); + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, sizeof(buf), &localhost6); + printf("Checking osmo_sockaddr_to_str_buf long IPv6\n"); + OSMO_ASSERT(!strncmp("[2003:1234:5678:90ab:cdef:1234:4321:4321]", result, sizeof(buf))); + + localhost6.u.sin6.sin6_port = htons(23420); + memset(&buf[0], 0, sizeof(buf)); + result = osmo_sockaddr_to_str_buf(buf, sizeof(buf), &localhost6); + printf("Checking osmo_sockaddr_to_str_buf long IPv6 port\n"); + OSMO_ASSERT(!strncmp("[2003:1234:5678:90ab:cdef:1234:4321:4321]:23420", result, sizeof(buf))); + + result = osmo_sockaddr_to_str(&localhost6); + printf("Checking osmo_sockaddr_to_str_buf long IPv6 port static buffer\n"); + OSMO_ASSERT(!strncmp("[2003:1234:5678:90ab:cdef:1234:4321:4321]:23420", result, sizeof(buf))); + + printf("Checking osmo_sockaddr_from_str_and_uint for 0.0.0.0\n"); + OSMO_ASSERT(osmo_sockaddr_from_str_and_uint(&osa, "0.0.0.0", 1234) == 0); + OSMO_ASSERT(osmo_sockaddr_is_any(&osa)); + + printf("Checking osmo_sockaddr_from_str_and_uint for ::\n"); + OSMO_ASSERT(osmo_sockaddr_from_str_and_uint(&osa, "::", 1234) == 0); + OSMO_ASSERT(osmo_sockaddr_is_any(&osa)); + + printf("Checking osmo_sockaddr_from_str_and_uint for 1.2.3.4\n"); + OSMO_ASSERT(osmo_sockaddr_from_str_and_uint(&osa, "1.2.3.4", 1234) == 0); + OSMO_ASSERT(!osmo_sockaddr_is_any(&osa)); +} + +static void test_osa_netmask_prefixlen(void) +{ + struct osmo_sockaddr ipv4; + struct osmo_sockaddr ipv6; + int rc; + + ipv4.u.sin = (struct sockaddr_in){ + .sin_family = AF_INET, + }; + + ipv4.u.sin.sin_addr.s_addr = inet_addr("0.0.0.0"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + OSMO_ASSERT(rc == 0); + + ipv4.u.sin.sin_addr.s_addr = inet_addr("255.0.0.0"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + OSMO_ASSERT(rc == 8); + + ipv4.u.sin.sin_addr.s_addr = inet_addr("255.255.0.0"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + OSMO_ASSERT(rc == 16); + + ipv4.u.sin.sin_addr.s_addr = inet_addr("255.255.255.0"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + OSMO_ASSERT(rc == 24); + + ipv4.u.sin.sin_addr.s_addr = inet_addr("255.255.255.255"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + OSMO_ASSERT(rc == 32); + + ipv4.u.sin.sin_addr.s_addr = inet_addr("0.255.0.0"); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv4); + /* FIXME: This shows the implementation is not that robust checking validity of input netmask: */ + OSMO_ASSERT(rc == 8); + + ipv6.u.sin6 = (struct sockaddr_in6){ + .sin6_family = AF_INET6, + }; + + inet_pton(AF_INET6, "fe::", &ipv6.u.sin6.sin6_addr); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv6); + OSMO_ASSERT(rc == 7); + + inet_pton(AF_INET6, "ff::", &ipv6.u.sin6.sin6_addr); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv6); + OSMO_ASSERT(rc == 8); + + inet_pton(AF_INET6, "ff:ff::", &ipv6.u.sin6.sin6_addr); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv6); + OSMO_ASSERT(rc == 16); + + inet_pton(AF_INET6, "ff:ff::ff", &ipv6.u.sin6.sin6_addr); + rc = osmo_sockaddr_netmask_to_prefixlen(&ipv6); + /* FIXME: This shows the implementation is not that robust checking validity of input netmask: */ + OSMO_ASSERT(rc == 24); +} const struct log_info_cat default_categories[] = { }; @@ -139,10 +468,16 @@ int main(int argc, char *argv[]) ctx = talloc_named_const(NULL, 0, "socket_test"); osmo_init_logging2(ctx, &info); log_set_use_color(osmo_stderr_target, 0); - log_set_print_filename(osmo_stderr_target, 0); + log_set_print_filename2(osmo_stderr_target, LOG_FILENAME_NONE); + log_set_print_category(osmo_stderr_target, 0); + log_set_print_category_hex(osmo_stderr_target, 0); test_sockinit(); test_sockinit2(); + test_get_ip_and_port(); + test_sockinit_osa(); + test_osa_str(); + test_osa_netmask_prefixlen(); return EXIT_SUCCESS; } diff --git a/tests/socket/socket_test.err b/tests/socket/socket_test.err index ed6e1865..3c198ac3 100644 --- a/tests/socket/socket_test.err +++ b/tests/socket/socket_test.err @@ -1,2 +1,8 @@ invalid: both bind and connect flags set: 0.0.0.0:0 invalid: you have to specify either BIND or CONNECT flags +Unable to find a common protocol (IPv4 or IPv6) for local host: 127.0.0.1 and remote host: ::1. +Unable to find a common protocol (IPv4 or IPv6) for local host: ::1 and remote host: 127.0.0.1. +invalid: you have to specify either BIND or CONNECT flags +invalid: the family for local and remote endpoint must be same. +invalid: the family for local and remote endpoint must be same. +invalid: the family for local and remote endpoint must be same. diff --git a/tests/socket/socket_test.ok b/tests/socket/socket_test.ok index 4b24fbce..2b1c1006 100644 --- a/tests/socket/socket_test.ok +++ b/tests/socket/socket_test.ok @@ -5,3 +5,32 @@ Checking osmo_sock_init2() with bind to a random local UDP port Checking osmo_sock_init2() for OSMO_SOCK_F_NONBLOCK Checking osmo_sock_init2() for invalid flags Checking osmo_sock_init2() for combined BIND + CONNECT +Checking osmo_sock_init2(AF_UNSPEC) must fail on mixed IPv4 & IPv6 +Checking osmo_sock_init2(AF_UNSPEC) must fail on mixed IPv6 & IPv4 +Checking osmo_sock_init2(AF_UNSPEC) BIND + CONNECT on IPv4 +Checking osmo_sock_init2(AF_UNSPEC) BIND + CONNECT on IPv6 +Checking osmo_sock_init2(AF_UNSPEC) BIND on IPv4 +Checking test_get_ip_and_port() for combined BIND + CONNECT on IPv4 +Checking test_get_ip_and_port() for combined BIND + CONNECT on IPv6 +Checking osmo_sock_init_osa() with bind to a random local UDP port +Checking osmo_sock_init_osa() IPv4 for OSMO_SOCK_F_NONBLOCK +Checking osmo_sock_init_osa() IPv6 for OSMO_SOCK_F_NONBLOCK +Checking osmo_sock_init_osa() for invalid flags +Checking osmo_sock_init_osa() for combined BIND + CONNECT on IPv4 +Checking osmo_sock_init_osa() for combined BIND + CONNECT on IPv6 +Checking osmo_sock_init_osa() must fail on mixed IPv4 & IPv6 +Checking osmo_sock_init_osa() must fail on mixed IPv6 & IPv4 +Checking osmo_sock_init_osa() must fail on invalid osmo_sockaddr +Checking osmo_sockaddr_to_str_buf to small IPv4 +Checking osmo_sockaddr_to_str_buf IPv4 +Checking osmo_sockaddr_to_str_buf IPv6 +Checking osmo_sockaddr_to_str_buf too short IPv6 +Checking osmo_sockaddr_to_str_buf too short IPv6 +Checking osmo_sockaddr_to_str_buf only 5 bytes IPv6 +Checking osmo_sockaddr_to_str_buf only 5 bytes IPv6 +Checking osmo_sockaddr_to_str_buf long IPv6 +Checking osmo_sockaddr_to_str_buf long IPv6 port +Checking osmo_sockaddr_to_str_buf long IPv6 port static buffer +Checking osmo_sockaddr_from_str_and_uint for 0.0.0.0 +Checking osmo_sockaddr_from_str_and_uint for :: +Checking osmo_sockaddr_from_str_and_uint for 1.2.3.4 |