libuv/test/test-udp-recvmsg-unreachable-error.c
Juan José Arboleda d984dea65c linux: add UDP support for MSG_ERRQUEUE on Linux IPv4/IPv6
Fixes: https://github.com/libuv/libuv/issues/4447
Co-authored-by: theanarkh <theratliter@gmail.com>
Signed-off-by: Juan José Arboleda <soyjuanarbol@gmail.com>
2025-10-04 17:54:47 -05:00

126 lines
4.3 KiB
C

/* Copyright libuv project and contributors. All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
#include "uv.h"
#include "task.h"
#include <stdio.h>
#include <stdlib.h>
#define CLIENT_TEST_PORT 9123
#define SERVER_TEST_PORT 9124
#define RECV_CB_MAX_CALL 3 /* ECONNREFUSED, EAGAIN/EWOULDBLOCK, ICMP delivery */
static int recv_cb_called = 0;
static void udp_send_cb(uv_udp_send_t* req, int status) {
ASSERT_EQ(status, 0);
}
static void alloc_cb(uv_handle_t* handle,
size_t suggested_size,
uv_buf_t* buf) {
static char storage[4]; /* "PING" */
buf->base = storage;
buf->len = sizeof(storage);
}
static void read_cb(uv_udp_t* handle,
ssize_t nread,
const uv_buf_t* buf,
const struct sockaddr* addr,
unsigned flags) {
ASSERT(flags == 0 || (flags & UV_UDP_LINUX_RECVERR));
recv_cb_called++;
}
static void timer_cb(uv_timer_t* handle) {
uv_udp_t* udp = handle->data;
uv_close((uv_handle_t*) udp, NULL);
}
TEST_IMPL(udp_recvmsg_unreachable_error) {
#if !defined(__linux__)
RETURN_SKIP("This test is Linux-specific");
#endif
struct sockaddr_in server_addr, client_addr;
uv_udp_t client;
uv_timer_t timer;
uv_udp_send_t send_req;
uv_buf_t buf = uv_buf_init("PING", 4);
ASSERT_OK(uv_ip4_addr("127.0.0.1", CLIENT_TEST_PORT, &client_addr));
ASSERT_OK(uv_ip4_addr("127.0.0.1", SERVER_TEST_PORT, &server_addr));
ASSERT_OK(uv_udp_init(uv_default_loop(), &client));
ASSERT_OK(uv_timer_init(uv_default_loop(), &timer));
ASSERT_OK(uv_udp_bind(&client,
(const struct sockaddr*) &client_addr,
UV_UDP_LINUX_RECVERR));
ASSERT_OK(uv_udp_recv_start(&client, alloc_cb, read_cb));
timer.data = &client;
ASSERT_OK(uv_timer_start(&timer, timer_cb, 3000, 0));
ASSERT_OK(uv_udp_send(&send_req,
&client,
&buf,
1,
(const struct sockaddr*) &server_addr,
udp_send_cb));
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
ASSERT_EQ(recv_cb_called, RECV_CB_MAX_CALL);
return 0;
}
TEST_IMPL(udp_recvmsg_unreachable_error6) {
#if !defined(__linux__)
RETURN_SKIP("This test is Linux-specific");
#endif
struct sockaddr_in6 server_addr, client_addr;
uv_udp_t client;
uv_timer_t timer;
uv_udp_send_t send_req;
uv_buf_t buf = uv_buf_init("PING", 4);
ASSERT_OK(uv_ip6_addr("::1", CLIENT_TEST_PORT, &client_addr));
ASSERT_OK(uv_ip6_addr("::1", SERVER_TEST_PORT, &server_addr));
ASSERT_OK(uv_udp_init(uv_default_loop(), &client));
ASSERT_OK(uv_timer_init(uv_default_loop(), &timer));
ASSERT_OK(uv_udp_bind(&client,
(const struct sockaddr*) &client_addr,
UV_UDP_LINUX_RECVERR));
ASSERT_OK(uv_udp_recv_start(&client, alloc_cb, read_cb));
timer.data = &client;
ASSERT_OK(uv_timer_start(&timer, timer_cb, 3000, 0));
ASSERT_OK(uv_udp_send(&send_req,
&client,
&buf,
1,
(const struct sockaddr*) &server_addr,
udp_send_cb));
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
ASSERT_EQ(recv_cb_called, RECV_CB_MAX_CALL);
return 0;
}