uring-sys2 0.11.0

liburing bindings
Documentation
/* SPDX-License-Identifier: MIT */
/*
 * Description: run various timeout tests
 *
 */
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/time.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <assert.h>

#include "helpers.h"
#include "liburing.h"
#include "../src/syscall.h"

#define IO_NSEC_PER_SEC			1000000000ULL

static bool support_abs = false;
static bool support_clock = false;

static unsigned long long timespec_to_ns(struct timespec *ts)
{
	return ts->tv_nsec + ts->tv_sec * IO_NSEC_PER_SEC;
}
static struct timespec ns_to_timespec(unsigned long long t)
{
	struct timespec ts;

	ts.tv_sec = t / IO_NSEC_PER_SEC;
	ts.tv_nsec = t - ts.tv_sec * IO_NSEC_PER_SEC;
	return ts;
}

static long long ns_since(struct timespec *ts)
{
	struct timespec now;
	int ret;

	ret = clock_gettime(CLOCK_MONOTONIC, &now);
	if (ret) {
		fprintf(stderr, "clock_gettime failed\n");
		exit(T_EXIT_FAIL);
	}

	return timespec_to_ns(&now) - timespec_to_ns(ts);

}

static int t_io_uring_wait(struct io_uring *ring, int nr, unsigned enter_flags,
			   struct timespec *ts)
{
	struct __kernel_timespec kts = {
		.tv_sec = ts->tv_sec,
		.tv_nsec = ts->tv_nsec
	};
	struct io_uring_getevents_arg arg = {
		.sigmask	= 0,
		.sigmask_sz	= _NSIG / 8,
		.ts		= (unsigned long) &kts
	};
	int ret;

	enter_flags |= IORING_ENTER_GETEVENTS | IORING_ENTER_EXT_ARG;
	ret = io_uring_enter2(ring->ring_fd, 0, nr, enter_flags,
			      (void *)&arg, sizeof(arg));
	return ret;
}

static int probe_timers(void)
{
	struct io_uring_clock_register cr = { .clockid = CLOCK_MONOTONIC, };
	struct io_uring ring;
	struct timespec ts;
	int ret;

	ret = io_uring_queue_init(8, &ring, 0);
	if (ret) {
		fprintf(stderr, "probe ring setup failed: %d\n", ret);
		return ret;
	}

	ret = clock_gettime(CLOCK_MONOTONIC, &ts);
	if (ret) {
		fprintf(stderr, "clock_gettime failed\n");
		return ret;
	}

	ret = t_io_uring_wait(&ring, 0, IORING_ENTER_ABS_TIMER, &ts);
	if (!ret) {
		support_abs = true;
	} else if (ret != -EINVAL) {
		fprintf(stderr, "wait failed %i\n", ret);
		return ret;
	}

	ret = io_uring_register_clock(&ring, &cr);
	if (!ret) {
		support_clock = true;
	} else if (ret != -EINVAL) {
		fprintf(stderr, "io_uring_register_clock %i\n", ret);
		return ret;
	}

	io_uring_queue_exit(&ring);
	return 0;
}

