sigtrap.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Basic test for sigtrap support.
  4. *
  5. * Copyright (C) 2021, Google LLC.
  6. */
  7. #include <errno.h>
  8. #include <stdint.h>
  9. #include <stdlib.h>
  10. #include <linux/hw_breakpoint.h>
  11. #include <linux/string.h>
  12. #include <pthread.h>
  13. #include <signal.h>
  14. #include <sys/ioctl.h>
  15. #include <sys/syscall.h>
  16. #include <unistd.h>
  17. #include "cloexec.h"
  18. #include "debug.h"
  19. #include "event.h"
  20. #include "tests.h"
  21. #include "../perf-sys.h"
  22. #define NUM_THREADS 5
  23. static struct {
  24. int tids_want_signal; /* Which threads still want a signal. */
  25. int signal_count; /* Sanity check number of signals received. */
  26. volatile int iterate_on; /* Variable to set breakpoint on. */
  27. siginfo_t first_siginfo; /* First observed siginfo_t. */
  28. } ctx;
  29. #define TEST_SIG_DATA (~(unsigned long)(&ctx.iterate_on))
  30. static struct perf_event_attr make_event_attr(void)
  31. {
  32. struct perf_event_attr attr = {
  33. .type = PERF_TYPE_BREAKPOINT,
  34. .size = sizeof(attr),
  35. .sample_period = 1,
  36. .disabled = 1,
  37. .bp_addr = (unsigned long)&ctx.iterate_on,
  38. .bp_type = HW_BREAKPOINT_RW,
  39. .bp_len = HW_BREAKPOINT_LEN_1,
  40. .inherit = 1, /* Children inherit events ... */
  41. .inherit_thread = 1, /* ... but only cloned with CLONE_THREAD. */
  42. .remove_on_exec = 1, /* Required by sigtrap. */
  43. .sigtrap = 1, /* Request synchronous SIGTRAP on event. */
  44. .sig_data = TEST_SIG_DATA,
  45. .exclude_kernel = 1, /* To allow */
  46. .exclude_hv = 1, /* running as !root */
  47. };
  48. return attr;
  49. }
  50. #ifdef HAVE_BPF_SKEL
  51. #include <bpf/btf.h>
  52. #include <util/btf.h>
  53. static struct btf *btf;
  54. static bool btf__available(void)
  55. {
  56. if (btf == NULL)
  57. btf = btf__load_vmlinux_btf();
  58. return btf != NULL;
  59. }
  60. static void btf__exit(void)
  61. {
  62. btf__free(btf);
  63. btf = NULL;
  64. }
  65. static bool attr_has_sigtrap(void)
  66. {
  67. int id;
  68. if (!btf__available()) {
  69. /* should be an old kernel */
  70. return false;
  71. }
  72. id = btf__find_by_name_kind(btf, "perf_event_attr", BTF_KIND_STRUCT);
  73. if (id < 0)
  74. return false;
  75. return __btf_type__find_member_by_name(btf, id, "sigtrap") != NULL;
  76. }
  77. static bool kernel_with_sleepable_spinlocks(void)
  78. {
  79. const struct btf_member *member;
  80. const struct btf_type *type;
  81. const char *type_name;
  82. int id;
  83. if (!btf__available())
  84. return false;
  85. id = btf__find_by_name_kind(btf, "spinlock", BTF_KIND_STRUCT);
  86. if (id < 0)
  87. return false;
  88. // Only RT has a "lock" member for "struct spinlock"
  89. member = __btf_type__find_member_by_name(btf, id, "lock");
  90. if (member == NULL)
  91. return false;
  92. // But check its type as well
  93. type = btf__type_by_id(btf, member->type);
  94. if (!type || !btf_is_struct(type))
  95. return false;
  96. type_name = btf__name_by_offset(btf, type->name_off);
  97. return type_name && !strcmp(type_name, "rt_mutex_base");
  98. }
  99. #else /* !HAVE_BPF_SKEL */
  100. static bool attr_has_sigtrap(void)
  101. {
  102. struct perf_event_attr attr = {
  103. .type = PERF_TYPE_SOFTWARE,
  104. .config = PERF_COUNT_SW_DUMMY,
  105. .size = sizeof(attr),
  106. .remove_on_exec = 1, /* Required by sigtrap. */
  107. .sigtrap = 1, /* Request synchronous SIGTRAP on event. */
  108. };
  109. int fd;
  110. bool ret = false;
  111. fd = sys_perf_event_open(&attr, 0, -1, -1, perf_event_open_cloexec_flag());
  112. if (fd >= 0) {
  113. ret = true;
  114. close(fd);
  115. }
  116. return ret;
  117. }
  118. static bool kernel_with_sleepable_spinlocks(void)
  119. {
  120. return false;
  121. }
  122. static void btf__exit(void)
  123. {
  124. }
  125. #endif /* HAVE_BPF_SKEL */
  126. static void
  127. sigtrap_handler(int signum __maybe_unused, siginfo_t *info, void *ucontext __maybe_unused)
  128. {
  129. if (!__atomic_fetch_add(&ctx.signal_count, 1, __ATOMIC_RELAXED))
  130. ctx.first_siginfo = *info;
  131. __atomic_fetch_sub(&ctx.tids_want_signal, syscall(SYS_gettid), __ATOMIC_RELAXED);
  132. }
  133. static void *test_thread(void *arg)
  134. {
  135. pthread_barrier_t *barrier = (pthread_barrier_t *)arg;
  136. pid_t tid = syscall(SYS_gettid);
  137. int i;
  138. pthread_barrier_wait(barrier);
  139. __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED);
  140. for (i = 0; i < ctx.iterate_on - 1; i++)
  141. __atomic_fetch_add(&ctx.tids_want_signal, tid, __ATOMIC_RELAXED);
  142. return NULL;
  143. }
  144. static int run_test_threads(pthread_t *threads, pthread_barrier_t *barrier)
  145. {
  146. int i;
  147. pthread_barrier_wait(barrier);
  148. for (i = 0; i < NUM_THREADS; i++)
  149. TEST_ASSERT_EQUAL("pthread_join() failed", pthread_join(threads[i], NULL), 0);
  150. return TEST_OK;
  151. }
  152. static int run_stress_test(int fd, pthread_t *threads, pthread_barrier_t *barrier)
  153. {
  154. int ret, expected_sigtraps;
  155. ctx.iterate_on = 3000;
  156. TEST_ASSERT_EQUAL("misfired signal?", ctx.signal_count, 0);
  157. TEST_ASSERT_EQUAL("enable failed", ioctl(fd, PERF_EVENT_IOC_ENABLE, 0), 0);
  158. ret = run_test_threads(threads, barrier);
  159. TEST_ASSERT_EQUAL("disable failed", ioctl(fd, PERF_EVENT_IOC_DISABLE, 0), 0);
  160. expected_sigtraps = NUM_THREADS * ctx.iterate_on;
  161. if (ctx.signal_count < expected_sigtraps && kernel_with_sleepable_spinlocks()) {
  162. pr_debug("Expected %d sigtraps, got %d, running on a kernel with sleepable spinlocks.\n",
  163. expected_sigtraps, ctx.signal_count);
  164. pr_debug("See https://lore.kernel.org/all/e368f2c848d77fbc8d259f44e2055fe469c219cf.camel@gmx.de/\n");
  165. return TEST_SKIP;
  166. } else
  167. TEST_ASSERT_EQUAL("unexpected sigtraps", ctx.signal_count, expected_sigtraps);
  168. TEST_ASSERT_EQUAL("missing signals or incorrectly delivered", ctx.tids_want_signal, 0);
  169. TEST_ASSERT_VAL("unexpected si_addr", ctx.first_siginfo.si_addr == &ctx.iterate_on);
  170. #if 0 /* FIXME: enable when libc's signal.h has si_perf_{type,data} */
  171. TEST_ASSERT_EQUAL("unexpected si_perf_type", ctx.first_siginfo.si_perf_type,
  172. PERF_TYPE_BREAKPOINT);
  173. TEST_ASSERT_EQUAL("unexpected si_perf_data", ctx.first_siginfo.si_perf_data,
  174. TEST_SIG_DATA);
  175. #endif
  176. return ret;
  177. }
  178. static int test__sigtrap(struct test_suite *test __maybe_unused, int subtest __maybe_unused)
  179. {
  180. struct perf_event_attr attr = make_event_attr();
  181. struct sigaction action = {};
  182. struct sigaction oldact;
  183. pthread_t threads[NUM_THREADS];
  184. pthread_barrier_t barrier;
  185. char sbuf[STRERR_BUFSIZE];
  186. int i, fd, ret = TEST_FAIL;
  187. if (!BP_SIGNAL_IS_SUPPORTED) {
  188. pr_debug("Test not supported on this architecture");
  189. return TEST_SKIP;
  190. }
  191. pthread_barrier_init(&barrier, NULL, NUM_THREADS + 1);
  192. action.sa_flags = SA_SIGINFO | SA_NODEFER;
  193. action.sa_sigaction = sigtrap_handler;
  194. sigemptyset(&action.sa_mask);
  195. if (sigaction(SIGTRAP, &action, &oldact)) {
  196. pr_debug("FAILED sigaction(): %s\n", str_error_r(errno, sbuf, sizeof(sbuf)));
  197. goto out;
  198. }
  199. fd = sys_perf_event_open(&attr, 0, -1, -1, perf_event_open_cloexec_flag());
  200. if (fd < 0) {
  201. if (attr_has_sigtrap()) {
  202. pr_debug("FAILED sys_perf_event_open(): %s\n",
  203. str_error_r(errno, sbuf, sizeof(sbuf)));
  204. } else {
  205. pr_debug("perf_event_attr doesn't have sigtrap\n");
  206. ret = TEST_SKIP;
  207. }
  208. goto out_restore_sigaction;
  209. }
  210. for (i = 0; i < NUM_THREADS; i++) {
  211. if (pthread_create(&threads[i], NULL, test_thread, &barrier)) {
  212. pr_debug("FAILED pthread_create(): %s\n", str_error_r(errno, sbuf, sizeof(sbuf)));
  213. goto out_close_perf_event;
  214. }
  215. }
  216. ret = run_stress_test(fd, threads, &barrier);
  217. out_close_perf_event:
  218. close(fd);
  219. out_restore_sigaction:
  220. sigaction(SIGTRAP, &oldact, NULL);
  221. out:
  222. pthread_barrier_destroy(&barrier);
  223. btf__exit();
  224. return ret;
  225. }
  226. DEFINE_SUITE("Sigtrap", sigtrap);