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
TEST(init_namespace_inodes)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