231 lines
4.2 KiB
C
231 lines
4.2 KiB
C
/* SPDX-License-Identifier: MIT */
|
|
/*
|
|
* Description: test io_uring link io with drain io
|
|
*
|
|
*/
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
|
|
#include "helpers.h"
|
|
#include "liburing.h"
|
|
|
|
static int test_link_drain_one(struct io_uring *ring)
|
|
{
|
|
struct io_uring_cqe *cqe;
|
|
struct io_uring_sqe *sqe[5];
|
|
struct iovec iovecs;
|
|
int i, fd, ret;
|
|
off_t off = 0;
|
|
char data[5] = {0};
|
|
char expect[5] = {0, 1, 2, 3, 4};
|
|
|
|
fd = open("testfile", O_WRONLY | O_CREAT, 0644);
|
|
if (fd < 0) {
|
|
perror("open");
|
|
return 1;
|
|
}
|
|
|
|
iovecs.iov_base = t_malloc(4096);
|
|
iovecs.iov_len = 4096;
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
sqe[i] = io_uring_get_sqe(ring);
|
|
if (!sqe[i]) {
|
|
printf("get sqe failed\n");
|
|
goto err;
|
|
}
|
|
}
|
|
|
|
/* normal heavy io */
|
|
io_uring_prep_writev(sqe[0], fd, &iovecs, 1, off);
|
|
sqe[0]->user_data = 0;
|
|
|
|
/* link io */
|
|
io_uring_prep_nop(sqe[1]);
|
|
sqe[1]->flags |= IOSQE_IO_LINK;
|
|
sqe[1]->user_data = 1;
|
|
|
|
/* link drain io */
|
|
io_uring_prep_nop(sqe[2]);
|
|
sqe[2]->flags |= (IOSQE_IO_LINK | IOSQE_IO_DRAIN);
|
|
sqe[2]->user_data = 2;
|
|
|
|
/* link io */
|
|
io_uring_prep_nop(sqe[3]);
|
|
sqe[3]->user_data = 3;
|
|
|
|
/* normal nop io */
|
|
io_uring_prep_nop(sqe[4]);
|
|
sqe[4]->user_data = 4;
|
|
|
|
ret = io_uring_submit(ring);
|
|
if (ret < 0) {
|
|
printf("sqe submit failed\n");
|
|
goto err;
|
|
} else if (ret < 5) {
|
|
printf("Submitted only %d\n", ret);
|
|
goto err;
|
|
}
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
ret = io_uring_wait_cqe(ring, &cqe);
|
|
if (ret < 0) {
|
|
printf("child: wait completion %d\n", ret);
|
|
goto err;
|
|
}
|
|
|
|
data[i] = cqe->user_data;
|
|
io_uring_cqe_seen(ring, cqe);
|
|
}
|
|
|
|
if (memcmp(data, expect, 5) != 0)
|
|
goto err;
|
|
|
|
free(iovecs.iov_base);
|
|
close(fd);
|
|
unlink("testfile");
|
|
return 0;
|
|
err:
|
|
free(iovecs.iov_base);
|
|
close(fd);
|
|
unlink("testfile");
|
|
return 1;
|
|
}
|
|
|
|
int test_link_drain_multi(struct io_uring *ring)
|
|
{
|
|
struct io_uring_cqe *cqe;
|
|
struct io_uring_sqe *sqe[9];
|
|
struct iovec iovecs;
|
|
int i, fd, ret;
|
|
off_t off = 0;
|
|
char data[9] = {0};
|
|
char expect[9] = {0, 1, 2, 3, 4, 5, 6, 7, 8};
|
|
|
|
fd = open("testfile", O_WRONLY | O_CREAT, 0644);
|
|
if (fd < 0) {
|
|
perror("open");
|
|
return 1;
|
|
}
|
|
|
|
iovecs.iov_base = t_malloc(4096);
|
|
iovecs.iov_len = 4096;
|
|
|
|
for (i = 0; i < 9; i++) {
|
|
sqe[i] = io_uring_get_sqe(ring);
|
|
if (!sqe[i]) {
|
|
printf("get sqe failed\n");
|
|
goto err;
|
|
}
|
|
}
|
|
|
|
/* normal heavy io */
|
|
io_uring_prep_writev(sqe[0], fd, &iovecs, 1, off);
|
|
sqe[0]->user_data = 0;
|
|
|
|
/* link1 io head */
|
|
io_uring_prep_nop(sqe[1]);
|
|
sqe[1]->flags |= IOSQE_IO_LINK;
|
|
sqe[1]->user_data = 1;
|
|
|
|
/* link1 drain io */
|
|
io_uring_prep_nop(sqe[2]);
|
|
sqe[2]->flags |= (IOSQE_IO_LINK | IOSQE_IO_DRAIN);
|
|
sqe[2]->user_data = 2;
|
|
|
|
/* link1 io end*/
|
|
io_uring_prep_nop(sqe[3]);
|
|
sqe[3]->user_data = 3;
|
|
|
|
/* link2 io head */
|
|
io_uring_prep_nop(sqe[4]);
|
|
sqe[4]->flags |= IOSQE_IO_LINK;
|
|
sqe[4]->user_data = 4;
|
|
|
|
/* link2 io */
|
|
io_uring_prep_nop(sqe[5]);
|
|
sqe[5]->flags |= IOSQE_IO_LINK;
|
|
sqe[5]->user_data = 5;
|
|
|
|
/* link2 drain io */
|
|
io_uring_prep_writev(sqe[6], fd, &iovecs, 1, off);
|
|
sqe[6]->flags |= (IOSQE_IO_LINK | IOSQE_IO_DRAIN);
|
|
sqe[6]->user_data = 6;
|
|
|
|
/* link2 io end */
|
|
io_uring_prep_nop(sqe[7]);
|
|
sqe[7]->user_data = 7;
|
|
|
|
/* normal io */
|
|
io_uring_prep_nop(sqe[8]);
|
|
sqe[8]->user_data = 8;
|
|
|
|
ret = io_uring_submit(ring);
|
|
if (ret < 0) {
|
|
printf("sqe submit failed\n");
|
|
goto err;
|
|
} else if (ret < 9) {
|
|
printf("Submitted only %d\n", ret);
|
|
goto err;
|
|
}
|
|
|
|
for (i = 0; i < 9; i++) {
|
|
ret = io_uring_wait_cqe(ring, &cqe);
|
|
if (ret < 0) {
|
|
printf("child: wait completion %d\n", ret);
|
|
goto err;
|
|
}
|
|
|
|
data[i] = cqe->user_data;
|
|
io_uring_cqe_seen(ring, cqe);
|
|
}
|
|
|
|
if (memcmp(data, expect, 9) != 0)
|
|
goto err;
|
|
|
|
free(iovecs.iov_base);
|
|
close(fd);
|
|
unlink("testfile");
|
|
return 0;
|
|
err:
|
|
free(iovecs.iov_base);
|
|
close(fd);
|
|
unlink("testfile");
|
|
return 1;
|
|
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
struct io_uring ring;
|
|
int i, ret;
|
|
|
|
if (argc > 1)
|
|
return 0;
|
|
|
|
ret = io_uring_queue_init(100, &ring, 0);
|
|
if (ret) {
|
|
printf("ring setup failed\n");
|
|
return 1;
|
|
}
|
|
|
|
for (i = 0; i < 1000; i++) {
|
|
ret = test_link_drain_one(&ring);
|
|
if (ret) {
|
|
fprintf(stderr, "test_link_drain_one failed\n");
|
|
break;
|
|
}
|
|
ret = test_link_drain_multi(&ring);
|
|
if (ret) {
|
|
fprintf(stderr, "test_link_drain_multi failed\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|