1*d093090eSChristian Brauner // SPDX-License-Identifier: GPL-2.0-or-later 2*d093090eSChristian Brauner // Copyright (c) 2025 Christian Brauner <brauner@kernel.org> 3*d093090eSChristian Brauner 4*d093090eSChristian Brauner #define _GNU_SOURCE 5*d093090eSChristian Brauner #include <fcntl.h> 6*d093090eSChristian Brauner #include <stdio.h> 7*d093090eSChristian Brauner #include <stdlib.h> 8*d093090eSChristian Brauner #include <sys/stat.h> 9*d093090eSChristian Brauner #include <unistd.h> 10*d093090eSChristian Brauner #include <errno.h> 11*d093090eSChristian Brauner #include <string.h> 12*d093090eSChristian Brauner #include <linux/nsfs.h> 13*d093090eSChristian Brauner 14*d093090eSChristian Brauner #include "../kselftest_harness.h" 15*d093090eSChristian Brauner 16*d093090eSChristian Brauner struct ns_info { 17*d093090eSChristian Brauner const char *name; 18*d093090eSChristian Brauner const char *proc_path; 19*d093090eSChristian Brauner unsigned int expected_ino; 20*d093090eSChristian Brauner }; 21*d093090eSChristian Brauner 22*d093090eSChristian Brauner static struct ns_info namespaces[] = { 23*d093090eSChristian Brauner { "ipc", "/proc/1/ns/ipc", IPC_NS_INIT_INO }, 24*d093090eSChristian Brauner { "uts", "/proc/1/ns/uts", UTS_NS_INIT_INO }, 25*d093090eSChristian Brauner { "user", "/proc/1/ns/user", USER_NS_INIT_INO }, 26*d093090eSChristian Brauner { "pid", "/proc/1/ns/pid", PID_NS_INIT_INO }, 27*d093090eSChristian Brauner { "cgroup", "/proc/1/ns/cgroup", CGROUP_NS_INIT_INO }, 28*d093090eSChristian Brauner { "time", "/proc/1/ns/time", TIME_NS_INIT_INO }, 29*d093090eSChristian Brauner { "net", "/proc/1/ns/net", NET_NS_INIT_INO }, 30*d093090eSChristian Brauner { "mnt", "/proc/1/ns/mnt", MNT_NS_INIT_INO }, 31*d093090eSChristian Brauner }; 32*d093090eSChristian Brauner 33*d093090eSChristian Brauner TEST(init_namespace_inodes) 34*d093090eSChristian Brauner { 35*d093090eSChristian Brauner struct stat st; 36*d093090eSChristian Brauner 37*d093090eSChristian Brauner for (int i = 0; i < sizeof(namespaces) / sizeof(namespaces[0]); i++) { 38*d093090eSChristian Brauner int ret = stat(namespaces[i].proc_path, &st); 39*d093090eSChristian Brauner 40*d093090eSChristian Brauner /* Some namespaces might not be available (e.g., time namespace on older kernels) */ 41*d093090eSChristian Brauner if (ret < 0) { 42*d093090eSChristian Brauner if (errno == ENOENT) { 43*d093090eSChristian Brauner ksft_test_result_skip("%s namespace not available\n", 44*d093090eSChristian Brauner namespaces[i].name); 45*d093090eSChristian Brauner continue; 46*d093090eSChristian Brauner } 47*d093090eSChristian Brauner ASSERT_GE(ret, 0) 48*d093090eSChristian Brauner TH_LOG("Failed to stat %s: %s", 49*d093090eSChristian Brauner namespaces[i].proc_path, strerror(errno)); 50*d093090eSChristian Brauner } 51*d093090eSChristian Brauner 52*d093090eSChristian Brauner ASSERT_EQ(st.st_ino, namespaces[i].expected_ino) 53*d093090eSChristian Brauner TH_LOG("Namespace %s has inode 0x%lx, expected 0x%x", 54*d093090eSChristian Brauner namespaces[i].name, st.st_ino, namespaces[i].expected_ino); 55*d093090eSChristian Brauner 56*d093090eSChristian Brauner ksft_print_msg("Namespace %s: inode 0x%lx matches expected 0x%x\n", 57*d093090eSChristian Brauner namespaces[i].name, st.st_ino, namespaces[i].expected_ino); 58*d093090eSChristian Brauner } 59*d093090eSChristian Brauner } 60*d093090eSChristian Brauner 61*d093090eSChristian Brauner TEST_HARNESS_MAIN 62