321 lines
7.8 KiB
C
321 lines
7.8 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
/* Basic per-epoll context busy poll test.
|
|
*
|
|
* Only tests the ioctls, but should be expanded to test two connected hosts in
|
|
* the future
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include <error.h>
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <limits.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include <sys/capability.h>
|
|
|
|
#include <sys/epoll.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
|
|
#include "../kselftest_harness.h"
|
|
|
|
/* if the headers haven't been updated, we need to define some things */
|
|
#if !defined(EPOLL_IOC_TYPE)
|
|
struct epoll_params {
|
|
uint32_t busy_poll_usecs;
|
|
uint16_t busy_poll_budget;
|
|
uint8_t prefer_busy_poll;
|
|
|
|
/* pad the struct to a multiple of 64bits */
|
|
uint8_t __pad;
|
|
};
|
|
|
|
#define EPOLL_IOC_TYPE 0x8A
|
|
#define EPIOCSPARAMS _IOW(EPOLL_IOC_TYPE, 0x01, struct epoll_params)
|
|
#define EPIOCGPARAMS _IOR(EPOLL_IOC_TYPE, 0x02, struct epoll_params)
|
|
#endif
|
|
|
|
FIXTURE(invalid_fd)
|
|
{
|
|
int invalid_fd;
|
|
struct epoll_params params;
|
|
};
|
|
|
|
FIXTURE_SETUP(invalid_fd)
|
|
{
|
|
int ret;
|
|
|
|
ret = socket(AF_UNIX, SOCK_DGRAM, 0);
|
|
EXPECT_NE(-1, ret)
|
|
TH_LOG("error creating unix socket");
|
|
|
|
self->invalid_fd = ret;
|
|
}
|
|
|
|
FIXTURE_TEARDOWN(invalid_fd)
|
|
{
|
|
int ret;
|
|
|
|
ret = close(self->invalid_fd);
|
|
EXPECT_EQ(0, ret);
|
|
}
|
|
|
|
TEST_F(invalid_fd, test_invalid_fd)
|
|
{
|
|
int ret;
|
|
|
|
ret = ioctl(self->invalid_fd, EPIOCGPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCGPARAMS on invalid epoll FD should error");
|
|
|
|
EXPECT_EQ(ENOTTY, errno)
|
|
TH_LOG("EPIOCGPARAMS on invalid epoll FD should set errno to ENOTTY");
|
|
|
|
memset(&self->params, 0, sizeof(struct epoll_params));
|
|
|
|
ret = ioctl(self->invalid_fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS on invalid epoll FD should error");
|
|
|
|
EXPECT_EQ(ENOTTY, errno)
|
|
TH_LOG("EPIOCSPARAMS on invalid epoll FD should set errno to ENOTTY");
|
|
}
|
|
|
|
FIXTURE(epoll_busy_poll)
|
|
{
|
|
int fd;
|
|
struct epoll_params params;
|
|
struct epoll_params *invalid_params;
|
|
cap_t caps;
|
|
};
|
|
|
|
FIXTURE_SETUP(epoll_busy_poll)
|
|
{
|
|
int ret;
|
|
|
|
ret = epoll_create1(0);
|
|
EXPECT_NE(-1, ret)
|
|
TH_LOG("epoll_create1 failed?");
|
|
|
|
self->fd = ret;
|
|
|
|
self->caps = cap_get_proc();
|
|
EXPECT_NE(NULL, self->caps);
|
|
}
|
|
|
|
FIXTURE_TEARDOWN(epoll_busy_poll)
|
|
{
|
|
int ret;
|
|
|
|
ret = close(self->fd);
|
|
EXPECT_EQ(0, ret);
|
|
|
|
ret = cap_free(self->caps);
|
|
EXPECT_NE(-1, ret)
|
|
TH_LOG("unable to free capabilities");
|
|
}
|
|
|
|
TEST_F(epoll_busy_poll, test_get_params)
|
|
{
|
|
/* begin by getting the epoll params from the kernel
|
|
*
|
|
* the default should be default and all fields should be zero'd by the
|
|
* kernel, so set params fields to garbage to test this.
|
|
*/
|
|
int ret = 0;
|
|
|
|
self->params.busy_poll_usecs = 0xff;
|
|
self->params.busy_poll_budget = 0xff;
|
|
self->params.prefer_busy_poll = 1;
|
|
self->params.__pad = 0xf;
|
|
|
|
ret = ioctl(self->fd, EPIOCGPARAMS, &self->params);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("ioctl EPIOCGPARAMS should succeed");
|
|
|
|
EXPECT_EQ(0, self->params.busy_poll_usecs)
|
|
TH_LOG("EPIOCGPARAMS busy_poll_usecs should have been 0");
|
|
|
|
EXPECT_EQ(0, self->params.busy_poll_budget)
|
|
TH_LOG("EPIOCGPARAMS busy_poll_budget should have been 0");
|
|
|
|
EXPECT_EQ(0, self->params.prefer_busy_poll)
|
|
TH_LOG("EPIOCGPARAMS prefer_busy_poll should have been 0");
|
|
|
|
EXPECT_EQ(0, self->params.__pad)
|
|
TH_LOG("EPIOCGPARAMS __pad should have been 0");
|
|
|
|
self->invalid_params = (struct epoll_params *)0xdeadbeef;
|
|
ret = ioctl(self->fd, EPIOCGPARAMS, self->invalid_params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCGPARAMS should error with invalid params");
|
|
|
|
EXPECT_EQ(EFAULT, errno)
|
|
TH_LOG("EPIOCGPARAMS with invalid params should set errno to EFAULT");
|
|
}
|
|
|
|
TEST_F(epoll_busy_poll, test_set_invalid)
|
|
{
|
|
int ret;
|
|
|
|
memset(&self->params, 0, sizeof(struct epoll_params));
|
|
|
|
self->params.__pad = 1;
|
|
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS non-zero __pad should error");
|
|
|
|
EXPECT_EQ(EINVAL, errno)
|
|
TH_LOG("EPIOCSPARAMS non-zero __pad errno should be EINVAL");
|
|
|
|
self->params.__pad = 0;
|
|
self->params.busy_poll_usecs = (uint32_t)INT_MAX + 1;
|
|
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS should error busy_poll_usecs > S32_MAX");
|
|
|
|
EXPECT_EQ(EINVAL, errno)
|
|
TH_LOG("EPIOCSPARAMS busy_poll_usecs > S32_MAX errno should be EINVAL");
|
|
|
|
self->params.__pad = 0;
|
|
self->params.busy_poll_usecs = 32;
|
|
self->params.prefer_busy_poll = 2;
|
|
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS should error prefer_busy_poll > 1");
|
|
|
|
EXPECT_EQ(EINVAL, errno)
|
|
TH_LOG("EPIOCSPARAMS prefer_busy_poll > 1 errno should be EINVAL");
|
|
|
|
self->params.__pad = 0;
|
|
self->params.busy_poll_usecs = 32;
|
|
self->params.prefer_busy_poll = 1;
|
|
|
|
/* set budget well above kernel's NAPI_POLL_WEIGHT of 64 */
|
|
self->params.busy_poll_budget = UINT16_MAX;
|
|
|
|
/* test harness should run with CAP_NET_ADMIN, but let's make sure */
|
|
cap_flag_value_t tmp;
|
|
|
|
ret = cap_get_flag(self->caps, CAP_NET_ADMIN, CAP_EFFECTIVE, &tmp);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("unable to get CAP_NET_ADMIN cap flag");
|
|
|
|
EXPECT_EQ(CAP_SET, tmp)
|
|
TH_LOG("expecting CAP_NET_ADMIN to be set for the test harness");
|
|
|
|
/* at this point we know CAP_NET_ADMIN is available, so setting the
|
|
* params with a busy_poll_budget > NAPI_POLL_WEIGHT should succeed
|
|
*/
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("EPIOCSPARAMS should allow busy_poll_budget > NAPI_POLL_WEIGHT");
|
|
|
|
/* remove CAP_NET_ADMIN from our effective set */
|
|
cap_value_t net_admin[] = { CAP_NET_ADMIN };
|
|
|
|
ret = cap_set_flag(self->caps, CAP_EFFECTIVE, 1, net_admin, CAP_CLEAR);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("couldn't clear CAP_NET_ADMIN");
|
|
|
|
ret = cap_set_proc(self->caps);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("cap_set_proc should drop CAP_NET_ADMIN");
|
|
|
|
/* this is now expected to fail */
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS should error busy_poll_budget > NAPI_POLL_WEIGHT");
|
|
|
|
EXPECT_EQ(EPERM, errno)
|
|
TH_LOG("EPIOCSPARAMS errno should be EPERM busy_poll_budget > NAPI_POLL_WEIGHT");
|
|
|
|
/* restore CAP_NET_ADMIN to our effective set */
|
|
ret = cap_set_flag(self->caps, CAP_EFFECTIVE, 1, net_admin, CAP_SET);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("couldn't restore CAP_NET_ADMIN");
|
|
|
|
ret = cap_set_proc(self->caps);
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("cap_set_proc should set CAP_NET_ADMIN");
|
|
|
|
self->invalid_params = (struct epoll_params *)0xdeadbeef;
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, self->invalid_params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("EPIOCSPARAMS should error when epoll_params is invalid");
|
|
|
|
EXPECT_EQ(EFAULT, errno)
|
|
TH_LOG("EPIOCSPARAMS should set errno to EFAULT when epoll_params is invalid");
|
|
}
|
|
|
|
TEST_F(epoll_busy_poll, test_set_and_get_valid)
|
|
{
|
|
int ret;
|
|
|
|
memset(&self->params, 0, sizeof(struct epoll_params));
|
|
|
|
self->params.busy_poll_usecs = 25;
|
|
self->params.busy_poll_budget = 16;
|
|
self->params.prefer_busy_poll = 1;
|
|
|
|
ret = ioctl(self->fd, EPIOCSPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("EPIOCSPARAMS with valid params should not error");
|
|
|
|
/* check that the kernel returns the same values back */
|
|
|
|
memset(&self->params, 0, sizeof(struct epoll_params));
|
|
|
|
ret = ioctl(self->fd, EPIOCGPARAMS, &self->params);
|
|
|
|
EXPECT_EQ(0, ret)
|
|
TH_LOG("EPIOCGPARAMS should not error");
|
|
|
|
EXPECT_EQ(25, self->params.busy_poll_usecs)
|
|
TH_LOG("params.busy_poll_usecs incorrect");
|
|
|
|
EXPECT_EQ(16, self->params.busy_poll_budget)
|
|
TH_LOG("params.busy_poll_budget incorrect");
|
|
|
|
EXPECT_EQ(1, self->params.prefer_busy_poll)
|
|
TH_LOG("params.prefer_busy_poll incorrect");
|
|
|
|
EXPECT_EQ(0, self->params.__pad)
|
|
TH_LOG("params.__pad was not 0");
|
|
}
|
|
|
|
TEST_F(epoll_busy_poll, test_invalid_ioctl)
|
|
{
|
|
int invalid_ioctl = EPIOCGPARAMS + 10;
|
|
int ret;
|
|
|
|
ret = ioctl(self->fd, invalid_ioctl, &self->params);
|
|
|
|
EXPECT_EQ(-1, ret)
|
|
TH_LOG("invalid ioctl should return error");
|
|
|
|
EXPECT_EQ(EINVAL, errno)
|
|
TH_LOG("invalid ioctl should set errno to EINVAL");
|
|
}
|
|
|
|
TEST_HARNESS_MAIN
|