1*1031c584SApple OSS Distributions #include <pthread.h>
2*1031c584SApple OSS Distributions #include <stdbool.h>
3*1031c584SApple OSS Distributions #include <signal.h>
4*1031c584SApple OSS Distributions #include <stdio.h>
5*1031c584SApple OSS Distributions #include <string.h>
6*1031c584SApple OSS Distributions #include <unistd.h>
7*1031c584SApple OSS Distributions #include <sys/time.h>
8*1031c584SApple OSS Distributions #include <mach/mach_time.h>
9*1031c584SApple OSS Distributions #include <dispatch/dispatch.h>
10*1031c584SApple OSS Distributions
11*1031c584SApple OSS Distributions #include <darwintest.h>
12*1031c584SApple OSS Distributions
13*1031c584SApple OSS Distributions T_GLOBAL_META(
14*1031c584SApple OSS Distributions T_META_NAMESPACE("xnu.workq"),
15*1031c584SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
16*1031c584SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("workq"),
17*1031c584SApple OSS Distributions T_META_RUN_CONCURRENTLY(true));
18*1031c584SApple OSS Distributions
19*1031c584SApple OSS Distributions
20*1031c584SApple OSS Distributions static pthread_t workq_thread;
21*1031c584SApple OSS Distributions static bool signal_received;
22*1031c584SApple OSS Distributions
23*1031c584SApple OSS Distributions static void
signal_handler(int sig __unused,siginfo_t * b __unused,void * unused __unused)24*1031c584SApple OSS Distributions signal_handler(int sig __unused, siginfo_t *b __unused, void* unused __unused)
25*1031c584SApple OSS Distributions {
26*1031c584SApple OSS Distributions if (pthread_self() == workq_thread) {
27*1031c584SApple OSS Distributions signal_received = true;
28*1031c584SApple OSS Distributions }
29*1031c584SApple OSS Distributions }
30*1031c584SApple OSS Distributions
31*1031c584SApple OSS Distributions static void
workq_block(void * unused __unused)32*1031c584SApple OSS Distributions workq_block(void *unused __unused)
33*1031c584SApple OSS Distributions {
34*1031c584SApple OSS Distributions workq_thread = pthread_self();
35*1031c584SApple OSS Distributions
36*1031c584SApple OSS Distributions /*
37*1031c584SApple OSS Distributions * sigset_t set;
38*1031c584SApple OSS Distributions * sigemptyset(&set);
39*1031c584SApple OSS Distributions * sigaddset(&set, SIGPROF);
40*1031c584SApple OSS Distributions * pthread_sigmask(SIG_UNBLOCK, &set, NULL);
41*1031c584SApple OSS Distributions */
42*1031c584SApple OSS Distributions
43*1031c584SApple OSS Distributions uint64_t spin_start = mach_absolute_time();
44*1031c584SApple OSS Distributions while (mach_absolute_time() - spin_start < 30 * NSEC_PER_SEC) {
45*1031c584SApple OSS Distributions if (signal_received) {
46*1031c584SApple OSS Distributions T_PASS("Got SIGPROF!");
47*1031c584SApple OSS Distributions T_END;
48*1031c584SApple OSS Distributions }
49*1031c584SApple OSS Distributions }
50*1031c584SApple OSS Distributions }
51*1031c584SApple OSS Distributions
52*1031c584SApple OSS Distributions T_DECL(workq_sigprof, "test that workqueue threads can receive sigprof")
53*1031c584SApple OSS Distributions {
54*1031c584SApple OSS Distributions struct sigaction sa = {
55*1031c584SApple OSS Distributions .sa_sigaction = signal_handler
56*1031c584SApple OSS Distributions };
57*1031c584SApple OSS Distributions sigfillset(&sa.sa_mask);
58*1031c584SApple OSS Distributions T_ASSERT_POSIX_ZERO(sigaction(SIGPROF, &sa, NULL), NULL);
59*1031c584SApple OSS Distributions
60*1031c584SApple OSS Distributions dispatch_queue_t q = dispatch_get_global_queue(0, 0);
61*1031c584SApple OSS Distributions dispatch_async_f(q, NULL, workq_block);
62*1031c584SApple OSS Distributions
63*1031c584SApple OSS Distributions struct itimerval timerval = {
64*1031c584SApple OSS Distributions .it_interval = {.tv_usec = 10000},
65*1031c584SApple OSS Distributions .it_value = {.tv_usec = 10000}
66*1031c584SApple OSS Distributions };
67*1031c584SApple OSS Distributions T_ASSERT_POSIX_ZERO(setitimer(ITIMER_PROF, &timerval, NULL), NULL);
68*1031c584SApple OSS Distributions
69*1031c584SApple OSS Distributions dispatch_main();
70*1031c584SApple OSS Distributions }
71