[PATCH liburing] IORING_OP_GETDENTS: add opcode, prep function, test, man page section

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Signed-off-by: Lennert Buytenhek <buytenh@xxxxxxxxxxxxxx>
---
 man/io_uring_enter.2            |  26 +++++
 src/include/liburing.h          |   7 ++
 src/include/liburing/io_uring.h |   1 +
 test/Makefile                   |   1 +
 test/getdents.c                 | 180 ++++++++++++++++++++++++++++++++
 5 files changed, 215 insertions(+)
 create mode 100644 test/getdents.c

diff --git a/man/io_uring_enter.2 b/man/io_uring_enter.2
index 086207d..e0bd638 100644
--- a/man/io_uring_enter.2
+++ b/man/io_uring_enter.2
@@ -759,6 +759,32 @@ being passed in to
 .BR unlinkat(2).
 Available since 5.11.
 
+.TP
+.B IORING_OP_GETDENTS
+Issue the equivalent of an
+.BR lseek(2)
+system call plus a
+.BR getdents64(2)
+system call.
+.I fd
+should be set to the fd of the directory being operated on,
+.I off
+should be set to the offset in the directory to start reading from,
+.I addr
+should be set to the
+.I dirp,
+and
+.I len
+should be set to the
+.I count.
+
+.B IORING_OP_GETDENTS
+may or may not change the specified directory's file offset, and the
+file offset should not be relied upon having any particular value during
+or after an
+.B IORING_OP_GETDENTS
+operation.
+
 .PP
 The
 .I flags
diff --git a/src/include/liburing.h b/src/include/liburing.h
index 5b96e02..1769cda 100644
--- a/src/include/liburing.h
+++ b/src/include/liburing.h
@@ -535,6 +535,13 @@ static inline void io_uring_prep_sync_file_range(struct io_uring_sqe *sqe,
 	sqe->sync_range_flags = flags;
 }
 
+static inline void io_uring_prep_getdents(struct io_uring_sqe *sqe, int fd,
+					  void *buf, unsigned int count,
+					  uint64_t off)
+{
+	io_uring_prep_rw(IORING_OP_GETDENTS, sqe, fd, buf, count, off);
+}
+
 /*
  * Returns number of unconsumed (if SQPOLL) or unsubmitted entries exist in
  * the SQ ring
diff --git a/src/include/liburing/io_uring.h b/src/include/liburing/io_uring.h
index 1d47389..8abc0af 100644
--- a/src/include/liburing/io_uring.h
+++ b/src/include/liburing/io_uring.h
@@ -142,6 +142,7 @@ enum {
 	IORING_OP_RENAMEAT,
 	IORING_OP_UNLINKAT,
 	IORING_OP_MKDIRAT,
+	IORING_OP_GETDENTS,
 
 	/* this goes last, obviously */
 	IORING_OP_LAST,
diff --git a/test/Makefile b/test/Makefile
index 7751eff..c76987a 100644
--- a/test/Makefile
+++ b/test/Makefile
@@ -55,6 +55,7 @@ test_targets += \
 	files-exit-hang-timeout \
 	fixed-link \
 	fsync \
+	getdents \
 	io-cancel \
 	io_uring_enter \
 	io_uring_register \
