blob: ebf848c859ec0b54811064f3a14abf7af6d8fe4f [file] [log] [blame]
/* SPDX-License-Identifier: MIT */
/*
* Description: test ring messaging with flags command
*
*/
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <pthread.h>
#include "liburing.h"
#include "helpers.h"
#define CUSTOM_FLAG 0x42
#define USER_DATA 0x5aa5
#define LEN 0x20
#define ID 0x1
struct data {
pthread_barrier_t barrier;
int fd;
};
static int recv_msg(struct io_uring *ring)
{
struct io_uring_cqe *cqe;
int ret;
ret = io_uring_wait_cqe(ring, &cqe);
if (ret) {
fprintf(stderr, "wait cqe %d\n", ret);
return T_EXIT_FAIL;
}
if (cqe->user_data != USER_DATA) {
fprintf(stderr, "user_data %llx\n", (long long) cqe->user_data);
return T_EXIT_FAIL;
}
if (cqe->res != LEN) {
fprintf(stderr, "len %x\n", cqe->res);
return T_EXIT_FAIL;
}
if (cqe->flags != CUSTOM_FLAG) {
fprintf(stderr, "flags %x\n", cqe->flags);
return T_EXIT_FAIL;
}
return T_EXIT_PASS;
}
static int send_msg(struct io_uring *ring, int target_fd)
{
struct io_uring_cqe *cqe;
struct io_uring_sqe *sqe;
int ret;
sqe = io_uring_get_sqe(ring);
if (!sqe) {
fprintf(stderr, "get sqe failed\n");
return T_EXIT_FAIL;
}
io_uring_prep_msg_ring_cqe_flags(sqe, target_fd, LEN, USER_DATA,
0, CUSTOM_FLAG);
sqe->user_data = ID;
ret = io_uring_submit(ring);
if (ret <= 0) {
if (ret == -EINVAL)
return T_EXIT_SKIP;
fprintf(stderr, "sqe submit failed: %d\n", ret);
return T_EXIT_FAIL;
}
ret = io_uring_wait_cqe(ring, &cqe);
if (ret < 0) {
fprintf(stderr, "wait completion %d\n", ret);
return T_EXIT_FAIL;
}
if (cqe->res != 0) {
if (cqe->res == -EINVAL)
return T_EXIT_SKIP;
fprintf(stderr, "cqe res %d\n", cqe->res);
return T_EXIT_FAIL;
}
if (cqe->user_data != ID) {
fprintf(stderr, "user_data %llx\n", (long long) cqe->user_data);
return T_EXIT_FAIL;
}
io_uring_cqe_seen(ring, cqe);
return T_EXIT_PASS;
}
static void *thread_fn(void *data)
{
struct data *d = data;
struct io_uring ring;
int ret;
ret = io_uring_queue_init(2, &ring, IORING_SETUP_DEFER_TASKRUN | IORING_SETUP_SINGLE_ISSUER);
if (ret) {
fprintf(stderr, "ring init failed %d\n", ret);
pthread_barrier_wait(&d->barrier);
return NULL;
}
d->fd = ring.ring_fd;
pthread_barrier_wait(&d->barrier);
if (recv_msg(&ring))
return (void *) 1;
return NULL;
}
static int test(int ring_flags)
{
struct io_uring ring, ring2;
pthread_t thread;
struct data d;
void *ret2;
int ret, i;
ret = io_uring_queue_init(2, &ring, ring_flags);
if (ret) {
fprintf(stderr, "io_uring_queue_init failed for ring1: %d\n", ret);
return T_EXIT_FAIL;
}
ret = io_uring_queue_init(2, &ring2, ring_flags);
if (ret) {
fprintf(stderr, "io_uring_queue_init failed for ring2: %d\n", ret);
return T_EXIT_FAIL;
}
ret = send_msg(&ring, ring2.ring_fd);
if (ret) {
if (ret != T_EXIT_SKIP)
fprintf(stderr, "send_msg failed: %d\n", ret);
return ret;
}
ret = recv_msg(&ring2);
if (ret) {
fprintf(stderr, "recv_msg failed: %d\n", ret);
return ret;
}
for (i = 0; i < 8; i++) {
ret = send_msg(&ring, ring2.ring_fd);
if (ret) {
if (ret != T_EXIT_SKIP)
fprintf(stderr, "send_msg failed: %d\n", ret);
return ret;
}
}
for (i = 0; i < 8; i++) {
ret = recv_msg(&ring2);
if (ret) {
fprintf(stderr, "recv_msg failed: %d\n", ret);
return ret;
}
}
pthread_barrier_init(&d.barrier, NULL, 2);
d.fd = -1;
pthread_create(&thread, NULL, thread_fn, &d);
pthread_barrier_wait(&d.barrier);
if (d.fd == -1)
return T_EXIT_FAIL;
ret = send_msg(&ring, d.fd);
if (ret) {
fprintf(stderr, "send msg failed: %d\n", ret);
return ret;
}
pthread_join(thread, &ret2);
if (ret2) {
fprintf(stderr, "Remote test failed\n");
return T_EXIT_FAIL;
}
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 ring_flags 0 failed\n");
return ret;
} else if (ret == T_EXIT_SKIP)
return ret;
ret = test(IORING_SETUP_SINGLE_ISSUER|IORING_SETUP_DEFER_TASKRUN);
if (ret == T_EXIT_FAIL) {
fprintf(stderr, "test ring_flags defer failed\n");
return ret;
}
return ret;
}