diff options
author | Pau Espin Pedrol <pespin@sysmocom.de> | 2020-08-19 17:25:04 +0200 |
---|---|---|
committer | pespin <pespin@sysmocom.de> | 2020-08-24 08:29:19 +0000 |
commit | cd133316cfc735a3006d499dae6b2f693c3c741c (patch) | |
tree | 7d2e1b2878229dd7686a47cbcbcf8d046eac7333 /tests/socket | |
parent | 796c6513726af46e997518fa18afc7a3477c951a (diff) |
socket: multiaddr: Support IPv4 + IPv6 addresses in SCTP associations
The function is improved to support AF_INET:v4->v4, AF_INET6:v6->v6 and
AF_UNSPEC:v4+v6->v4+v6.
Unit tests for the function are added to make sure function behaves
correctly in several scenarios.
Change-Id: I36d8ab85d92bba4d6adb83bc1875eb61094ed2ef
Diffstat (limited to 'tests/socket')
-rw-r--r-- | tests/socket/socket_test.c | 176 | ||||
-rw-r--r-- | tests/socket/socket_test.err | 8 | ||||
-rw-r--r-- | tests/socket/socket_test.ok | 24 |
3 files changed, 208 insertions, 0 deletions
diff --git a/tests/socket/socket_test.c b/tests/socket/socket_test.c index e68243c6..1d4253c4 100644 --- a/tests/socket/socket_test.c +++ b/tests/socket/socket_test.c @@ -23,6 +23,7 @@ #include <unistd.h> #include <string.h> #include <fcntl.h> +#include <inttypes.h> #include <sys/socket.h> #include <arpa/inet.h> @@ -32,6 +33,7 @@ #include <osmocom/core/utils.h> #include <osmocom/core/socket.h> #include <osmocom/core/logging.h> +#include <osmocom/core/bits.h> #include "../config.h" @@ -146,6 +148,175 @@ static int test_sockinit2(void) return 0; } +#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); + + 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); + + 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(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[] = { }; @@ -164,6 +335,11 @@ int main(int argc, char *argv[]) test_sockinit(); test_sockinit2(); +#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_test.err b/tests/socket/socket_test.err index 0f0f8da8..3c5dac50 100644 --- a/tests/socket/socket_test.err +++ b/tests/socket/socket_test.err @@ -2,3 +2,11 @@ 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 v4 vs v6 in local vs remote addresses +Invalid v4 vs v6 in local vs remote addresses +invalid: you have to specify either BIND or CONNECT flags +Invalid v4 vs v6 in local vs remote addresses +Invalid v4 vs v6 in local vs remote addresses +getaddrinfo returned NULL: ::1:0: Transport endpoint is not connected +getaddrinfo returned NULL: 127.0.0.1:0: Transport endpoint is not connected diff --git a/tests/socket/socket_test.ok b/tests/socket/socket_test.ok index 4265be8d..959fa84d 100644 --- a/tests/socket/socket_test.ok +++ b/tests/socket/socket_test.ok @@ -9,3 +9,27 @@ 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_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 |