|
Packit |
d3489f |
// SPDX-License-Identifier: MIT
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
#define _GNU_SOURCE 1
|
|
Packit |
d3489f |
#define _FILE_OFFSET_BITS 64
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
// Test program for io_uring IORING_OP_CLOSE with O_PATH file.
|
|
Packit |
d3489f |
// Author: Clayton Harris <bugs@claycon.org>, 2020-06-07
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
// linux 5.6.14-300.fc32.x86_64
|
|
Packit |
d3489f |
// gcc 10.1.1-1.fc32
|
|
Packit |
d3489f |
// liburing.x86_64 0.5-1.fc32
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
// gcc -O2 -Wall -Wextra -std=c11 -o close_opath close_opath.c -luring
|
|
Packit |
d3489f |
// ./close_opath testfilepath
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
#include <errno.h>
|
|
Packit |
d3489f |
#include <fcntl.h>
|
|
Packit |
d3489f |
#include <liburing.h>
|
|
Packit |
d3489f |
#include <sys/stat.h>
|
|
Packit |
d3489f |
#include <stdio.h>
|
|
Packit |
d3489f |
#include <string.h>
|
|
Packit |
d3489f |
#include <unistd.h>
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
typedef struct
|
|
Packit |
d3489f |
{
|
|
Packit |
d3489f |
const char *const flnames;
|
|
Packit |
d3489f |
const int oflags;
|
|
Packit |
d3489f |
} oflgs_t;
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
static int test_io_uring_close(struct io_uring *ring, int fd)
|
|
Packit |
d3489f |
{
|
|
Packit |
d3489f |
struct io_uring_sqe *sqe;
|
|
Packit |
d3489f |
struct io_uring_cqe *cqe;
|
|
Packit |
d3489f |
int ret;
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
sqe = io_uring_get_sqe(ring);
|
|
Packit |
d3489f |
if (!sqe) {
|
|
Packit |
d3489f |
fprintf(stderr, "io_uring_get_sqe() failed\n");
|
|
Packit |
d3489f |
return -ENOENT;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
io_uring_prep_close(sqe, fd);
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
ret = io_uring_submit(ring);
|
|
Packit |
d3489f |
if (ret < 0) {
|
|
Packit |
d3489f |
fprintf(stderr, "io_uring_submit() failed, errno %d: %s\n",
|
|
Packit |
d3489f |
-ret, strerror(-ret));
|
|
Packit |
d3489f |
return ret;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
ret = io_uring_wait_cqe(ring, &cqe);
|
|
Packit |
d3489f |
if (ret < 0) {
|
|
Packit |
d3489f |
fprintf(stderr, "io_uring_wait_cqe() failed, errno %d: %s\n",
|
|
Packit |
d3489f |
-ret, strerror(-ret));
|
|
Packit |
d3489f |
return ret;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
ret = cqe->res;
|
|
Packit |
d3489f |
io_uring_cqe_seen(ring, cqe);
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
if (ret < 0 && ret != -EOPNOTSUPP) {
|
|
Packit |
d3489f |
fprintf(stderr, "io_uring close() failed, errno %d: %s\n",
|
|
Packit |
d3489f |
-ret, strerror(-ret));
|
|
Packit |
d3489f |
return ret;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
return 0;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
static int open_file(const char *path, const oflgs_t *oflgs)
|
|
Packit |
d3489f |
{
|
|
Packit |
d3489f |
int fd;
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
fd = openat(AT_FDCWD, path, oflgs->oflags, 0);
|
|
Packit |
d3489f |
if (fd < 0) {
|
|
Packit |
d3489f |
int err = errno;
|
|
Packit |
d3489f |
fprintf(stderr, "openat(%s, %s) failed, errno %d: %s\n",
|
|
Packit |
d3489f |
path, oflgs->flnames, err, strerror(err));
|
|
Packit |
d3489f |
return -err;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
return fd;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
int main(int argc, char *argv[])
|
|
Packit |
d3489f |
{
|
|
Packit |
d3489f |
const char *fname = ".";
|
|
Packit |
d3489f |
struct io_uring ring;
|
|
Packit |
d3489f |
int ret, i;
|
|
Packit |
d3489f |
static const oflgs_t oflgs[] = {
|
|
Packit |
d3489f |
{ "O_RDONLY", O_RDONLY },
|
|
Packit |
d3489f |
{ "O_PATH", O_PATH }
|
|
Packit |
d3489f |
};
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
ret = io_uring_queue_init(2, &ring, 0);
|
|
Packit |
d3489f |
if (ret < 0) {
|
|
Packit |
d3489f |
fprintf(stderr, "io_uring_queue_init() failed, errno %d: %s\n",
|
|
Packit |
d3489f |
-ret, strerror(-ret));
|
|
Packit |
d3489f |
return 0x02;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
#define OFLGS_SIZE (sizeof(oflgs) / sizeof(oflgs[0]))
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
ret = 0;
|
|
Packit |
d3489f |
for (i = 0; i < OFLGS_SIZE; i++) {
|
|
Packit |
d3489f |
int fd;
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
fd = open_file(fname, &oflgs[i]);
|
|
Packit |
d3489f |
if (fd < 0) {
|
|
Packit |
d3489f |
ret |= 0x02;
|
|
Packit |
d3489f |
break;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
/* Should always succeed */
|
|
Packit |
d3489f |
if (test_io_uring_close(&ring, fd) < 0)
|
|
Packit |
d3489f |
ret |= 0x04 << i;
|
|
Packit |
d3489f |
}
|
|
Packit |
d3489f |
#undef OFLGS_SIZE
|
|
Packit |
d3489f |
|
|
Packit |
d3489f |
io_uring_queue_exit(&ring);
|
|
Packit |
d3489f |
return ret;
|
|
Packit |
d3489f |
}
|