// SPDX-License-Identifier: GPL-2.0+ #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <sys/stat.h> #include <sys/sysmacros.h> #include <sys/types.h> #include "../kselftest_harness.h" /* Remove a file, ignoring the result if it didn't exist. */ void rm(struct __test_metadata *_metadata, const char *pathname, int is_dir) { int rc; if (is_dir) rc = rmdir(pathname); else rc = unlink(pathname); if (rc < 0) { ASSERT_EQ(errno, ENOENT) { TH_LOG("Not ENOENT: %s", pathname); } } else { ASSERT_EQ(rc, 0) { TH_LOG("Failed to remove: %s", pathname); } } } FIXTURE(file) { char *pathname; int is_dir; }; FIXTURE_VARIANT(file) { const char *name; int expected; int is_dir; void (*setup)(struct __test_metadata *_metadata, FIXTURE_DATA(file) *self, const FIXTURE_VARIANT(file) *variant); int major, minor, mode; /* for mknod() */ }; void setup_link(struct __test_metadata *_metadata, FIXTURE_DATA(file) *self, const FIXTURE_VARIANT(file) *variant) { const char * const paths[] = { "/bin/true", "/usr/bin/true", }; int i; for (i = 0; i < ARRAY_SIZE(paths); i++) { if (access(paths[i], X_OK) == 0) { ASSERT_EQ(symlink(paths[i], self->pathname), 0); return; } } ASSERT_EQ(1, 0) { TH_LOG("Could not find viable 'true' binary"); } } FIXTURE_VARIANT_ADD(file, S_IFLNK) { .name = "S_IFLNK", .expected = ELOOP, .setup = setup_link, }; void setup_dir(struct __test_metadata *_metadata, FIXTURE_DATA(file) *self, const FIXTURE_VARIANT(file) *variant) { ASSERT_EQ(mkdir(self->pathname, 0755), 0); } FIXTURE_VARIANT_ADD(file, S_IFDIR) { .name = "S_IFDIR", .is_dir = 1, .expected = EACCES, .setup = setup_dir, }; void setup_node(struct __test_metadata *_metadata, FIXTURE_DATA(file) *self, const FIXTURE_VARIANT(file) *variant) { dev_t dev; int rc; dev = makedev(variant->major, variant->minor); rc = mknod(self->pathname, 0755 | variant->mode, dev); ASSERT_EQ(rc, 0) { if (errno == EPERM) SKIP(return, "Please run as root; cannot mknod(%s)", variant->name); } } FIXTURE_VARIANT_ADD(file, S_IFBLK) { .name = "S_IFBLK", .expected = EACCES, .setup = setup_node, /* /dev/loop0 */ .major = 7, .minor = 0, .mode = S_IFBLK, }; FIXTURE_VARIANT_ADD(file, S_IFCHR) { .name = "S_IFCHR", .expected = EACCES, .setup = setup_node, /* /dev/zero */ .major = 1, .minor = 5, .mode = S_IFCHR, }; void setup_fifo(struct __test_metadata *_metadata, FIXTURE_DATA(file) *self, const FIXTURE_VARIANT(file) *variant) { ASSERT_EQ(mkfifo(self->pathname, 0755), 0); } FIXTURE_VARIANT_ADD(file, S_IFIFO) { .name = "S_IFIFO", .expected = EACCES, .setup = setup_fifo, }; FIXTURE_SETUP(file) { ASSERT_GT(asprintf(&self->pathname, "%s.test", variant->name), 6); self->is_dir = variant->is_dir; rm(_metadata, self->pathname, variant->is_dir); variant->setup(_metadata, self, variant); } FIXTURE_TEARDOWN(file) { rm(_metadata, self->pathname, self->is_dir); } TEST_F(file, exec_errno) { char * const argv[2] = { (char * const)self->pathname, NULL }; EXPECT_LT(execv(argv[0], argv), 0); EXPECT_EQ(errno, variant->expected); } /* S_IFSOCK */ FIXTURE(sock) { int fd; }; FIXTURE_SETUP(sock) { self->fd = socket(AF_INET, SOCK_STREAM, 0); ASSERT_GE(self->fd, 0); } FIXTURE_TEARDOWN(sock) { if (self->fd >= 0) ASSERT_EQ(close(self->fd), 0); } TEST_F(sock, exec_errno) { char * const argv[2] = { " magic socket ", NULL }; char * const envp[1] = { NULL }; EXPECT_LT(fexecve(self->fd, argv, envp), 0); EXPECT_EQ(errno, EACCES); } TEST_HARNESS_MAIN