uring-sys2 0.11.0

liburing bindings
Documentation
/* SPDX-License-Identifier: MIT */
/*
 * Description: test reading a normal file with incremental buffer
 *		consumption. Some kernels had a bug where the initial part
 *		of the buffer got skipped, test for that.
 *
 */
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include "liburing.h"
#include "helpers.h"

#define BUF_BGID	4
#define BUF_BID		8

static void arm_read(struct io_uring *ring, int fd, int offset)
{
	struct io_uring_sqe *sqe;

	sqe = io_uring_get_sqe(ring);
	io_uring_prep_read(sqe, fd, NULL, 80, offset);
	sqe->flags = IOSQE_BUFFER_SELECT;
	sqe->buf_group = BUF_BGID;
	io_uring_submit(ring);
}

static int create_test_file(const char *fname)
{
	char buf[80], c;
	int fd, i, ret;

	fd = open(fname, O_WRONLY | O_CREAT | O_TRUNC, 0644);
	if (fd < 0) {
		perror("open");
		return T_EXIT_FAIL;
	}

	c = 'a';
	for (i = 0; i < 8; i++) {
		memset(buf, c, sizeof(buf));
		ret = write(fd, buf, sizeof(buf));
		if (ret < 0) {
			perror("write");
			unlink(fname);
			return T_EXIT_FAIL;
		} else if (ret != sizeof(buf)) {
			fprintf(stderr, "Short write: %d\n", ret);
			unlink(fname);
			return T_EXIT_FAIL;
		}
		c++;
	}

	close(fd);
	return 0;
}

int main(int argc, char *argv[])
{
	struct io_uring_buf_ring *br;
	struct io_uring_params p = { };
	struct io_uring_cqe *cqe;
	struct io_uring ring;
	int tret, ret, fd, i;
	char fname[64];
	char c = 'a';
	char *buf;
	void *ptr;

	if (argc > 1)
		return T_EXIT_SKIP;

	sprintf(fname, ".buf-inc-file.%d", getpid());
	if (create_test_file(fname))
		return T_EXIT_FAIL;

	fd = open(fname, O_RDONLY);
	if (fd < 0) {
		perror("open");
		goto err;
	}

	ret = io_uring_queue_init_params(64, &ring, &p);
	if (ret) {
		fprintf(stderr, "ring setup failed: %d\n", ret);
		goto err;
	}

	if (posix_memalign((void **) &buf, 4096, 65536))
		goto err;

	tret = T_EXIT_SKIP;
	br = io_uring_setup_buf_ring(&ring, 32, BUF_BGID, IOU_PBUF_RING_INC, &ret);
	if (!br) {
		if (ret == -EINVAL)
			goto out;
		fprintf(stderr, "Buffer ring register failed %d\n", ret);
		goto err;
	}

	tret = T_EXIT_PASS;
	io_uring_buf_ring_add(br, buf, 65536, BUF_BID, 31, 0);
	io_uring_buf_ring_advance(br, 1);

	memset(buf, 0, 65536);

	ptr = buf;
	for (i = 0; i < 4; i++) {
		int bid;

		arm_read(&ring, fd, i * 80);
		ret = io_uring_wait_cqe(&ring, &cqe);
		if (ret) {
			fprintf(stderr, "wait %d\n", ret);
			goto err;
		}
		if (!(cqe->flags & IORING_CQE_F_BUFFER)) {
			fprintf(stderr, "buffer not assigned\n");
			goto err;
		}
		bid = cqe->flags >> IORING_CQE_BUFFER_SHIFT;
		if (bid != BUF_BID) {
			fprintf(stderr, "got wrong buffer bid %d\n", bid);
			goto err;
		}
		if (cqe->res != 80) {
			fprintf(stderr, "bad read size %d\n", ret);
			goto err;
		}
		io_uring_cqe_seen(&ring, cqe);
		if (!memchr(ptr, c, cqe->res)) {
			fprintf(stderr, "fail buffer check loop %d\n", i);
			goto err;
		}
		c++;
		ptr += cqe->res;
	}

	io_uring_free_buf_ring(&ring, br, 32, BUF_BGID);
	io_uring_queue_exit(&ring);
out:
	free(buf);
	unlink(fname);
	return tret;
err:
	unlink(fname);
	return T_EXIT_FAIL;
}