blob: c830876446549ce353d7ec20e832214a1695674a [file] [log] [blame]
/*
* Copyright (c) 2020 Friedt Professional Engineering Services, Inc
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <fcntl.h>
#include <logging/log.h>
LOG_MODULE_DECLARE(net_test, CONFIG_NET_SOCKETS_LOG_LEVEL);
#include <string.h>
#include <net/socket.h>
#include <sys/util.h>
#include <posix/unistd.h>
#include <ztest_assert.h>
#undef read
#define read(fd, buf, len) zsock_recv(fd, buf, len, 0)
#undef write
#define write(fd, buf, len) zsock_send(fd, buf, len, 0)
static void happy_path(
const int family, const char *family_s,
const int type, const char *type_s,
const int proto, const char *proto_s
)
{
int res;
int sv[2] = {-1, -1};
const char *expected_msg = "Hello, socketpair(2) world!";
const unsigned int expected_msg_len = strlen(expected_msg);
char actual_msg[32];
size_t actual_msg_len;
struct iovec iovec;
struct msghdr msghdr;
LOG_DBG("calling socketpair(%u, %u, %u, %p)", family, type, proto, sv);
res = socketpair(family, type, proto, sv);
zassert_true(res == -1 || res == 0,
"socketpair returned an unspecified value");
zassert_equal(res, 0, "socketpair failed");
LOG_DBG("sv: {%d, %d}", sv[0], sv[1]);
socklen_t len;
/* sockets are bidirectional. test functions from both ends */
for (int i = 0; i < 2; ++i) {
/*
* Test with write(2) / read(2)
*/
LOG_DBG("calling write(%d, '%s', %u)", sv[i], expected_msg,
expected_msg_len);
res = write(sv[i], expected_msg, expected_msg_len);
zassert_not_equal(res, -1, "write(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"did not write entire message");
memset(actual_msg, 0, sizeof(actual_msg));
LOG_DBG("calling read(%d, %p, %u)", sv[i], actual_msg,
(unsigned int)sizeof(actual_msg));
res = read(sv[(!i) & 1], actual_msg, sizeof(actual_msg));
zassert_not_equal(res, -1, "read(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"wrong return value");
zassert_true(strncmp(expected_msg, actual_msg,
actual_msg_len) == 0,
"the wrong message was passed through the socketpair");
/*
* Test with send(2) / recv(2)
*/
res = send(sv[i], expected_msg, expected_msg_len, 0);
zassert_not_equal(res, -1, "send(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"did not send entire message");
memset(actual_msg, 0, sizeof(actual_msg));
res = recv(sv[(!i) & 1], actual_msg, sizeof(actual_msg), 0);
zassert_not_equal(res, -1, "recv(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"wrong return value");
zassert_true(strncmp(expected_msg, actual_msg,
actual_msg_len) == 0,
"the wrong message was passed through the socketpair");
/*
* Test with sendto(2) / recvfrom(2)
*/
res = sendto(sv[i], expected_msg, expected_msg_len, 0, NULL, 0);
zassert_not_equal(res, -1, "sendto(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"did not sendto entire message");
memset(actual_msg, 0, sizeof(actual_msg));
len = 0;
res = recvfrom(sv[(!i) & 1], actual_msg, sizeof(actual_msg), 0,
NULL, &len);
zassert_true(res >= 0, "recvfrom(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"wrong return value");
zassert_true(strncmp(expected_msg, actual_msg,
actual_msg_len) == 0,
"the wrong message was passed through the socketpair");
/*
* Test with sendmsg(2) / recv(2) - Zephyr lacks recvmsg atm
*/
memset(&msghdr, 0, sizeof(msghdr));
msghdr.msg_iov = &iovec;
msghdr.msg_iovlen = 1;
iovec.iov_base = (void *)expected_msg;
iovec.iov_len = expected_msg_len;
res = sendmsg(sv[i], &msghdr, 0);
zassert_not_equal(res, -1, "sendmsg(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"did not sendmsg entire message");
res = read(sv[(!i) & 1], actual_msg, sizeof(actual_msg));
zassert_not_equal(res, -1, "read(2) failed: %d", errno);
actual_msg_len = res;
zassert_equal(actual_msg_len, expected_msg_len,
"wrong return value");
zassert_true(strncmp(expected_msg, actual_msg,
actual_msg_len) == 0,
"the wrong message was passed through the socketpair");
}
res = close(sv[0]);
zassert_equal(res, 0, "close failed");
res = close(sv[1]);
zassert_equal(res, 0, "close failed");
}
void test_socketpair_AF_LOCAL__SOCK_STREAM__0(void)
{
happy_path(
AF_LOCAL, "AF_LOCAL",
SOCK_STREAM, "SOCK_STREAM",
0, "0"
);
}
void test_socketpair_AF_UNIX__SOCK_STREAM__0(void)
{
happy_path(
AF_UNIX, "AF_UNIX",
SOCK_STREAM, "SOCK_STREAM",
0, "0"
);
}