Commit | Line | Data |
---|---|---|
b2441318 | 1 | /* SPDX-License-Identifier: GPL-2.0 */ |
acce292c CLG |
2 | #ifndef _LINUX_USER_NAMESPACE_H |
3 | #define _LINUX_USER_NAMESPACE_H | |
4 | ||
5 | #include <linux/kref.h> | |
6 | #include <linux/nsproxy.h> | |
435d5f4b | 7 | #include <linux/ns_common.h> |
acce292c | 8 | #include <linux/sched.h> |
b2d5bfea | 9 | #include <linux/workqueue.h> |
cd9c513b | 10 | #include <linux/rwsem.h> |
cc5efc23 | 11 | #include <linux/sysctl.h> |
77ec739d | 12 | #include <linux/err.h> |
acce292c | 13 | |
6397fac4 CB |
14 | #define UID_GID_MAP_MAX_BASE_EXTENTS 5 |
15 | #define UID_GID_MAP_MAX_EXTENTS 340 | |
22d917d8 | 16 | |
aa4bf44d CB |
17 | struct uid_gid_extent { |
18 | u32 first; | |
19 | u32 lower_first; | |
20 | u32 count; | |
21 | }; | |
22 | ||
6397fac4 | 23 | struct uid_gid_map { /* 64 bytes -- 1 cache line */ |
22d917d8 | 24 | u32 nr_extents; |
aa4bf44d | 25 | union { |
6397fac4 | 26 | struct uid_gid_extent extent[UID_GID_MAP_MAX_BASE_EXTENTS]; |
aa4bf44d CB |
27 | struct { |
28 | struct uid_gid_extent *forward; | |
29 | struct uid_gid_extent *reverse; | |
30 | }; | |
31 | }; | |
22d917d8 EB |
32 | }; |
33 | ||
9cc46516 EB |
34 | #define USERNS_SETGROUPS_ALLOWED 1UL |
35 | ||
36 | #define USERNS_INIT_FLAGS USERNS_SETGROUPS_ALLOWED | |
37 | ||
f6b2db1a | 38 | struct ucounts; |
25f9c081 EB |
39 | |
40 | enum ucount_type { | |
41 | UCOUNT_USER_NAMESPACES, | |
f333c700 | 42 | UCOUNT_PID_NAMESPACES, |
f7af3d1c | 43 | UCOUNT_UTS_NAMESPACES, |
aba35661 | 44 | UCOUNT_IPC_NAMESPACES, |
70328660 | 45 | UCOUNT_NET_NAMESPACES, |
537f7ccb | 46 | UCOUNT_MNT_NAMESPACES, |
d08311dd | 47 | UCOUNT_CGROUP_NAMESPACES, |
769071ac | 48 | UCOUNT_TIME_NAMESPACES, |
1cce1eea NB |
49 | #ifdef CONFIG_INOTIFY_USER |
50 | UCOUNT_INOTIFY_INSTANCES, | |
51 | UCOUNT_INOTIFY_WATCHES, | |
52 | #endif | |
25f9c081 EB |
53 | UCOUNT_COUNTS, |
54 | }; | |
55 | ||
acce292c | 56 | struct user_namespace { |
22d917d8 EB |
57 | struct uid_gid_map uid_map; |
58 | struct uid_gid_map gid_map; | |
f76d207a | 59 | struct uid_gid_map projid_map; |
c61a2810 | 60 | atomic_t count; |
aeb3ae9d | 61 | struct user_namespace *parent; |
8742f229 | 62 | int level; |
783291e6 EB |
63 | kuid_t owner; |
64 | kgid_t group; | |
435d5f4b | 65 | struct ns_common ns; |
9cc46516 | 66 | unsigned long flags; |
f36f8c75 | 67 | |
b206f281 | 68 | #ifdef CONFIG_KEYS |
0f44e4d9 DH |
69 | /* List of joinable keyrings in this namespace. Modification access of |
70 | * these pointers is controlled by keyring_sem. Once | |
71 | * user_keyring_register is set, it won't be changed, so it can be | |
72 | * accessed directly with READ_ONCE(). | |
73 | */ | |
b206f281 | 74 | struct list_head keyring_name_list; |
0f44e4d9 DH |
75 | struct key *user_keyring_register; |
76 | struct rw_semaphore keyring_sem; | |
b206f281 DH |
77 | #endif |
78 | ||
f36f8c75 DH |
79 | /* Register of per-UID persistent keyrings for this namespace */ |
80 | #ifdef CONFIG_PERSISTENT_KEYRINGS | |
81 | struct key *persistent_keyring_register; | |
f36f8c75 | 82 | #endif |
b032132c | 83 | struct work_struct work; |
dbec2846 EB |
84 | #ifdef CONFIG_SYSCTL |
85 | struct ctl_table_set set; | |
86 | struct ctl_table_header *sysctls; | |
87 | #endif | |
f6b2db1a | 88 | struct ucounts *ucounts; |
25f9c081 | 89 | int ucount_max[UCOUNT_COUNTS]; |
3859a271 | 90 | } __randomize_layout; |
f6b2db1a EB |
91 | |
92 | struct ucounts { | |
93 | struct hlist_node node; | |
94 | struct user_namespace *ns; | |
95 | kuid_t uid; | |
040757f7 | 96 | int count; |
25f9c081 | 97 | atomic_t ucount[UCOUNT_COUNTS]; |
acce292c CLG |
98 | }; |
99 | ||
100 | extern struct user_namespace init_user_ns; | |
f6b2db1a EB |
101 | |
102 | bool setup_userns_sysctls(struct user_namespace *ns); | |
103 | void retire_userns_sysctls(struct user_namespace *ns); | |
25f9c081 EB |
104 | struct ucounts *inc_ucount(struct user_namespace *ns, kuid_t uid, enum ucount_type type); |
105 | void dec_ucount(struct ucounts *ucounts, enum ucount_type type); | |
acce292c CLG |
106 | |
107 | #ifdef CONFIG_USER_NS | |
108 | ||
109 | static inline struct user_namespace *get_user_ns(struct user_namespace *ns) | |
110 | { | |
111 | if (ns) | |
c61a2810 | 112 | atomic_inc(&ns->count); |
acce292c CLG |
113 | return ns; |
114 | } | |
115 | ||
18b6e041 | 116 | extern int create_user_ns(struct cred *new); |
b2e0d987 | 117 | extern int unshare_userns(unsigned long unshare_flags, struct cred **new_cred); |
b032132c | 118 | extern void __put_user_ns(struct user_namespace *ns); |
acce292c CLG |
119 | |
120 | static inline void put_user_ns(struct user_namespace *ns) | |
121 | { | |
c61a2810 | 122 | if (ns && atomic_dec_and_test(&ns->count)) |
b032132c | 123 | __put_user_ns(ns); |
acce292c CLG |
124 | } |
125 | ||
22d917d8 | 126 | struct seq_operations; |
ccf94f1b FF |
127 | extern const struct seq_operations proc_uid_seq_operations; |
128 | extern const struct seq_operations proc_gid_seq_operations; | |
129 | extern const struct seq_operations proc_projid_seq_operations; | |
22d917d8 EB |
130 | extern ssize_t proc_uid_map_write(struct file *, const char __user *, size_t, loff_t *); |
131 | extern ssize_t proc_gid_map_write(struct file *, const char __user *, size_t, loff_t *); | |
f76d207a | 132 | extern ssize_t proc_projid_map_write(struct file *, const char __user *, size_t, loff_t *); |
9cc46516 EB |
133 | extern ssize_t proc_setgroups_write(struct file *, const char __user *, size_t, loff_t *); |
134 | extern int proc_setgroups_show(struct seq_file *m, void *v); | |
273d2c67 | 135 | extern bool userns_may_setgroups(const struct user_namespace *ns); |
a2b42626 EB |
136 | extern bool in_userns(const struct user_namespace *ancestor, |
137 | const struct user_namespace *child); | |
d07b846f | 138 | extern bool current_in_userns(const struct user_namespace *target_ns); |
bcac25a5 | 139 | struct ns_common *ns_get_owner(struct ns_common *ns); |
acce292c CLG |
140 | #else |
141 | ||
142 | static inline struct user_namespace *get_user_ns(struct user_namespace *ns) | |
143 | { | |
144 | return &init_user_ns; | |
145 | } | |
146 | ||
18b6e041 | 147 | static inline int create_user_ns(struct cred *new) |
acce292c | 148 | { |
18b6e041 | 149 | return -EINVAL; |
acce292c CLG |
150 | } |
151 | ||
b2e0d987 EB |
152 | static inline int unshare_userns(unsigned long unshare_flags, |
153 | struct cred **new_cred) | |
154 | { | |
155 | if (unshare_flags & CLONE_NEWUSER) | |
156 | return -EINVAL; | |
157 | return 0; | |
158 | } | |
159 | ||
acce292c CLG |
160 | static inline void put_user_ns(struct user_namespace *ns) |
161 | { | |
162 | } | |
163 | ||
273d2c67 EB |
164 | static inline bool userns_may_setgroups(const struct user_namespace *ns) |
165 | { | |
166 | return true; | |
167 | } | |
d07b846f | 168 | |
a2b42626 EB |
169 | static inline bool in_userns(const struct user_namespace *ancestor, |
170 | const struct user_namespace *child) | |
171 | { | |
172 | return true; | |
173 | } | |
174 | ||
d07b846f SF |
175 | static inline bool current_in_userns(const struct user_namespace *target_ns) |
176 | { | |
177 | return true; | |
178 | } | |
bcac25a5 AV |
179 | |
180 | static inline struct ns_common *ns_get_owner(struct ns_common *ns) | |
181 | { | |
182 | return ERR_PTR(-EPERM); | |
183 | } | |
22d917d8 EB |
184 | #endif |
185 | ||
acce292c | 186 | #endif /* _LINUX_USER_H */ |