2025-01-25 12:59:11 +09:00
|
|
|
package seccomp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
2025-02-03 18:10:29 +09:00
|
|
|
"runtime"
|
|
|
|
"sync"
|
2025-01-25 12:59:11 +09:00
|
|
|
)
|
|
|
|
|
2025-02-03 18:10:29 +09:00
|
|
|
type exporter struct {
|
|
|
|
opts SyscallOpts
|
|
|
|
r, w *os.File
|
|
|
|
|
|
|
|
prepareOnce sync.Once
|
|
|
|
prepareErr error
|
2025-02-13 22:49:16 +09:00
|
|
|
closeOnce sync.Once
|
|
|
|
closeErr error
|
2025-02-03 18:10:29 +09:00
|
|
|
exportErr <-chan error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *exporter) prepare() error {
|
|
|
|
e.prepareOnce.Do(func() {
|
|
|
|
if r, w, err := os.Pipe(); err != nil {
|
|
|
|
e.prepareErr = err
|
|
|
|
return
|
|
|
|
} else {
|
|
|
|
e.r, e.w = r, w
|
|
|
|
}
|
|
|
|
|
|
|
|
ec := make(chan error, 1)
|
2025-02-13 10:40:51 +09:00
|
|
|
go func(fd uintptr) { ec <- exportFilter(fd, e.opts); close(ec); _ = e.closeWrite() }(e.w.Fd())
|
2025-02-03 18:10:29 +09:00
|
|
|
e.exportErr = ec
|
|
|
|
runtime.SetFinalizer(e, (*exporter).closeWrite)
|
|
|
|
})
|
|
|
|
return e.prepareErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *exporter) closeWrite() error {
|
2025-02-13 22:49:16 +09:00
|
|
|
e.closeOnce.Do(func() {
|
|
|
|
if e.w == nil {
|
|
|
|
panic("closeWrite called on invalid exporter")
|
|
|
|
}
|
|
|
|
e.closeErr = e.w.Close()
|
|
|
|
|
|
|
|
// no need for a finalizer anymore
|
|
|
|
runtime.SetFinalizer(e, nil)
|
|
|
|
})
|
|
|
|
|
|
|
|
return e.closeErr
|
2025-02-03 18:10:29 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
func newExporter(opts SyscallOpts) *exporter {
|
|
|
|
return &exporter{opts: opts}
|
2025-01-25 12:59:11 +09:00
|
|
|
}
|