Blame test/io_uring_setup.c

Packit d3489f
/* SPDX-License-Identifier: MIT */
Packit d3489f
/*
Packit d3489f
 * io_uring_setup.c
Packit d3489f
 *
Packit d3489f
 * Description: Unit tests for the io_uring_setup system call.
Packit d3489f
 *
Packit d3489f
 * Copyright 2019, Red Hat, Inc.
Packit d3489f
 * Author: Jeff Moyer <jmoyer@redhat.com>
Packit d3489f
 */
Packit d3489f
#include <stdio.h>
Packit d3489f
#include <fcntl.h>
Packit d3489f
#include <string.h>
Packit d3489f
#include <stdlib.h>
Packit d3489f
#include <unistd.h>
Packit d3489f
#include <errno.h>
Packit d3489f
#include <sys/sysinfo.h>
Packit d3489f
#include "liburing.h"
Packit d3489f
Packit d3489f
#include "../syscall.h"
Packit d3489f
Packit d3489f
char *features_string(struct io_uring_params *p)
Packit d3489f
{
Packit d3489f
	static char flagstr[64];
Packit d3489f
Packit d3489f
	if (!p || !p->features)
Packit d3489f
		return "none";
Packit d3489f
Packit d3489f
	if (p->features & ~IORING_FEAT_SINGLE_MMAP) {
Packit d3489f
		snprintf(flagstr, 64, "0x%.8x", p->features);
Packit d3489f
		return flagstr;
Packit d3489f
	}
Packit d3489f
Packit d3489f
	if (p->features & IORING_FEAT_SINGLE_MMAP)
Packit d3489f
		strncat(flagstr, "IORING_FEAT_SINGLE_MMAP", 64 - strlen(flagstr));
Packit d3489f
Packit d3489f
	return flagstr;
Packit d3489f
}
Packit d3489f
Packit d3489f
/*
Packit d3489f
 * Attempt the call with the given args.  Return 0 when expect matches
Packit d3489f
 * the return value of the system call, 1 otherwise.
Packit d3489f
 */
Packit d3489f
char *
Packit d3489f
flags_string(struct io_uring_params *p)
Packit d3489f
{
Packit d3489f
	static char flagstr[64];
Packit d3489f
	int add_pipe = 0;
Packit d3489f
Packit d3489f
	memset(flagstr, 0, sizeof(flagstr));
Packit d3489f
Packit d3489f
	if (!p || p->flags == 0)
Packit d3489f
		return "none";
Packit d3489f
Packit d3489f
	/*
Packit d3489f
	 * If unsupported flags are present, just print the bitmask.
Packit d3489f
	 */
Packit d3489f
	if (p->flags & ~(IORING_SETUP_IOPOLL | IORING_SETUP_SQPOLL |
Packit d3489f
			 IORING_SETUP_SQ_AFF)) {
Packit d3489f
		snprintf(flagstr, 64, "0x%.8x", p->flags);
Packit d3489f
		return flagstr;
Packit d3489f
	}
Packit d3489f
Packit d3489f
	if (p->flags & IORING_SETUP_IOPOLL) {
Packit d3489f
		strncat(flagstr, "IORING_SETUP_IOPOLL", 64 - strlen(flagstr));
Packit d3489f
		add_pipe = 1;
Packit d3489f
	}
Packit d3489f
	if (p->flags & IORING_SETUP_SQPOLL) {
Packit d3489f
		if (add_pipe)
Packit d3489f
			strncat(flagstr, "|", 64 - strlen(flagstr));
Packit d3489f
		else
Packit d3489f
			add_pipe = 1;
Packit d3489f
		strncat(flagstr, "IORING_SETUP_SQPOLL", 64 - strlen(flagstr));
Packit d3489f
	}
Packit d3489f
	if (p->flags & IORING_SETUP_SQ_AFF) {
Packit d3489f
		if (add_pipe)
Packit d3489f
			strncat(flagstr, "|", 64 - strlen(flagstr));
Packit d3489f
		strncat(flagstr, "IORING_SETUP_SQ_AFF", 64 - strlen(flagstr));
Packit d3489f
	}
Packit d3489f
Packit d3489f
	return flagstr;
Packit d3489f
}
Packit d3489f
Packit d3489f
char *
Packit d3489f
dump_resv(struct io_uring_params *p)
Packit d3489f
{
Packit d3489f
	static char resvstr[4096];
Packit d3489f
Packit d3489f
	if (!p)
Packit d3489f
		return "";
Packit d3489f
Packit d3489f
	sprintf(resvstr, "0x%.8x 0x%.8x 0x%.8x", p->resv[0],
Packit d3489f
		p->resv[1], p->resv[2]);
Packit d3489f
Packit d3489f
	return resvstr;
Packit d3489f
}
Packit d3489f
Packit d3489f
/* bogus: setup returns a valid fd on success... expect can't predict the
Packit d3489f
   fd we'll get, so this really only takes 1 parameter: error */
