// SPDX-License-Identifier: GPL-2.0 /* * Test for perf events with SIGTRAP across all threads. * * Copyright (C) 2021, Google LLC. */ #define _GNU_SOURCE /* We need the latest siginfo from the kernel repo. */ #include <sys/types.h> #include <asm/siginfo.h> #define __have_siginfo_t 1 #define __have_sigval_t 1 #define __have_sigevent_t 1 #define __siginfo_t_defined #define __sigval_t_defined #define __sigevent_t_defined #define _BITS_SIGINFO_CONSTS_H 1 #define _BITS_SIGEVENT_CONSTS_H 1 #include <stdbool.h> #include <stddef.h> #include <stdint.h> #include <stdio.h> #include <linux/hw_breakpoint.h> #include <linux/perf_event.h> #include <pthread.h> #include <signal.h> #include <sys/ioctl.h> #include <sys/syscall.h> #include <unistd.h> #include "../kselftest_harness.h" #define NUM_THREADS 5 /* Data shared between test body, threads, and signal handler. */ static struct { int tids_want_signal; /* Which threads still want a signal. */ int signal_count; /* Sanity check number of signals received. */ volatile int iterate_on; /* Variable to set breakpoint on. */ siginfo_t first_siginfo; /* First observed siginfo_t. */ } ctx; /* Unique value to check si_perf_data is correctly set from perf_event_attr::sig_data. */ #define TEST_SIG_DATA(addr, id) (~(unsigned long)(addr) + id) static struct perf_event_attr make_event_attr(bool enabled, volatile void *addr, unsigned long id) { struct perf_event_attr attr = { .type = PERF_TYPE_BREAKPOINT, .size = sizeof(attr), .sample_period = 1, .disabled = !enabled, .bp_addr = (unsigned long)addr, .bp_type = HW_BREAKPOINT_RW, .bp_len = HW_BREAKPOINT_LEN_1, .inherit = 1, /* Children inherit events ... */ .inherit_thread = 1, /* ... but only cloned with CLONE_THREAD. */ .remove_on_exec = 1, /* Required by sigtrap. */ .sigtrap = 1, /* Request synchronous SIGTRAP on event. */ .sig_data = TEST_SIG_DATA(addr, id), .exclude_kernel = 1, /* To allow */ .exclude_hv = 1, /* running as !root */ }; return attr; } static void sigtrap_handler(int signum, siginfo_t *info, void *ucontext) { if (info->si_code != TRAP_PERF) { fprintf(stderr, "%s: unexpected si_code %d\n", __func__, info->si_code); return; } /* * The data in siginfo_t we're interested in should all be the same * across threads. */ if (!__atomic_fetch_add(&ctx.signal_count, 1, __ATOMIC_RELAXED)) ctx.first_siginfo = *info; __atomic_fetch_sub(&ctx.tids_want_signal, syscall(__NR_gettid), __ATOMIC_RELAXED); } static void *test_thread(void *arg) { pthread_barrier_t *barrier = (pthread_barrier_t *)arg; pid_t tid = syscall(__NR_gettid); int iter; int i; pthread_barrier_wait(barrier); __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED); iter = ctx.iterate_on; /* read */ if (iter >= 0) { for (i = 0; i < iter - 1; i++) { __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED); ctx.iterate_on = iter; /* idempotent write */ } } else { while (ctx.iterate_on); } return NULL; } FIXTURE(sigtrap_threads) { struct sigaction oldact; pthread_t threads[NUM_THREADS]; pthread_barrier_t barrier; int fd; }; FIXTURE_SETUP(sigtrap_threads) { struct perf_event_attr attr = make_event_attr(false, &ctx.iterate_on, 0); struct sigaction action = {}; int i; memset(&ctx, 0, sizeof(ctx)); /* Initialize sigtrap handler. */ action.sa_flags = SA_SIGINFO | SA_NODEFER; action.sa_sigaction = sigtrap_handler; sigemptyset(&action.sa_mask); ASSERT_EQ(sigaction(SIGTRAP, &action, &self->oldact), 0); /* Initialize perf event. */ self->fd = syscall(__NR_perf_event_open, &attr, 0, -1, -1, PERF_FLAG_FD_CLOEXEC); ASSERT_NE(self->fd, -1); /* Spawn threads inheriting perf event. */ pthread_barrier_init(&self->barrier, NULL, NUM_THREADS + 1); for (i = 0; i < NUM_THREADS; i++) ASSERT_EQ(pthread_create(&self->threads[i], NULL, test_thread, &self->barrier), 0); } FIXTURE_TEARDOWN(sigtrap_threads) { pthread_barrier_destroy(&self->barrier); close(self->fd); sigaction(SIGTRAP, &self->oldact, NULL); } static void run_test_threads(struct __test_metadata *_metadata, FIXTURE_DATA(sigtrap_threads) *self) { int i; pthread_barrier_wait(&self->barrier); for (i = 0; i < NUM_THREADS; i++) ASSERT_EQ(pthread_join(self->threads[i], NULL), 0); } TEST_F(sigtrap_threads, remain_disabled) { run_test_threads(_metadata, self); EXPECT_EQ(ctx.signal_count, 0); EXPECT_NE(ctx.tids_want_signal, 0); } TEST_F(sigtrap_threads, enable_event) { EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0); run_test_threads(_metadata, self); EXPECT_EQ(ctx.signal_count, NUM_THREADS); EXPECT_EQ(ctx.tids_want_signal, 0); EXPECT_EQ(ctx.first_siginfo.si_addr, &ctx.iterate_on); EXPECT_EQ(ctx.first_siginfo.si_perf_type, PERF_TYPE_BREAKPOINT); EXPECT_EQ(ctx.first_siginfo.si_perf_data, TEST_SIG_DATA(&ctx.iterate_on, 0)); /* Check enabled for parent. */ ctx.iterate_on = 0; EXPECT_EQ(ctx.signal_count, NUM_THREADS + 1); } /* Test that modification propagates to all inherited events. */ TEST_F(sigtrap_threads, modify_and_enable_event) { struct perf_event_attr new_attr = make_event_attr(true, &ctx.iterate_on, 42); EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_MODIFY_ATTRIBUTES, &new_attr), 0); run_test_threads(_metadata, self); EXPECT_EQ(ctx.signal_count, NUM_THREADS); EXPECT_EQ(ctx.tids_want_signal, 0); EXPECT_EQ(ctx.first_siginfo.si_addr, &ctx.iterate_on); EXPECT_EQ(ctx.first_siginfo.si_perf_type, PERF_TYPE_BREAKPOINT); EXPECT_EQ(ctx.first_siginfo.si_perf_data, TEST_SIG_DATA(&ctx.iterate_on, 42)); /* Check enabled for parent. */ ctx.iterate_on = 0; EXPECT_EQ(ctx.signal_count, NUM_THREADS + 1); } /* Stress test event + signal handling. */ TEST_F(sigtrap_threads, signal_stress) { ctx.iterate_on = 3000; EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0); run_test_threads(_metadata, self); EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_DISABLE, 0), 0); EXPECT_EQ(ctx.signal_count, NUM_THREADS * ctx.iterate_on); EXPECT_EQ(ctx.tids_want_signal, 0); EXPECT_EQ(ctx.first_siginfo.si_addr, &ctx.iterate_on); EXPECT_EQ(ctx.first_siginfo.si_perf_type, PERF_TYPE_BREAKPOINT); EXPECT_EQ(ctx.first_siginfo.si_perf_data, TEST_SIG_DATA(&ctx.iterate_on, 0)); } TEST_F(sigtrap_threads, signal_stress_with_disable) { const int target_count = NUM_THREADS * 3000; int i; ctx.iterate_on = -1; EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0); pthread_barrier_wait(&self->barrier); while (__atomic_load_n(&ctx.signal_count, __ATOMIC_RELAXED) < target_count) { EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_DISABLE, 0), 0); EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0); } ctx.iterate_on = 0; for (i = 0; i < NUM_THREADS; i++) ASSERT_EQ(pthread_join(self->threads[i], NULL), 0); EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_DISABLE, 0), 0); EXPECT_EQ(ctx.first_siginfo.si_addr, &ctx.iterate_on); EXPECT_EQ(ctx.first_siginfo.si_perf_type, PERF_TYPE_BREAKPOINT); EXPECT_EQ(ctx.first_siginfo.si_perf_data, TEST_SIG_DATA(&ctx.iterate_on, 0)); } TEST_HARNESS_MAIN