blob: 50d5501f7d2d7305943a9396f9254eb251c58ae0 [file]
/* SPDX-License-Identifier: MIT */
/*
* Description: test multishot POLL_ADD on an eventfd that is also the
* CQ notification eventfd of the same ring. Triggering the poll
* posts a CQE, which notifies the eventfd, which re-enters the
* poll wake path via EPOLL_URING_WAKE. The multishot poll must
* keep delivering events for subsequent wakeups; a regression
* here leaves the poll request permanently stuck with
* IORING_CQE_F_MORE set but no further CQEs.
*/
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <poll.h>
#include <sys/eventfd.h>
#include "liburing.h"
#include "helpers.h"
#define NR_LOOPS 2
static int test(void)
{
struct __kernel_timespec ts;
struct io_uring_cqe *cqe;
struct io_uring_sqe *sqe;
struct io_uring ring;
int ret, evfd, i;
uint64_t val;
evfd = eventfd(0, EFD_NONBLOCK);
if (evfd < 0) {
perror("eventfd");
return T_EXIT_FAIL;
}
ret = io_uring_queue_init(8, &ring, 0);
if (ret) {
fprintf(stderr, "queue_init: %d\n", ret);
close(evfd);
return T_EXIT_FAIL;
}
ret = io_uring_register_eventfd(&ring, evfd);
if (ret) {
fprintf(stderr, "register_eventfd: %d\n", ret);
close(evfd);
io_uring_queue_exit(&ring);
return T_EXIT_FAIL;
}
sqe = io_uring_get_sqe(&ring);
io_uring_prep_poll_multishot(sqe, evfd, POLLIN);
sqe->user_data = 1;
ret = io_uring_submit(&ring);
if (ret != 1) {
fprintf(stderr, "submit: %d\n", ret);
goto err;
}
for (i = 0; i < NR_LOOPS; i++) {
val = 1;
ret = write(evfd, &val, sizeof(val));
if (ret != sizeof(val)) {
perror("write");
goto err;
}
ret = read(evfd, &val, sizeof(val));
if (ret != sizeof(val)) {
perror("read");
goto err;
}
ts.tv_sec = 1;
ts.tv_nsec = 0;
ret = io_uring_wait_cqe_timeout(&ring, &cqe, &ts);
if (ret == -ETIME) {
fprintf(stderr, "poll stuck: no CQE after iteration %d\n", i);
goto err;
}
if (ret < 0) {
fprintf(stderr, "wait_cqe: %d\n", ret);
goto err;
}
if (cqe->user_data != 1) {
fprintf(stderr, "unexpected user_data %llx\n",
(unsigned long long) cqe->user_data);
io_uring_cqe_seen(&ring, cqe);
goto err;
}
if (cqe->res < 0) {
fprintf(stderr, "cqe res: %d\n", cqe->res);
io_uring_cqe_seen(&ring, cqe);
goto err;
}
if (!i) {
if (!(cqe->flags & IORING_CQE_F_MORE)) {
fprintf(stderr, "IORING_CQE_F_MORE not set\n");
goto err;
}
} else {
if (cqe->flags & IORING_CQE_F_MORE) {
fprintf(stderr, "IORING_CQE_F_MORE set\n");
goto err;
}
}
io_uring_cqe_seen(&ring, cqe);
}
close(evfd);
io_uring_queue_exit(&ring);
return T_EXIT_PASS;
err:
close(evfd);
io_uring_queue_exit(&ring);
return T_EXIT_FAIL;
}
int main(int argc, char *argv[])
{
if (argc > 1)
return T_EXIT_SKIP;
return test();
}