/* SPDX-License-Identifier: LGPL-2.1-or-later */ #include #include #include "alloc-util.h" #include "capability-util.h" #include "fd-util.h" #include "fileio.h" #include "fs-util.h" #include "missing_magic.h" #include "missing_mount.h" #include "mkdir.h" #include "mount-util.h" #include "mountpoint-util.h" #include "namespace-util.h" #include "path-util.h" #include "process-util.h" #include "rm-rf.h" #include "stat-util.h" #include "string-util.h" #include "strv.h" #include "tests.h" #include "tmpfile-util.h" TEST(remount_and_move_sub_mounts) { int r; if (geteuid() != 0 || have_effective_cap(CAP_SYS_ADMIN) <= 0) return (void) log_tests_skipped("not running privileged"); r = safe_fork("(remount-and-move-sub-mounts)", FORK_RESET_SIGNALS | FORK_CLOSE_ALL_FDS | FORK_DEATHSIG | FORK_WAIT | FORK_REOPEN_LOG | FORK_LOG | FORK_NEW_MOUNTNS | FORK_MOUNTNS_SLAVE, NULL); assert_se(r >= 0); if (r == 0) { _cleanup_free_ char *d = NULL, *fn = NULL; assert_se(mkdtemp_malloc(NULL, &d) >= 0); assert_se(mount_nofollow_verbose(LOG_DEBUG, "tmpfs", d, "tmpfs", MS_NOSUID|MS_NODEV, NULL) >= 0); assert_se(fn = path_join(d, "memo")); assert_se(write_string_file(fn, d, WRITE_STRING_FILE_CREATE|WRITE_STRING_FILE_AVOID_NEWLINE) >= 0); assert_se(access(fn, F_OK) >= 0); /* Create fs tree */ FOREACH_STRING(p, "sub1", "sub1/hoge", "sub1/foo", "sub2", "sub2/aaa", "sub2/bbb") { _cleanup_free_ char *where = NULL, *filename = NULL; assert_se(where = path_join(d, p)); assert_se(mkdir_p(where, 0755) >= 0); assert_se(mount_nofollow_verbose(LOG_DEBUG, "tmpfs", where, "tmpfs", MS_NOSUID|MS_NODEV, NULL) >= 0); assert_se(filename = path_join(where, "memo")); assert_se(write_string_file(filename, where, WRITE_STRING_FILE_CREATE|WRITE_STRING_FILE_AVOID_NEWLINE) >= 0); assert_se(access(filename, F_OK) >= 0); } /* Hide sub1. */ FOREACH_STRING(p, "sub1", "sub1/hogehoge", "sub1/foofoo") { _cleanup_free_ char *where = NULL, *filename = NULL; assert_se(where = path_join(d, p)); assert_se(mkdir_p(where, 0755) >= 0); assert_se(mount_nofollow_verbose(LOG_DEBUG, "tmpfs", where, "tmpfs", MS_NOSUID|MS_NODEV, NULL) >= 0); assert_se(filename = path_join(where, "memo")); assert_se(write_string_file(filename, where, WRITE_STRING_FILE_CREATE|WRITE_STRING_FILE_AVOID_NEWLINE) >= 0); assert_se(access(filename, F_OK) >= 0); } /* Remount the main fs. */ r = remount_and_move_sub_mounts("tmpfs", d, "tmpfs", MS_NOSUID|MS_NODEV, NULL); if (r == -EINVAL || (r < 0 && ERRNO_IS_NOT_SUPPORTED(r))) { log_tests_skipped_errno(r, "The kernel seems too old: %m"); _exit(EXIT_SUCCESS); } /* Check the file in the main fs does not exist. */ assert_se(access(fn, F_OK) < 0 && errno == ENOENT); /* Check the files in sub-mounts are kept. */ FOREACH_STRING(p, "sub1", "sub1/hogehoge", "sub1/foofoo", "sub2", "sub2/aaa", "sub2/bbb") { _cleanup_free_ char *where = NULL, *filename = NULL, *content = NULL; assert_se(where = path_join(d, p)); assert_se(filename = path_join(where, "memo")); assert_se(read_full_file(filename, &content, NULL) >= 0); assert_se(streq(content, where)); } /* umount sub1, and check if the previously hidden sub-mounts are dropped. */ FOREACH_STRING(p, "sub1/hoge", "sub1/foo") { _cleanup_free_ char *where = NULL; assert_se(where = path_join(d, p)); assert_se(access(where, F_OK) < 0 && errno == ENOENT); } _exit(EXIT_SUCCESS); } } TEST(remount_sysfs) { int r; if (geteuid() != 0 || have_effective_cap(CAP_SYS_ADMIN) <= 0) return (void) log_tests_skipped("not running privileged"); if (path_is_fs_type("/sys", SYSFS_MAGIC) <= 0) return (void) log_tests_skipped("sysfs is not mounted on /sys"); if (access("/sys/class/net/dummy-test-mnt", F_OK) < 0) return (void) log_tests_skipped_errno(errno, "The network interface dummy-test-mnt does not exit"); r = safe_fork("(remount-sysfs)", FORK_RESET_SIGNALS | FORK_CLOSE_ALL_FDS | FORK_DEATHSIG | FORK_WAIT | FORK_REOPEN_LOG | FORK_LOG | FORK_NEW_MOUNTNS | FORK_MOUNTNS_SLAVE, NULL); assert_se(r >= 0); if (r == 0) { assert_se(unshare(CLONE_NEWNET) >= 0); /* Even unshare()ed, the interfaces in the main namespace can be accessed through sysfs. */ assert_se(access("/sys/class/net/lo", F_OK) >= 0); assert_se(access("/sys/class/net/dummy-test-mnt", F_OK) >= 0); r = remount_sysfs("/sys"); if (r == -EINVAL || (r < 0 && ERRNO_IS_NOT_SUPPORTED(r))) { log_tests_skipped_errno(r, "The kernel seems too old: %m"); _exit(EXIT_SUCCESS); } /* After remounting sysfs, the interfaces in the main namespace cannot be accessed. */ assert_se(access("/sys/class/net/lo", F_OK) >= 0); assert_se(access("/sys/class/net/dummy-test-mnt", F_OK) < 0 && errno == ENOENT); _exit(EXIT_SUCCESS); } } TEST(mount_option_mangle) { char *opts = NULL; unsigned long f; assert_se(mount_option_mangle(NULL, MS_RDONLY|MS_NOSUID, &f, &opts) == 0); assert_se(f == (MS_RDONLY|MS_NOSUID)); assert_se(opts == NULL); assert_se(mount_option_mangle("", MS_RDONLY|MS_NOSUID, &f, &opts) == 0); assert_se(f == (MS_RDONLY|MS_NOSUID)); assert_se(opts == NULL); assert_se(mount_option_mangle("ro,nosuid,nodev,noexec", 0, &f, &opts) == 0); assert_se(f == (MS_RDONLY|MS_NOSUID|MS_NODEV|MS_NOEXEC)); assert_se(opts == NULL); assert_se(mount_option_mangle("ro,nosuid,nodev,noexec,mode=0755", 0, &f, &opts) == 0); assert_se(f == (MS_RDONLY|MS_NOSUID|MS_NODEV|MS_NOEXEC)); assert_se(streq(opts, "mode=0755")); opts = mfree(opts); assert_se(mount_option_mangle("rw,nosuid,foo,hogehoge,nodev,mode=0755", 0, &f, &opts) == 0); assert_se(f == (MS_NOSUID|MS_NODEV)); assert_se(streq(opts, "foo,hogehoge,mode=0755")); opts = mfree(opts); assert_se(mount_option_mangle("rw,nosuid,nodev,noexec,relatime,net_cls,net_prio", MS_RDONLY, &f, &opts) == 0); assert_se(f == (MS_NOSUID|MS_NODEV|MS_NOEXEC|MS_RELATIME)); assert_se(streq(opts, "net_cls,net_prio")); opts = mfree(opts); assert_se(mount_option_mangle("rw,nosuid,nodev,relatime,size=1630748k,mode=0700,uid=1000,gid=1000", MS_RDONLY, &f, &opts) == 0); assert_se(f == (MS_NOSUID|MS_NODEV|MS_RELATIME)); assert_se(streq(opts, "size=1630748k,mode=0700,uid=1000,gid=1000")); opts = mfree(opts); assert_se(mount_option_mangle("size=1630748k,rw,gid=1000,,,nodev,relatime,,mode=0700,nosuid,uid=1000", MS_RDONLY, &f, &opts) == 0); assert_se(f == (MS_NOSUID|MS_NODEV|MS_RELATIME)); assert_se(streq(opts, "size=1630748k,gid=1000,mode=0700,uid=1000")); opts = mfree(opts); assert_se(mount_option_mangle("rw,exec,size=8143984k,nr_inodes=2035996,mode=0755", MS_RDONLY|MS_NOSUID|MS_NOEXEC|MS_NODEV, &f, &opts) == 0); assert_se(f == (MS_NOSUID|MS_NODEV)); assert_se(streq(opts, "size=8143984k,nr_inodes=2035996,mode=0755")); opts = mfree(opts); assert_se(mount_option_mangle("rw,relatime,fmask=0022,,,dmask=0022", MS_RDONLY, &f, &opts) == 0); assert_se(f == MS_RELATIME); assert_se(streq(opts, "fmask=0022,dmask=0022")); opts = mfree(opts); assert_se(mount_option_mangle("rw,relatime,fmask=0022,dmask=0022,\"hogehoge", MS_RDONLY, &f, &opts) < 0); assert_se(mount_option_mangle("mode=01777,size=10%,nr_inodes=400k,uid=496107520,gid=496107520,context=\"system_u:object_r:svirt_sandbox_file_t:s0:c0,c1\"", 0, &f, &opts) == 0); assert_se(f == 0); assert_se(streq(opts, "mode=01777,size=10%,nr_inodes=400k,uid=496107520,gid=496107520,context=\"system_u:object_r:svirt_sandbox_file_t:s0:c0,c1\"")); opts = mfree(opts); } static void test_mount_flags_to_string_one(unsigned long flags, const char *expected) { _cleanup_free_ char *x = NULL; int r; r = mount_flags_to_string(flags, &x); log_info("flags: %#lX → %d/\"%s\"", flags, r, strnull(x)); assert_se(r >= 0); assert_se(streq(x, expected)); } TEST(mount_flags_to_string) { test_mount_flags_to_string_one(0, "0"); test_mount_flags_to_string_one(MS_RDONLY, "MS_RDONLY"); test_mount_flags_to_string_one(MS_NOSUID, "MS_NOSUID"); test_mount_flags_to_string_one(MS_NODEV, "MS_NODEV"); test_mount_flags_to_string_one(MS_NOEXEC, "MS_NOEXEC"); test_mount_flags_to_string_one(MS_SYNCHRONOUS, "MS_SYNCHRONOUS"); test_mount_flags_to_string_one(MS_REMOUNT, "MS_REMOUNT"); test_mount_flags_to_string_one(MS_MANDLOCK, "MS_MANDLOCK"); test_mount_flags_to_string_one(MS_DIRSYNC, "MS_DIRSYNC"); test_mount_flags_to_string_one(MS_NOSYMFOLLOW, "MS_NOSYMFOLLOW"); test_mount_flags_to_string_one(MS_NOATIME, "MS_NOATIME"); test_mount_flags_to_string_one(MS_NODIRATIME, "MS_NODIRATIME"); test_mount_flags_to_string_one(MS_BIND, "MS_BIND"); test_mount_flags_to_string_one(MS_MOVE, "MS_MOVE"); test_mount_flags_to_string_one(MS_REC, "MS_REC"); test_mount_flags_to_string_one(MS_SILENT, "MS_SILENT"); test_mount_flags_to_string_one(MS_POSIXACL, "MS_POSIXACL"); test_mount_flags_to_string_one(MS_UNBINDABLE, "MS_UNBINDABLE"); test_mount_flags_to_string_one(MS_PRIVATE, "MS_PRIVATE"); test_mount_flags_to_string_one(MS_SLAVE, "MS_SLAVE"); test_mount_flags_to_string_one(MS_SHARED, "MS_SHARED"); test_mount_flags_to_string_one(MS_RELATIME, "MS_RELATIME"); test_mount_flags_to_string_one(MS_KERNMOUNT, "MS_KERNMOUNT"); test_mount_flags_to_string_one(MS_I_VERSION, "MS_I_VERSION"); test_mount_flags_to_string_one(MS_STRICTATIME, "MS_STRICTATIME"); test_mount_flags_to_string_one(MS_LAZYTIME, "MS_LAZYTIME"); test_mount_flags_to_string_one(MS_LAZYTIME|MS_STRICTATIME, "MS_STRICTATIME|MS_LAZYTIME"); test_mount_flags_to_string_one(UINT_MAX, "MS_RDONLY|MS_NOSUID|MS_NODEV|MS_NOEXEC|MS_SYNCHRONOUS|MS_REMOUNT|" "MS_MANDLOCK|MS_DIRSYNC|MS_NOSYMFOLLOW|MS_NOATIME|MS_NODIRATIME|" "MS_BIND|MS_MOVE|MS_REC|MS_SILENT|MS_POSIXACL|MS_UNBINDABLE|" "MS_PRIVATE|MS_SLAVE|MS_SHARED|MS_RELATIME|MS_KERNMOUNT|" "MS_I_VERSION|MS_STRICTATIME|MS_LAZYTIME|fc000200"); } TEST(bind_remount_recursive) { _cleanup_(rm_rf_physical_and_freep) char *tmp = NULL; _cleanup_free_ char *subdir = NULL; if (geteuid() != 0 || have_effective_cap(CAP_SYS_ADMIN) <= 0) { (void) log_tests_skipped("not running privileged"); return; } assert_se(mkdtemp_malloc("/tmp/XXXXXX", &tmp) >= 0); subdir = path_join(tmp, "subdir"); assert_se(subdir); assert_se(mkdir(subdir, 0755) >= 0); FOREACH_STRING(p, "/usr", "/sys", "/", tmp) { pid_t pid; pid = fork(); assert_se(pid >= 0); if (pid == 0) { struct statvfs svfs; /* child */ assert_se(detach_mount_namespace() >= 0); /* Check that the subdir is writable (it must be because it's in /tmp) */ assert_se(statvfs(subdir, &svfs) >= 0); assert_se(!FLAGS_SET(svfs.f_flag, ST_RDONLY)); /* Make the subdir a bind mount */ assert_se(mount_nofollow(subdir, subdir, NULL, MS_BIND|MS_REC, NULL) >= 0); /* Ensure it's still writable */ assert_se(statvfs(subdir, &svfs) >= 0); assert_se(!FLAGS_SET(svfs.f_flag, ST_RDONLY)); /* Now mark the path we currently run for read-only */ assert_se(bind_remount_recursive(p, MS_RDONLY, MS_RDONLY, path_equal(p, "/sys") ? STRV_MAKE("/sys/kernel") : NULL) >= 0); /* Ensure that this worked on the top-level */ assert_se(statvfs(p, &svfs) >= 0); assert_se(FLAGS_SET(svfs.f_flag, ST_RDONLY)); /* And ensure this had an effect on the subdir exactly if we are talking about a path above the subdir */ assert_se(statvfs(subdir, &svfs) >= 0); assert_se(FLAGS_SET(svfs.f_flag, ST_RDONLY) == !!path_startswith(subdir, p)); _exit(EXIT_SUCCESS); } assert_se(wait_for_terminate_and_check("test-remount-rec", pid, WAIT_LOG) == EXIT_SUCCESS); } } TEST(bind_remount_one) { pid_t pid; if (geteuid() != 0 || have_effective_cap(CAP_SYS_ADMIN) <= 0) { (void) log_tests_skipped("not running privileged"); return; } pid = fork(); assert_se(pid >= 0); if (pid == 0) { /* child */ _cleanup_fclose_ FILE *proc_self_mountinfo = NULL; assert_se(detach_mount_namespace() >= 0); assert_se(fopen_unlocked("/proc/self/mountinfo", "re", &proc_self_mountinfo) >= 0); assert_se(bind_remount_one_with_mountinfo("/run", MS_RDONLY, MS_RDONLY, proc_self_mountinfo) >= 0); assert_se(bind_remount_one_with_mountinfo("/run", MS_NOEXEC, MS_RDONLY|MS_NOEXEC, proc_self_mountinfo) >= 0); assert_se(bind_remount_one_with_mountinfo("/proc/idontexist", MS_RDONLY, MS_RDONLY, proc_self_mountinfo) == -ENOENT); assert_se(bind_remount_one_with_mountinfo("/proc/self", MS_RDONLY, MS_RDONLY, proc_self_mountinfo) == -EINVAL); assert_se(bind_remount_one_with_mountinfo("/", MS_RDONLY, MS_RDONLY, proc_self_mountinfo) >= 0); _exit(EXIT_SUCCESS); } assert_se(wait_for_terminate_and_check("test-remount-one", pid, WAIT_LOG) == EXIT_SUCCESS); } TEST(make_mount_point_inode) { _cleanup_(rm_rf_physical_and_freep) char *d = NULL; const char *src_file, *src_dir, *dst_file, *dst_dir; struct stat st; assert_se(mkdtemp_malloc(NULL, &d) >= 0); src_file = strjoina(d, "/src/file"); src_dir = strjoina(d, "/src/dir"); dst_file = strjoina(d, "/dst/file"); dst_dir = strjoina(d, "/dst/dir"); assert_se(mkdir_p(src_dir, 0755) >= 0); assert_se(mkdir_parents(dst_file, 0755) >= 0); assert_se(touch(src_file) >= 0); assert_se(make_mount_point_inode_from_path(src_file, dst_file, 0755) >= 0); assert_se(make_mount_point_inode_from_path(src_dir, dst_dir, 0755) >= 0); assert_se(stat(dst_dir, &st) == 0); assert_se(S_ISDIR(st.st_mode)); assert_se(stat(dst_file, &st) == 0); assert_se(S_ISREG(st.st_mode)); assert_se(!(S_IXUSR & st.st_mode)); assert_se(!(S_IXGRP & st.st_mode)); assert_se(!(S_IXOTH & st.st_mode)); assert_se(unlink(dst_file) == 0); assert_se(rmdir(dst_dir) == 0); assert_se(stat(src_file, &st) == 0); assert_se(make_mount_point_inode_from_stat(&st, dst_file, 0755) >= 0); assert_se(stat(src_dir, &st) == 0); assert_se(make_mount_point_inode_from_stat(&st, dst_dir, 0755) >= 0); assert_se(stat(dst_dir, &st) == 0); assert_se(S_ISDIR(st.st_mode)); assert_se(stat(dst_file, &st) == 0); assert_se(S_ISREG(st.st_mode)); assert_se(!(S_IXUSR & st.st_mode)); assert_se(!(S_IXGRP & st.st_mode)); assert_se(!(S_IXOTH & st.st_mode)); } static int intro(void) { /* Create a dummy network interface for testing remount_sysfs(). */ (void) system("ip link add dummy-test-mnt type dummy"); return 0; } static int outro(void) { (void) system("ip link del dummy-test-mnt"); return 0; } DEFINE_TEST_MAIN_FULL(LOG_DEBUG, intro, outro);