Skip to content

Commit

Permalink
Merge pull request #4337 from AkihiroSuda/fix-4328
Browse files Browse the repository at this point in the history
Revert "libcontainer: seccomp: pass around *os.File for notifyfd"
  • Loading branch information
AkihiroSuda authored Jul 9, 2024
2 parents 8a324d6 + a5e660c commit 3778ae6
Show file tree
Hide file tree
Showing 5 changed files with 35 additions and 33 deletions.
8 changes: 4 additions & 4 deletions libcontainer/init_linux.go
Original file line number Diff line number Diff line change
Expand Up @@ -451,11 +451,11 @@ func syncParentHooks(pipe *syncSocket) error {

// syncParentSeccomp sends the fd associated with the seccomp file descriptor
// to the parent, and wait for the parent to do pidfd_getfd() to grab a copy.
func syncParentSeccomp(pipe *syncSocket, seccompFd *os.File) error {
if seccompFd == nil {
func syncParentSeccomp(pipe *syncSocket, seccompFd int) error {
if seccompFd == -1 {
return nil
}
defer seccompFd.Close()
defer unix.Close(seccompFd)

// Tell parent to grab our fd.
//
Expand All @@ -466,7 +466,7 @@ func syncParentSeccomp(pipe *syncSocket, seccompFd *os.File) error {
// before the parent gets the file descriptor would deadlock "runc init" if
// we allowed it for SCMP_ACT_NOTIFY). See seccomp.InitSeccomp() for more
// details.
if err := writeSyncArg(pipe, procSeccomp, seccompFd.Fd()); err != nil {
if err := writeSyncArg(pipe, procSeccomp, seccompFd); err != nil {
return err
}
// Wait for parent to tell us they've grabbed the seccompfd.
Expand Down
13 changes: 7 additions & 6 deletions libcontainer/seccomp/patchbpf/enosys_linux.go
Original file line number Diff line number Diff line change
Expand Up @@ -704,32 +704,33 @@ func sysSeccompSetFilter(flags uint, filter []unix.SockFilter) (fd int, err erro
// patches said filter to handle -ENOSYS in a much nicer manner than the
// default libseccomp default action behaviour, and loads the patched filter
// into the kernel for the current process.
func PatchAndLoad(config *configs.Seccomp, filter *libseccomp.ScmpFilter) (*os.File, error) {
func PatchAndLoad(config *configs.Seccomp, filter *libseccomp.ScmpFilter) (int, error) {
// Generate a patched filter.
fprog, err := enosysPatchFilter(config, filter)
if err != nil {
return nil, fmt.Errorf("error patching filter: %w", err)
return -1, fmt.Errorf("error patching filter: %w", err)
}

// Get the set of libseccomp flags set.
seccompFlags, noNewPrivs, err := filterFlags(config, filter)
if err != nil {
return nil, fmt.Errorf("unable to fetch seccomp filter flags: %w", err)
return -1, fmt.Errorf("unable to fetch seccomp filter flags: %w", err)
}

// Set no_new_privs if it was requested, though in runc we handle
// no_new_privs separately so warn if we hit this path.
if noNewPrivs {
logrus.Warnf("potentially misconfigured filter -- setting no_new_privs in seccomp path")
if err := unix.Prctl(unix.PR_SET_NO_NEW_PRIVS, 1, 0, 0, 0); err != nil {
return nil, fmt.Errorf("error enabling no_new_privs bit: %w", err)
return -1, fmt.Errorf("error enabling no_new_privs bit: %w", err)
}
}

// Finally, load the filter.
fd, err := sysSeccompSetFilter(seccompFlags, fprog)
if err != nil {
return nil, fmt.Errorf("error loading seccomp filter: %w", err)
return -1, fmt.Errorf("error loading seccomp filter: %w", err)
}
return os.NewFile(uintptr(fd), "[seccomp filter]"), nil

return fd, nil
}
35 changes: 18 additions & 17 deletions libcontainer/seccomp/seccomp_linux.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ package seccomp
import (
"errors"
"fmt"
"os"

libseccomp "github.com/seccomp/libseccomp-golang"
"github.com/sirupsen/logrus"
Expand All @@ -27,24 +26,25 @@ const (
)

// InitSeccomp installs the seccomp filters to be used in the container as
// specified in config. Returns the seccomp file descriptor if any of the
// filters include a SCMP_ACT_NOTIFY action.
func InitSeccomp(config *configs.Seccomp) (*os.File, error) {
// specified in config.
// Returns the seccomp file descriptor if any of the filters include a
// SCMP_ACT_NOTIFY action, otherwise returns -1.
func InitSeccomp(config *configs.Seccomp) (int, error) {
if config == nil {
return nil, errors.New("cannot initialize Seccomp - nil config passed")
return -1, errors.New("cannot initialize Seccomp - nil config passed")
}

defaultAction, err := getAction(config.DefaultAction, config.DefaultErrnoRet)
if err != nil {
return nil, errors.New("error initializing seccomp - invalid default action")
return -1, errors.New("error initializing seccomp - invalid default action")
}

// Ignore the error since pre-2.4 libseccomp is treated as API level 0.
apiLevel, _ := libseccomp.GetAPI()
for _, call := range config.Syscalls {
if call.Action == configs.Notify {
if apiLevel < 6 {
return nil, fmt.Errorf("seccomp notify unsupported: API level: got %d, want at least 6. Please try with libseccomp >= 2.5.0 and Linux >= 5.7", apiLevel)
return -1, fmt.Errorf("seccomp notify unsupported: API level: got %d, want at least 6. Please try with libseccomp >= 2.5.0 and Linux >= 5.7", apiLevel)
}

// We can't allow the write syscall to notify to the seccomp agent.
Expand All @@ -60,36 +60,36 @@ func InitSeccomp(config *configs.Seccomp) (*os.File, error) {
// agent allows those syscalls to proceed, initialization works just fine and the agent can
// handle future read()/close() syscalls as it wanted.
if call.Name == "write" {
return nil, errors.New("SCMP_ACT_NOTIFY cannot be used for the write syscall")
return -1, errors.New("SCMP_ACT_NOTIFY cannot be used for the write syscall")
}
}
}

// See comment on why write is not allowed. The same reason applies, as this can mean handling write too.
if defaultAction == libseccomp.ActNotify {
return nil, errors.New("SCMP_ACT_NOTIFY cannot be used as default action")
return -1, errors.New("SCMP_ACT_NOTIFY cannot be used as default action")
}

filter, err := libseccomp.NewFilter(defaultAction)
if err != nil {
return nil, fmt.Errorf("error creating filter: %w", err)
return -1, fmt.Errorf("error creating filter: %w", err)
}

// Add extra architectures
for _, arch := range config.Architectures {
scmpArch, err := libseccomp.GetArchFromString(arch)
if err != nil {
return nil, fmt.Errorf("error validating Seccomp architecture: %w", err)
return -1, fmt.Errorf("error validating Seccomp architecture: %w", err)
}
if err := filter.AddArch(scmpArch); err != nil {
return nil, fmt.Errorf("error adding architecture to seccomp filter: %w", err)
return -1, fmt.Errorf("error adding architecture to seccomp filter: %w", err)
}
}

// Add extra flags.
for _, flag := range config.Flags {
if err := setFlag(filter, flag); err != nil {
return nil, err
return -1, err
}
}

Expand All @@ -109,24 +109,25 @@ func InitSeccomp(config *configs.Seccomp) (*os.File, error) {

// Unset no new privs bit
if err := filter.SetNoNewPrivsBit(false); err != nil {
return nil, fmt.Errorf("error setting no new privileges: %w", err)
return -1, fmt.Errorf("error setting no new privileges: %w", err)
}

// Add a rule for each syscall
for _, call := range config.Syscalls {
if call == nil {
return nil, errors.New("encountered nil syscall while initializing Seccomp")
return -1, errors.New("encountered nil syscall while initializing Seccomp")
}

if err := matchCall(filter, call, defaultAction); err != nil {
return nil, err
return -1, err
}
}

seccompFd, err := patchbpf.PatchAndLoad(config, filter)
if err != nil {
return nil, fmt.Errorf("error loading seccomp filter into kernel: %w", err)
return -1, fmt.Errorf("error loading seccomp filter into kernel: %w", err)
}

return seccompFd, nil
}

Expand Down
7 changes: 3 additions & 4 deletions libcontainer/seccomp/seccomp_unsupported.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ package seccomp

import (
"errors"
"os"

"github.com/opencontainers/runc/libcontainer/configs"
"github.com/opencontainers/runtime-spec/specs-go"
Expand All @@ -13,11 +12,11 @@ import (
var ErrSeccompNotEnabled = errors.New("seccomp: config provided but seccomp not supported")

// InitSeccomp does nothing because seccomp is not supported.
func InitSeccomp(config *configs.Seccomp) (*os.File, error) {
func InitSeccomp(config *configs.Seccomp) (int, error) {
if config != nil {
return nil, ErrSeccompNotEnabled
return -1, ErrSeccompNotEnabled
}
return nil, nil
return -1, nil
}

// FlagSupported tells if a provided seccomp flag is supported.
Expand Down
5 changes: 3 additions & 2 deletions tests/integration/seccomp-notify.bats
Original file line number Diff line number Diff line change
Expand Up @@ -83,16 +83,17 @@ function scmp_act_notify_template() {
}

# Test important syscalls (some might be executed by runc) work fine when handled by the agent. noNewPrivileges FALSE.
# fcntl: https://github.com/opencontainers/runc/issues/4328
@test "runc run [seccomp] (SCMP_ACT_NOTIFY important syscalls noNewPrivileges false)" {
scmp_act_notify_template "/bin/true" false '"execve","openat","open","read","close"'
scmp_act_notify_template "/bin/true" false '"execve","openat","open","read","close","fcntl"'

runc run test_busybox
[ "$status" -eq 0 ]
}

# Test important syscalls (some might be executed by runc) work fine when handled by the agent. noNewPrivileges TRUE.
@test "runc run [seccomp] (SCMP_ACT_NOTIFY important syscalls noNewPrivileges true)" {
scmp_act_notify_template "/bin/true" true '"execve","openat","open","read","close"'
scmp_act_notify_template "/bin/true" true '"execve","openat","open","read","close","fcntl"'

runc run test_busybox
[ "$status" -eq 0 ]
Expand Down

0 comments on commit 3778ae6

Please sign in to comment.