Files
LearnGO/go/pkg/mod/github.com/cilium/ebpf@v0.11.0/link/link_test.go
T
2024-09-19 21:38:24 -04:00

274 lines
5.8 KiB
Go

package link
import (
"errors"
"math"
"os"
"path/filepath"
"reflect"
"testing"
"github.com/cilium/ebpf"
"github.com/cilium/ebpf/asm"
"github.com/cilium/ebpf/internal/sys"
"github.com/cilium/ebpf/internal/testutils"
"github.com/cilium/ebpf/internal/testutils/fdtrace"
"github.com/cilium/ebpf/internal/unix"
qt "github.com/frankban/quicktest"
)
func TestMain(m *testing.M) {
fdtrace.TestMain(m)
}
func TestRawLink(t *testing.T) {
cgroup, prog := mustCgroupFixtures(t)
link, err := AttachRawLink(RawLinkOptions{
Target: int(cgroup.Fd()),
Program: prog,
Attach: ebpf.AttachCGroupInetEgress,
})
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal("Can't create raw link:", err)
}
info, err := link.Info()
if err != nil {
t.Fatal("Can't get link info:", err)
}
pi, err := prog.Info()
if err != nil {
t.Fatal("Can't get program info:", err)
}
progID, ok := pi.ID()
if !ok {
t.Fatal("Program ID not available in program info")
}
if info.Program != progID {
t.Error("Link program ID doesn't match program ID")
}
testLink(t, &linkCgroup{*link}, prog)
}
func TestUnpinRawLink(t *testing.T) {
cgroup, prog := mustCgroupFixtures(t)
link, _ := newPinnedRawLink(t, cgroup, prog)
defer link.Close()
qt.Assert(t, link.IsPinned(), qt.IsTrue)
if err := link.Unpin(); err != nil {
t.Fatal(err)
}
qt.Assert(t, link.IsPinned(), qt.IsFalse)
}
func TestRawLinkLoadPinnedWithOptions(t *testing.T) {
cgroup, prog := mustCgroupFixtures(t)
link, path := newPinnedRawLink(t, cgroup, prog)
defer link.Close()
qt.Assert(t, link.IsPinned(), qt.IsTrue)
// It seems like the kernel ignores BPF_F_RDONLY when updating a link,
// so we can't test this.
_, err := loadPinnedRawLink(path, &ebpf.LoadPinOptions{
Flags: math.MaxUint32,
})
if !errors.Is(err, unix.EINVAL) {
t.Fatal("Invalid flags don't trigger an error:", err)
}
}
func newPinnedRawLink(t *testing.T, cgroup *os.File, prog *ebpf.Program) (*RawLink, string) {
t.Helper()
link, err := AttachRawLink(RawLinkOptions{
Target: int(cgroup.Fd()),
Program: prog,
Attach: ebpf.AttachCGroupInetEgress,
})
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal("Can't create raw link:", err)
}
path := filepath.Join(testutils.TempBPFFS(t), "link")
err = link.Pin(path)
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal(err)
}
return link, path
}
func mustCgroupFixtures(t *testing.T) (*os.File, *ebpf.Program) {
t.Helper()
testutils.SkipIfNotSupported(t, haveProgAttach())
return testutils.CreateCgroup(t), mustLoadProgram(t, ebpf.CGroupSKB, 0, "")
}
func testLink(t *testing.T, link Link, prog *ebpf.Program) {
t.Helper()
tmp, err := os.MkdirTemp("/sys/fs/bpf", "ebpf-test")
if err != nil {
t.Fatal(err)
}
defer os.RemoveAll(tmp)
t.Run("link/pinning", func(t *testing.T) {
path := filepath.Join(tmp, "link")
err = link.Pin(path)
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatalf("Can't pin %T: %s", link, err)
}
link2, err := LoadPinnedLink(path, nil)
if err != nil {
t.Fatalf("Can't load pinned %T: %s", link, err)
}
link2.Close()
if reflect.TypeOf(link) != reflect.TypeOf(link2) {
t.Errorf("Loading a pinned %T returns a %T", link, link2)
}
_, err = LoadPinnedLink(path, &ebpf.LoadPinOptions{
Flags: math.MaxUint32,
})
if !errors.Is(err, unix.EINVAL) {
t.Errorf("Loading a pinned %T doesn't respect flags", link)
}
})
t.Run("link/update", func(t *testing.T) {
err := link.Update(prog)
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal("Update returns an error:", err)
}
func() {
// Panicking is OK
defer func() {
_ = recover()
}()
if err := link.Update(nil); err == nil {
t.Fatalf("%T.Update accepts nil program", link)
}
}()
})
t.Run("link/info", func(t *testing.T) {
info, err := link.Info()
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal("Link info returns an error:", err)
}
if info.Type == 0 {
t.Fatal("Failed to get link info type")
}
switch info.Type {
case sys.BPF_LINK_TYPE_TRACING:
if info.Tracing() == nil {
t.Fatalf("Failed to get link tracing extra info")
}
case sys.BPF_LINK_TYPE_CGROUP:
cg := info.Cgroup()
if cg.CgroupId == 0 {
t.Fatalf("Failed to get link Cgroup extra info")
}
case sys.BPF_LINK_TYPE_NETNS:
netns := info.NetNs()
if netns.AttachType == 0 {
t.Fatalf("Failed to get link NetNs extra info")
}
case sys.BPF_LINK_TYPE_XDP:
xdp := info.XDP()
if xdp.Ifindex == 0 {
t.Fatalf("Failed to get link XDP extra info")
}
}
})
type FDer interface {
FD() int
}
t.Run("from fd", func(t *testing.T) {
fder, ok := link.(FDer)
if !ok {
t.Skip("Link doesn't allow retrieving FD")
}
// We need to dup the FD since NewLinkFromFD takes
// ownership.
dupFD, err := unix.FcntlInt(uintptr(fder.FD()), unix.F_DUPFD_CLOEXEC, 1)
if err != nil {
t.Fatal("Can't dup link FD:", err)
}
defer unix.Close(dupFD)
newLink, err := NewLinkFromFD(dupFD)
testutils.SkipIfNotSupported(t, err)
if err != nil {
t.Fatal("Can't create new link from dup link FD:", err)
}
defer newLink.Close()
if reflect.TypeOf(newLink) != reflect.TypeOf(link) {
t.Fatalf("Expected type %T, got %T", link, newLink)
}
})
if err := link.Close(); err != nil {
t.Fatalf("%T.Close returns an error: %s", link, err)
}
}
func mustLoadProgram(tb testing.TB, typ ebpf.ProgramType, attachType ebpf.AttachType, attachTo string) *ebpf.Program {
tb.Helper()
license := "MIT"
switch typ {
case ebpf.RawTracepoint, ebpf.LSM:
license = "GPL"
}
prog, err := ebpf.NewProgram(&ebpf.ProgramSpec{
Type: typ,
AttachType: attachType,
AttachTo: attachTo,
License: license,
Instructions: asm.Instructions{
asm.Mov.Imm(asm.R0, 0),
asm.Return(),
},
})
if err != nil {
tb.Fatal(err)
}
tb.Cleanup(func() {
prog.Close()
})
return prog
}