diff --git a/link/uprobe_multi_test.go b/link/uprobe_multi_test.go new file mode 100644 index 000000000..65f0e4303 --- /dev/null +++ b/link/uprobe_multi_test.go @@ -0,0 +1,246 @@ +package link + +import ( + "errors" + "math" + "os" + "os/exec" + "testing" + + "github.com/cilium/ebpf" + "github.com/cilium/ebpf/internal/testutils" + "github.com/go-quicktest/qt" +) + +func TestUprobeMulti(t *testing.T) { + testutils.SkipIfNotSupported(t, haveBPFLinkUprobeMulti()) + + prog := mustLoadProgram(t, ebpf.Kprobe, ebpf.AttachTraceUprobeMulti, "") + + // uprobe + um, err := bashEx.UprobeMulti(bashSyms, prog, nil) + if err != nil { + t.Fatal(err) + } + + testLink(t, um, prog) + _ = um.Close() + + // uretprobe + um, err = bashEx.UretprobeMulti(bashSyms, prog, nil) + if err != nil { + t.Fatal(err) + } + + testLink(t, um, prog) + _ = um.Close() +} + +func TestUprobeMultiInput(t *testing.T) { + testutils.SkipIfNotSupported(t, haveBPFLinkUprobeMulti()) + + prog := mustLoadProgram(t, ebpf.Kprobe, ebpf.AttachTraceUprobeMulti, "") + + // Always doing same test for both uprobe and uretprobe + + // One of symbols or offsets must be given. + _, err := bashEx.UprobeMulti([]string{}, prog, nil) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + _, err = bashEx.UretprobeMulti([]string{}, prog, nil) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // One address, two cookies. + _, err = bashEx.UprobeMulti([]string{}, prog, &UprobeMultiOptions{ + Addresses: []uint64{1}, + Cookies: []uint64{2, 3}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + _, err = bashEx.UretprobeMulti([]string{}, prog, &UprobeMultiOptions{ + Addresses: []uint64{1}, + Cookies: []uint64{2, 3}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // Two addresses, one refctr offset. + _, err = bashEx.UprobeMulti([]string{}, prog, &UprobeMultiOptions{ + Addresses: []uint64{1, 2}, + RefCtrOffsets: []uint64{4}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + _, err = bashEx.UretprobeMulti([]string{}, prog, &UprobeMultiOptions{ + Addresses: []uint64{1, 2}, + RefCtrOffsets: []uint64{4}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // It's either symbols or addresses. + _, err = bashEx.UprobeMulti(bashSyms, prog, &UprobeMultiOptions{ + Addresses: []uint64{1}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + _, err = bashEx.UretprobeMulti(bashSyms, prog, &UprobeMultiOptions{ + Addresses: []uint64{1}, + }) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // No addresses and no symbols + _, err = bashEx.UprobeMulti([]string{}, prog, nil) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + _, err = bashEx.UretprobeMulti([]string{}, prog, nil) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // PID not found + _, err = bashEx.UprobeMulti(bashSyms, prog, &UprobeMultiOptions{ + PID: math.MaxUint32, + }) + qt.Assert(t, qt.ErrorIs(err, os.ErrNotExist)) + + _, err = bashEx.UretprobeMulti(bashSyms, prog, &UprobeMultiOptions{ + PID: math.MaxUint32, + }) + qt.Assert(t, qt.ErrorIs(err, os.ErrNotExist)) +} + +func TestUprobeMultiResolveOk(t *testing.T) { + addrSym1, err := bashEx.address(bashSyms[0], 0, 0) + qt.Assert(t, qt.IsNil(err)) + addrSym2, err := bashEx.address(bashSyms[1], 0, 0) + qt.Assert(t, qt.IsNil(err)) + addrSym3, err := bashEx.address(bashSyms[2], 0, 0) + qt.Assert(t, qt.IsNil(err)) + + addrs, err := bashEx.addresses(bashSyms, nil, nil) + qt.Assert(t, qt.IsNil(err)) + qt.Assert(t, qt.DeepEquals(addrs, []uint64{addrSym1, addrSym2, addrSym3})) + + addrs, err = bashEx.addresses(bashSyms, nil, []uint64{5, 10, 11}) + qt.Assert(t, qt.IsNil(err)) + qt.Assert(t, qt.DeepEquals(addrs, []uint64{addrSym1 + 5, addrSym2 + 10, addrSym3 + 11})) + + addrs, err = bashEx.addresses(bashSyms, []uint64{1, 2, 3}, nil) + qt.Assert(t, qt.IsNil(err)) + + qt.Assert(t, qt.DeepEquals(addrs, []uint64{1, 2, 3})) +} + +func TestUprobeMultiResolveFail(t *testing.T) { + // No input + _, err := bashEx.addresses(nil, nil, nil) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // Different dimensions for Addresses and Offsets + _, err = bashEx.addresses(nil, []uint64{100, 200}, []uint64{5, 10, 11}) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) + + // Different dimensions for symbols and Offsets + _, err = bashEx.addresses(bashSyms, nil, []uint64{5, 10}) + qt.Assert(t, qt.ErrorIs(err, errInvalidInput)) +} + +func TestUprobeMultiCookie(t *testing.T) { + testutils.SkipIfNotSupported(t, haveBPFLinkUprobeMulti()) + + prog := mustLoadProgram(t, ebpf.Kprobe, ebpf.AttachTraceUprobeMulti, "") + + // uprobe + um, err := bashEx.UprobeMulti(bashSyms, prog, + &UprobeMultiOptions{ + Cookies: []uint64{1, 2, 3}, + }) + if err != nil { + t.Fatal(err) + } + _ = um.Close() + + // uretprobe + um, err = bashEx.UretprobeMulti(bashSyms, prog, + &UprobeMultiOptions{ + Cookies: []uint64{3, 2, 1}, + }) + if err != nil { + t.Fatal(err) + } + _ = um.Close() +} + +func TestUprobeMultiProgramCall(t *testing.T) { + testutils.SkipIfNotSupported(t, haveBPFLinkUprobeMulti()) + + // We execute 'bash --help' + args := []string{"--help"} + elf := "/bin/bash" + + test := func(retprobe bool, expected uint32) { + m, p := newUpdaterMapProg(t, ebpf.Kprobe, ebpf.AttachTraceUprobeMulti) + + var err error + + // Load the executable. + ex, err := OpenExecutable(elf) + if err != nil { + t.Fatal(err) + } + + var um Link + + // Open UprobeMulti on the executable for the given symbol + // and attach it to the ebpf program created above. + if retprobe { + um, err = ex.UretprobeMulti(bashSyms, p, nil) + } else { + um, err = ex.UprobeMulti(bashSyms, p, nil) + } + if errors.Is(err, ErrNoSymbol) { + // Assume bash_Syms symbols always exist and skip the test + // if the symbol can't be found as certain OS (eg. Debian) + // strip binaries. + t.Skipf("executable %s appear to be stripped, skipping", elf) + } + if err != nil { + t.Fatal(err) + } + + // Trigger ebpf program call. + trigger := func(t *testing.T) { + if err := exec.Command(elf, args...).Run(); err != nil { + t.Fatal(err) + } + } + trigger(t) + + // Detach link. + if err := um.Close(); err != nil { + t.Fatal(err) + } + + assertMapValueGE(t, m, 0, expected) + + // Reset map value to 0 at index 0. + if err := m.Update(uint32(0), uint32(0), ebpf.UpdateExist); err != nil { + t.Fatal(err) + } + + // Retrigger the ebpf program call. + trigger(t) + + // Assert that this time the value has not been updated. + assertMapValue(t, m, 0, 0) + } + + // all 3 uprobes should trigger for entry uprobes + test(false, 3) + + // We have return uprobe installed on main, _start and check_dev_tty + // functions, but only check_dev_tty is triggered, because 'bash --help' + // calls exit(0). + test(true, 1) +} + +func TestHaveBPFLinkUprobeMulti(t *testing.T) { + testutils.CheckFeatureTest(t, haveBPFLinkUprobeMulti) +} diff --git a/link/uprobe_test.go b/link/uprobe_test.go index 101f6a299..bc49a1f9f 100644 --- a/link/uprobe_test.go +++ b/link/uprobe_test.go @@ -18,7 +18,8 @@ import ( var ( bashEx, _ = OpenExecutable("/bin/bash") - bashSym = "main" + bashSyms = []string{"main", "_start", "check_dev_tty"} + bashSym = bashSyms[0] ) func TestExecutable(t *testing.T) {