/* 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 #include #include #include #include #include #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; }