forked from lxc/lxcri
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathseccomp.go
134 lines (120 loc) · 3.65 KB
/
seccomp.go
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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
package lxcri
import (
"bufio"
"fmt"
"os"
"strings"
"golang.org/x/sys/unix"
"github.com/opencontainers/runtime-spec/specs-go"
)
var seccompAction = map[specs.LinuxSeccompAction]string{
specs.ActKill: "kill",
specs.ActTrap: "trap",
specs.ActErrno: "errno",
specs.ActAllow: "allow",
//specs.ActTrace: "trace",
//specs.ActLog: "log",
//specs.ActKillProcess: "kill_process",
}
// Note seccomp flags (see `man 2 seccomp`) are currently not supported
// https://github.com/opencontainers/runtime-spec/blob/v1.0.2/config-linux.md#seccomp
func writeSeccompProfile(profilePath string, seccomp *specs.LinuxSeccomp) error {
// #nosec
profile, err := os.OpenFile(profilePath, os.O_WRONLY|os.O_TRUNC|os.O_CREATE, 0440)
if err != nil {
return err
}
// #nosec
defer profile.Close()
w := bufio.NewWriter(profile)
// #nosec
w.WriteString("2\n")
action, err := defaultAction(seccomp)
if err != nil {
return err
}
fmt.Fprintf(w, "allowlist %s\n", action)
platformArchs, err := seccompArchs(seccomp)
if err != nil {
return fmt.Errorf("failed to detect platform architecture: %w", err)
}
for _, arch := range platformArchs {
fmt.Fprintf(w, "[%s]\n", arch)
for _, sc := range seccomp.Syscalls {
if err := writeSeccompSyscall(w, sc); err != nil {
return err
}
}
}
// ensure profile is written to disk without errors
if err := w.Flush(); err != nil {
return err
}
return profile.Sync()
}
func defaultAction(seccomp *specs.LinuxSeccomp) (string, error) {
switch seccomp.DefaultAction {
case specs.ActKill:
return "kill", nil
case specs.ActTrap:
return "trap", nil
case specs.ActErrno:
return "errno 0", nil
case specs.ActAllow:
return "allow", nil
case specs.ActTrace, specs.ActLog: // Not (yet) supported by lxc
fallthrough
//case specs.ActKillProcess: fallthrough // specs > 1.0.2
default:
return "kill", fmt.Errorf("unsupported seccomp default action %q", seccomp.DefaultAction)
}
}
func seccompArchs(seccomp *specs.LinuxSeccomp) ([]string, error) {
var uts unix.Utsname
if err := unix.Uname(&uts); err != nil {
return nil, err
}
nativeArch := nullTerminatedString(uts.Machine[:])
if len(seccomp.Architectures) == 0 {
return []string{nativeArch}, nil
}
archs := make([]string, len(seccomp.Architectures))
for _, a := range seccomp.Architectures {
s := strings.ToLower(strings.TrimPrefix(string(a), "SCMP_ARCH_"))
if strings.ToLower(nativeArch) == s {
// lxc seccomp code automatically adds syscalls to compat architectures
return []string{nativeArch}, nil
}
archs = append(archs, s)
}
return archs, nil
}
func writeSeccompSyscall(w *bufio.Writer, sc specs.LinuxSyscall) error {
for _, name := range sc.Names {
action, ok := seccompAction[sc.Action]
if !ok {
return fmt.Errorf("unsupported seccomp action: %s", sc.Action)
}
if sc.Action == specs.ActErrno {
var ret uint = 0
if sc.ErrnoRet != nil {
ret = *sc.ErrnoRet
}
action = fmt.Sprintf("%s %d", action, ret)
}
if len(sc.Args) == 0 {
fmt.Fprintf(w, "%s %s\n", name, action)
} else {
// Only write a single argument per line - this is required when the same arg.Index is used multiple times.
// from `man 7 seccomp_rule_add_exact_array`
// "When adding syscall argument comparisons to the filter it is important to remember
// that while it is possible to have multiple comparisons in a single rule,
// you can only compare each argument once in a single rule.
// In other words, you can not have multiple comparisons of the 3rd syscall argument in a single rule."
for _, arg := range sc.Args {
fmt.Fprintf(w, "%s %s [%d,%d,%s,%d]\n", name, action, arg.Index, arg.Value, arg.Op, arg.ValueTwo)
}
}
}
return nil
}