diff options
Diffstat (limited to 'tools/testing/selftests/filesystems')
9 files changed, 458 insertions, 20 deletions
| diff --git a/tools/testing/selftests/filesystems/.gitignore b/tools/testing/selftests/filesystems/.gitignore index fcbdb1297e24..64ac0dfa46b7 100644 --- a/tools/testing/selftests/filesystems/.gitignore +++ b/tools/testing/selftests/filesystems/.gitignore @@ -1,6 +1,7 @@  # SPDX-License-Identifier: GPL-2.0-only  dnotify_test  devpts_pts +fclog  file_stressor  anon_inode_test  kernfs_test diff --git a/tools/testing/selftests/filesystems/Makefile b/tools/testing/selftests/filesystems/Makefile index 73d4650af1a5..85427d7f19b9 100644 --- a/tools/testing/selftests/filesystems/Makefile +++ b/tools/testing/selftests/filesystems/Makefile @@ -1,7 +1,7 @@  # SPDX-License-Identifier: GPL-2.0  CFLAGS += $(KHDR_INCLUDES) -TEST_GEN_PROGS := devpts_pts file_stressor anon_inode_test kernfs_test +TEST_GEN_PROGS := devpts_pts file_stressor anon_inode_test kernfs_test fclog  TEST_GEN_PROGS_EXTENDED := dnotify_test  include ../lib.mk diff --git a/tools/testing/selftests/filesystems/fclog.c b/tools/testing/selftests/filesystems/fclog.c new file mode 100644 index 000000000000..912a8b755c3b --- /dev/null +++ b/tools/testing/selftests/filesystems/fclog.c @@ -0,0 +1,130 @@ +// SPDX-License-Identifier: GPL-2.0-or-later +/* + * Author: Aleksa Sarai <cyphar@cyphar.com> + * Copyright (C) 2025 SUSE LLC. + */ + +#include <assert.h> +#include <errno.h> +#include <sched.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <unistd.h> +#include <sys/mount.h> + +#include "../kselftest_harness.h" + +#define ASSERT_ERRNO(expected, _t, seen)				\ +	__EXPECT(expected, #expected,					\ +		({__typeof__(seen) _tmp_seen = (seen);			\ +		  _tmp_seen >= 0 ? _tmp_seen : -errno; }), #seen, _t, 1) + +#define ASSERT_ERRNO_EQ(expected, seen) \ +	ASSERT_ERRNO(expected, ==, seen) + +#define ASSERT_SUCCESS(seen) \ +	ASSERT_ERRNO(0, <=, seen) + +FIXTURE(ns) +{ +	int host_mntns; +}; + +FIXTURE_SETUP(ns) +{ +	/* Stash the old mntns. */ +	self->host_mntns = open("/proc/self/ns/mnt", O_RDONLY|O_CLOEXEC); +	ASSERT_SUCCESS(self->host_mntns); + +	/* Create a new mount namespace and make it private. */ +	ASSERT_SUCCESS(unshare(CLONE_NEWNS)); +	ASSERT_SUCCESS(mount(NULL, "/", NULL, MS_PRIVATE|MS_REC, NULL)); +} + +FIXTURE_TEARDOWN(ns) +{ +	ASSERT_SUCCESS(setns(self->host_mntns, CLONE_NEWNS)); +	ASSERT_SUCCESS(close(self->host_mntns)); +} + +TEST_F(ns, fscontext_log_enodata) +{ +	int fsfd = fsopen("tmpfs", FSOPEN_CLOEXEC); +	ASSERT_SUCCESS(fsfd); + +	/* A brand new fscontext has no log entries. */ +	char buf[128] = {}; +	for (int i = 0; i < 16; i++) +		ASSERT_ERRNO_EQ(-ENODATA, read(fsfd, buf, sizeof(buf))); + +	ASSERT_SUCCESS(close(fsfd)); +} + +TEST_F(ns, fscontext_log_errorfc) +{ +	int fsfd = fsopen("tmpfs", FSOPEN_CLOEXEC); +	ASSERT_SUCCESS(fsfd); + +	ASSERT_ERRNO_EQ(-EINVAL, fsconfig(fsfd, FSCONFIG_SET_STRING, "invalid-arg", "123", 0)); + +	char buf[128] = {}; +	ASSERT_SUCCESS(read(fsfd, buf, sizeof(buf))); +	EXPECT_STREQ("e tmpfs: Unknown parameter 'invalid-arg'\n", buf); + +	/* The message has been consumed. */ +	ASSERT_ERRNO_EQ(-ENODATA, read(fsfd, buf, sizeof(buf))); +	ASSERT_SUCCESS(close(fsfd)); +} + +TEST_F(ns, fscontext_log_errorfc_after_fsmount) +{ +	int fsfd = fsopen("tmpfs", FSOPEN_CLOEXEC); +	ASSERT_SUCCESS(fsfd); + +	ASSERT_ERRNO_EQ(-EINVAL, fsconfig(fsfd, FSCONFIG_SET_STRING, "invalid-arg", "123", 0)); + +	ASSERT_SUCCESS(fsconfig(fsfd, FSCONFIG_CMD_CREATE, NULL, NULL, 0)); +	int mfd = fsmount(fsfd, FSMOUNT_CLOEXEC, MOUNT_ATTR_NOEXEC | MOUNT_ATTR_NOSUID); +	ASSERT_SUCCESS(mfd); +	ASSERT_SUCCESS(move_mount(mfd, "", AT_FDCWD, "/tmp", MOVE_MOUNT_F_EMPTY_PATH)); + +	/* +	 * The fscontext log should still contain data even after +	 * FSCONFIG_CMD_CREATE and fsmount(). +	 */ +	char buf[128] = {}; +	ASSERT_SUCCESS(read(fsfd, buf, sizeof(buf))); +	EXPECT_STREQ("e tmpfs: Unknown parameter 'invalid-arg'\n", buf); + +	/* The message has been consumed. */ +	ASSERT_ERRNO_EQ(-ENODATA, read(fsfd, buf, sizeof(buf))); +	ASSERT_SUCCESS(close(fsfd)); +} + +TEST_F(ns, fscontext_log_emsgsize) +{ +	int fsfd = fsopen("tmpfs", FSOPEN_CLOEXEC); +	ASSERT_SUCCESS(fsfd); + +	ASSERT_ERRNO_EQ(-EINVAL, fsconfig(fsfd, FSCONFIG_SET_STRING, "invalid-arg", "123", 0)); + +	char buf[128] = {}; +	/* +	 * Attempting to read a message with too small a buffer should not +	 * result in the message getting consumed. +	 */ +	ASSERT_ERRNO_EQ(-EMSGSIZE, read(fsfd, buf, 0)); +	ASSERT_ERRNO_EQ(-EMSGSIZE, read(fsfd, buf, 1)); +	for (int i = 0; i < 16; i++) +		ASSERT_ERRNO_EQ(-EMSGSIZE, read(fsfd, buf, 16)); + +	ASSERT_SUCCESS(read(fsfd, buf, sizeof(buf))); +	EXPECT_STREQ("e tmpfs: Unknown parameter 'invalid-arg'\n", buf); + +	/* The message has been consumed. */ +	ASSERT_ERRNO_EQ(-ENODATA, read(fsfd, buf, sizeof(buf))); +	ASSERT_SUCCESS(close(fsfd)); +} + +TEST_HARNESS_MAIN diff --git a/tools/testing/selftests/filesystems/fuse/.gitignore b/tools/testing/selftests/filesystems/fuse/.gitignore new file mode 100644 index 000000000000..3e72e742d08e --- /dev/null +++ b/tools/testing/selftests/filesystems/fuse/.gitignore @@ -0,0 +1,3 @@ +# SPDX-License-Identifier: GPL-2.0-only +fuse_mnt +fusectl_test diff --git a/tools/testing/selftests/filesystems/fuse/Makefile b/tools/testing/selftests/filesystems/fuse/Makefile new file mode 100644 index 000000000000..612aad69a93a --- /dev/null +++ b/tools/testing/selftests/filesystems/fuse/Makefile @@ -0,0 +1,21 @@ +# SPDX-License-Identifier: GPL-2.0-or-later + +CFLAGS += -Wall -O2 -g $(KHDR_INCLUDES) + +TEST_GEN_PROGS := fusectl_test +TEST_GEN_FILES := fuse_mnt + +include ../../lib.mk + +VAR_CFLAGS := $(shell pkg-config fuse --cflags 2>/dev/null) +ifeq ($(VAR_CFLAGS),) +VAR_CFLAGS := -D_FILE_OFFSET_BITS=64 -I/usr/include/fuse +endif + +VAR_LDLIBS := $(shell pkg-config fuse --libs 2>/dev/null) +ifeq ($(VAR_LDLIBS),) +VAR_LDLIBS := -lfuse -pthread +endif + +$(OUTPUT)/fuse_mnt: CFLAGS += $(VAR_CFLAGS) +$(OUTPUT)/fuse_mnt: LDLIBS += $(VAR_LDLIBS) diff --git a/tools/testing/selftests/filesystems/fuse/fuse_mnt.c b/tools/testing/selftests/filesystems/fuse/fuse_mnt.c new file mode 100644 index 000000000000..d12b17f30fad --- /dev/null +++ b/tools/testing/selftests/filesystems/fuse/fuse_mnt.c @@ -0,0 +1,146 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * fusectl test file-system + * Creates a simple FUSE filesystem with a single read-write file (/test) + */ + +#define FUSE_USE_VERSION 26 + +#include <fuse.h> +#include <stdio.h> +#include <string.h> +#include <errno.h> +#include <fcntl.h> +#include <stdlib.h> +#include <unistd.h> + +#define MAX(a, b) ((a) > (b) ? (a) : (b)) + +static char *content; +static size_t content_size = 0; +static const char test_path[] = "/test"; + +static int test_getattr(const char *path, struct stat *st) +{ +	memset(st, 0, sizeof(*st)); + +	if (!strcmp(path, "/")) { +		st->st_mode = S_IFDIR | 0755; +		st->st_nlink = 2; +		return 0; +	} + +	if (!strcmp(path, test_path)) { +		st->st_mode = S_IFREG | 0664; +		st->st_nlink = 1; +		st->st_size = content_size; +		return 0; +	} + +	return -ENOENT; +} + +static int test_readdir(const char *path, void *buf, fuse_fill_dir_t filler, +			off_t offset, struct fuse_file_info *fi) +{ +	if (strcmp(path, "/")) +		return -ENOENT; + +	filler(buf, ".", NULL, 0); +	filler(buf, "..", NULL, 0); +	filler(buf, test_path + 1, NULL, 0); + +	return 0; +} + +static int test_open(const char *path, struct fuse_file_info *fi) +{ +	if (strcmp(path, test_path)) +		return -ENOENT; + +	return 0; +} + +static int test_read(const char *path, char *buf, size_t size, off_t offset, +		     struct fuse_file_info *fi) +{ +	if (strcmp(path, test_path) != 0) +		return -ENOENT; + +	if (!content || content_size == 0) +		return 0; + +	if (offset >= content_size) +		return 0; + +	if (offset + size > content_size) +		size = content_size - offset; + +	memcpy(buf, content + offset, size); + +	return size; +} + +static int test_write(const char *path, const char *buf, size_t size, +		      off_t offset, struct fuse_file_info *fi) +{ +	size_t new_size; + +	if (strcmp(path, test_path) != 0) +		return -ENOENT; + +	if(offset > content_size) +		return -EINVAL; + +	new_size = MAX(offset + size, content_size); + +	if (new_size > content_size) +		content = realloc(content, new_size); + +	content_size = new_size; + +	if (!content) +		return -ENOMEM; + +	memcpy(content + offset, buf, size); + +	return size; +} + +static int test_truncate(const char *path, off_t size) +{ +	if (strcmp(path, test_path) != 0) +		return -ENOENT; + +	if (size == 0) { +		free(content); +		content = NULL; +		content_size = 0; +		return 0; +	} + +	content = realloc(content, size); + +	if (!content) +		return -ENOMEM; + +	if (size > content_size) +		memset(content + content_size, 0, size - content_size); + +	content_size = size; +	return 0; +} + +static struct fuse_operations memfd_ops = { +	.getattr = test_getattr, +	.readdir = test_readdir, +	.open = test_open, +	.read = test_read, +	.write = test_write, +	.truncate = test_truncate, +}; + +int main(int argc, char *argv[]) +{ +	return fuse_main(argc, argv, &memfd_ops, NULL); +} diff --git a/tools/testing/selftests/filesystems/fuse/fusectl_test.c b/tools/testing/selftests/filesystems/fuse/fusectl_test.c new file mode 100644 index 000000000000..8d124d1cacb2 --- /dev/null +++ b/tools/testing/selftests/filesystems/fuse/fusectl_test.c @@ -0,0 +1,140 @@ +// SPDX-License-Identifier: GPL-2.0-or-later +// Copyright (c) 2025 Chen Linxuan <chenlinxuan@uniontech.com> + +#define _GNU_SOURCE + +#include <errno.h> +#include <fcntl.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/mount.h> +#include <sys/stat.h> +#include <sys/types.h> +#include <sys/wait.h> +#include <unistd.h> +#include <dirent.h> +#include <sched.h> +#include <linux/limits.h> + +#include "../../kselftest_harness.h" + +#define FUSECTL_MOUNTPOINT "/sys/fs/fuse/connections" +#define FUSE_MOUNTPOINT "/tmp/fuse_mnt_XXXXXX" +#define FUSE_DEVICE "/dev/fuse" +#define FUSECTL_TEST_VALUE "1" + +static void write_file(struct __test_metadata *const _metadata, +		       const char *path, const char *val) +{ +	int fd = open(path, O_WRONLY); +	size_t len = strlen(val); + +	ASSERT_GE(fd, 0); +	ASSERT_EQ(write(fd, val, len), len); +	ASSERT_EQ(close(fd), 0); +} + +FIXTURE(fusectl){ +	char fuse_mountpoint[sizeof(FUSE_MOUNTPOINT)]; +	int connection; +}; + +FIXTURE_SETUP(fusectl) +{ +	const char *fuse_mnt_prog = "./fuse_mnt"; +	int status, pid; +	struct stat statbuf; +	uid_t uid = getuid(); +	gid_t gid = getgid(); +	char buf[32]; + +	/* Setup userns */ +	ASSERT_EQ(unshare(CLONE_NEWNS|CLONE_NEWUSER), 0); +	sprintf(buf, "0 %d 1", uid); +	write_file(_metadata, "/proc/self/uid_map", buf); +	write_file(_metadata, "/proc/self/setgroups", "deny"); +	sprintf(buf, "0 %d 1", gid); +	write_file(_metadata, "/proc/self/gid_map", buf); +	ASSERT_EQ(mount("", "/", NULL, MS_REC|MS_PRIVATE, NULL), 0); + +	strcpy(self->fuse_mountpoint, FUSE_MOUNTPOINT); + +	if (!mkdtemp(self->fuse_mountpoint)) +		SKIP(return, +		     "Failed to create FUSE mountpoint %s", +		     strerror(errno)); + +	if (access(FUSECTL_MOUNTPOINT, F_OK)) +		SKIP(return, +		     "FUSE control filesystem not mounted"); + +	pid = fork(); +	if (pid < 0) +		SKIP(return, +		     "Failed to fork FUSE daemon process: %s", +		     strerror(errno)); + +	if (pid == 0) { +		execlp(fuse_mnt_prog, fuse_mnt_prog, self->fuse_mountpoint, NULL); +		exit(errno); +	} + +	waitpid(pid, &status, 0); +	if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) { +		SKIP(return, +		     "Failed to start FUSE daemon %s", +		     strerror(WEXITSTATUS(status))); +	} + +	if (stat(self->fuse_mountpoint, &statbuf)) +		SKIP(return, +		     "Failed to stat FUSE mountpoint %s", +		     strerror(errno)); + +	self->connection = statbuf.st_dev; +} + +FIXTURE_TEARDOWN(fusectl) +{ +	umount2(self->fuse_mountpoint, MNT_DETACH); +	rmdir(self->fuse_mountpoint); +} + +TEST_F(fusectl, abort) +{ +	char path_buf[PATH_MAX]; +	int abort_fd, test_fd, ret; + +	sprintf(path_buf, "/sys/fs/fuse/connections/%d/abort", self->connection); + +	ASSERT_EQ(0, access(path_buf, F_OK)); + +	abort_fd = open(path_buf, O_WRONLY); +	ASSERT_GE(abort_fd, 0); + +	sprintf(path_buf, "%s/test", self->fuse_mountpoint); + +	test_fd = open(path_buf, O_RDWR); +	ASSERT_GE(test_fd, 0); + +	ret = read(test_fd, path_buf, sizeof(path_buf)); +	ASSERT_EQ(ret, 0); + +	ret = write(test_fd, "test", sizeof("test")); +	ASSERT_EQ(ret, sizeof("test")); + +	ret = lseek(test_fd, 0, SEEK_SET); +	ASSERT_GE(ret, 0); + +	ret = write(abort_fd, FUSECTL_TEST_VALUE, sizeof(FUSECTL_TEST_VALUE)); +	ASSERT_GT(ret, 0); + +	close(abort_fd); + +	ret = read(test_fd, path_buf, sizeof(path_buf)); +	ASSERT_EQ(ret, -1); +	ASSERT_EQ(errno, ENOTCONN); +} + +TEST_HARNESS_MAIN diff --git a/tools/testing/selftests/filesystems/mount-notify/mount-notify_test.c b/tools/testing/selftests/filesystems/mount-notify/mount-notify_test.c index 63ce708d93ed..e4b7c2b457ee 100644 --- a/tools/testing/selftests/filesystems/mount-notify/mount-notify_test.c +++ b/tools/testing/selftests/filesystems/mount-notify/mount-notify_test.c @@ -2,6 +2,13 @@  // Copyright (c) 2025 Miklos Szeredi <miklos@szeredi.hu>  #define _GNU_SOURCE + +// Needed for linux/fanotify.h +typedef struct { +	int	val[2]; +} __kernel_fsid_t; +#define __kernel_fsid_t __kernel_fsid_t +  #include <fcntl.h>  #include <sched.h>  #include <stdio.h> @@ -10,20 +17,12 @@  #include <sys/mount.h>  #include <unistd.h>  #include <sys/syscall.h> +#include <sys/fanotify.h>  #include "../../kselftest_harness.h"  #include "../statmount/statmount.h"  #include "../utils.h" -// Needed for linux/fanotify.h -#ifndef __kernel_fsid_t -typedef struct { -	int	val[2]; -} __kernel_fsid_t; -#endif - -#include <sys/fanotify.h> -  static const char root_mntpoint_templ[] = "/tmp/mount-notify_test_root.XXXXXX";  static const int mark_cmds[] = { diff --git a/tools/testing/selftests/filesystems/mount-notify/mount-notify_test_ns.c b/tools/testing/selftests/filesystems/mount-notify/mount-notify_test_ns.c index 090a5ca65004..9f57ca46e3af 100644 --- a/tools/testing/selftests/filesystems/mount-notify/mount-notify_test_ns.c +++ b/tools/testing/selftests/filesystems/mount-notify/mount-notify_test_ns.c @@ -2,6 +2,13 @@  // Copyright (c) 2025 Miklos Szeredi <miklos@szeredi.hu>  #define _GNU_SOURCE + +// Needed for linux/fanotify.h +typedef struct { +	int	val[2]; +} __kernel_fsid_t; +#define __kernel_fsid_t __kernel_fsid_t +  #include <fcntl.h>  #include <sched.h>  #include <stdio.h> @@ -10,21 +17,12 @@  #include <sys/mount.h>  #include <unistd.h>  #include <sys/syscall.h> +#include <sys/fanotify.h>  #include "../../kselftest_harness.h" -#include "../../pidfd/pidfd.h"  #include "../statmount/statmount.h"  #include "../utils.h" -// Needed for linux/fanotify.h -#ifndef __kernel_fsid_t -typedef struct { -	int	val[2]; -} __kernel_fsid_t; -#endif - -#include <sys/fanotify.h> -  static const char root_mntpoint_templ[] = "/tmp/mount-notify_test_root.XXXXXX";  static const int mark_types[] = { | 
