// SPDX-License-Identifier: GPL-2.0 #define _GNU_SOURCE #include #include "test_deny_namespace.skel.h" #include #include "cap_helpers.h" #include static int wait_for_pid(pid_t pid) { int status, ret; again: ret = waitpid(pid, &status, 0); if (ret == -1) { if (errno == EINTR) goto again; return -1; } if (!WIFEXITED(status)) return -1; return WEXITSTATUS(status); } /* negative return value -> some internal error * positive return value -> userns creation failed * 0 -> userns creation succeeded */ static int create_user_ns(void) { pid_t pid; pid = fork(); if (pid < 0) return -1; if (pid == 0) { if (unshare(CLONE_NEWUSER)) _exit(EXIT_FAILURE); _exit(EXIT_SUCCESS); } return wait_for_pid(pid); } static void test_userns_create_bpf(void) { __u32 cap_mask = 1ULL << CAP_SYS_ADMIN; __u64 old_caps = 0; cap_enable_effective(cap_mask, &old_caps); ASSERT_OK(create_user_ns(), "priv new user ns"); cap_disable_effective(cap_mask, &old_caps); ASSERT_EQ(create_user_ns(), EPERM, "unpriv new user ns"); if (cap_mask & old_caps) cap_enable_effective(cap_mask, NULL); } static void test_unpriv_userns_create_no_bpf(void) { __u32 cap_mask = 1ULL << CAP_SYS_ADMIN; __u64 old_caps = 0; cap_disable_effective(cap_mask, &old_caps); ASSERT_OK(create_user_ns(), "no-bpf unpriv new user ns"); if (cap_mask & old_caps) cap_enable_effective(cap_mask, NULL); } void test_deny_namespace(void) { struct test_deny_namespace *skel = NULL; int err; if (test__start_subtest("unpriv_userns_create_no_bpf")) test_unpriv_userns_create_no_bpf(); skel = test_deny_namespace__open_and_load(); if (!ASSERT_OK_PTR(skel, "skel load")) goto close_prog; err = test_deny_namespace__attach(skel); if (!ASSERT_OK(err, "attach")) goto close_prog; if (test__start_subtest("userns_create_bpf")) test_userns_create_bpf(); test_deny_namespace__detach(skel); close_prog: test_deny_namespace__destroy(skel); }