Packit d3489f
int
Packit d3489f
try_io_uring_setup(unsigned entries, struct io_uring_params *p, int expect, int error)
Packit d3489f
{
Packit d3489f
	int ret, __errno;
Packit d3489f
Packit d3489f
	printf("io_uring_setup(%u, %p), flags: %s, feat: %s, resv: %s, sq_thread_cpu: %u\n",
Packit d3489f
	       entries, p, flags_string(p), features_string(p), dump_resv(p),
Packit d3489f
	       p ? p->sq_thread_cpu : 0);
Packit d3489f
Packit d3489f
	ret = __sys_io_uring_setup(entries, p);
Packit d3489f
	if (ret != expect) {
Packit d3489f
		printf("expected %d, got %d\n", expect, ret);
Packit d3489f
		/* if we got a valid uring, close it */
Packit d3489f
		if (ret > 0)
Packit d3489f
			close(ret);
Packit d3489f
		return 1;
Packit d3489f
	}
Packit d3489f
	__errno = errno;
Packit d3489f
	if (expect == -1 && error != __errno) {
Packit d3489f
		if (__errno == EPERM && geteuid() != 0) {
Packit d3489f
			printf("Needs root, not flagging as an error\n");
Packit d3489f
			return 0;
Packit d3489f
		}
Packit d3489f
		printf("expected errno %d, got %d\n", error, __errno);
Packit d3489f
		return 1;
Packit d3489f
	}
Packit d3489f
Packit d3489f
	return 0;
Packit d3489f
}
Packit d3489f
Packit d3489f
int
Packit d3489f
main(int argc, char **argv)
Packit d3489f
{
Packit d3489f
	int fd;
Packit d3489f
	unsigned int status = 0;
Packit d3489f
	struct io_uring_params p;
Packit d3489f
Packit d3489f
	if (argc > 1)
Packit d3489f
		return 0;
Packit d3489f
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	status |= try_io_uring_setup(0, &p, -1, EINVAL);
Packit d3489f
	status |= try_io_uring_setup(1, NULL, -1, EFAULT);
Packit d3489f
Packit d3489f
	/* resv array is non-zero */
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	p.resv[0] = p.resv[1] = p.resv[2] = 1;
Packit d3489f
	status |= try_io_uring_setup(1, &p, -1, EINVAL);
Packit d3489f
Packit d3489f
	/* invalid flags */
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	p.flags = ~0U;
Packit d3489f
	status |= try_io_uring_setup(1, &p, -1, EINVAL);
Packit d3489f
Packit d3489f
	/* IORING_SETUP_SQ_AFF set but not IORING_SETUP_SQPOLL */
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	p.flags = IORING_SETUP_SQ_AFF;
Packit d3489f
	status |= try_io_uring_setup(1, &p, -1, EINVAL);
Packit d3489f
Packit d3489f
	/* attempt to bind to invalid cpu */
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	p.flags = IORING_SETUP_SQPOLL | IORING_SETUP_SQ_AFF;
Packit d3489f
	p.sq_thread_cpu = get_nprocs_conf();
Packit d3489f
	status |= try_io_uring_setup(1, &p, -1, EINVAL);
Packit d3489f
Packit d3489f
	/* I think we can limit a process to a set of cpus.  I assume
Packit d3489f
	 * we shouldn't be able to setup a kernel thread outside of that.
Packit d3489f
	 * try to do that. (task->cpus_allowed) */
Packit d3489f
Packit d3489f
	/* read/write on io_uring_fd */
Packit d3489f
	memset(&p, 0, sizeof(p));
Packit d3489f
	fd = __sys_io_uring_setup(1, &p);
Packit d3489f
	if (fd < 0) {
Packit d3489f
		printf("io_uring_setup failed with %d, expected success\n",
Packit d3489f
		       errno);
Packit d3489f
		status = 1;
Packit d3489f
	} else {
Packit d3489f
		char buf[4096];
Packit d3489f
		int ret;
Packit d3489f
		ret = read(fd, buf, 4096);
Packit d3489f
		if (ret >= 0) {
Packit d3489f
			printf("read from io_uring fd succeeded.  expected fail\n");
Packit d3489f
			status = 1;
Packit d3489f
		}
Packit d3489f
	}
Packit d3489f
Packit d3489f
	if (!status) {
Packit d3489f
		printf("PASS\n");
Packit d3489f
		return 0;
Packit d3489f
	}
Packit d3489f
Packit d3489f
	printf("FAIL\n");
Packit d3489f
	return -1;
Packit d3489f
}