diff --git a/test/getdents.c b/test/getdents.c
new file mode 100644
index 0000000..3ca7b05
--- /dev/null
+++ b/test/getdents.c
@@ -0,0 +1,180 @@
+/* SPDX-License-Identifier: MIT */
+/*
+ * Description: run various getdents tests
+ *
+ */
+#include <errno.h>
+#include <stdio.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <fcntl.h>
+
+#include "liburing.h"
+
+/*
+ * This struct isn't exported via include/uapi/ , so we define it by hand.
+ */
+struct linux_dirent64 {
+	uint64_t	d_ino;
+	int64_t		d_off;
+	uint16_t	d_reclen;
+	uint8_t		d_type;
+	char		d_name[];
+};
+
+#define BUFSZ		65536
+
+static int dirfd;
+
+static int test_getdents(struct io_uring *ring, void *buf,
+			 unsigned int count, uint64_t off)
+{
+	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");
+		goto err;
+	}
+	io_uring_prep_getdents(sqe, dirfd, buf, count, off);
+
+	ret = io_uring_submit(ring);
+	if (ret <= 0) {
+		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;
+	}
+
+	ret = cqe->res;
+
+	io_uring_cqe_seen(ring, cqe);
+
+	return ret;
+
+err:
+	return -1;
+}
+
+static void dump_dents(const uint8_t *buf, int len)
+{
+	const uint8_t *end;
+
+	fprintf(stderr, "inode offset type path\n");
+
+	end = buf + len;
+	while (buf < end) {
+		struct linux_dirent64 *dent;
+
+		dent = (struct linux_dirent64 *)buf;
+
+		fprintf(stderr, "%" PRId64 " %" PRId64 " %d %s\n", dent->d_ino,
+			dent->d_off, dent->d_type, dent->d_name);
+
+		buf += dent->d_reclen;
+	}
+
+	fprintf(stderr, "\n");
+}
+
+int main(int argc, char *argv[])
+{
+	struct io_uring ring;
+	int ret;
+	uint8_t buf[BUFSZ];
+	bool found_dot;
+	bool found_dotdot;
+	uint8_t *bufp;
+	uint8_t *end;
+
+	if (argc > 1)
+		return 0;
+
+	ret = io_uring_queue_init(1, &ring, 0);
+	if (ret) {
+		fprintf(stderr, "ring setup failed: %d\n", ret);
+		return 1;
+	}
+
+	dirfd = open(".", O_DIRECTORY);
+	if (dirfd < 0) {
+		fprintf(stderr, "opening \".\" failed: %s\n", strerror(errno));
+		return 1;
+	}
+
+	memset(buf, 0, sizeof(buf));
+
+	ret = test_getdents(&ring, buf, sizeof(buf), 0);
+	if (ret < 0) {
+		if (ret == -EINVAL) {
+			fprintf(stdout, "getdents not supported, skipping\n");
+			return 0;
+		}
+		fprintf(stderr, "getdents: %s\n", strerror(-ret));
+		return 1;
+	}
+
+	found_dot = false;
+	found_dotdot = false;
+
+	bufp = buf;
+	end = bufp + ret;
+
+	while (bufp < end) {
+		struct linux_dirent64 *dent;
+		uint8_t buf2[BUFSZ];
+
+		dent = (struct linux_dirent64 *)bufp;
+
+		if (!found_dot && !strcmp(dent->d_name, "."))
+			found_dot = true;
+		else if (!found_dotdot && !strcmp(dent->d_name, ".."))
+			found_dotdot = true;
+
+		bufp += dent->d_reclen;
+
+		/*
+		 * Now try to read the directory starting from the given
+		 * offset, and make sure we end up with the same data.
+		 */
+		memset(buf2, 0, sizeof(buf2));
+
+		ret = test_getdents(&ring, buf2, sizeof(buf2), dent->d_off);
+		if (ret < 0) {
+			fprintf(stderr, "getdents: %s\n", strerror(-ret));
+			return 1;
+		}
+
+		if (ret != end - bufp || memcmp(bufp, buf2, ret)) {
+			fprintf(stderr, "getdents: read from offset "
+					"%" PRId64 " returned unexpected "
+					"data\n\n", (uint64_t)dent->d_off);
+
+			fprintf(stderr, "read from offset zero:\n");
+			dump_dents(bufp, end - bufp);
+
+			fprintf(stderr, "offsetted read:\n");
+			dump_dents(buf2, ret);
+
+			return 1;
+		}
+	}
+
+	if (!found_dot)
+		fprintf(stderr, "getdents didn't return \".\" entry\n");
+
+	if (!found_dotdot)
+		fprintf(stderr, "getdents didn't return \"..\" entry\n");
+
+	if (!found_dot || !found_dotdot)
+		return 1;
+
+	return 0;
+}
-- 
2.29.2



[Index of Archives]     [Linux Samsung SoC]     [Linux Rockchip SoC]     [Linux Actions SoC]     [Linux for Synopsys ARC Processors]     [Linux NFS]     [Linux NILFS]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]


  Powered by Linux