xref: /linux/tools/testing/selftests/namespaces/init_ino_test.c (revision 18b19abc3709b109676ffd1f48dcd332c2e477d4)
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