mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
b79dec8138
Signed-off-by: Akihiro Suda <akihiro.suda.cz@hco.ntt.co.jp>
761 lines
20 KiB
Go
761 lines
20 KiB
Go
package ebpf
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"math"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/cilium/ebpf/asm"
|
|
"github.com/cilium/ebpf/internal"
|
|
"github.com/cilium/ebpf/internal/btf"
|
|
"github.com/cilium/ebpf/internal/unix"
|
|
)
|
|
|
|
// ErrNotSupported is returned whenever the kernel doesn't support a feature.
|
|
var ErrNotSupported = internal.ErrNotSupported
|
|
|
|
var errUnsatisfiedReference = errors.New("unsatisfied reference")
|
|
|
|
// ProgramID represents the unique ID of an eBPF program.
|
|
type ProgramID uint32
|
|
|
|
const (
|
|
// Number of bytes to pad the output buffer for BPF_PROG_TEST_RUN.
|
|
// This is currently the maximum of spare space allocated for SKB
|
|
// and XDP programs, and equal to XDP_PACKET_HEADROOM + NET_IP_ALIGN.
|
|
outputPad = 256 + 2
|
|
)
|
|
|
|
// DefaultVerifierLogSize is the default number of bytes allocated for the
|
|
// verifier log.
|
|
const DefaultVerifierLogSize = 64 * 1024
|
|
|
|
// ProgramOptions control loading a program into the kernel.
|
|
type ProgramOptions struct {
|
|
// Controls the detail emitted by the kernel verifier. Set to non-zero
|
|
// to enable logging.
|
|
LogLevel uint32
|
|
// Controls the output buffer size for the verifier. Defaults to
|
|
// DefaultVerifierLogSize.
|
|
LogSize int
|
|
// An ELF containing the target BTF for this program. It is used both to
|
|
// find the correct function to trace and to apply CO-RE relocations.
|
|
// This is useful in environments where the kernel BTF is not available
|
|
// (containers) or where it is in a non-standard location. Defaults to
|
|
// use the kernel BTF from a well-known location.
|
|
TargetBTF io.ReaderAt
|
|
}
|
|
|
|
// ProgramSpec defines a Program.
|
|
type ProgramSpec struct {
|
|
// Name is passed to the kernel as a debug aid. Must only contain
|
|
// alpha numeric and '_' characters.
|
|
Name string
|
|
|
|
// Type determines at which hook in the kernel a program will run.
|
|
Type ProgramType
|
|
AttachType AttachType
|
|
// Name of a kernel data structure or function to attach to. Its
|
|
// interpretation depends on Type and AttachType.
|
|
AttachTo string
|
|
// The program to attach to. Must be provided manually.
|
|
AttachTarget *Program
|
|
Instructions asm.Instructions
|
|
|
|
// Flags is passed to the kernel and specifies additional program
|
|
// load attributes.
|
|
Flags uint32
|
|
|
|
// License of the program. Some helpers are only available if
|
|
// the license is deemed compatible with the GPL.
|
|
//
|
|
// See https://www.kernel.org/doc/html/latest/process/license-rules.html#id1
|
|
License string
|
|
|
|
// Version used by Kprobe programs.
|
|
//
|
|
// Deprecated on kernels 5.0 and later. Leave empty to let the library
|
|
// detect this value automatically.
|
|
KernelVersion uint32
|
|
|
|
// The BTF associated with this program. Changing Instructions
|
|
// will most likely invalidate the contained data, and may
|
|
// result in errors when attempting to load it into the kernel.
|
|
BTF *btf.Program
|
|
|
|
// The byte order this program was compiled for, may be nil.
|
|
ByteOrder binary.ByteOrder
|
|
}
|
|
|
|
// Copy returns a copy of the spec.
|
|
func (ps *ProgramSpec) Copy() *ProgramSpec {
|
|
if ps == nil {
|
|
return nil
|
|
}
|
|
|
|
cpy := *ps
|
|
cpy.Instructions = make(asm.Instructions, len(ps.Instructions))
|
|
copy(cpy.Instructions, ps.Instructions)
|
|
return &cpy
|
|
}
|
|
|
|
// Tag calculates the kernel tag for a series of instructions.
|
|
//
|
|
// Use asm.Instructions.Tag if you need to calculate for non-native endianness.
|
|
func (ps *ProgramSpec) Tag() (string, error) {
|
|
return ps.Instructions.Tag(internal.NativeEndian)
|
|
}
|
|
|
|
// Program represents BPF program loaded into the kernel.
|
|
//
|
|
// It is not safe to close a Program which is used by other goroutines.
|
|
type Program struct {
|
|
// Contains the output of the kernel verifier if enabled,
|
|
// otherwise it is empty.
|
|
VerifierLog string
|
|
|
|
fd *internal.FD
|
|
name string
|
|
pinnedPath string
|
|
typ ProgramType
|
|
}
|
|
|
|
// NewProgram creates a new Program.
|
|
//
|
|
// Loading a program for the first time will perform
|
|
// feature detection by loading small, temporary programs.
|
|
func NewProgram(spec *ProgramSpec) (*Program, error) {
|
|
return NewProgramWithOptions(spec, ProgramOptions{})
|
|
}
|
|
|
|
// NewProgramWithOptions creates a new Program.
|
|
//
|
|
// Loading a program for the first time will perform
|
|
// feature detection by loading small, temporary programs.
|
|
func NewProgramWithOptions(spec *ProgramSpec, opts ProgramOptions) (*Program, error) {
|
|
handles := newHandleCache()
|
|
defer handles.close()
|
|
|
|
prog, err := newProgramWithOptions(spec, opts, handles)
|
|
if errors.Is(err, errUnsatisfiedReference) {
|
|
return nil, fmt.Errorf("cannot load program without loading its whole collection: %w", err)
|
|
}
|
|
return prog, err
|
|
}
|
|
|
|
func newProgramWithOptions(spec *ProgramSpec, opts ProgramOptions, handles *handleCache) (*Program, error) {
|
|
if len(spec.Instructions) == 0 {
|
|
return nil, errors.New("instructions cannot be empty")
|
|
}
|
|
|
|
if spec.ByteOrder != nil && spec.ByteOrder != internal.NativeEndian {
|
|
return nil, fmt.Errorf("can't load %s program on %s", spec.ByteOrder, internal.NativeEndian)
|
|
}
|
|
|
|
// Kernels before 5.0 (6c4fc209fcf9 "bpf: remove useless version check for prog load")
|
|
// require the version field to be set to the value of the KERNEL_VERSION
|
|
// macro for kprobe-type programs.
|
|
// Overwrite Kprobe program version if set to zero or the magic version constant.
|
|
kv := spec.KernelVersion
|
|
if spec.Type == Kprobe && (kv == 0 || kv == internal.MagicKernelVersion) {
|
|
v, err := internal.KernelVersion()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("detecting kernel version: %w", err)
|
|
}
|
|
kv = v.Kernel()
|
|
}
|
|
|
|
attr := &internal.BPFProgLoadAttr{
|
|
ProgType: uint32(spec.Type),
|
|
ProgFlags: spec.Flags,
|
|
ExpectedAttachType: uint32(spec.AttachType),
|
|
License: internal.NewStringPointer(spec.License),
|
|
KernelVersion: kv,
|
|
}
|
|
|
|
if haveObjName() == nil {
|
|
attr.ProgName = internal.NewBPFObjName(spec.Name)
|
|
}
|
|
|
|
var err error
|
|
var targetBTF *btf.Spec
|
|
if opts.TargetBTF != nil {
|
|
targetBTF, err = handles.btfSpec(opts.TargetBTF)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("load target BTF: %w", err)
|
|
}
|
|
}
|
|
|
|
var btfDisabled bool
|
|
var core btf.COREFixups
|
|
if spec.BTF != nil {
|
|
core, err = spec.BTF.Fixups(targetBTF)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("CO-RE relocations: %w", err)
|
|
}
|
|
|
|
handle, err := handles.btfHandle(spec.BTF.Spec())
|
|
btfDisabled = errors.Is(err, btf.ErrNotSupported)
|
|
if err != nil && !btfDisabled {
|
|
return nil, fmt.Errorf("load BTF: %w", err)
|
|
}
|
|
|
|
if handle != nil {
|
|
attr.ProgBTFFd = uint32(handle.FD())
|
|
|
|
recSize, bytes, err := spec.BTF.LineInfos()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get BTF line infos: %w", err)
|
|
}
|
|
attr.LineInfoRecSize = recSize
|
|
attr.LineInfoCnt = uint32(uint64(len(bytes)) / uint64(recSize))
|
|
attr.LineInfo = internal.NewSlicePointer(bytes)
|
|
|
|
recSize, bytes, err = spec.BTF.FuncInfos()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get BTF function infos: %w", err)
|
|
}
|
|
attr.FuncInfoRecSize = recSize
|
|
attr.FuncInfoCnt = uint32(uint64(len(bytes)) / uint64(recSize))
|
|
attr.FuncInfo = internal.NewSlicePointer(bytes)
|
|
}
|
|
}
|
|
|
|
insns, err := core.Apply(spec.Instructions)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("CO-RE fixup: %w", err)
|
|
}
|
|
|
|
if err := fixupJumpsAndCalls(insns); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
buf := bytes.NewBuffer(make([]byte, 0, len(spec.Instructions)*asm.InstructionSize))
|
|
err = insns.Marshal(buf, internal.NativeEndian)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bytecode := buf.Bytes()
|
|
attr.Instructions = internal.NewSlicePointer(bytecode)
|
|
attr.InsCount = uint32(len(bytecode) / asm.InstructionSize)
|
|
|
|
if spec.AttachTo != "" {
|
|
if spec.AttachTarget != nil {
|
|
info, err := spec.AttachTarget.Info()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("load target BTF: %w", err)
|
|
}
|
|
|
|
btfID, ok := info.BTFID()
|
|
if !ok {
|
|
return nil, fmt.Errorf("load target BTF: no BTF info available")
|
|
}
|
|
btfHandle, err := btf.NewHandleFromID(btfID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("load target BTF: %w", err)
|
|
}
|
|
defer btfHandle.Close()
|
|
|
|
targetBTF = btfHandle.Spec()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("load target BTF: %w", err)
|
|
}
|
|
}
|
|
|
|
target, err := resolveBTFType(targetBTF, spec.AttachTo, spec.Type, spec.AttachType)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if target != nil {
|
|
attr.AttachBTFID = uint32(target.ID())
|
|
}
|
|
if spec.AttachTarget != nil {
|
|
attr.AttachProgFd = uint32(spec.AttachTarget.FD())
|
|
}
|
|
}
|
|
|
|
logSize := DefaultVerifierLogSize
|
|
if opts.LogSize > 0 {
|
|
logSize = opts.LogSize
|
|
}
|
|
|
|
var logBuf []byte
|
|
if opts.LogLevel > 0 {
|
|
logBuf = make([]byte, logSize)
|
|
attr.LogLevel = opts.LogLevel
|
|
attr.LogSize = uint32(len(logBuf))
|
|
attr.LogBuf = internal.NewSlicePointer(logBuf)
|
|
}
|
|
|
|
fd, err := internal.BPFProgLoad(attr)
|
|
if err == nil {
|
|
return &Program{internal.CString(logBuf), fd, spec.Name, "", spec.Type}, nil
|
|
}
|
|
|
|
logErr := err
|
|
if opts.LogLevel == 0 && opts.LogSize >= 0 {
|
|
// Re-run with the verifier enabled to get better error messages.
|
|
logBuf = make([]byte, logSize)
|
|
attr.LogLevel = 1
|
|
attr.LogSize = uint32(len(logBuf))
|
|
attr.LogBuf = internal.NewSlicePointer(logBuf)
|
|
|
|
fd, logErr = internal.BPFProgLoad(attr)
|
|
if logErr == nil {
|
|
fd.Close()
|
|
}
|
|
}
|
|
|
|
if errors.Is(logErr, unix.EPERM) && logBuf[0] == 0 {
|
|
// EPERM due to RLIMIT_MEMLOCK happens before the verifier, so we can
|
|
// check that the log is empty to reduce false positives.
|
|
return nil, fmt.Errorf("load program: %w (MEMLOCK bay be too low, consider rlimit.RemoveMemlock)", logErr)
|
|
}
|
|
|
|
err = internal.ErrorWithLog(err, logBuf, logErr)
|
|
if btfDisabled {
|
|
return nil, fmt.Errorf("load program without BTF: %w", err)
|
|
}
|
|
return nil, fmt.Errorf("load program: %w", err)
|
|
}
|
|
|
|
// NewProgramFromFD creates a program from a raw fd.
|
|
//
|
|
// You should not use fd after calling this function.
|
|
//
|
|
// Requires at least Linux 4.10.
|
|
func NewProgramFromFD(fd int) (*Program, error) {
|
|
if fd < 0 {
|
|
return nil, errors.New("invalid fd")
|
|
}
|
|
|
|
return newProgramFromFD(internal.NewFD(uint32(fd)))
|
|
}
|
|
|
|
// NewProgramFromID returns the program for a given id.
|
|
//
|
|
// Returns ErrNotExist, if there is no eBPF program with the given id.
|
|
func NewProgramFromID(id ProgramID) (*Program, error) {
|
|
fd, err := internal.BPFObjGetFDByID(internal.BPF_PROG_GET_FD_BY_ID, uint32(id))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get program by id: %w", err)
|
|
}
|
|
|
|
return newProgramFromFD(fd)
|
|
}
|
|
|
|
func newProgramFromFD(fd *internal.FD) (*Program, error) {
|
|
info, err := newProgramInfoFromFd(fd)
|
|
if err != nil {
|
|
fd.Close()
|
|
return nil, fmt.Errorf("discover program type: %w", err)
|
|
}
|
|
|
|
return &Program{"", fd, "", "", info.Type}, nil
|
|
}
|
|
|
|
func (p *Program) String() string {
|
|
if p.name != "" {
|
|
return fmt.Sprintf("%s(%s)#%v", p.typ, p.name, p.fd)
|
|
}
|
|
return fmt.Sprintf("%s(%v)", p.typ, p.fd)
|
|
}
|
|
|
|
// Type returns the underlying type of the program.
|
|
func (p *Program) Type() ProgramType {
|
|
return p.typ
|
|
}
|
|
|
|
// Info returns metadata about the program.
|
|
//
|
|
// Requires at least 4.10.
|
|
func (p *Program) Info() (*ProgramInfo, error) {
|
|
return newProgramInfoFromFd(p.fd)
|
|
}
|
|
|
|
// FD gets the file descriptor of the Program.
|
|
//
|
|
// It is invalid to call this function after Close has been called.
|
|
func (p *Program) FD() int {
|
|
fd, err := p.fd.Value()
|
|
if err != nil {
|
|
// Best effort: -1 is the number most likely to be an
|
|
// invalid file descriptor.
|
|
return -1
|
|
}
|
|
|
|
return int(fd)
|
|
}
|
|
|
|
// Clone creates a duplicate of the Program.
|
|
//
|
|
// Closing the duplicate does not affect the original, and vice versa.
|
|
//
|
|
// Cloning a nil Program returns nil.
|
|
func (p *Program) Clone() (*Program, error) {
|
|
if p == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
dup, err := p.fd.Dup()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't clone program: %w", err)
|
|
}
|
|
|
|
return &Program{p.VerifierLog, dup, p.name, "", p.typ}, nil
|
|
}
|
|
|
|
// Pin persists the Program on the BPF virtual file system past the lifetime of
|
|
// the process that created it
|
|
//
|
|
// Calling Pin on a previously pinned program will overwrite the path, except when
|
|
// the new path already exists. Re-pinning across filesystems is not supported.
|
|
//
|
|
// This requires bpffs to be mounted above fileName. See https://docs.cilium.io/en/k8s-doc/admin/#admin-mount-bpffs
|
|
func (p *Program) Pin(fileName string) error {
|
|
if err := internal.Pin(p.pinnedPath, fileName, p.fd); err != nil {
|
|
return err
|
|
}
|
|
p.pinnedPath = fileName
|
|
return nil
|
|
}
|
|
|
|
// Unpin removes the persisted state for the Program from the BPF virtual filesystem.
|
|
//
|
|
// Failed calls to Unpin will not alter the state returned by IsPinned.
|
|
//
|
|
// Unpinning an unpinned Program returns nil.
|
|
func (p *Program) Unpin() error {
|
|
if err := internal.Unpin(p.pinnedPath); err != nil {
|
|
return err
|
|
}
|
|
p.pinnedPath = ""
|
|
return nil
|
|
}
|
|
|
|
// IsPinned returns true if the Program has a non-empty pinned path.
|
|
func (p *Program) IsPinned() bool {
|
|
return p.pinnedPath != ""
|
|
}
|
|
|
|
// Close unloads the program from the kernel.
|
|
func (p *Program) Close() error {
|
|
if p == nil {
|
|
return nil
|
|
}
|
|
|
|
return p.fd.Close()
|
|
}
|
|
|
|
// Test runs the Program in the kernel with the given input and returns the
|
|
// value returned by the eBPF program. outLen may be zero.
|
|
//
|
|
// Note: the kernel expects at least 14 bytes input for an ethernet header for
|
|
// XDP and SKB programs.
|
|
//
|
|
// This function requires at least Linux 4.12.
|
|
func (p *Program) Test(in []byte) (uint32, []byte, error) {
|
|
ret, out, _, err := p.testRun(in, 1, nil)
|
|
if err != nil {
|
|
return ret, nil, fmt.Errorf("can't test program: %w", err)
|
|
}
|
|
return ret, out, nil
|
|
}
|
|
|
|
// Benchmark runs the Program with the given input for a number of times
|
|
// and returns the time taken per iteration.
|
|
//
|
|
// Returns the result of the last execution of the program and the time per
|
|
// run or an error. reset is called whenever the benchmark syscall is
|
|
// interrupted, and should be set to testing.B.ResetTimer or similar.
|
|
//
|
|
// Note: profiling a call to this function will skew it's results, see
|
|
// https://github.com/cilium/ebpf/issues/24
|
|
//
|
|
// This function requires at least Linux 4.12.
|
|
func (p *Program) Benchmark(in []byte, repeat int, reset func()) (uint32, time.Duration, error) {
|
|
ret, _, total, err := p.testRun(in, repeat, reset)
|
|
if err != nil {
|
|
return ret, total, fmt.Errorf("can't benchmark program: %w", err)
|
|
}
|
|
return ret, total, nil
|
|
}
|
|
|
|
var haveProgTestRun = internal.FeatureTest("BPF_PROG_TEST_RUN", "4.12", func() error {
|
|
prog, err := NewProgram(&ProgramSpec{
|
|
Type: SocketFilter,
|
|
Instructions: asm.Instructions{
|
|
asm.LoadImm(asm.R0, 0, asm.DWord),
|
|
asm.Return(),
|
|
},
|
|
License: "MIT",
|
|
})
|
|
if err != nil {
|
|
// This may be because we lack sufficient permissions, etc.
|
|
return err
|
|
}
|
|
defer prog.Close()
|
|
|
|
// Programs require at least 14 bytes input
|
|
in := make([]byte, 14)
|
|
attr := bpfProgTestRunAttr{
|
|
fd: uint32(prog.FD()),
|
|
dataSizeIn: uint32(len(in)),
|
|
dataIn: internal.NewSlicePointer(in),
|
|
}
|
|
|
|
err = bpfProgTestRun(&attr)
|
|
if errors.Is(err, unix.EINVAL) {
|
|
// Check for EINVAL specifically, rather than err != nil since we
|
|
// otherwise misdetect due to insufficient permissions.
|
|
return internal.ErrNotSupported
|
|
}
|
|
if errors.Is(err, unix.EINTR) {
|
|
// We know that PROG_TEST_RUN is supported if we get EINTR.
|
|
return nil
|
|
}
|
|
return err
|
|
})
|
|
|
|
func (p *Program) testRun(in []byte, repeat int, reset func()) (uint32, []byte, time.Duration, error) {
|
|
if uint(repeat) > math.MaxUint32 {
|
|
return 0, nil, 0, fmt.Errorf("repeat is too high")
|
|
}
|
|
|
|
if len(in) == 0 {
|
|
return 0, nil, 0, fmt.Errorf("missing input")
|
|
}
|
|
|
|
if uint(len(in)) > math.MaxUint32 {
|
|
return 0, nil, 0, fmt.Errorf("input is too long")
|
|
}
|
|
|
|
if err := haveProgTestRun(); err != nil {
|
|
return 0, nil, 0, err
|
|
}
|
|
|
|
// Older kernels ignore the dataSizeOut argument when copying to user space.
|
|
// Combined with things like bpf_xdp_adjust_head() we don't really know what the final
|
|
// size will be. Hence we allocate an output buffer which we hope will always be large
|
|
// enough, and panic if the kernel wrote past the end of the allocation.
|
|
// See https://patchwork.ozlabs.org/cover/1006822/
|
|
out := make([]byte, len(in)+outputPad)
|
|
|
|
fd, err := p.fd.Value()
|
|
if err != nil {
|
|
return 0, nil, 0, err
|
|
}
|
|
|
|
attr := bpfProgTestRunAttr{
|
|
fd: fd,
|
|
dataSizeIn: uint32(len(in)),
|
|
dataSizeOut: uint32(len(out)),
|
|
dataIn: internal.NewSlicePointer(in),
|
|
dataOut: internal.NewSlicePointer(out),
|
|
repeat: uint32(repeat),
|
|
}
|
|
|
|
for {
|
|
err = bpfProgTestRun(&attr)
|
|
if err == nil {
|
|
break
|
|
}
|
|
|
|
if errors.Is(err, unix.EINTR) {
|
|
if reset != nil {
|
|
reset()
|
|
}
|
|
continue
|
|
}
|
|
|
|
return 0, nil, 0, fmt.Errorf("can't run test: %w", err)
|
|
}
|
|
|
|
if int(attr.dataSizeOut) > cap(out) {
|
|
// Houston, we have a problem. The program created more data than we allocated,
|
|
// and the kernel wrote past the end of our buffer.
|
|
panic("kernel wrote past end of output buffer")
|
|
}
|
|
out = out[:int(attr.dataSizeOut)]
|
|
|
|
total := time.Duration(attr.duration) * time.Nanosecond
|
|
return attr.retval, out, total, nil
|
|
}
|
|
|
|
func unmarshalProgram(buf []byte) (*Program, error) {
|
|
if len(buf) != 4 {
|
|
return nil, errors.New("program id requires 4 byte value")
|
|
}
|
|
|
|
// Looking up an entry in a nested map or prog array returns an id,
|
|
// not an fd.
|
|
id := internal.NativeEndian.Uint32(buf)
|
|
return NewProgramFromID(ProgramID(id))
|
|
}
|
|
|
|
func marshalProgram(p *Program, length int) ([]byte, error) {
|
|
if length != 4 {
|
|
return nil, fmt.Errorf("can't marshal program to %d bytes", length)
|
|
}
|
|
|
|
value, err := p.fd.Value()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
buf := make([]byte, 4)
|
|
internal.NativeEndian.PutUint32(buf, value)
|
|
return buf, nil
|
|
}
|
|
|
|
// Attach a Program.
|
|
//
|
|
// Deprecated: use link.RawAttachProgram instead.
|
|
func (p *Program) Attach(fd int, typ AttachType, flags AttachFlags) error {
|
|
if fd < 0 {
|
|
return errors.New("invalid fd")
|
|
}
|
|
|
|
pfd, err := p.fd.Value()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
attr := internal.BPFProgAttachAttr{
|
|
TargetFd: uint32(fd),
|
|
AttachBpfFd: pfd,
|
|
AttachType: uint32(typ),
|
|
AttachFlags: uint32(flags),
|
|
}
|
|
|
|
return internal.BPFProgAttach(&attr)
|
|
}
|
|
|
|
// Detach a Program.
|
|
//
|
|
// Deprecated: use link.RawDetachProgram instead.
|
|
func (p *Program) Detach(fd int, typ AttachType, flags AttachFlags) error {
|
|
if fd < 0 {
|
|
return errors.New("invalid fd")
|
|
}
|
|
|
|
if flags != 0 {
|
|
return errors.New("flags must be zero")
|
|
}
|
|
|
|
pfd, err := p.fd.Value()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
attr := internal.BPFProgDetachAttr{
|
|
TargetFd: uint32(fd),
|
|
AttachBpfFd: pfd,
|
|
AttachType: uint32(typ),
|
|
}
|
|
|
|
return internal.BPFProgDetach(&attr)
|
|
}
|
|
|
|
// LoadPinnedProgram loads a Program from a BPF file.
|
|
//
|
|
// Requires at least Linux 4.11.
|
|
func LoadPinnedProgram(fileName string, opts *LoadPinOptions) (*Program, error) {
|
|
fd, err := internal.BPFObjGet(fileName, opts.Marshal())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
info, err := newProgramInfoFromFd(fd)
|
|
if err != nil {
|
|
_ = fd.Close()
|
|
return nil, fmt.Errorf("info for %s: %w", fileName, err)
|
|
}
|
|
|
|
return &Program{"", fd, filepath.Base(fileName), fileName, info.Type}, nil
|
|
}
|
|
|
|
// SanitizeName replaces all invalid characters in name with replacement.
|
|
// Passing a negative value for replacement will delete characters instead
|
|
// of replacing them. Use this to automatically generate valid names for maps
|
|
// and programs at runtime.
|
|
//
|
|
// The set of allowed characters depends on the running kernel version.
|
|
// Dots are only allowed as of kernel 5.2.
|
|
func SanitizeName(name string, replacement rune) string {
|
|
return strings.Map(func(char rune) rune {
|
|
if invalidBPFObjNameChar(char) {
|
|
return replacement
|
|
}
|
|
return char
|
|
}, name)
|
|
}
|
|
|
|
// ProgramGetNextID returns the ID of the next eBPF program.
|
|
//
|
|
// Returns ErrNotExist, if there is no next eBPF program.
|
|
func ProgramGetNextID(startID ProgramID) (ProgramID, error) {
|
|
id, err := objGetNextID(internal.BPF_PROG_GET_NEXT_ID, uint32(startID))
|
|
return ProgramID(id), err
|
|
}
|
|
|
|
// ID returns the systemwide unique ID of the program.
|
|
//
|
|
// Deprecated: use ProgramInfo.ID() instead.
|
|
func (p *Program) ID() (ProgramID, error) {
|
|
info, err := bpfGetProgInfoByFD(p.fd, nil)
|
|
if err != nil {
|
|
return ProgramID(0), err
|
|
}
|
|
return ProgramID(info.id), nil
|
|
}
|
|
|
|
func resolveBTFType(spec *btf.Spec, name string, progType ProgramType, attachType AttachType) (btf.Type, error) {
|
|
type match struct {
|
|
p ProgramType
|
|
a AttachType
|
|
}
|
|
|
|
var typeName, featureName string
|
|
switch (match{progType, attachType}) {
|
|
case match{LSM, AttachLSMMac}:
|
|
typeName = "bpf_lsm_" + name
|
|
featureName = name + " LSM hook"
|
|
case match{Tracing, AttachTraceIter}:
|
|
typeName = "bpf_iter_" + name
|
|
featureName = name + " iterator"
|
|
case match{Extension, AttachNone}:
|
|
typeName = name
|
|
featureName = fmt.Sprintf("freplace %s", name)
|
|
default:
|
|
return nil, nil
|
|
}
|
|
|
|
if spec == nil {
|
|
var err error
|
|
spec, err = btf.LoadKernelSpec()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("load kernel spec: %w", err)
|
|
}
|
|
}
|
|
|
|
var target *btf.Func
|
|
err := spec.FindType(typeName, &target)
|
|
if errors.Is(err, btf.ErrNotFound) {
|
|
return nil, &internal.UnsupportedFeatureError{
|
|
Name: featureName,
|
|
}
|
|
}
|
|
if err != nil {
|
|
return nil, fmt.Errorf("resolve BTF for %s: %w", featureName, err)
|
|
}
|
|
|
|
return target, nil
|
|
}
|