Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Marco Elver | 765 | 58.76% | 1 | 7.69% |
Arnaldo Carvalho de Melo | 331 | 25.42% | 6 | 46.15% |
Namhyung Kim | 179 | 13.75% | 1 | 7.69% |
Thomas Richter | 20 | 1.54% | 1 | 7.69% |
Adrian Hunter | 3 | 0.23% | 1 | 7.69% |
Jiri Olsa | 2 | 0.15% | 1 | 7.69% |
John Garry | 1 | 0.08% | 1 | 7.69% |
Greg Kroah-Hartman | 1 | 0.08% | 1 | 7.69% |
Total | 1302 | 13 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
// SPDX-License-Identifier: GPL-2.0 /* * Basic test for sigtrap support. * * Copyright (C) 2021, Google LLC. */ #include <errno.h> #include <stdint.h> #include <stdlib.h> #include <linux/hw_breakpoint.h> #include <linux/string.h> #include <pthread.h> #include <signal.h> #include <sys/ioctl.h> #include <sys/syscall.h> #include <unistd.h> #include "cloexec.h" #include "debug.h" #include "event.h" #include "tests.h" #include "../perf-sys.h" #define NUM_THREADS 5 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; #define TEST_SIG_DATA (~(unsigned long)(&ctx.iterate_on)) static struct perf_event_attr make_event_attr(void) { struct perf_event_attr attr = { .type = PERF_TYPE_BREAKPOINT, .size = sizeof(attr), .sample_period = 1, .disabled = 1, .bp_addr = (unsigned long)&ctx.iterate_on, .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, .exclude_kernel = 1, /* To allow */ .exclude_hv = 1, /* running as !root */ }; return attr; } #ifdef HAVE_BPF_SKEL #include <bpf/btf.h> static struct btf *btf; static bool btf__available(void) { if (btf == NULL) btf = btf__load_vmlinux_btf(); return btf != NULL; } static void btf__exit(void) { btf__free(btf); btf = NULL; } static const struct btf_member *__btf_type__find_member_by_name(int type_id, const char *member_name) { const struct btf_type *t = btf__type_by_id(btf, type_id); const struct btf_member *m; int i; for (i = 0, m = btf_members(t); i < btf_vlen(t); i++, m++) { const char *current_member_name = btf__name_by_offset(btf, m->name_off); if (!strcmp(current_member_name, member_name)) return m; } return NULL; } static bool attr_has_sigtrap(void) { int id; if (!btf__available()) { /* should be an old kernel */ return false; } id = btf__find_by_name_kind(btf, "perf_event_attr", BTF_KIND_STRUCT); if (id < 0) return false; return __btf_type__find_member_by_name(id, "sigtrap") != NULL; } static bool kernel_with_sleepable_spinlocks(void) { const struct btf_member *member; const struct btf_type *type; const char *type_name; int id; if (!btf__available()) return false; id = btf__find_by_name_kind(btf, "spinlock", BTF_KIND_STRUCT); if (id < 0) return false; // Only RT has a "lock" member for "struct spinlock" member = __btf_type__find_member_by_name(id, "lock"); if (member == NULL) return false; // But check its type as well type = btf__type_by_id(btf, member->type); if (!type || !btf_is_struct(type)) return false; type_name = btf__name_by_offset(btf, type->name_off); return type_name && !strcmp(type_name, "rt_mutex_base"); } #else /* !HAVE_BPF_SKEL */ static bool attr_has_sigtrap(void) { struct perf_event_attr attr = { .type = PERF_TYPE_SOFTWARE, .config = PERF_COUNT_SW_DUMMY, .size = sizeof(attr), .remove_on_exec = 1, /* Required by sigtrap. */ .sigtrap = 1, /* Request synchronous SIGTRAP on event. */ }; int fd; bool ret = false; fd = sys_perf_event_open(&attr, 0, -1, -1, perf_event_open_cloexec_flag()); if (fd >= 0) { ret = true; close(fd); } return ret; } static bool kernel_with_sleepable_spinlocks(void) { return false; } static void btf__exit(void) { } #endif /* HAVE_BPF_SKEL */ static void sigtrap_handler(int signum __maybe_unused, siginfo_t *info, void *ucontext __maybe_unused) { if (!__atomic_fetch_add(&ctx.signal_count, 1, __ATOMIC_RELAXED)) ctx.first_siginfo = *info; __atomic_fetch_sub(&ctx.tids_want_signal, syscall(SYS_gettid), __ATOMIC_RELAXED); } static void *test_thread(void *arg) { pthread_barrier_t *barrier = (pthread_barrier_t *)arg; pid_t tid = syscall(SYS_gettid); int i; pthread_barrier_wait(barrier); __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED); for (i = 0; i < ctx.iterate_on - 1; i++) __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED); return NULL; } static int run_test_threads(pthread_t *threads, pthread_barrier_t *barrier) { int i; pthread_barrier_wait(barrier); for (i = 0; i < NUM_THREADS; i++) TEST_ASSERT_EQUAL("pthread_join() failed", pthread_join(threads[i], NULL), 0); return TEST_OK; } static int run_stress_test(int fd, pthread_t *threads, pthread_barrier_t *barrier) { int ret, expected_sigtraps; ctx.iterate_on = 3000; TEST_ASSERT_EQUAL("misfired signal?", ctx.signal_count, 0); TEST_ASSERT_EQUAL("enable failed", ioctl(fd, PERF_EVENT_IOC_ENABLE, 0), 0); ret = run_test_threads(threads, barrier); TEST_ASSERT_EQUAL("disable failed", ioctl(fd, PERF_EVENT_IOC_DISABLE, 0), 0); expected_sigtraps = NUM_THREADS * ctx.iterate_on; if (ctx.signal_count < expected_sigtraps && kernel_with_sleepable_spinlocks()) { pr_debug("Expected %d sigtraps, got %d, running on a kernel with sleepable spinlocks.\n", expected_sigtraps, ctx.signal_count); pr_debug("See https://lore.kernel.org/all/e368f2c848d77fbc8d259f44e2055fe469c219cf.camel@gmx.de/\n"); return TEST_SKIP; } else TEST_ASSERT_EQUAL("unexpected sigtraps", ctx.signal_count, expected_sigtraps); TEST_ASSERT_EQUAL("missing signals or incorrectly delivered", ctx.tids_want_signal, 0); TEST_ASSERT_VAL("unexpected si_addr", ctx.first_siginfo.si_addr == &ctx.iterate_on); #if 0 /* FIXME: enable when libc's signal.h has si_perf_{type,data} */ TEST_ASSERT_EQUAL("unexpected si_perf_type", ctx.first_siginfo.si_perf_type, PERF_TYPE_BREAKPOINT); TEST_ASSERT_EQUAL("unexpected si_perf_data", ctx.first_siginfo.si_perf_data, TEST_SIG_DATA); #endif return ret; } static int test__sigtrap(struct test_suite *test __maybe_unused, int subtest __maybe_unused) { struct perf_event_attr attr = make_event_attr(); struct sigaction action = {}; struct sigaction oldact; pthread_t threads[NUM_THREADS]; pthread_barrier_t barrier; char sbuf[STRERR_BUFSIZE]; int i, fd, ret = TEST_FAIL; if (!BP_SIGNAL_IS_SUPPORTED) { pr_debug("Test not supported on this architecture"); return TEST_SKIP; } pthread_barrier_init(&barrier, NULL, NUM_THREADS + 1); action.sa_flags = SA_SIGINFO | SA_NODEFER; action.sa_sigaction = sigtrap_handler; sigemptyset(&action.sa_mask); if (sigaction(SIGTRAP, &action, &oldact)) { pr_debug("FAILED sigaction(): %s\n", str_error_r(errno, sbuf, sizeof(sbuf))); goto out; } fd = sys_perf_event_open(&attr, 0, -1, -1, perf_event_open_cloexec_flag()); if (fd < 0) { if (attr_has_sigtrap()) { pr_debug("FAILED sys_perf_event_open(): %s\n", str_error_r(errno, sbuf, sizeof(sbuf))); } else { pr_debug("perf_event_attr doesn't have sigtrap\n"); ret = TEST_SKIP; } goto out_restore_sigaction; } for (i = 0; i < NUM_THREADS; i++) { if (pthread_create(&threads[i], NULL, test_thread, &barrier)) { pr_debug("FAILED pthread_create(): %s\n", str_error_r(errno, sbuf, sizeof(sbuf))); goto out_close_perf_event; } } ret = run_stress_test(fd, threads, &barrier); out_close_perf_event: close(fd); out_restore_sigaction: sigaction(SIGTRAP, &oldact, NULL); out: pthread_barrier_destroy(&barrier); btf__exit(); return ret; } DEFINE_SUITE("Sigtrap", sigtrap);
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1