123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205 |
- #include "../config-host.h"
- /* SPDX-License-Identifier: MIT */
- /*
- * Check that repeated IORING_OP_CONNECT to a socket without a listener keeps
- * yielding -ECONNREFUSED rather than -ECONNABORTED. Based on a reproducer
- * from:
- *
- * https://github.com/axboe/liburing/issues/828
- *
- * and adopted to our usual test cases. Other changes made like looping,
- * using different ring types, adding a memset() for reuse, etc.
- *
- */
- #include <stdio.h>
- #include <netinet/in.h>
- #include <string.h>
- #include <unistd.h>
- #include <stdlib.h>
- #include <arpa/inet.h>
- #include "liburing.h"
- #include "helpers.h"
- static unsigned long ud;
- static int init_test_server(struct sockaddr_in *serv_addr)
- {
- socklen_t servaddr_len = sizeof(struct sockaddr_in);
- int fd;
- /* Init server socket. Bind but don't listen */
- fd = socket(AF_INET, SOCK_STREAM, 0);
- if (fd < 0) {
- perror("socket");
- return -1;
- }
- serv_addr->sin_family = AF_INET;
- serv_addr->sin_addr.s_addr = inet_addr("127.0.0.1");
- if (bind(fd, (struct sockaddr *) serv_addr, servaddr_len) < 0) {
- perror("bind");
- return -1;
- }
- /*
- * Get the addresses the socket is bound to because the port is chosen
- * by the network stack.
- */
- if (getsockname(fd, (struct sockaddr *)serv_addr, &servaddr_len) < 0) {
- perror("getsockname");
- return -1;
- }
- return fd;
- }
- static int init_test_client(void)
- {
- socklen_t addr_len = sizeof(struct sockaddr_in);
- struct sockaddr_in client_addr = {};
- int clientfd;
- clientfd = socket(AF_INET, SOCK_STREAM, 0);
- if (clientfd < 0) {
- perror("socket");
- return -1;
- }
- client_addr.sin_family = AF_INET;
- client_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
- if (bind(clientfd, (struct sockaddr *)&client_addr, addr_len) < 0) {
- perror("bind");
- close(clientfd);
- return -1;
- }
- /*
- * Get the addresses the socket is bound to because the port is chosen
- * by the network stack.
- */
- if (getsockname(clientfd, (struct sockaddr *)&client_addr, &addr_len) < 0) {
- perror("getsockname");
- close(clientfd);
- return -1;
- }
- return clientfd;
- }
- static int get_completion_and_print(struct io_uring *ring)
- {
- struct io_uring_cqe *cqe;
- int ret, res;
- ret = io_uring_wait_cqe(ring, &cqe);
- if (ret < 0) {
- fprintf(stderr, "wait_cqe=%d\n", ret);
- return -1;
- }
- /* Mark this completion as seen */
- res = cqe->res;
- io_uring_cqe_seen(ring, cqe);
- return res;
- }
- static int test_connect(struct io_uring *ring,
- int clientfd, struct sockaddr_in *serv_addr)
- {
- struct sockaddr_in local_sa;
- struct io_uring_sqe *sqe;
- int ret;
- sqe = io_uring_get_sqe(ring);
- io_uring_prep_connect(sqe, clientfd, (const struct sockaddr *)serv_addr,
- sizeof(struct sockaddr_in));
- sqe->user_data = ++ud;
- memcpy(&local_sa, serv_addr, sizeof(local_sa));
- ret = io_uring_submit_and_wait(ring, 1);
- if (ret != 1) {
- fprintf(stderr, "submit=%d\n", ret);
- return T_EXIT_FAIL;
- }
- /* check for reuse at the same time */
- memset(&local_sa, 0xff, sizeof(local_sa));
- ret = get_completion_and_print(ring);
- if (ret != -ECONNREFUSED) {
- fprintf(stderr, "Connect got %d\n", ret);
- return T_EXIT_FAIL;
- }
- return T_EXIT_PASS;
- }
- static int test(int flags)
- {
- struct io_uring_params params = { .flags = flags, };
- struct sockaddr_in serv_addr = {};
- struct io_uring ring;
- int ret, clientfd, s_fd, i;
- if (flags & IORING_SETUP_SQPOLL)
- params.sq_thread_idle = 50;
- ret = io_uring_queue_init_params(8, &ring, ¶ms);
- if (ret < 0) {
- fprintf(stderr, "Queue init: %d\n", ret);
- return T_EXIT_FAIL;
- }
- s_fd = init_test_server(&serv_addr);
- if (s_fd < 0)
- return T_EXIT_FAIL;
- clientfd = init_test_client();
- if (clientfd < 0) {
- close(s_fd);
- return T_EXIT_FAIL;
- }
- /* make sure SQPOLL thread is sleeping */
- if (flags & IORING_SETUP_SQPOLL)
- usleep(100000);
- for (i = 0; i < 32; i++) {
- ret = test_connect(&ring, clientfd, &serv_addr);
- if (ret == T_EXIT_SKIP)
- return T_EXIT_SKIP;
- else if (ret == T_EXIT_PASS)
- continue;
- return T_EXIT_FAIL;
- }
- close(s_fd);
- close(clientfd);
- return T_EXIT_PASS;
- }
- int main(int argc, char *argv[])
- {
- int ret;
- if (argc > 1)
- return T_EXIT_SKIP;
- ret = test(0);
- if (ret == T_EXIT_FAIL) {
- fprintf(stderr, "test(0) failed\n");
- return T_EXIT_FAIL;
- }
- ret = test(IORING_SETUP_SQPOLL);
- if (ret == T_EXIT_FAIL) {
- fprintf(stderr, "test(SQPOLL) failed\n");
- return T_EXIT_FAIL;
- }
- return 0;
- }
|