1*4f1223e8SApple OSS Distributions #include <assert.h>
2*4f1223e8SApple OSS Distributions #include <errno.h>
3*4f1223e8SApple OSS Distributions #include <stdio.h>
4*4f1223e8SApple OSS Distributions #include <stdlib.h>
5*4f1223e8SApple OSS Distributions #include <string.h>
6*4f1223e8SApple OSS Distributions #include <unistd.h>
7*4f1223e8SApple OSS Distributions #include <sys/types.h>
8*4f1223e8SApple OSS Distributions #include <sys/sysctl.h>
9*4f1223e8SApple OSS Distributions #include <sys/qos.h>
10*4f1223e8SApple OSS Distributions
11*4f1223e8SApple OSS Distributions #include <dispatch/dispatch.h>
12*4f1223e8SApple OSS Distributions #include <os/lock.h>
13*4f1223e8SApple OSS Distributions #include <mach/mach.h>
14*4f1223e8SApple OSS Distributions #include <mach/mach_time.h>
15*4f1223e8SApple OSS Distributions
16*4f1223e8SApple OSS Distributions #include <pthread/workqueue_private.h>
17*4f1223e8SApple OSS Distributions #include <pthread/qos_private.h>
18*4f1223e8SApple OSS Distributions
19*4f1223e8SApple OSS Distributions static dispatch_group_t group;
20*4f1223e8SApple OSS Distributions static mach_timebase_info_data_t timebase_info;
21*4f1223e8SApple OSS Distributions
22*4f1223e8SApple OSS Distributions static void
req_cooperative_wq_threads(qos_class_t qos,size_t num_threads)23*4f1223e8SApple OSS Distributions req_cooperative_wq_threads(qos_class_t qos, size_t num_threads)
24*4f1223e8SApple OSS Distributions {
25*4f1223e8SApple OSS Distributions int ret;
26*4f1223e8SApple OSS Distributions
27*4f1223e8SApple OSS Distributions for (size_t i = 0; i < num_threads; i++) {
28*4f1223e8SApple OSS Distributions dispatch_group_enter(group);
29*4f1223e8SApple OSS Distributions
30*4f1223e8SApple OSS Distributions ret = _pthread_workqueue_add_cooperativethreads(1,
31*4f1223e8SApple OSS Distributions _pthread_qos_class_encode(qos, 0, 0));
32*4f1223e8SApple OSS Distributions assert(ret == 0);
33*4f1223e8SApple OSS Distributions }
34*4f1223e8SApple OSS Distributions }
35*4f1223e8SApple OSS Distributions
36*4f1223e8SApple OSS Distributions static void
req_wq_threads(qos_class_t qos,size_t num_threads,bool overcommit)37*4f1223e8SApple OSS Distributions req_wq_threads(qos_class_t qos, size_t num_threads, bool overcommit)
38*4f1223e8SApple OSS Distributions {
39*4f1223e8SApple OSS Distributions int ret;
40*4f1223e8SApple OSS Distributions
41*4f1223e8SApple OSS Distributions for (size_t i = 0; i < num_threads; i++) {
42*4f1223e8SApple OSS Distributions dispatch_group_enter(group);
43*4f1223e8SApple OSS Distributions
44*4f1223e8SApple OSS Distributions ret = _pthread_workqueue_addthreads(1,
45*4f1223e8SApple OSS Distributions _pthread_qos_class_encode(qos, 0,
46*4f1223e8SApple OSS Distributions (overcommit ? _PTHREAD_PRIORITY_OVERCOMMIT_FLAG : 0)));
47*4f1223e8SApple OSS Distributions assert(ret == 0);
48*4f1223e8SApple OSS Distributions }
49*4f1223e8SApple OSS Distributions }
50*4f1223e8SApple OSS Distributions
51*4f1223e8SApple OSS Distributions static uint32_t
ncpus(void)52*4f1223e8SApple OSS Distributions ncpus(void)
53*4f1223e8SApple OSS Distributions {
54*4f1223e8SApple OSS Distributions static uint32_t num_cpus;
55*4f1223e8SApple OSS Distributions if (!num_cpus) {
56*4f1223e8SApple OSS Distributions uint32_t n;
57*4f1223e8SApple OSS Distributions size_t s = sizeof(n);
58*4f1223e8SApple OSS Distributions sysctlbyname("hw.ncpu", &n, &s, NULL, 0);
59*4f1223e8SApple OSS Distributions num_cpus = n;
60*4f1223e8SApple OSS Distributions }
61*4f1223e8SApple OSS Distributions return num_cpus;
62*4f1223e8SApple OSS Distributions }
63*4f1223e8SApple OSS Distributions
64*4f1223e8SApple OSS Distributions static inline bool
thread_is_overcommit(pthread_priority_t priority)65*4f1223e8SApple OSS Distributions thread_is_overcommit(pthread_priority_t priority)
66*4f1223e8SApple OSS Distributions {
67*4f1223e8SApple OSS Distributions return (priority & _PTHREAD_PRIORITY_OVERCOMMIT_FLAG) != 0;
68*4f1223e8SApple OSS Distributions }
69*4f1223e8SApple OSS Distributions
70*4f1223e8SApple OSS Distributions static inline bool
thread_is_nonovercommit(pthread_priority_t priority)71*4f1223e8SApple OSS Distributions thread_is_nonovercommit(pthread_priority_t priority)
72*4f1223e8SApple OSS Distributions {
73*4f1223e8SApple OSS Distributions return (priority & (_PTHREAD_PRIORITY_OVERCOMMIT_FLAG | _PTHREAD_PRIORITY_COOPERATIVE_FLAG)) != 0;
74*4f1223e8SApple OSS Distributions }
75*4f1223e8SApple OSS Distributions
76*4f1223e8SApple OSS Distributions static inline bool
thread_is_cooperative(pthread_priority_t priority)77*4f1223e8SApple OSS Distributions thread_is_cooperative(pthread_priority_t priority)
78*4f1223e8SApple OSS Distributions {
79*4f1223e8SApple OSS Distributions return (priority & _PTHREAD_PRIORITY_COOPERATIVE_FLAG) != 0;
80*4f1223e8SApple OSS Distributions }
81*4f1223e8SApple OSS Distributions
82*4f1223e8SApple OSS Distributions qos_class_t
thread_has_qos(pthread_priority_t pri)83*4f1223e8SApple OSS Distributions thread_has_qos(pthread_priority_t pri)
84*4f1223e8SApple OSS Distributions {
85*4f1223e8SApple OSS Distributions return _pthread_qos_class_decode(pri, NULL, NULL);
86*4f1223e8SApple OSS Distributions }
87*4f1223e8SApple OSS Distributions
88*4f1223e8SApple OSS Distributions char *
qos_to_str(qos_class_t qos)89*4f1223e8SApple OSS Distributions qos_to_str(qos_class_t qos)
90*4f1223e8SApple OSS Distributions {
91*4f1223e8SApple OSS Distributions switch (qos) {
92*4f1223e8SApple OSS Distributions case QOS_CLASS_MAINTENANCE:
93*4f1223e8SApple OSS Distributions return "MT";
94*4f1223e8SApple OSS Distributions case QOS_CLASS_BACKGROUND:
95*4f1223e8SApple OSS Distributions return "BG";
96*4f1223e8SApple OSS Distributions case QOS_CLASS_UTILITY:
97*4f1223e8SApple OSS Distributions return "UT";
98*4f1223e8SApple OSS Distributions case QOS_CLASS_DEFAULT:
99*4f1223e8SApple OSS Distributions return "DEF";
100*4f1223e8SApple OSS Distributions case QOS_CLASS_USER_INITIATED:
101*4f1223e8SApple OSS Distributions return "IN";
102*4f1223e8SApple OSS Distributions case QOS_CLASS_USER_INTERACTIVE:
103*4f1223e8SApple OSS Distributions return "UI";
104*4f1223e8SApple OSS Distributions }
105*4f1223e8SApple OSS Distributions }
106*4f1223e8SApple OSS Distributions
107*4f1223e8SApple OSS Distributions /*
108*4f1223e8SApple OSS Distributions * Test that we handle cooperative requests first and then overcommit if they
109*4f1223e8SApple OSS Distributions * are at the same QoS
110*4f1223e8SApple OSS Distributions */
111*4f1223e8SApple OSS Distributions
112*4f1223e8SApple OSS Distributions static bool overcommit_thread_request_handled = false;
113*4f1223e8SApple OSS Distributions static bool cooperative_thread_request_handled = false;
114*4f1223e8SApple OSS Distributions
115*4f1223e8SApple OSS Distributions static void
worker_cooperative_then_overcommit(pthread_priority_t priority)116*4f1223e8SApple OSS Distributions worker_cooperative_then_overcommit(pthread_priority_t priority)
117*4f1223e8SApple OSS Distributions {
118*4f1223e8SApple OSS Distributions if (thread_is_cooperative(priority)) {
119*4f1223e8SApple OSS Distributions assert(!overcommit_thread_request_handled);
120*4f1223e8SApple OSS Distributions cooperative_thread_request_handled = true;
121*4f1223e8SApple OSS Distributions } else if (thread_is_overcommit(priority)) {
122*4f1223e8SApple OSS Distributions assert(cooperative_thread_request_handled);
123*4f1223e8SApple OSS Distributions overcommit_thread_request_handled = true;
124*4f1223e8SApple OSS Distributions }
125*4f1223e8SApple OSS Distributions
126*4f1223e8SApple OSS Distributions dispatch_group_leave(group);
127*4f1223e8SApple OSS Distributions }
128*4f1223e8SApple OSS Distributions
129*4f1223e8SApple OSS Distributions int
do_cooperative_then_overcommit()130*4f1223e8SApple OSS Distributions do_cooperative_then_overcommit()
131*4f1223e8SApple OSS Distributions {
132*4f1223e8SApple OSS Distributions int ret = _pthread_workqueue_init(worker_cooperative_then_overcommit, 0, 0);
133*4f1223e8SApple OSS Distributions assert(ret == 0);
134*4f1223e8SApple OSS Distributions
135*4f1223e8SApple OSS Distributions req_wq_threads(QOS_CLASS_USER_INITIATED, 1, true);
136*4f1223e8SApple OSS Distributions req_cooperative_wq_threads(QOS_CLASS_USER_INITIATED, 1);
137*4f1223e8SApple OSS Distributions
138*4f1223e8SApple OSS Distributions dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
139*4f1223e8SApple OSS Distributions return 0;
140*4f1223e8SApple OSS Distributions }
141*4f1223e8SApple OSS Distributions
142*4f1223e8SApple OSS Distributions /*
143*4f1223e8SApple OSS Distributions * Test thread reuse from cooperative requests
144*4f1223e8SApple OSS Distributions */
145*4f1223e8SApple OSS Distributions
146*4f1223e8SApple OSS Distributions bool test_should_end = false;
147*4f1223e8SApple OSS Distributions
148*4f1223e8SApple OSS Distributions qos_class_t
get_rand_qos_class(void)149*4f1223e8SApple OSS Distributions get_rand_qos_class(void)
150*4f1223e8SApple OSS Distributions {
151*4f1223e8SApple OSS Distributions switch (rand() % 6) {
152*4f1223e8SApple OSS Distributions case 0:
153*4f1223e8SApple OSS Distributions return QOS_CLASS_MAINTENANCE;
154*4f1223e8SApple OSS Distributions case 1:
155*4f1223e8SApple OSS Distributions return QOS_CLASS_BACKGROUND;
156*4f1223e8SApple OSS Distributions case 2:
157*4f1223e8SApple OSS Distributions return QOS_CLASS_UTILITY;
158*4f1223e8SApple OSS Distributions case 3:
159*4f1223e8SApple OSS Distributions return QOS_CLASS_DEFAULT;
160*4f1223e8SApple OSS Distributions case 4:
161*4f1223e8SApple OSS Distributions return QOS_CLASS_USER_INITIATED;
162*4f1223e8SApple OSS Distributions case 5:
163*4f1223e8SApple OSS Distributions return QOS_CLASS_USER_INTERACTIVE;
164*4f1223e8SApple OSS Distributions }
165*4f1223e8SApple OSS Distributions }
166*4f1223e8SApple OSS Distributions
167*4f1223e8SApple OSS Distributions int
get_rand_num_thread_requests(void)168*4f1223e8SApple OSS Distributions get_rand_num_thread_requests(void)
169*4f1223e8SApple OSS Distributions {
170*4f1223e8SApple OSS Distributions return rand() % (ncpus() * 2);
171*4f1223e8SApple OSS Distributions }
172*4f1223e8SApple OSS Distributions
173*4f1223e8SApple OSS Distributions uint64_t
get_rand_spin_duration_nsecs(void)174*4f1223e8SApple OSS Distributions get_rand_spin_duration_nsecs(void)
175*4f1223e8SApple OSS Distributions {
176*4f1223e8SApple OSS Distributions /* Spin for at most half a second */
177*4f1223e8SApple OSS Distributions return rand() % (NSEC_PER_SEC / 2);
178*4f1223e8SApple OSS Distributions }
179*4f1223e8SApple OSS Distributions
180*4f1223e8SApple OSS Distributions void
spin(uint64_t spin_duration_nsecs)181*4f1223e8SApple OSS Distributions spin(uint64_t spin_duration_nsecs)
182*4f1223e8SApple OSS Distributions {
183*4f1223e8SApple OSS Distributions uint64_t duration = spin_duration_nsecs * timebase_info.denom / timebase_info.numer;
184*4f1223e8SApple OSS Distributions uint64_t deadline = mach_absolute_time() + duration;
185*4f1223e8SApple OSS Distributions while (mach_absolute_time() < deadline) {
186*4f1223e8SApple OSS Distributions ;
187*4f1223e8SApple OSS Distributions }
188*4f1223e8SApple OSS Distributions }
189*4f1223e8SApple OSS Distributions
190*4f1223e8SApple OSS Distributions static void
worker_cb_stress(pthread_priority_t priority)191*4f1223e8SApple OSS Distributions worker_cb_stress(pthread_priority_t priority)
192*4f1223e8SApple OSS Distributions {
193*4f1223e8SApple OSS Distributions if (test_should_end) {
194*4f1223e8SApple OSS Distributions dispatch_group_leave(group);
195*4f1223e8SApple OSS Distributions return;
196*4f1223e8SApple OSS Distributions }
197*4f1223e8SApple OSS Distributions
198*4f1223e8SApple OSS Distributions if (thread_is_cooperative(priority)) {
199*4f1223e8SApple OSS Distributions printf("\t Cooperative thread of QoS %s\n", qos_to_str(thread_has_qos(priority)));
200*4f1223e8SApple OSS Distributions spin(get_rand_spin_duration_nsecs());
201*4f1223e8SApple OSS Distributions req_wq_threads(get_rand_qos_class(), get_rand_num_thread_requests(), false);
202*4f1223e8SApple OSS Distributions } else if (thread_is_nonovercommit(priority)) {
203*4f1223e8SApple OSS Distributions printf("\t Nonovercommit thread of QoS %s\n", qos_to_str(thread_has_qos(priority)));
204*4f1223e8SApple OSS Distributions
205*4f1223e8SApple OSS Distributions spin(get_rand_spin_duration_nsecs());
206*4f1223e8SApple OSS Distributions req_cooperative_wq_threads(get_rand_qos_class(), get_rand_num_thread_requests());
207*4f1223e8SApple OSS Distributions } else {
208*4f1223e8SApple OSS Distributions printf("\t Overcommit thread of QoS %s\n", qos_to_str(thread_has_qos(priority)));
209*4f1223e8SApple OSS Distributions req_wq_threads(get_rand_qos_class(), get_rand_num_thread_requests(), true);
210*4f1223e8SApple OSS Distributions spin(get_rand_spin_duration_nsecs());
211*4f1223e8SApple OSS Distributions }
212*4f1223e8SApple OSS Distributions
213*4f1223e8SApple OSS Distributions dispatch_group_leave(group);
214*4f1223e8SApple OSS Distributions }
215*4f1223e8SApple OSS Distributions
216*4f1223e8SApple OSS Distributions int
do_stress_test()217*4f1223e8SApple OSS Distributions do_stress_test()
218*4f1223e8SApple OSS Distributions {
219*4f1223e8SApple OSS Distributions int ret = _pthread_workqueue_init(worker_cb_stress, 0, 0);
220*4f1223e8SApple OSS Distributions assert(ret == 0);
221*4f1223e8SApple OSS Distributions
222*4f1223e8SApple OSS Distributions req_wq_threads(QOS_CLASS_DEFAULT, ncpus() / 2, true);
223*4f1223e8SApple OSS Distributions req_cooperative_wq_threads(QOS_CLASS_USER_INITIATED, ncpus());
224*4f1223e8SApple OSS Distributions req_wq_threads(QOS_CLASS_DEFAULT, ncpus(), false);
225*4f1223e8SApple OSS Distributions
226*4f1223e8SApple OSS Distributions sleep(10);
227*4f1223e8SApple OSS Distributions
228*4f1223e8SApple OSS Distributions test_should_end = true;
229*4f1223e8SApple OSS Distributions
230*4f1223e8SApple OSS Distributions dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
231*4f1223e8SApple OSS Distributions printf("\n All thread requests completed\n");
232*4f1223e8SApple OSS Distributions return 0;
233*4f1223e8SApple OSS Distributions }
234*4f1223e8SApple OSS Distributions
235*4f1223e8SApple OSS Distributions int
main(int argc,char * argv[])236*4f1223e8SApple OSS Distributions main(int argc, char * argv[])
237*4f1223e8SApple OSS Distributions {
238*4f1223e8SApple OSS Distributions int ret = 0;
239*4f1223e8SApple OSS Distributions
240*4f1223e8SApple OSS Distributions if (argc < 2) {
241*4f1223e8SApple OSS Distributions return EINVAL;
242*4f1223e8SApple OSS Distributions }
243*4f1223e8SApple OSS Distributions
244*4f1223e8SApple OSS Distributions const char *cmd = argv[1];
245*4f1223e8SApple OSS Distributions
246*4f1223e8SApple OSS Distributions group = dispatch_group_create();
247*4f1223e8SApple OSS Distributions mach_timebase_info(&timebase_info);
248*4f1223e8SApple OSS Distributions
249*4f1223e8SApple OSS Distributions if (strcmp(cmd, "cooperative_then_overcommit") == 0) {
250*4f1223e8SApple OSS Distributions return do_cooperative_then_overcommit();
251*4f1223e8SApple OSS Distributions }
252*4f1223e8SApple OSS Distributions
253*4f1223e8SApple OSS Distributions if (strcmp(cmd, "stress_test") == 0) {
254*4f1223e8SApple OSS Distributions return do_stress_test();
255*4f1223e8SApple OSS Distributions }
256*4f1223e8SApple OSS Distributions
257*4f1223e8SApple OSS Distributions return -1;
258*4f1223e8SApple OSS Distributions }
259