summaryrefslogtreecommitdiff
path: root/internal/eventloop_security_test.go
diff options
context:
space:
mode:
Diffstat (limited to 'internal/eventloop_security_test.go')
-rw-r--r--internal/eventloop_security_test.go63
1 files changed, 63 insertions, 0 deletions
diff --git a/internal/eventloop_security_test.go b/internal/eventloop_security_test.go
index 0dd9ae7..e00b98a 100644
--- a/internal/eventloop_security_test.go
+++ b/internal/eventloop_security_test.go
@@ -43,6 +43,69 @@ func TestHandlePerfOpenExitTracksReturnedFd(t *testing.T) {
}
}
+// TestHandlePerfOpenExitFailedReturnNoFd locks in the audit finding that
+// perf_event_open(2) returns a new fd on success or -1 on error. A negative
+// return (e.g. EPERM, which is common for perf_event_open under a restrictive
+// perf_event_paranoid setting) must not be recorded as an fd, since it is not
+// a real descriptor. The return is an fd number, never a byte count.
+func TestHandlePerfOpenExitFailedReturnNoFd(t *testing.T) {
+ el := mustNewEventLoop(t, eventLoopConfig{})
+
+ enter := &types.PerfOpenEvent{
+ EventType: types.ENTER_PERF_OPEN_EVENT,
+ TraceId: types.SYS_ENTER_PERF_EVENT_OPEN,
+ Time: 100,
+ Pid: 300,
+ Tid: 301,
+ AttrType: 1,
+ AttrSize: 64,
+ Config: 2,
+ TargetPid: -1,
+ Cpu: 0,
+ GroupFd: -1,
+ Flags: 0,
+ }
+ exit := &types.RetEvent{
+ EventType: types.EXIT_RET_EVENT,
+ TraceId: types.SYS_EXIT_PERF_EVENT_OPEN,
+ Time: 200,
+ Ret: -1, // -EPERM mapped to -1 return
+ Pid: 300,
+ Tid: 301,
+ }
+ ep := &event.Pair{EnterEv: enter, ExitEv: exit}
+
+ if ok := el.handlePerfOpenExit(ep, enter); !ok {
+ t.Fatal("handlePerfOpenExit returned false")
+ }
+ if ep.File != nil {
+ t.Fatalf("expected no fd recorded for failed perf_event_open, got file=%v", ep.File)
+ }
+ if _, ok := el.fdState().get(-1); ok {
+ t.Fatal("failed perf_event_open must not register an fd in fdState")
+ }
+}
+
+// TestPerfDescriptorNameFormat pins the descriptor format produced for a
+// successful perf_event_open. The audit confirmed args[0] (attr pointer) is
+// captured as the attr struct's type/config, not as an fd, and args[1] is the
+// monitored pid. The descriptor encodes attr_type, config, target pid, cpu,
+// and group_fd under the "perf:" prefix (matched by the integration test).
+func TestPerfDescriptorNameFormat(t *testing.T) {
+ ev := &types.PerfOpenEvent{
+ AttrType: 4,
+ Config: 7,
+ TargetPid: 0,
+ Cpu: -1,
+ GroupFd: -1,
+ }
+ got := perfDescriptorName(ev)
+ const want = "perf:4:7:0:-1:-1"
+ if got != want {
+ t.Fatalf("perfDescriptorName = %q, want %q", got, want)
+ }
+}
+
func TestHandlePerfOpenExitAppliesPairFilter(t *testing.T) {
el := mustNewEventLoop(t, eventLoopConfig{
filter: globalfilter.Filter{