-
Notifications
You must be signed in to change notification settings - Fork 31
/
Copy pathseccomp_filter.h
81 lines (69 loc) · 3.21 KB
/
seccomp_filter.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
#ifdef __aarch64__
#define __ARCH_WANT_SYSCALL_NO_AT
#endif
#include <linux/audit.h>
#include <linux/filter.h>
#include <linux/seccomp.h>
#include <sys/syscall.h>
#include <elf.h>
#include "common.h"
#ifndef SECCOMP_FILTER_FLAG_SPEC_ALLOW
# define SECCOMP_FILTER_FLAG_SPEC_ALLOW 4
#endif
#ifndef AUDIT_ARCH_AARCH64
# define AUDIT_ARCH_AARCH64 (EM_AARCH64|__AUDIT_ARCH_64BIT|__AUDIT_ARCH_LE)
#endif
/*
* This seccomp filter is necessary to allow users to install packages inside the
* container. Another common option is to assign each user a range of subordinate uids
* and gids, but this can prove challenging to setup in a cluster environment with a
* large number of users:
* - it requires careful setup to avoid overlap between users on different machines
* - it requires using setuid binaries newuidmap(1) and newgidmap(1)
* - writing to a shared filesystem with a subordinate uid/gid can create surprising
* results for the user.
* This code was taken from enroot: https://github.com/NVIDIA/enroot/blob/77bb892ff1738fd65c2dfc437f82fb7d57888c0b/bin/enroot-nsenter.c#L93-L129
*/
static struct sock_filter filter[] = {
/* Check for the syscall architecture (x86_64 and aarch64 ABIs). */
BPF_STMT(BPF_LD|BPF_W|BPF_ABS, offsetof(struct seccomp_data, arch)),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, AUDIT_ARCH_X86_64, 3, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, AUDIT_ARCH_AARCH64, 2, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, AUDIT_ARCH_PPC64LE, 1, 0),
/* FIXME We do not support x86, x32 and aarch32, allow all syscalls for now. */
BPF_STMT(BPF_RET|BPF_K, SECCOMP_RET_ALLOW),
/* Load the syscall number. */
BPF_STMT(BPF_LD|BPF_W|BPF_ABS, offsetof(struct seccomp_data, nr)),
/* Return success on all the following syscalls. */
#if defined(SYS_chown) && defined(SYS_lchown)
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_chown, 15, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_lchown, 14, 0),
#endif
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setuid, 13, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setgid, 12, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setreuid, 11, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setregid, 10, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setresuid, 9, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setresgid, 8, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setgroups, 7, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_fchownat, 6, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_fchown, 5, 0),
/* For setfsuid/setfsgid we only return success if the uid/gid argument is not -1. */
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setfsuid, 1, 0),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, SYS_setfsgid, 0, 2),
BPF_STMT(BPF_LD|BPF_W|BPF_ABS, offsetof(struct seccomp_data, args[0])),
BPF_JUMP(BPF_JMP|BPF_JEQ|BPF_K, (uint32_t)-1, 0, 1),
/* Execute the syscall as usual. */
BPF_STMT(BPF_RET|BPF_K, SECCOMP_RET_ALLOW),
/* Return success. */
BPF_STMT(BPF_RET|BPF_K, SECCOMP_RET_ERRNO|(SECCOMP_RET_DATA & 0x0)),
};
static inline int seccomp_set_filter(void)
{
if ((int)syscall(SYS_seccomp, SECCOMP_SET_MODE_FILTER, SECCOMP_FILTER_FLAG_SPEC_ALLOW, &(const struct sock_fprog){ARRAY_SIZE(filter), filter}) == 0)
return (0);
else if (errno != EINVAL)
return (-1);
return syscall(SYS_seccomp, SECCOMP_SET_MODE_FILTER, 0,
&(const struct sock_fprog){ARRAY_SIZE(filter), filter});
}