static int test_timeout(bool abs, bool set_clock)
{
	unsigned enter_flags = abs ? IORING_ENTER_ABS_TIMER : 0;
	struct io_uring ring;
	struct timespec start, end, ts;
	long long dt;
	int ret;

	ret = io_uring_queue_init(8, &ring, 0);
	if (ret) {
		fprintf(stderr, "ring setup failed: %d\n", ret);
		return 1;
	}

	if (set_clock) {
		struct io_uring_clock_register cr = {};

		cr.clockid = CLOCK_BOOTTIME;
		ret = io_uring_register_clock(&ring, &cr);
		if (ret) {
			fprintf(stderr, "io_uring_register_clock failed\n");
			return 1;
		}
	}

	/* pass current time */
	ret = clock_gettime(CLOCK_MONOTONIC, &start);
	assert(ret == 0);

	ts = abs ? start : ns_to_timespec(0);
	ret = t_io_uring_wait(&ring, 1, enter_flags, &ts);
	if (ret != -ETIME) {
		fprintf(stderr, "wait current time failed, %i\n", ret);
		return 1;
	}

	if (ns_since(&start) >= IO_NSEC_PER_SEC) {
		fprintf(stderr, "current time test failed\n");
		return 1;
	}

	if (abs) {
		/* expired time */
		ret = clock_gettime(CLOCK_MONOTONIC, &start);
		assert(ret == 0);
		ts = ns_to_timespec(timespec_to_ns(&start) - IO_NSEC_PER_SEC);

		ret = t_io_uring_wait(&ring, 1, enter_flags, &ts);
		if (ret != -ETIME) {
			fprintf(stderr, "expired timeout wait failed, %i\n", ret);
			return 1;
		}

		ret = clock_gettime(CLOCK_MONOTONIC, &end);
		assert(ret == 0);

		if (ns_since(&start) >= IO_NSEC_PER_SEC) {
			fprintf(stderr, "expired timer test failed\n");
			return 1;
		}
	}

	/* 1s wait */
	ret = clock_gettime(CLOCK_MONOTONIC, &start);
	assert(ret == 0);

	dt = 2 * IO_NSEC_PER_SEC + (abs ? timespec_to_ns(&start) : 0);
	ts = ns_to_timespec(dt);
	ret = t_io_uring_wait(&ring, 1, enter_flags, &ts);
	if (ret != -ETIME) {
		fprintf(stderr, "wait timeout failed, %i\n", ret);
		return 1;
	}

	dt = ns_since(&start);
	if (dt < IO_NSEC_PER_SEC || dt > 3 * IO_NSEC_PER_SEC) {
		fprintf(stderr, "early wake up, %lld\n", dt);
		return 1;
	}
	return 0;
}

static int test_clock_setup(void)
{
	struct io_uring ring;
	struct io_uring_clock_register cr = {};
	int ret;

	ret = io_uring_queue_init(8, &ring, 0);
	if (ret) {
		fprintf(stderr, "ring setup failed: %d\n", ret);
		return T_EXIT_FAIL;
	}

	ret = __sys_io_uring_register(ring.ring_fd, IORING_REGISTER_CLOCK, NULL, 0);
	if (!ret) {
		fprintf(stderr, "invalid null clock registration %i\n", ret);
		return T_EXIT_FAIL;
	}

	cr.clockid = -1;
	ret = __sys_io_uring_register(ring.ring_fd, IORING_REGISTER_CLOCK, &cr, 0);
	if (ret != -EINVAL) {
		fprintf(stderr, "invalid clockid registration %i\n", ret);
		return T_EXIT_FAIL;
	}

	cr.clockid = CLOCK_MONOTONIC;
	ret = __sys_io_uring_register(ring.ring_fd, IORING_REGISTER_CLOCK, &cr, 0);
	if (ret) {
		fprintf(stderr, "clock monotonic registration failed %i\n", ret);
		return T_EXIT_FAIL;
	}

	cr.clockid = CLOCK_BOOTTIME;
	ret = __sys_io_uring_register(ring.ring_fd, IORING_REGISTER_CLOCK, &cr, 0);
	if (ret) {
		fprintf(stderr, "clock boottime registration failed %i\n", ret);
		return T_EXIT_FAIL;
	}

	cr.clockid = CLOCK_MONOTONIC;
	ret = __sys_io_uring_register(ring.ring_fd, IORING_REGISTER_CLOCK, &cr, 0);
	if (ret) {
		fprintf(stderr, "2nd clock monotonic registration failed %i\n", ret);
		return T_EXIT_FAIL;
	}

	io_uring_queue_exit(&ring);
	return 0;
}

int main(int argc, char *argv[])
{
	int ret, i;

	if (argc > 1)
		return 0;

	ret = probe_timers();
	if (ret) {
		fprintf(stderr, "probe failed\n");
		return T_EXIT_FAIL;
	}
	if (!support_abs && !support_clock)
		return T_EXIT_SKIP;

	if (support_clock) {
		ret = test_clock_setup();
		if (ret) {
			fprintf(stderr, "test_clock_setup failed\n");
			return T_EXIT_FAIL;
		}
	}

	for (i = 0; i < 4; i++) {
		bool abs = i & 1;
		bool clock = i & 2;

		if (abs && !support_abs)
			continue;
		if (clock && !support_clock)
			continue;

		ret = test_timeout(abs, clock);
		if (ret) {
			fprintf(stderr, "test_timeout failed %i %i\n",
					abs, clock);
			return ret;
		}
	}

	return 0;
}