Annotation of embedaddon/bird2/sysdep/linux/syspriv.h, revision 1.1
1.1 ! misho 1: #ifndef _BIRD_SYSPRIV_H_
! 2: #define _BIRD_SYSPRIV_H_
! 3:
! 4: #ifndef _GNU_SOURCE
! 5: #define _GNU_SOURCE
! 6: #endif
! 7:
! 8: #include <unistd.h>
! 9: #include <sys/prctl.h>
! 10: #include <linux/capability.h>
! 11:
! 12: #ifndef _LINUX_CAPABILITY_VERSION_3
! 13: #define _LINUX_CAPABILITY_VERSION_3 0x20080522
! 14: #define _LINUX_CAPABILITY_U32S_3 2
! 15: #endif
! 16:
! 17: /* CAP_TO_MASK is missing in CentOS header files */
! 18: #ifndef CAP_TO_MASK
! 19: #define CAP_TO_MASK(x) (1 << ((x) & 31))
! 20: #endif
! 21:
! 22: /* capset() prototype is missing ... */
! 23: int capset(cap_user_header_t hdrp, const cap_user_data_t datap);
! 24:
! 25: static inline int
! 26: set_capabilities(u32 caps)
! 27: {
! 28: struct __user_cap_header_struct cap_hdr;
! 29: struct __user_cap_data_struct cap_dat[_LINUX_CAPABILITY_U32S_3];
! 30: int err;
! 31:
! 32: cap_hdr.version = _LINUX_CAPABILITY_VERSION_3;
! 33: cap_hdr.pid = 0;
! 34:
! 35: memset(cap_dat, 0, sizeof(cap_dat));
! 36: cap_dat[0].effective = cap_dat[0].permitted = caps;
! 37:
! 38: err = capset(&cap_hdr, cap_dat);
! 39: if (!err)
! 40: return 0;
! 41:
! 42: /* Kernel may support do not support our version of capability interface.
! 43: The last call returned supported version so we just retry it. */
! 44: if (errno == EINVAL)
! 45: {
! 46: err = capset(&cap_hdr, cap_dat);
! 47: if (!err)
! 48: return 0;
! 49: }
! 50:
! 51: return -1;
! 52: }
! 53:
! 54: static void
! 55: drop_uid(uid_t uid)
! 56: {
! 57: u32 caps =
! 58: CAP_TO_MASK(CAP_NET_BIND_SERVICE) |
! 59: CAP_TO_MASK(CAP_NET_BROADCAST) |
! 60: CAP_TO_MASK(CAP_NET_ADMIN) |
! 61: CAP_TO_MASK(CAP_NET_RAW);
! 62:
! 63: /* change effective user ID to be able to switch to that
! 64: user ID completely after dropping CAP_SETUID */
! 65: if (seteuid(uid) < 0)
! 66: die("seteuid: %m");
! 67:
! 68: /* restrict the capabilities */
! 69: if (set_capabilities(caps) < 0)
! 70: die("capset: %m");
! 71:
! 72: /* keep the capabilities after dropping root ID */
! 73: if (prctl(PR_SET_KEEPCAPS, 1) < 0)
! 74: die("prctl: %m");
! 75:
! 76: /* completely switch to the unprivileged user ID */
! 77: if (setresuid(uid, uid, uid) < 0)
! 78: die("setresuid: %m");
! 79: }
! 80:
! 81: #endif /* _BIRD_SYSPRIV_H_ */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>