commit 783a170d62f30b3415f1746db6f264af85bd81f0
Author: Yawning Angel <yawning(a)schwanenlied.me>
Date: Mon Dec 5 08:18:14 2016 +0000
More seccomp whitelist improvements.
* (General) Explicitly SIGKILL processes that try to use an unexpected
ABI (eg: X32, x86_64 on x86 and vice versa).
* (tor) Filter args to `fcntl[64]`, `accept4`.
* (tor) `src/common/sandbox.c` things we need AF_NETLINK. Fuck that.
---
src/cmd/gen-seccomp/seccomp.go | 3 ++
src/cmd/gen-seccomp/seccomp_tor.go | 97 ++++++++++++++++++++++++++++++++++----
2 files changed, 92 insertions(+), 8 deletions(-)
diff --git a/src/cmd/gen-seccomp/seccomp.go b/src/cmd/gen-seccomp/seccomp.go
index 6c7366f..62b286d 100644
--- a/src/cmd/gen-seccomp/seccomp.go
+++ b/src/cmd/gen-seccomp/seccomp.go
@@ -84,6 +84,9 @@ func newWhitelist(is386 bool) (*seccomp.ScmpFilter, error) {
f.Release()
return nil, err
}
+ if err = f.SetBadArchAction(seccomp.ActKill); err != nil {
+ return nil, err
+ }
return f, nil
}
diff --git a/src/cmd/gen-seccomp/seccomp_tor.go b/src/cmd/gen-seccomp/seccomp_tor.go
index 73ed3c0..2b01656 100644
--- a/src/cmd/gen-seccomp/seccomp_tor.go
+++ b/src/cmd/gen-seccomp/seccomp_tor.go
@@ -43,7 +43,6 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
"eventfd2",
"pipe2",
"pipe",
- "fcntl",
"fstat",
"getdents",
"getdents64",
@@ -84,10 +83,6 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
"sendto",
"unlink",
- // XXX: Calls that should be filtered by arg, but aren't yet.
- "rt_sigaction",
- "accept4",
-
// Calls that tor can filter, but I can't due to not being in
// the tor daemon's process space.
"chown",
@@ -105,6 +100,7 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
"restart_syscall",
"set_tid_address",
"unshare",
+ "rt_sigaction", // Tor filters this but libc does more.
}
if is386 {
allowedNoArgs386 := []string{
@@ -115,12 +111,11 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
"getuid32",
"_llseek",
"sigreturn",
- "fcntl64", // XXX: Filter by arg.
"recv",
"send",
"stat64",
- "socketcall", // Sigh... (see accept4 in the tor code)
+ "socketcall", // Sigh...
"ugetrlimit",
"set_thread_area",
@@ -161,6 +156,9 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
if err = allowCmpEq(f, "mremap", 3, mremapMaymove); err != nil {
return err
}
+ if err = torFilterAccept4(f, is386); err != nil {
+ return err
+ }
if err = torFilterPoll(f); err != nil {
return err
}
@@ -179,6 +177,9 @@ func compileTorSeccompProfile(fd *os.File, useBridges bool, is386 bool) error {
if err = torFilterMmap(f, is386); err != nil {
return err
}
+ if err = torFilterFcntl(f, is386); err != nil {
+ return err
+ }
if useBridges {
// XXX: One day, all the PTs will live in their own containers.
@@ -246,6 +247,25 @@ func torFilterPrctl(f *seccomp.ScmpFilter) error {
return f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{isPrSetDeathsig})
}
+func torFilterAccept4(f *seccomp.ScmpFilter, is386 bool) error {
+ scall, err := seccomp.GetSyscallFromName("accept4")
+ if err != nil {
+ return err
+ }
+ if is386 {
+ // XXX: The tor common/sandbox.c file, explcitly allows socketcall()
+ // by arg for this call, and only this call. ??????
+ return f.AddRule(scall, seccomp.ActAllow)
+ }
+
+ cond, err := seccomp.MakeCondition(3, seccomp.CompareMaskedEqual, 0, syscall.SOCK_CLOEXEC|syscall.SOCK_NONBLOCK)
+ if err != nil {
+ return nil
+ }
+
+ return f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{cond})
+}
+
func torFilterPoll(f *seccomp.ScmpFilter) error {
scall, err := seccomp.GetSyscallFromName("poll")
if err != nil {
@@ -273,7 +293,7 @@ func torFilterSocket(f *seccomp.ScmpFilter, is386 bool) error {
}
// XXX: Tighten this some more.
- return allowCmpEq(f, "socket", 0, syscall.AF_UNIX, syscall.AF_INET, syscall.AF_INET6, syscall.AF_NETLINK)
+ return allowCmpEq(f, "socket", 0, syscall.AF_UNIX, syscall.AF_INET, syscall.AF_INET6 /*, syscall.AF_NETLINK */)
}
func torFilterSetsockopt(f *seccomp.ScmpFilter, is386 bool) error {
@@ -447,6 +467,67 @@ func torFilterMmap(f *seccomp.ScmpFilter, is386 bool) error {
return nil
}
+func torFilterFcntl(f *seccomp.ScmpFilter, is386 bool) error {
+ scallFcntl, err := seccomp.GetSyscallFromName("fcntl")
+ if err != nil {
+ return err
+ }
+ scalls := []seccomp.ScmpSyscall{scallFcntl}
+ if is386 {
+ scallFcntl64, err := seccomp.GetSyscallFromName("fcntl64")
+ if err != nil {
+ return err
+ }
+ scalls = append(scalls, scallFcntl64)
+ }
+
+ isFGetfl, err := seccomp.MakeCondition(1, seccomp.CompareEqual, syscall.F_GETFL)
+ if err != nil {
+ return err
+ }
+ isFGetfd, err := seccomp.MakeCondition(1, seccomp.CompareEqual, syscall.F_GETFD)
+ if err != nil {
+ return err
+ }
+
+ isFSetfl, err := seccomp.MakeCondition(1, seccomp.CompareEqual, syscall.F_SETFL)
+ if err != nil {
+ return err
+ }
+ isFSetflFlags, err := seccomp.MakeCondition(2, seccomp.CompareEqual, syscall.O_RDWR|syscall.O_NONBLOCK)
+ if err != nil {
+ return err
+ }
+
+ isFSetfd, err := seccomp.MakeCondition(1, seccomp.CompareEqual, syscall.F_SETFD)
+ if err != nil {
+ return err
+ }
+ isFdCloexec, err := seccomp.MakeCondition(2, seccomp.CompareEqual, syscall.FD_CLOEXEC)
+ if err != nil {
+ return err
+ }
+
+ for _, scall := range scalls {
+ if err = f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{isFGetfl}); err != nil {
+ return err
+ }
+ if err = f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{isFGetfd}); err != nil {
+ return err
+ }
+
+ if err = f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{isFSetfl, isFSetflFlags}); err != nil {
+ return err
+ }
+
+ if err = f.AddRuleConditional(scall, seccomp.ActAllow, []seccomp.ScmpCondition{isFSetfd, isFdCloexec}); err != nil {
+ return err
+ }
+ }
+
+ return nil
+}
+
func obfsFilterSetsockopt(f *seccomp.ScmpFilter, is386 bool) error {
// 386 already blindly allows all setsockopt() calls.
if is386 {