1*43a90889SApple OSS Distributions /*
2*43a90889SApple OSS Distributions * turnstiles_test: Tests turnstile kernel primitive.
3*43a90889SApple OSS Distributions */
4*43a90889SApple OSS Distributions
5*43a90889SApple OSS Distributions #ifdef T_NAMESPACE
6*43a90889SApple OSS Distributions #undef T_NAMESPACE
7*43a90889SApple OSS Distributions #endif
8*43a90889SApple OSS Distributions
9*43a90889SApple OSS Distributions #include <darwintest.h>
10*43a90889SApple OSS Distributions #include <darwintest_multiprocess.h>
11*43a90889SApple OSS Distributions
12*43a90889SApple OSS Distributions #include <pthread.h>
13*43a90889SApple OSS Distributions #include <launch.h>
14*43a90889SApple OSS Distributions #include <servers/bootstrap.h>
15*43a90889SApple OSS Distributions #include <stdlib.h>
16*43a90889SApple OSS Distributions #include <sys/event.h>
17*43a90889SApple OSS Distributions #include <unistd.h>
18*43a90889SApple OSS Distributions #include <crt_externs.h>
19*43a90889SApple OSS Distributions #include <sys/sysctl.h>
20*43a90889SApple OSS Distributions #include <sys/types.h>
21*43a90889SApple OSS Distributions
22*43a90889SApple OSS Distributions #define SYSCTL_TURNSTILE_TEST_USER_DEFAULT 1
23*43a90889SApple OSS Distributions #define SYSCTL_TURNSTILE_TEST_USER_HASHTABLE 2
24*43a90889SApple OSS Distributions #define SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT 3
25*43a90889SApple OSS Distributions #define SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE 4
26*43a90889SApple OSS Distributions
27*43a90889SApple OSS Distributions T_GLOBAL_META(T_META_NAMESPACE("xnu.turnstiles_test"));
28*43a90889SApple OSS Distributions
29*43a90889SApple OSS Distributions static void
thread_create_at_qos(qos_class_t qos,void * (* function)(void *),int type)30*43a90889SApple OSS Distributions thread_create_at_qos(qos_class_t qos, void * (*function)(void *), int type)
31*43a90889SApple OSS Distributions {
32*43a90889SApple OSS Distributions qos_class_t qos_thread;
33*43a90889SApple OSS Distributions pthread_t thread;
34*43a90889SApple OSS Distributions pthread_attr_t attr;
35*43a90889SApple OSS Distributions int ret;
36*43a90889SApple OSS Distributions
37*43a90889SApple OSS Distributions ret = setpriority(PRIO_DARWIN_ROLE, 0, PRIO_DARWIN_ROLE_UI_FOCAL);
38*43a90889SApple OSS Distributions if (ret != 0) {
39*43a90889SApple OSS Distributions T_LOG("set priority failed\n");
40*43a90889SApple OSS Distributions }
41*43a90889SApple OSS Distributions
42*43a90889SApple OSS Distributions pthread_attr_init(&attr);
43*43a90889SApple OSS Distributions pthread_attr_set_qos_class_np(&attr, qos, 0);
44*43a90889SApple OSS Distributions pthread_create(&thread, &attr, function, (void *)type);
45*43a90889SApple OSS Distributions
46*43a90889SApple OSS Distributions T_LOG("pthread created\n");
47*43a90889SApple OSS Distributions pthread_get_qos_class_np(thread, &qos_thread, NULL);
48*43a90889SApple OSS Distributions T_EXPECT_EQ(qos_thread, (qos_class_t)qos, NULL);
49*43a90889SApple OSS Distributions }
50*43a90889SApple OSS Distributions
51*43a90889SApple OSS Distributions static int
get_sched_pri(thread_t thread_port)52*43a90889SApple OSS Distributions get_sched_pri(thread_t thread_port)
53*43a90889SApple OSS Distributions {
54*43a90889SApple OSS Distributions kern_return_t kr;
55*43a90889SApple OSS Distributions
56*43a90889SApple OSS Distributions thread_extended_info_data_t extended_info;
57*43a90889SApple OSS Distributions mach_msg_type_number_t count = THREAD_EXTENDED_INFO_COUNT;
58*43a90889SApple OSS Distributions kr = thread_info(thread_port, THREAD_EXTENDED_INFO,
59*43a90889SApple OSS Distributions (thread_info_t)&extended_info, &count);
60*43a90889SApple OSS Distributions
61*43a90889SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "thread_info");
62*43a90889SApple OSS Distributions return extended_info.pth_curpri;
63*43a90889SApple OSS Distributions }
64*43a90889SApple OSS Distributions
65*43a90889SApple OSS Distributions static int
get_base_pri(thread_t thread_port)66*43a90889SApple OSS Distributions get_base_pri(thread_t thread_port)
67*43a90889SApple OSS Distributions {
68*43a90889SApple OSS Distributions kern_return_t kr;
69*43a90889SApple OSS Distributions
70*43a90889SApple OSS Distributions thread_extended_info_data_t extended_info;
71*43a90889SApple OSS Distributions mach_msg_type_number_t count = THREAD_EXTENDED_INFO_COUNT;
72*43a90889SApple OSS Distributions kr = thread_info(thread_port, THREAD_EXTENDED_INFO,
73*43a90889SApple OSS Distributions (thread_info_t)&extended_info, &count);
74*43a90889SApple OSS Distributions
75*43a90889SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "thread_info");
76*43a90889SApple OSS Distributions return extended_info.pth_priority;
77*43a90889SApple OSS Distributions }
78*43a90889SApple OSS Distributions
79*43a90889SApple OSS Distributions static void
turnstile_prim_lock(int type)80*43a90889SApple OSS Distributions turnstile_prim_lock(int type)
81*43a90889SApple OSS Distributions {
82*43a90889SApple OSS Distributions int ret;
83*43a90889SApple OSS Distributions uint64_t tid;
84*43a90889SApple OSS Distributions int in_val = type;
85*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
86*43a90889SApple OSS Distributions T_LOG("sysctlbyname lock type %d called from thread %llu \n", type, tid);
87*43a90889SApple OSS Distributions ret = sysctlbyname("kern.turnstiles_test_lock", NULL, 0, &in_val, sizeof(in_val));
88*43a90889SApple OSS Distributions T_LOG("sysctlbyname lock returned from thread %llu with value %d \n", tid, ret);
89*43a90889SApple OSS Distributions }
90*43a90889SApple OSS Distributions
91*43a90889SApple OSS Distributions static void
turnstile_prim_unlock(int type)92*43a90889SApple OSS Distributions turnstile_prim_unlock(int type)
93*43a90889SApple OSS Distributions {
94*43a90889SApple OSS Distributions int ret;
95*43a90889SApple OSS Distributions uint64_t tid;
96*43a90889SApple OSS Distributions int in_val = type;
97*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
98*43a90889SApple OSS Distributions T_LOG("sysctlbyname unlock type %d called from thread %llu \n", type, tid);
99*43a90889SApple OSS Distributions ret = sysctlbyname("kern.turnstiles_test_unlock", NULL, 0, &in_val, sizeof(in_val));
100*43a90889SApple OSS Distributions T_LOG("sysctlbyname unlock returned from thread %llu with value %d \n", tid, ret);
101*43a90889SApple OSS Distributions }
102*43a90889SApple OSS Distributions
103*43a90889SApple OSS Distributions struct thread_data {
104*43a90889SApple OSS Distributions int pri_to_set;
105*43a90889SApple OSS Distributions int lock1;
106*43a90889SApple OSS Distributions int lock2;
107*43a90889SApple OSS Distributions unsigned int sleep;
108*43a90889SApple OSS Distributions int sched_pri_to_check;
109*43a90889SApple OSS Distributions int base_pri_to_check;
110*43a90889SApple OSS Distributions };
111*43a90889SApple OSS Distributions
112*43a90889SApple OSS Distributions static void *
chain_locking(void * args)113*43a90889SApple OSS Distributions chain_locking(void* args)
114*43a90889SApple OSS Distributions {
115*43a90889SApple OSS Distributions struct thread_data* data = (struct thread_data*) args;
116*43a90889SApple OSS Distributions int policy, pri;
117*43a90889SApple OSS Distributions int ret;
118*43a90889SApple OSS Distributions struct sched_param param;
119*43a90889SApple OSS Distributions
120*43a90889SApple OSS Distributions /* Change our priority to pri_to_set */
121*43a90889SApple OSS Distributions ret = pthread_getschedparam(pthread_self(), &policy, ¶m);
122*43a90889SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(ret, "pthread_getschedparam");
123*43a90889SApple OSS Distributions
124*43a90889SApple OSS Distributions param.sched_priority = data->pri_to_set;
125*43a90889SApple OSS Distributions
126*43a90889SApple OSS Distributions /* this sets both sched and base pri */
127*43a90889SApple OSS Distributions ret = pthread_setschedparam(pthread_self(), policy, ¶m);
128*43a90889SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(ret, "pthread_setschedparam");
129*43a90889SApple OSS Distributions
130*43a90889SApple OSS Distributions pri = get_sched_pri(mach_thread_self());
131*43a90889SApple OSS Distributions
132*43a90889SApple OSS Distributions T_ASSERT_EQ(pri, data->pri_to_set, "Priority before holding locks");
133*43a90889SApple OSS Distributions
134*43a90889SApple OSS Distributions /* take lock1 */
135*43a90889SApple OSS Distributions if (data->lock1) {
136*43a90889SApple OSS Distributions turnstile_prim_lock(data->lock1);
137*43a90889SApple OSS Distributions }
138*43a90889SApple OSS Distributions
139*43a90889SApple OSS Distributions /* take lock2 */
140*43a90889SApple OSS Distributions if (data->lock2) {
141*43a90889SApple OSS Distributions turnstile_prim_lock(data->lock2);
142*43a90889SApple OSS Distributions }
143*43a90889SApple OSS Distributions
144*43a90889SApple OSS Distributions if (data->sleep) {
145*43a90889SApple OSS Distributions sleep(data->sleep);
146*43a90889SApple OSS Distributions }
147*43a90889SApple OSS Distributions
148*43a90889SApple OSS Distributions if (data->sched_pri_to_check) {
149*43a90889SApple OSS Distributions pri = get_sched_pri(mach_thread_self());
150*43a90889SApple OSS Distributions T_ASSERT_EQ(pri, data->sched_pri_to_check, "Sched priority while holding locks");
151*43a90889SApple OSS Distributions }
152*43a90889SApple OSS Distributions
153*43a90889SApple OSS Distributions if (data->base_pri_to_check) {
154*43a90889SApple OSS Distributions pri = get_base_pri(mach_thread_self());
155*43a90889SApple OSS Distributions T_ASSERT_EQ(pri, data->base_pri_to_check, "Base priority while holding locks");
156*43a90889SApple OSS Distributions }
157*43a90889SApple OSS Distributions
158*43a90889SApple OSS Distributions if (data->lock2) {
159*43a90889SApple OSS Distributions turnstile_prim_unlock(data->lock2);
160*43a90889SApple OSS Distributions }
161*43a90889SApple OSS Distributions
162*43a90889SApple OSS Distributions if (data->lock1) {
163*43a90889SApple OSS Distributions turnstile_prim_unlock(data->lock1);
164*43a90889SApple OSS Distributions }
165*43a90889SApple OSS Distributions
166*43a90889SApple OSS Distributions pri = get_sched_pri(mach_thread_self());
167*43a90889SApple OSS Distributions T_ASSERT_EQ(pri, data->pri_to_set, "Priority after releasing locks");
168*43a90889SApple OSS Distributions
169*43a90889SApple OSS Distributions return NULL;
170*43a90889SApple OSS Distributions }
171*43a90889SApple OSS Distributions
172*43a90889SApple OSS Distributions static void *
take_lock_check_priority(void * arg)173*43a90889SApple OSS Distributions take_lock_check_priority(void * arg)
174*43a90889SApple OSS Distributions {
175*43a90889SApple OSS Distributions int old_pri = get_base_pri(mach_thread_self());
176*43a90889SApple OSS Distributions int unboosted_pri;
177*43a90889SApple OSS Distributions int boosted_pri;
178*43a90889SApple OSS Distributions int after_unlock_pri;
179*43a90889SApple OSS Distributions uint64_t tid;
180*43a90889SApple OSS Distributions int type = (int)arg;
181*43a90889SApple OSS Distributions
182*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
183*43a90889SApple OSS Distributions
184*43a90889SApple OSS Distributions T_ASSERT_EQ(old_pri, 37, "thread(%llu) priority before acquiring the lock is %d\n", tid, old_pri);
185*43a90889SApple OSS Distributions
186*43a90889SApple OSS Distributions /* Take the test lock */
187*43a90889SApple OSS Distributions turnstile_prim_lock(type);
188*43a90889SApple OSS Distributions
189*43a90889SApple OSS Distributions unboosted_pri = get_base_pri(mach_thread_self());
190*43a90889SApple OSS Distributions T_ASSERT_EQ(unboosted_pri, 37, "thread(%llu) priority after acquiring the lock (uncontended) is %d\n", tid, unboosted_pri);
191*43a90889SApple OSS Distributions
192*43a90889SApple OSS Distributions sleep(8);
193*43a90889SApple OSS Distributions
194*43a90889SApple OSS Distributions /* Check for elevated priority */
195*43a90889SApple OSS Distributions boosted_pri = get_base_pri(mach_thread_self());
196*43a90889SApple OSS Distributions T_ASSERT_EQ(boosted_pri, 47, "thread(%llu) priority after contention by 47 thread is %d\n", tid, boosted_pri);
197*43a90889SApple OSS Distributions
198*43a90889SApple OSS Distributions /* Drop the lock */
199*43a90889SApple OSS Distributions turnstile_prim_unlock(type);
200*43a90889SApple OSS Distributions
201*43a90889SApple OSS Distributions /* Check for regular priority */
202*43a90889SApple OSS Distributions after_unlock_pri = get_base_pri(mach_thread_self());
203*43a90889SApple OSS Distributions T_ASSERT_EQ(after_unlock_pri, 37, "thread(%llu) priority after dropping lock is %d\n", tid, after_unlock_pri);
204*43a90889SApple OSS Distributions
205*43a90889SApple OSS Distributions return NULL;
206*43a90889SApple OSS Distributions }
207*43a90889SApple OSS Distributions
208*43a90889SApple OSS Distributions static void *
try_to_take_lock_and_unlock(void * arg)209*43a90889SApple OSS Distributions try_to_take_lock_and_unlock(void *arg)
210*43a90889SApple OSS Distributions {
211*43a90889SApple OSS Distributions uint64_t tid;
212*43a90889SApple OSS Distributions int type = (int)arg;
213*43a90889SApple OSS Distributions
214*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
215*43a90889SApple OSS Distributions sleep(4);
216*43a90889SApple OSS Distributions
217*43a90889SApple OSS Distributions int old_pri = get_base_pri(mach_thread_self());
218*43a90889SApple OSS Distributions T_ASSERT_EQ(old_pri, 47, "thread(%llu) priority before acquiring the lock is %d\n", tid, old_pri);
219*43a90889SApple OSS Distributions
220*43a90889SApple OSS Distributions /* Try taking the test lock */
221*43a90889SApple OSS Distributions turnstile_prim_lock(type);
222*43a90889SApple OSS Distributions sleep(2);
223*43a90889SApple OSS Distributions turnstile_prim_unlock(type);
224*43a90889SApple OSS Distributions return NULL;
225*43a90889SApple OSS Distributions }
226*43a90889SApple OSS Distributions
227*43a90889SApple OSS Distributions static void *
take_lock_and_exit(void * arg)228*43a90889SApple OSS Distributions take_lock_and_exit(void * arg)
229*43a90889SApple OSS Distributions {
230*43a90889SApple OSS Distributions int old_pri = get_base_pri(mach_thread_self());
231*43a90889SApple OSS Distributions int unboosted_pri;
232*43a90889SApple OSS Distributions int boosted_pri;
233*43a90889SApple OSS Distributions uint64_t tid;
234*43a90889SApple OSS Distributions int type = (int)arg;
235*43a90889SApple OSS Distributions
236*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
237*43a90889SApple OSS Distributions
238*43a90889SApple OSS Distributions T_ASSERT_EQ(old_pri, 37, "thread(%llu) priority before acquiring the lock is %d\n", tid, old_pri);
239*43a90889SApple OSS Distributions
240*43a90889SApple OSS Distributions /* Take the test lock */
241*43a90889SApple OSS Distributions turnstile_prim_lock(type);
242*43a90889SApple OSS Distributions
243*43a90889SApple OSS Distributions unboosted_pri = get_base_pri(mach_thread_self());
244*43a90889SApple OSS Distributions T_ASSERT_EQ(unboosted_pri, 37, "thread(%llu) priority after acquiring the lock (uncontended) is %d\n", tid, unboosted_pri);
245*43a90889SApple OSS Distributions
246*43a90889SApple OSS Distributions sleep(8);
247*43a90889SApple OSS Distributions
248*43a90889SApple OSS Distributions /* Check for elevated priority */
249*43a90889SApple OSS Distributions boosted_pri = get_base_pri(mach_thread_self());
250*43a90889SApple OSS Distributions T_ASSERT_EQ(boosted_pri, 47, "thread(%llu) priority after contention by 47 thread is %d\n", tid, boosted_pri);
251*43a90889SApple OSS Distributions
252*43a90889SApple OSS Distributions /* return without unlocking the lock */
253*43a90889SApple OSS Distributions return NULL;
254*43a90889SApple OSS Distributions }
255*43a90889SApple OSS Distributions
256*43a90889SApple OSS Distributions static void *
unlock_an_owner_exited_lock(void * arg)257*43a90889SApple OSS Distributions unlock_an_owner_exited_lock(void *arg)
258*43a90889SApple OSS Distributions {
259*43a90889SApple OSS Distributions uint64_t tid;
260*43a90889SApple OSS Distributions int type = (int)arg;
261*43a90889SApple OSS Distributions
262*43a90889SApple OSS Distributions pthread_threadid_np(NULL, &tid);
263*43a90889SApple OSS Distributions sleep(12);
264*43a90889SApple OSS Distributions
265*43a90889SApple OSS Distributions int old_pri = get_base_pri(mach_thread_self());
266*43a90889SApple OSS Distributions T_ASSERT_EQ(old_pri, 47, "thread(%llu) priority before acquiring the lock is %d\n", tid, old_pri);
267*43a90889SApple OSS Distributions
268*43a90889SApple OSS Distributions /* Unlock the test lock causing the turnstile code to call thread_deallocate_safe */
269*43a90889SApple OSS Distributions turnstile_prim_unlock(type);
270*43a90889SApple OSS Distributions return NULL;
271*43a90889SApple OSS Distributions }
272*43a90889SApple OSS Distributions
273*43a90889SApple OSS Distributions /*
274*43a90889SApple OSS Distributions * Test 1: test if lock contended by a UI thread boosts the owner to UI qos.
275*43a90889SApple OSS Distributions */
276*43a90889SApple OSS Distributions static void
test1(int type)277*43a90889SApple OSS Distributions test1(int type)
278*43a90889SApple OSS Distributions {
279*43a90889SApple OSS Distributions T_LOG("Test 1: test if lock contended by a UI thread boosts the owner to UI qos");
280*43a90889SApple OSS Distributions
281*43a90889SApple OSS Distributions /* Create a thread at IN and take lock */
282*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INITIATED, &take_lock_check_priority, type);
283*43a90889SApple OSS Distributions
284*43a90889SApple OSS Distributions /* Create a thread at UI and try to take lock */
285*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INTERACTIVE, &try_to_take_lock_and_unlock, type);
286*43a90889SApple OSS Distributions
287*43a90889SApple OSS Distributions sleep(12);
288*43a90889SApple OSS Distributions return;
289*43a90889SApple OSS Distributions }
290*43a90889SApple OSS Distributions
291*43a90889SApple OSS Distributions /*
292*43a90889SApple OSS Distributions * Test 2: test if lock contended by a 2 UI thread boosts the owner to UI qos.
293*43a90889SApple OSS Distributions */
294*43a90889SApple OSS Distributions static void
test2(int type)295*43a90889SApple OSS Distributions test2(int type)
296*43a90889SApple OSS Distributions {
297*43a90889SApple OSS Distributions T_LOG("Test 2: test if lock contended by a 2 UI thread boosts the owner to UI qos");
298*43a90889SApple OSS Distributions
299*43a90889SApple OSS Distributions /* Create a thread at IN and take lock */
300*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INITIATED, &take_lock_check_priority, type);
301*43a90889SApple OSS Distributions
302*43a90889SApple OSS Distributions /* Create a thread at UI and try to take lock */
303*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INTERACTIVE, &try_to_take_lock_and_unlock, type);
304*43a90889SApple OSS Distributions
305*43a90889SApple OSS Distributions /* Create a thread at UI and try to take lock */
306*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INTERACTIVE, &try_to_take_lock_and_unlock, type);
307*43a90889SApple OSS Distributions
308*43a90889SApple OSS Distributions sleep(16);
309*43a90889SApple OSS Distributions return;
310*43a90889SApple OSS Distributions }
311*43a90889SApple OSS Distributions
312*43a90889SApple OSS Distributions /*
313*43a90889SApple OSS Distributions * Test 3: test if lock owner thread exiting without unlocking allows turnstile to work correctly.
314*43a90889SApple OSS Distributions */
315*43a90889SApple OSS Distributions static void
test3(int type)316*43a90889SApple OSS Distributions test3(int type)
317*43a90889SApple OSS Distributions {
318*43a90889SApple OSS Distributions T_LOG("Test 3: test if lock owner thread exiting without unlocking allows turnstile to work correctly");
319*43a90889SApple OSS Distributions
320*43a90889SApple OSS Distributions /* Create a thread at IN and take lock */
321*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INITIATED, &take_lock_and_exit, type);
322*43a90889SApple OSS Distributions
323*43a90889SApple OSS Distributions /* Create a thread at UI and try to take lock */
324*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INTERACTIVE, &try_to_take_lock_and_unlock, type);
325*43a90889SApple OSS Distributions
326*43a90889SApple OSS Distributions /* Create a thread at UI and try to take lock */
327*43a90889SApple OSS Distributions thread_create_at_qos(QOS_CLASS_USER_INTERACTIVE, &unlock_an_owner_exited_lock, type);
328*43a90889SApple OSS Distributions
329*43a90889SApple OSS Distributions sleep(16);
330*43a90889SApple OSS Distributions return;
331*43a90889SApple OSS Distributions }
332*43a90889SApple OSS Distributions
333*43a90889SApple OSS Distributions /*
334*43a90889SApple OSS Distributions * Test 4: test if a chain of user-space turnstile primitives followed by kernel primitives works correctly.
335*43a90889SApple OSS Distributions */
336*43a90889SApple OSS Distributions static void
test4(void)337*43a90889SApple OSS Distributions test4(void)
338*43a90889SApple OSS Distributions {
339*43a90889SApple OSS Distributions pthread_t threads[5] = {};
340*43a90889SApple OSS Distributions struct thread_data data[5] = {};
341*43a90889SApple OSS Distributions
342*43a90889SApple OSS Distributions T_LOG("Test 4: test if a chain of user-space turnstile primitives followed by kernel primitives works correctly");
343*43a90889SApple OSS Distributions
344*43a90889SApple OSS Distributions /*
345*43a90889SApple OSS Distributions * Chain: t4->ud->t3->uh->t2->kh->t1->kd->t0
346*43a90889SApple OSS Distributions * ud and uh (user space turnstiles) will push base pri and sched pri
347*43a90889SApple OSS Distributions * kd and kh (kernel space turnstiles) will push sched pri
348*43a90889SApple OSS Distributions * sched pri should be propagated up to the end
349*43a90889SApple OSS Distributions * kh is the breaking point of the chain for sched pri
350*43a90889SApple OSS Distributions */
351*43a90889SApple OSS Distributions
352*43a90889SApple OSS Distributions
353*43a90889SApple OSS Distributions /* Create a thread at priority 4 and take SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT lock */
354*43a90889SApple OSS Distributions data[0].pri_to_set = 4;
355*43a90889SApple OSS Distributions data[0].lock1 = SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT; /* this should be not locked */
356*43a90889SApple OSS Distributions data[0].lock2 = 0;
357*43a90889SApple OSS Distributions data[0].sleep = 10; /* long sleep, nothing is blocking this thread */
358*43a90889SApple OSS Distributions data[0].sched_pri_to_check = 60;
359*43a90889SApple OSS Distributions data[0].base_pri_to_check = 4;
360*43a90889SApple OSS Distributions pthread_create(&threads[0], NULL, chain_locking, (void *)&data[0]);
361*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
362*43a90889SApple OSS Distributions
363*43a90889SApple OSS Distributions /* Create a thread at priority 31 and take SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE lock followed by SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT */
364*43a90889SApple OSS Distributions data[1].pri_to_set = 31;
365*43a90889SApple OSS Distributions data[1].lock1 = SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE; /* this should be not locked */
366*43a90889SApple OSS Distributions data[1].lock2 = SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT; /* this should be locked */
367*43a90889SApple OSS Distributions data[1].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
368*43a90889SApple OSS Distributions data[1].sched_pri_to_check = 60;
369*43a90889SApple OSS Distributions data[1].base_pri_to_check = 31;
370*43a90889SApple OSS Distributions pthread_create(&threads[1], NULL, chain_locking, (void *)&data[1]);
371*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
372*43a90889SApple OSS Distributions
373*43a90889SApple OSS Distributions /* Create a thread at priority 40 and take SYSCTL_TURNSTILE_TEST_USER_HASHTABLE lock followed by SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE */
374*43a90889SApple OSS Distributions data[2].pri_to_set = 40;
375*43a90889SApple OSS Distributions data[2].lock1 = SYSCTL_TURNSTILE_TEST_USER_HASHTABLE; /* this should be not locked */
376*43a90889SApple OSS Distributions data[2].lock2 = SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE; /* this should be locked */
377*43a90889SApple OSS Distributions data[2].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
378*43a90889SApple OSS Distributions data[2].sched_pri_to_check = 60;
379*43a90889SApple OSS Distributions data[2].base_pri_to_check = 60;
380*43a90889SApple OSS Distributions pthread_create(&threads[2], NULL, chain_locking, (void *)&data[2]);
381*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
382*43a90889SApple OSS Distributions
383*43a90889SApple OSS Distributions /* Create a thread at priority 47 and take SYSCTL_TURNSTILE_TEST_USER_DEFAULT lock followed by SYSCTL_TURNSTILE_TEST_USER_HASHTABLE */
384*43a90889SApple OSS Distributions data[3].pri_to_set = 47;
385*43a90889SApple OSS Distributions data[3].lock1 = SYSCTL_TURNSTILE_TEST_USER_DEFAULT; /* this should be not locked */
386*43a90889SApple OSS Distributions data[3].lock2 = SYSCTL_TURNSTILE_TEST_USER_HASHTABLE; /* this should be locked */
387*43a90889SApple OSS Distributions data[3].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
388*43a90889SApple OSS Distributions data[3].sched_pri_to_check = 60;
389*43a90889SApple OSS Distributions data[3].base_pri_to_check = 60;
390*43a90889SApple OSS Distributions pthread_create(&threads[3], NULL, chain_locking, (void *)&data[3]);
391*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
392*43a90889SApple OSS Distributions
393*43a90889SApple OSS Distributions /* Create a thread at priority 60 and take SYSCTL_TURNSTILE_TEST_USER_DEFAULT */
394*43a90889SApple OSS Distributions data[4].pri_to_set = 60;
395*43a90889SApple OSS Distributions data[4].lock1 = SYSCTL_TURNSTILE_TEST_USER_DEFAULT; /* this should be locked */
396*43a90889SApple OSS Distributions data[4].lock2 = 0;
397*43a90889SApple OSS Distributions data[4].sleep = 0; /* no need to sleep, nothing should be pushing by the time it acquires the lock */
398*43a90889SApple OSS Distributions data[4].sched_pri_to_check = 60; /* this is its own priority */
399*43a90889SApple OSS Distributions data[4].base_pri_to_check = 60;
400*43a90889SApple OSS Distributions pthread_create(&threads[4], NULL, chain_locking, (void *)&data[4]);
401*43a90889SApple OSS Distributions
402*43a90889SApple OSS Distributions sleep(16);
403*43a90889SApple OSS Distributions return;
404*43a90889SApple OSS Distributions }
405*43a90889SApple OSS Distributions
406*43a90889SApple OSS Distributions /*
407*43a90889SApple OSS Distributions * Test 5: test if a chain of user-space turnstile primitives interleaved by kernel primitives works correctly.
408*43a90889SApple OSS Distributions */
409*43a90889SApple OSS Distributions static void
test5(void)410*43a90889SApple OSS Distributions test5(void)
411*43a90889SApple OSS Distributions {
412*43a90889SApple OSS Distributions pthread_t threads[5] = {};
413*43a90889SApple OSS Distributions struct thread_data data[5] = {};
414*43a90889SApple OSS Distributions
415*43a90889SApple OSS Distributions T_LOG("Test 5: test if a chain of user-space turnstile primitives interleaved by kernel primitives works correctly");
416*43a90889SApple OSS Distributions
417*43a90889SApple OSS Distributions /*
418*43a90889SApple OSS Distributions * Chain: t4->ud->t3->kh->t2->uh->t1->kd->t0
419*43a90889SApple OSS Distributions * ud and uh (user space turnstiles) will push base pri and sched pri
420*43a90889SApple OSS Distributions * kd and kh (kernel space turnstiles) will push sched pri
421*43a90889SApple OSS Distributions * uh is the breaking point of the chain for sched pri
422*43a90889SApple OSS Distributions */
423*43a90889SApple OSS Distributions
424*43a90889SApple OSS Distributions /* Create a thread at priority 4 and take SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT lock */
425*43a90889SApple OSS Distributions data[0].pri_to_set = 4;
426*43a90889SApple OSS Distributions data[0].lock1 = SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT; /* this should be not locked */
427*43a90889SApple OSS Distributions data[0].lock2 = 0;
428*43a90889SApple OSS Distributions data[0].sleep = 10; /* long sleep, nothing is blocking this thread */
429*43a90889SApple OSS Distributions data[0].sched_pri_to_check = 41;
430*43a90889SApple OSS Distributions data[0].base_pri_to_check = 4;
431*43a90889SApple OSS Distributions pthread_create(&threads[0], NULL, chain_locking, (void *)&data[0]);
432*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
433*43a90889SApple OSS Distributions
434*43a90889SApple OSS Distributions /* Create a thread at priority 31 and take SYSCTL_TURNSTILE_TEST_USER_HASHTABLE lock followed by SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT */
435*43a90889SApple OSS Distributions data[1].pri_to_set = 31;
436*43a90889SApple OSS Distributions data[1].lock1 = SYSCTL_TURNSTILE_TEST_USER_HASHTABLE; /* this should be not locked */
437*43a90889SApple OSS Distributions data[1].lock2 = SYSCTL_TURNSTILE_TEST_KERNEL_DEFAULT; /* this should be locked */
438*43a90889SApple OSS Distributions data[1].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
439*43a90889SApple OSS Distributions data[1].sched_pri_to_check = 41;
440*43a90889SApple OSS Distributions data[1].base_pri_to_check = 41;
441*43a90889SApple OSS Distributions pthread_create(&threads[1], NULL, chain_locking, (void *)&data[1]);
442*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
443*43a90889SApple OSS Distributions
444*43a90889SApple OSS Distributions /* Create a thread at priority 41 and take SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE lock followed by SYSCTL_TURNSTILE_TEST_USER_HASHTABLE */
445*43a90889SApple OSS Distributions data[2].pri_to_set = 41;
446*43a90889SApple OSS Distributions data[2].lock1 = SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE; /* this should be not locked */
447*43a90889SApple OSS Distributions data[2].lock2 = SYSCTL_TURNSTILE_TEST_USER_HASHTABLE; /* this should be locked */
448*43a90889SApple OSS Distributions data[2].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
449*43a90889SApple OSS Distributions data[2].sched_pri_to_check = 60;
450*43a90889SApple OSS Distributions data[2].base_pri_to_check = 41;
451*43a90889SApple OSS Distributions pthread_create(&threads[2], NULL, chain_locking, (void *)&data[2]);
452*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
453*43a90889SApple OSS Distributions
454*43a90889SApple OSS Distributions /* Create a thread at priority 47 and take SYSCTL_TURNSTILE_TEST_USER_DEFAULT lock followed by SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE */
455*43a90889SApple OSS Distributions data[3].pri_to_set = 47;
456*43a90889SApple OSS Distributions data[3].lock1 = SYSCTL_TURNSTILE_TEST_USER_DEFAULT; /* this should be not locked */
457*43a90889SApple OSS Distributions data[3].lock2 = SYSCTL_TURNSTILE_TEST_KERNEL_HASHTABLE; /* this should be locked */
458*43a90889SApple OSS Distributions data[3].sleep = 0; /* no need to sleep, everything should be pushing by the time it acquires the lock */
459*43a90889SApple OSS Distributions data[3].sched_pri_to_check = 60;
460*43a90889SApple OSS Distributions data[3].base_pri_to_check = 60;
461*43a90889SApple OSS Distributions pthread_create(&threads[3], NULL, chain_locking, (void *)&data[3]);
462*43a90889SApple OSS Distributions sleep(2); /* give the thread time to acquire the lock */
463*43a90889SApple OSS Distributions
464*43a90889SApple OSS Distributions /* Create a thread at priority 60 and take SYSCTL_TURNSTILE_TEST_USER_DEFAULT */
465*43a90889SApple OSS Distributions data[4].pri_to_set = 60;
466*43a90889SApple OSS Distributions data[4].lock1 = SYSCTL_TURNSTILE_TEST_USER_DEFAULT; /* this should be locked */
467*43a90889SApple OSS Distributions data[4].lock2 = 0;
468*43a90889SApple OSS Distributions data[4].sleep = 0; /* no need to sleep, nothing should be pushing by the time it acquires the lock */
469*43a90889SApple OSS Distributions data[4].sched_pri_to_check = 60; /* this is its own priority */
470*43a90889SApple OSS Distributions data[4].base_pri_to_check = 60;
471*43a90889SApple OSS Distributions pthread_create(&threads[4], NULL, chain_locking, (void *)&data[4]);
472*43a90889SApple OSS Distributions
473*43a90889SApple OSS Distributions sleep(16);
474*43a90889SApple OSS Distributions return;
475*43a90889SApple OSS Distributions }
476*43a90889SApple OSS Distributions
477*43a90889SApple OSS Distributions T_DECL(turnstile_test, "Turnstile test", T_META_ASROOT(YES), T_META_TAG_VM_PREFERRED)
478*43a90889SApple OSS Distributions {
479*43a90889SApple OSS Distributions test1(SYSCTL_TURNSTILE_TEST_USER_DEFAULT);
480*43a90889SApple OSS Distributions test2(SYSCTL_TURNSTILE_TEST_USER_DEFAULT);
481*43a90889SApple OSS Distributions test3(SYSCTL_TURNSTILE_TEST_USER_DEFAULT);
482*43a90889SApple OSS Distributions
483*43a90889SApple OSS Distributions test1(SYSCTL_TURNSTILE_TEST_USER_HASHTABLE);
484*43a90889SApple OSS Distributions test2(SYSCTL_TURNSTILE_TEST_USER_HASHTABLE);
485*43a90889SApple OSS Distributions test3(SYSCTL_TURNSTILE_TEST_USER_HASHTABLE);
486*43a90889SApple OSS Distributions
487*43a90889SApple OSS Distributions /*
488*43a90889SApple OSS Distributions * rdar://problem/46302128
489*43a90889SApple OSS Distributions * These tests are using a sysctl to lock a dummy kernel resource that uses turnstile.
490*43a90889SApple OSS Distributions * However a thread holding a kernel push from turnstile should never return in
491*43a90889SApple OSS Distributions * userspace, and rdar://problem/24194397 adds an assert for it.
492*43a90889SApple OSS Distributions */
493*43a90889SApple OSS Distributions //test4();
494*43a90889SApple OSS Distributions //test5();
495*43a90889SApple OSS Distributions }
496