1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
|
/* SPDX-License-Identifier: MIT */
/*
* Description: link <open file><read from file><close file> with an existing
* file present in the opened slot, verifying that we get the new file
* rather than the old one.
*
*/
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include "liburing.h"
#include "helpers.h"
#define MAX_FILES 8
#define FNAME1 ".slot.reuse.1"
#define FNAME2 ".slot.reuse.2"
#define PAT1 0xaa
#define PAT2 0x55
#define BSIZE 4096
static int test(struct io_uring *ring)
{
struct io_uring_cqe *cqe;
struct io_uring_sqe *sqe;
char buf[BSIZE];
int ret, i;
/* open FNAME1 in slot 0 */
sqe = io_uring_get_sqe(ring);
io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME1, O_RDONLY, 0, 0);
sqe->user_data = 1;
ret = io_uring_submit(ring);
if (ret != 1) {
fprintf(stderr, "sqe submit failed: %d\n", ret);
goto err;
}
ret = io_uring_wait_cqe(ring, &cqe);
if (ret < 0) {
fprintf(stderr, "wait completion %d\n", ret);
goto err;
}
if (cqe->res != 0) {
fprintf(stderr, "open res %d\n", ret);
goto err;
}
io_uring_cqe_seen(ring, cqe);
/*
* Now open FNAME2 in that same slot, verifying we get data from
* FNAME2 and not FNAME1.
*/
sqe = io_uring_get_sqe(ring);
io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME2, O_RDONLY, 0, 0);
sqe->flags |= IOSQE_IO_LINK;
sqe->user_data = 2;
sqe = io_uring_get_sqe(ring);
io_uring_prep_read(sqe, 0, buf, sizeof(buf), 0);
sqe->flags |= IOSQE_FIXED_FILE;
sqe->flags |= IOSQE_IO_LINK;
sqe->user_data = 3;
sqe = io_uring_get_sqe(ring);
io_uring_prep_close_direct(sqe, 0);
sqe->user_data = 4;
ret = io_uring_submit(ring);
if (ret != 3) {
fprintf(stderr, "sqe submit failed: %d\n", ret);
goto err;
}
for (i = 0; i < 3; i++) {
ret = io_uring_wait_cqe(ring, &cqe);
if (ret < 0) {
fprintf(stderr, "wait completion %d\n", ret);
goto err;
}
switch (cqe->user_data) {
case 2:
if (cqe->res) {
fprintf(stderr, "bad open %d\n", cqe->res);
goto err;
}
break;
case 3:
if (cqe->res != sizeof(buf)) {
fprintf(stderr, "bad read %d\n", cqe->res);
goto err;
}
break;
case 4:
if (cqe->res) {
fprintf(stderr, "bad close %d\n", cqe->res);
goto err;
}
break;
}
io_uring_cqe_seen(ring, cqe);
}
for (i = 0; i < sizeof(buf); i++) {
if (buf[i] == PAT2)
continue;
fprintf(stderr, "Bad pattern %x at %d\n", buf[i], i);
goto err;
}
return 0;
err:
return 1;
}
int main(int argc, char *argv[])
{
struct io_uring ring;
struct io_uring_params p = { };
int ret, files[MAX_FILES];
if (argc > 1)
return T_EXIT_SKIP;
ret = io_uring_queue_init_params(8, &ring, &p);
if (ret) {
fprintf(stderr, "ring setup failed: %d\n", ret);
return T_EXIT_FAIL;
}
if (!(p.features & IORING_FEAT_CQE_SKIP))
return T_EXIT_SKIP;
memset(files, -1, sizeof(files));
ret = io_uring_register_files(&ring, files, ARRAY_SIZE(files));
if (ret) {
fprintf(stderr, "Failed registering files\n");
return T_EXIT_FAIL;
}
t_create_file_pattern(FNAME1, 4096, PAT1);
t_create_file_pattern(FNAME2, 4096, PAT2);
ret = test(&ring);
if (ret) {
fprintf(stderr, "test failed\n");
goto err;
}
unlink(FNAME1);
unlink(FNAME2);
return T_EXIT_PASS;
err:
unlink(FNAME1);
unlink(FNAME2);
return T_EXIT_FAIL;
}
|