1*b2441318SGreg Kroah-Hartman /* SPDX-License-Identifier: GPL-2.0 */
2f76d207aSEric W. Biederman #ifndef _LINUX_PROJID_H
3f76d207aSEric W. Biederman #define _LINUX_PROJID_H
4f76d207aSEric W. Biederman
5f76d207aSEric W. Biederman /*
6f76d207aSEric W. Biederman * A set of types for the internal kernel types representing project ids.
7f76d207aSEric W. Biederman *
8f76d207aSEric W. Biederman * The types defined in this header allow distinguishing which project ids in
9f76d207aSEric W. Biederman * the kernel are values used by userspace and which project id values are
10f76d207aSEric W. Biederman * the internal kernel values. With the addition of user namespaces the values
11f76d207aSEric W. Biederman * can be different. Using the type system makes it possible for the compiler
12f76d207aSEric W. Biederman * to detect when we overlook these differences.
13f76d207aSEric W. Biederman *
14f76d207aSEric W. Biederman */
15f76d207aSEric W. Biederman #include <linux/types.h>
16f76d207aSEric W. Biederman
17f76d207aSEric W. Biederman struct user_namespace;
18f76d207aSEric W. Biederman extern struct user_namespace init_user_ns;
19f76d207aSEric W. Biederman
20f76d207aSEric W. Biederman typedef __kernel_uid32_t projid_t;
21f76d207aSEric W. Biederman
22f76d207aSEric W. Biederman typedef struct {
23f76d207aSEric W. Biederman projid_t val;
24f76d207aSEric W. Biederman } kprojid_t;
25f76d207aSEric W. Biederman
__kprojid_val(kprojid_t projid)26f76d207aSEric W. Biederman static inline projid_t __kprojid_val(kprojid_t projid)
27f76d207aSEric W. Biederman {
28f76d207aSEric W. Biederman return projid.val;
29f76d207aSEric W. Biederman }
30f76d207aSEric W. Biederman
31f76d207aSEric W. Biederman #define KPROJIDT_INIT(value) (kprojid_t){ value }
32f76d207aSEric W. Biederman
33f76d207aSEric W. Biederman #define INVALID_PROJID KPROJIDT_INIT(-1)
34f76d207aSEric W. Biederman #define OVERFLOW_PROJID 65534
35f76d207aSEric W. Biederman
projid_eq(kprojid_t left,kprojid_t right)36f76d207aSEric W. Biederman static inline bool projid_eq(kprojid_t left, kprojid_t right)
37f76d207aSEric W. Biederman {
38f76d207aSEric W. Biederman return __kprojid_val(left) == __kprojid_val(right);
39f76d207aSEric W. Biederman }
40f76d207aSEric W. Biederman
projid_lt(kprojid_t left,kprojid_t right)41f76d207aSEric W. Biederman static inline bool projid_lt(kprojid_t left, kprojid_t right)
42f76d207aSEric W. Biederman {
43f76d207aSEric W. Biederman return __kprojid_val(left) < __kprojid_val(right);
44f76d207aSEric W. Biederman }
45f76d207aSEric W. Biederman
projid_valid(kprojid_t projid)46f76d207aSEric W. Biederman static inline bool projid_valid(kprojid_t projid)
47f76d207aSEric W. Biederman {
48f76d207aSEric W. Biederman return !projid_eq(projid, INVALID_PROJID);
49f76d207aSEric W. Biederman }
50f76d207aSEric W. Biederman
51f76d207aSEric W. Biederman #ifdef CONFIG_USER_NS
52f76d207aSEric W. Biederman
53f76d207aSEric W. Biederman extern kprojid_t make_kprojid(struct user_namespace *from, projid_t projid);
54f76d207aSEric W. Biederman
55f76d207aSEric W. Biederman extern projid_t from_kprojid(struct user_namespace *to, kprojid_t projid);
56f76d207aSEric W. Biederman extern projid_t from_kprojid_munged(struct user_namespace *to, kprojid_t projid);
57f76d207aSEric W. Biederman
kprojid_has_mapping(struct user_namespace * ns,kprojid_t projid)58f76d207aSEric W. Biederman static inline bool kprojid_has_mapping(struct user_namespace *ns, kprojid_t projid)
59f76d207aSEric W. Biederman {
60f76d207aSEric W. Biederman return from_kprojid(ns, projid) != (projid_t)-1;
61f76d207aSEric W. Biederman }
62f76d207aSEric W. Biederman
63f76d207aSEric W. Biederman #else
64f76d207aSEric W. Biederman
make_kprojid(struct user_namespace * from,projid_t projid)65f76d207aSEric W. Biederman static inline kprojid_t make_kprojid(struct user_namespace *from, projid_t projid)
66f76d207aSEric W. Biederman {
67f76d207aSEric W. Biederman return KPROJIDT_INIT(projid);
68f76d207aSEric W. Biederman }
69f76d207aSEric W. Biederman
from_kprojid(struct user_namespace * to,kprojid_t kprojid)70f76d207aSEric W. Biederman static inline projid_t from_kprojid(struct user_namespace *to, kprojid_t kprojid)
71f76d207aSEric W. Biederman {
72f76d207aSEric W. Biederman return __kprojid_val(kprojid);
73f76d207aSEric W. Biederman }
74f76d207aSEric W. Biederman
from_kprojid_munged(struct user_namespace * to,kprojid_t kprojid)75f76d207aSEric W. Biederman static inline projid_t from_kprojid_munged(struct user_namespace *to, kprojid_t kprojid)
76f76d207aSEric W. Biederman {
77f76d207aSEric W. Biederman projid_t projid = from_kprojid(to, kprojid);
78f76d207aSEric W. Biederman if (projid == (projid_t)-1)
79f76d207aSEric W. Biederman projid = OVERFLOW_PROJID;
80f76d207aSEric W. Biederman return projid;
81f76d207aSEric W. Biederman }
82f76d207aSEric W. Biederman
kprojid_has_mapping(struct user_namespace * ns,kprojid_t projid)83f76d207aSEric W. Biederman static inline bool kprojid_has_mapping(struct user_namespace *ns, kprojid_t projid)
84f76d207aSEric W. Biederman {
85f76d207aSEric W. Biederman return true;
86f76d207aSEric W. Biederman }
87f76d207aSEric W. Biederman
88f76d207aSEric W. Biederman #endif /* CONFIG_USER_NS */
89f76d207aSEric W. Biederman
90f76d207aSEric W. Biederman #endif /* _LINUX_PROJID_H */
91