1*5e3eaea3SApple OSS Distributions /* test that the header doesn't implicitly depend on others */
2*5e3eaea3SApple OSS Distributions #include <sys/work_interval.h>
3*5e3eaea3SApple OSS Distributions
4*5e3eaea3SApple OSS Distributions #include <stdlib.h>
5*5e3eaea3SApple OSS Distributions #include <stdio.h>
6*5e3eaea3SApple OSS Distributions #include <unistd.h>
7*5e3eaea3SApple OSS Distributions #include <errno.h>
8*5e3eaea3SApple OSS Distributions #include <err.h>
9*5e3eaea3SApple OSS Distributions #include <string.h>
10*5e3eaea3SApple OSS Distributions #include <pthread.h>
11*5e3eaea3SApple OSS Distributions #include <sys/sysctl.h>
12*5e3eaea3SApple OSS Distributions
13*5e3eaea3SApple OSS Distributions #include <mach/mach.h>
14*5e3eaea3SApple OSS Distributions #include <mach/semaphore.h>
15*5e3eaea3SApple OSS Distributions
16*5e3eaea3SApple OSS Distributions #include <libkern/OSAtomic.h>
17*5e3eaea3SApple OSS Distributions
18*5e3eaea3SApple OSS Distributions #include <darwintest.h>
19*5e3eaea3SApple OSS Distributions #include "test_utils.h"
20*5e3eaea3SApple OSS Distributions
21*5e3eaea3SApple OSS Distributions T_GLOBAL_META(T_META_NAMESPACE("xnu.scheduler"),
22*5e3eaea3SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
23*5e3eaea3SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("scheduler"));
24*5e3eaea3SApple OSS Distributions
25*5e3eaea3SApple OSS Distributions
26*5e3eaea3SApple OSS Distributions static mach_timebase_info_data_t timebase_info;
27*5e3eaea3SApple OSS Distributions
28*5e3eaea3SApple OSS Distributions static uint64_t
nanos_to_abs(uint64_t nanos)29*5e3eaea3SApple OSS Distributions nanos_to_abs(uint64_t nanos)
30*5e3eaea3SApple OSS Distributions {
31*5e3eaea3SApple OSS Distributions mach_timebase_info(&timebase_info);
32*5e3eaea3SApple OSS Distributions return nanos * timebase_info.denom / timebase_info.numer;
33*5e3eaea3SApple OSS Distributions }
34*5e3eaea3SApple OSS Distributions
35*5e3eaea3SApple OSS Distributions static uint64_t
abs_to_nanos(uint64_t abs)36*5e3eaea3SApple OSS Distributions abs_to_nanos(uint64_t abs)
37*5e3eaea3SApple OSS Distributions {
38*5e3eaea3SApple OSS Distributions return abs * timebase_info.numer / timebase_info.denom;
39*5e3eaea3SApple OSS Distributions }
40*5e3eaea3SApple OSS Distributions
41*5e3eaea3SApple OSS Distributions static void
set_realtime(pthread_t thread,uint64_t interval_nanos)42*5e3eaea3SApple OSS Distributions set_realtime(pthread_t thread, uint64_t interval_nanos)
43*5e3eaea3SApple OSS Distributions {
44*5e3eaea3SApple OSS Distributions kern_return_t kr;
45*5e3eaea3SApple OSS Distributions thread_time_constraint_policy_data_t pol;
46*5e3eaea3SApple OSS Distributions
47*5e3eaea3SApple OSS Distributions mach_port_t target_thread = pthread_mach_thread_np(thread);
48*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GT(target_thread, 0, "pthread_mach_thread_np");
49*5e3eaea3SApple OSS Distributions
50*5e3eaea3SApple OSS Distributions /* 1s 100ms 10ms */
51*5e3eaea3SApple OSS Distributions pol.period = (uint32_t)nanos_to_abs(interval_nanos);
52*5e3eaea3SApple OSS Distributions pol.constraint = (uint32_t)nanos_to_abs(interval_nanos);
53*5e3eaea3SApple OSS Distributions pol.computation = (uint32_t)nanos_to_abs(interval_nanos - 1000000); // 1 ms of leeway
54*5e3eaea3SApple OSS Distributions
55*5e3eaea3SApple OSS Distributions pol.preemptible = 0; /* Ignored by OS */
56*5e3eaea3SApple OSS Distributions kr = thread_policy_set(target_thread, THREAD_TIME_CONSTRAINT_POLICY, (thread_policy_t) &pol,
57*5e3eaea3SApple OSS Distributions THREAD_TIME_CONSTRAINT_POLICY_COUNT);
58*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "thread_policy_set(THREAD_TIME_CONSTRAINT_POLICY)");
59*5e3eaea3SApple OSS Distributions }
60*5e3eaea3SApple OSS Distributions
61*5e3eaea3SApple OSS Distributions static void
create_coreaudio_work_interval(work_interval_t * wi_handle,work_interval_instance_t * wi_instance,mach_port_t * wi_port,bool enable_telemetry,uint32_t create_flags)62*5e3eaea3SApple OSS Distributions create_coreaudio_work_interval(work_interval_t *wi_handle, work_interval_instance_t *wi_instance,
63*5e3eaea3SApple OSS Distributions mach_port_t *wi_port, bool enable_telemetry, uint32_t create_flags)
64*5e3eaea3SApple OSS Distributions {
65*5e3eaea3SApple OSS Distributions int ret = 0;
66*5e3eaea3SApple OSS Distributions create_flags |= WORK_INTERVAL_FLAG_GROUP | WORK_INTERVAL_FLAG_JOINABLE | WORK_INTERVAL_TYPE_COREAUDIO;
67*5e3eaea3SApple OSS Distributions if (enable_telemetry) {
68*5e3eaea3SApple OSS Distributions create_flags |= WORK_INTERVAL_FLAG_ENABLE_TELEMETRY_DATA;
69*5e3eaea3SApple OSS Distributions }
70*5e3eaea3SApple OSS Distributions
71*5e3eaea3SApple OSS Distributions ret = work_interval_create(wi_handle, create_flags);
72*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "work_interval_create");
73*5e3eaea3SApple OSS Distributions
74*5e3eaea3SApple OSS Distributions ret = work_interval_copy_port(*wi_handle, wi_port);
75*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "work_interval_copy_port");
76*5e3eaea3SApple OSS Distributions
77*5e3eaea3SApple OSS Distributions *wi_instance = work_interval_instance_alloc(*wi_handle);
78*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NE(*wi_instance, NULL, "work_interval_instance_alloc");
79*5e3eaea3SApple OSS Distributions }
80*5e3eaea3SApple OSS Distributions
81*5e3eaea3SApple OSS Distributions static void
join_coreaudio_work_interval(mach_port_t * wi_port,uint64_t interval_nanos)82*5e3eaea3SApple OSS Distributions join_coreaudio_work_interval(mach_port_t *wi_port, uint64_t interval_nanos)
83*5e3eaea3SApple OSS Distributions {
84*5e3eaea3SApple OSS Distributions int ret = 0;
85*5e3eaea3SApple OSS Distributions
86*5e3eaea3SApple OSS Distributions set_realtime(pthread_self(), interval_nanos);
87*5e3eaea3SApple OSS Distributions
88*5e3eaea3SApple OSS Distributions ret = work_interval_join_port(*wi_port);
89*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "work_interval_join_port");
90*5e3eaea3SApple OSS Distributions }
91*5e3eaea3SApple OSS Distributions
92*5e3eaea3SApple OSS Distributions static pthread_mutex_t barrier_lock = PTHREAD_MUTEX_INITIALIZER;
93*5e3eaea3SApple OSS Distributions static pthread_cond_t barrier_cond = PTHREAD_COND_INITIALIZER;
94*5e3eaea3SApple OSS Distributions static uint32_t barrier_count[2];
95*5e3eaea3SApple OSS Distributions static unsigned int active_barrier_ind;
96*5e3eaea3SApple OSS Distributions static uint32_t total_thread_count;
97*5e3eaea3SApple OSS Distributions static uint32_t expected_cond_wakeups;
98*5e3eaea3SApple OSS Distributions
99*5e3eaea3SApple OSS Distributions /*
100*5e3eaea3SApple OSS Distributions * This implementation of a barrier using pthread_cond_t is
101*5e3eaea3SApple OSS Distributions * intended to control the number of thread sleeps/wakeups
102*5e3eaea3SApple OSS Distributions * that can occur, so that the reported wakeup counts from
103*5e3eaea3SApple OSS Distributions * the work interval data can be validated.
104*5e3eaea3SApple OSS Distributions * Each call to pthread_mutex_lock can produce 0 or 1 thread
105*5e3eaea3SApple OSS Distributions * wakeups, and each call to pthread_cond_wait produces 0 or
106*5e3eaea3SApple OSS Distributions * 1 wakeups.
107*5e3eaea3SApple OSS Distributions */
108*5e3eaea3SApple OSS Distributions static void
thread_barrier(void)109*5e3eaea3SApple OSS Distributions thread_barrier(void)
110*5e3eaea3SApple OSS Distributions {
111*5e3eaea3SApple OSS Distributions int ret = 0;
112*5e3eaea3SApple OSS Distributions ret = pthread_mutex_lock(&barrier_lock);
113*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_mutex_lock");
114*5e3eaea3SApple OSS Distributions
115*5e3eaea3SApple OSS Distributions barrier_count[active_barrier_ind]--;
116*5e3eaea3SApple OSS Distributions
117*5e3eaea3SApple OSS Distributions if (barrier_count[active_barrier_ind]) {
118*5e3eaea3SApple OSS Distributions unsigned int local_active_barrier_ind = active_barrier_ind;
119*5e3eaea3SApple OSS Distributions while (barrier_count[local_active_barrier_ind]) {
120*5e3eaea3SApple OSS Distributions expected_cond_wakeups++;
121*5e3eaea3SApple OSS Distributions ret = pthread_cond_wait(&barrier_cond, &barrier_lock);
122*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_cond_wait");
123*5e3eaea3SApple OSS Distributions }
124*5e3eaea3SApple OSS Distributions } else {
125*5e3eaea3SApple OSS Distributions ret = pthread_cond_broadcast(&barrier_cond);
126*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_cond_broadcast");
127*5e3eaea3SApple OSS Distributions active_barrier_ind = (active_barrier_ind + 1) % 2;
128*5e3eaea3SApple OSS Distributions barrier_count[active_barrier_ind] = total_thread_count;
129*5e3eaea3SApple OSS Distributions }
130*5e3eaea3SApple OSS Distributions
131*5e3eaea3SApple OSS Distributions ret = pthread_mutex_unlock(&barrier_lock);
132*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_mutex_unlock");
133*5e3eaea3SApple OSS Distributions }
134*5e3eaea3SApple OSS Distributions
135*5e3eaea3SApple OSS Distributions struct thread_data {
136*5e3eaea3SApple OSS Distributions work_interval_t wi_handle;
137*5e3eaea3SApple OSS Distributions mach_port_t *wi_port;
138*5e3eaea3SApple OSS Distributions unsigned int num_iterations;
139*5e3eaea3SApple OSS Distributions uint64_t interval_nanos;
140*5e3eaea3SApple OSS Distributions };
141*5e3eaea3SApple OSS Distributions
142*5e3eaea3SApple OSS Distributions static volatile int64_t work_sum;
143*5e3eaea3SApple OSS Distributions
144*5e3eaea3SApple OSS Distributions /*
145*5e3eaea3SApple OSS Distributions * This work performed in the work interval is designed to
146*5e3eaea3SApple OSS Distributions * require CPU compute so that CLPC perf-controls the work
147*5e3eaea3SApple OSS Distributions * interval as it typically would. It is also designed such that
148*5e3eaea3SApple OSS Distributions * the threads agree when the work interval work is done
149*5e3eaea3SApple OSS Distributions * (work_sum higher than a specified threshold), so that the
150*5e3eaea3SApple OSS Distributions * amount of work performed will be consistent between the
151*5e3eaea3SApple OSS Distributions * different work interval instances.
152*5e3eaea3SApple OSS Distributions */
153*5e3eaea3SApple OSS Distributions static void
contribute_to_work_sum(void)154*5e3eaea3SApple OSS Distributions contribute_to_work_sum(void)
155*5e3eaea3SApple OSS Distributions {
156*5e3eaea3SApple OSS Distributions volatile unsigned int x = 0;
157*5e3eaea3SApple OSS Distributions do {
158*5e3eaea3SApple OSS Distributions for (int i = 0; i < 1000; i++) {
159*5e3eaea3SApple OSS Distributions x = x * x - x - 1;
160*5e3eaea3SApple OSS Distributions }
161*5e3eaea3SApple OSS Distributions x %= 10;
162*5e3eaea3SApple OSS Distributions } while (OSAtomicAdd64(x, &work_sum) < 10000);
163*5e3eaea3SApple OSS Distributions }
164*5e3eaea3SApple OSS Distributions
165*5e3eaea3SApple OSS Distributions static void *
coreaudio_workload_fn(void * arg)166*5e3eaea3SApple OSS Distributions coreaudio_workload_fn(void *arg)
167*5e3eaea3SApple OSS Distributions {
168*5e3eaea3SApple OSS Distributions struct thread_data *info = (struct thread_data *)arg;
169*5e3eaea3SApple OSS Distributions
170*5e3eaea3SApple OSS Distributions join_coreaudio_work_interval(info->wi_port, info->interval_nanos);
171*5e3eaea3SApple OSS Distributions
172*5e3eaea3SApple OSS Distributions for (unsigned int i = 0; i < info->num_iterations; i++) {
173*5e3eaea3SApple OSS Distributions thread_barrier();
174*5e3eaea3SApple OSS Distributions contribute_to_work_sum();
175*5e3eaea3SApple OSS Distributions }
176*5e3eaea3SApple OSS Distributions
177*5e3eaea3SApple OSS Distributions int ret = work_interval_leave();
178*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "work_interval_leave");
179*5e3eaea3SApple OSS Distributions
180*5e3eaea3SApple OSS Distributions thread_barrier();
181*5e3eaea3SApple OSS Distributions
182*5e3eaea3SApple OSS Distributions return NULL;
183*5e3eaea3SApple OSS Distributions }
184*5e3eaea3SApple OSS Distributions
185*5e3eaea3SApple OSS Distributions static void
start_helper_threads(unsigned int num_threads,pthread_t * threads,struct thread_data * thread_datas,work_interval_t wi_handle,mach_port_t * wi_port,unsigned int num_iterations,uint64_t interval_nanos)186*5e3eaea3SApple OSS Distributions start_helper_threads(unsigned int num_threads, pthread_t *threads, struct thread_data *thread_datas,
187*5e3eaea3SApple OSS Distributions work_interval_t wi_handle, mach_port_t *wi_port, unsigned int num_iterations, uint64_t interval_nanos)
188*5e3eaea3SApple OSS Distributions {
189*5e3eaea3SApple OSS Distributions int ret = 0;
190*5e3eaea3SApple OSS Distributions for (unsigned int i = 0; i < num_threads; i++) {
191*5e3eaea3SApple OSS Distributions thread_datas[i].wi_handle = wi_handle;
192*5e3eaea3SApple OSS Distributions thread_datas[i].wi_port = wi_port;
193*5e3eaea3SApple OSS Distributions thread_datas[i].num_iterations = num_iterations;
194*5e3eaea3SApple OSS Distributions thread_datas[i].interval_nanos = interval_nanos;
195*5e3eaea3SApple OSS Distributions ret = pthread_create(&threads[i], NULL, coreaudio_workload_fn, &thread_datas[i]);
196*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_ZERO(ret, "pthread_create");
197*5e3eaea3SApple OSS Distributions }
198*5e3eaea3SApple OSS Distributions }
199*5e3eaea3SApple OSS Distributions
200*5e3eaea3SApple OSS Distributions static void
start_work_interval_instance(uint64_t interval_length_abs,work_interval_instance_t wi_instance,work_interval_data_t wi_data)201*5e3eaea3SApple OSS Distributions start_work_interval_instance(uint64_t interval_length_abs, work_interval_instance_t wi_instance,
202*5e3eaea3SApple OSS Distributions work_interval_data_t wi_data)
203*5e3eaea3SApple OSS Distributions {
204*5e3eaea3SApple OSS Distributions int ret = 0;
205*5e3eaea3SApple OSS Distributions uint64_t start = mach_absolute_time();
206*5e3eaea3SApple OSS Distributions
207*5e3eaea3SApple OSS Distributions work_interval_instance_clear(wi_instance);
208*5e3eaea3SApple OSS Distributions work_interval_instance_set_start(wi_instance, start);
209*5e3eaea3SApple OSS Distributions work_interval_instance_set_deadline(wi_instance, start + interval_length_abs);
210*5e3eaea3SApple OSS Distributions
211*5e3eaea3SApple OSS Distributions // Sanity assertions that the work interval creation flags and interval id are as expected
212*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_instance->wi_create_flags & WORK_INTERVAL_FLAG_IGNORED, 0, "ignored flag start");
213*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_instance->wi_create_flags & WORK_INTERVAL_TYPE_MASK, WORK_INTERVAL_TYPE_COREAUDIO, "coreaudio start");
214*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NE(wi_instance->wi_interval_id, 0ULL, "nonzero wi_interval_id");
215*5e3eaea3SApple OSS Distributions
216*5e3eaea3SApple OSS Distributions ret = work_interval_instance_start(wi_instance);
217*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_ZERO(ret, "work_interval_instance_start");
218*5e3eaea3SApple OSS Distributions
219*5e3eaea3SApple OSS Distributions if (wi_instance->wi_instance_id == 0ULL) {
220*5e3eaea3SApple OSS Distributions T_LOG("wi_instance_id is 0, which is an acceptable condition for devices running legacy CLPC");
221*5e3eaea3SApple OSS Distributions }
222*5e3eaea3SApple OSS Distributions
223*5e3eaea3SApple OSS Distributions work_interval_instance_get_telemetry_data(wi_instance, wi_data, sizeof(struct work_interval_data));
224*5e3eaea3SApple OSS Distributions }
225*5e3eaea3SApple OSS Distributions
226*5e3eaea3SApple OSS Distributions static uint64_t
finish_work_interval_instance(work_interval_instance_t wi_instance,work_interval_data_t wi_data)227*5e3eaea3SApple OSS Distributions finish_work_interval_instance(work_interval_instance_t wi_instance, work_interval_data_t wi_data)
228*5e3eaea3SApple OSS Distributions {
229*5e3eaea3SApple OSS Distributions int ret = 0;
230*5e3eaea3SApple OSS Distributions uint64_t finish = mach_absolute_time();
231*5e3eaea3SApple OSS Distributions work_interval_instance_set_finish(wi_instance, finish);
232*5e3eaea3SApple OSS Distributions
233*5e3eaea3SApple OSS Distributions // Sanity assertions that the work interval creation flags and interval id are as expected
234*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_instance->wi_create_flags & WORK_INTERVAL_FLAG_IGNORED, 0, "ignored flag");
235*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_instance->wi_create_flags & WORK_INTERVAL_TYPE_MASK, WORK_INTERVAL_TYPE_COREAUDIO, "coreaudio start");
236*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NE(wi_instance->wi_interval_id, 0ULL, "nonzero wi_interval_id");
237*5e3eaea3SApple OSS Distributions
238*5e3eaea3SApple OSS Distributions uint64_t remembered_start = wi_instance->wi_start;
239*5e3eaea3SApple OSS Distributions
240*5e3eaea3SApple OSS Distributions ret = work_interval_instance_finish(wi_instance);
241*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_ZERO(ret, "work_interval_instance_finish");
242*5e3eaea3SApple OSS Distributions
243*5e3eaea3SApple OSS Distributions work_interval_instance_get_telemetry_data(wi_instance, wi_data, sizeof(struct work_interval_data));
244*5e3eaea3SApple OSS Distributions
245*5e3eaea3SApple OSS Distributions return abs_to_nanos(finish - remembered_start);
246*5e3eaea3SApple OSS Distributions }
247*5e3eaea3SApple OSS Distributions
248*5e3eaea3SApple OSS Distributions static void
verify_monotonic_work_interval_data(struct work_interval_data * curr_data,struct work_interval_data * prev_data,bool supports_cpi)249*5e3eaea3SApple OSS Distributions verify_monotonic_work_interval_data(struct work_interval_data *curr_data, struct work_interval_data *prev_data, bool supports_cpi)
250*5e3eaea3SApple OSS Distributions {
251*5e3eaea3SApple OSS Distributions if (prev_data != NULL) {
252*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_external_wakeups, prev_data->wid_external_wakeups, "wid_external_wakeups");
253*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_total_wakeups, prev_data->wid_total_wakeups, "wid_external_wakeups");
254*5e3eaea3SApple OSS Distributions }
255*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_user_time_mach, prev_data == NULL ? 1 : prev_data->wid_user_time_mach, "monotonic wid_user_time_mach");
256*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_system_time_mach, prev_data == NULL ? 1 : prev_data->wid_system_time_mach, "monotonic wid_system_time_mach");
257*5e3eaea3SApple OSS Distributions if (supports_cpi) {
258*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_cycles, prev_data == NULL ? 1 : prev_data->wid_cycles, "monotonic wid_cycles");
259*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_GE(curr_data->wid_instructions, prev_data == NULL ? 1 : prev_data->wid_instructions, "monotonic wid_instructions");
260*5e3eaea3SApple OSS Distributions }
261*5e3eaea3SApple OSS Distributions }
262*5e3eaea3SApple OSS Distributions
263*5e3eaea3SApple OSS Distributions static void
verify_zero_work_interval_data(struct work_interval_data * wi_data,bool supports_cpi)264*5e3eaea3SApple OSS Distributions verify_zero_work_interval_data(struct work_interval_data *wi_data, bool supports_cpi)
265*5e3eaea3SApple OSS Distributions {
266*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_external_wakeups, 0, "zero wid_external_wakeups");
267*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_total_wakeups, 0, "zero wid_total_wakeups");
268*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_user_time_mach, 0ULL, "zero wid_user_time_mach");
269*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_system_time_mach, 0ULL, "zero wid_system_time_mach");
270*5e3eaea3SApple OSS Distributions if (supports_cpi) {
271*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_cycles, 0ULL, "zero wid_cycles");
272*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(wi_data->wid_instructions, 0ULL, "zero wid_instructions");
273*5e3eaea3SApple OSS Distributions }
274*5e3eaea3SApple OSS Distributions }
275*5e3eaea3SApple OSS Distributions
276*5e3eaea3SApple OSS Distributions static void
run_work_interval_data_test(unsigned int num_iterations,uint64_t interval_nanos,unsigned int thread_count,bool enable_telemetry,uint32_t flags)277*5e3eaea3SApple OSS Distributions run_work_interval_data_test(unsigned int num_iterations, uint64_t interval_nanos, unsigned int thread_count,
278*5e3eaea3SApple OSS Distributions bool enable_telemetry, uint32_t flags)
279*5e3eaea3SApple OSS Distributions {
280*5e3eaea3SApple OSS Distributions T_SETUPBEGIN;
281*5e3eaea3SApple OSS Distributions
282*5e3eaea3SApple OSS Distributions int ret = 0;
283*5e3eaea3SApple OSS Distributions
284*5e3eaea3SApple OSS Distributions int supports_cpi = 0;
285*5e3eaea3SApple OSS Distributions size_t supports_cpi_size = sizeof(supports_cpi);
286*5e3eaea3SApple OSS Distributions ret = sysctlbyname("kern.monotonic.supported", &supports_cpi, &supports_cpi_size, NULL, 0);
287*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "sysctl kern.monotonic.supported");
288*5e3eaea3SApple OSS Distributions
289*5e3eaea3SApple OSS Distributions work_interval_t wi_handle = NULL;
290*5e3eaea3SApple OSS Distributions work_interval_instance_t wi_instance = NULL;
291*5e3eaea3SApple OSS Distributions mach_port_t wi_port = MACH_PORT_NULL;
292*5e3eaea3SApple OSS Distributions
293*5e3eaea3SApple OSS Distributions create_coreaudio_work_interval(&wi_handle, &wi_instance, &wi_port, enable_telemetry, flags);
294*5e3eaea3SApple OSS Distributions join_coreaudio_work_interval(&wi_port, interval_nanos);
295*5e3eaea3SApple OSS Distributions
296*5e3eaea3SApple OSS Distributions total_thread_count = thread_count;
297*5e3eaea3SApple OSS Distributions expected_cond_wakeups = 0;
298*5e3eaea3SApple OSS Distributions unsigned int num_helper_threads = thread_count - 1;
299*5e3eaea3SApple OSS Distributions active_barrier_ind = 0;
300*5e3eaea3SApple OSS Distributions barrier_count[active_barrier_ind] = thread_count;
301*5e3eaea3SApple OSS Distributions pthread_t wi_threads[num_helper_threads];
302*5e3eaea3SApple OSS Distributions struct thread_data wi_thread_datas[num_helper_threads];
303*5e3eaea3SApple OSS Distributions
304*5e3eaea3SApple OSS Distributions start_helper_threads(num_helper_threads, wi_threads, wi_thread_datas, wi_handle, &wi_port, num_iterations, interval_nanos);
305*5e3eaea3SApple OSS Distributions
306*5e3eaea3SApple OSS Distributions T_SETUPEND;
307*5e3eaea3SApple OSS Distributions
308*5e3eaea3SApple OSS Distributions uint64_t interval_length_abs = nanos_to_abs(interval_nanos);
309*5e3eaea3SApple OSS Distributions uint64_t duration_sum = 0;
310*5e3eaea3SApple OSS Distributions struct work_interval_data start_data = {0};
311*5e3eaea3SApple OSS Distributions struct work_interval_data finish_data = {0};
312*5e3eaea3SApple OSS Distributions
313*5e3eaea3SApple OSS Distributions for (unsigned int i = 0; i < num_iterations; i++) {
314*5e3eaea3SApple OSS Distributions work_sum = 0;
315*5e3eaea3SApple OSS Distributions
316*5e3eaea3SApple OSS Distributions usleep(1000);
317*5e3eaea3SApple OSS Distributions
318*5e3eaea3SApple OSS Distributions start_work_interval_instance(interval_length_abs, wi_instance, &start_data);
319*5e3eaea3SApple OSS Distributions if (i == 0 && enable_telemetry) {
320*5e3eaea3SApple OSS Distributions verify_monotonic_work_interval_data(&start_data, NULL, supports_cpi);
321*5e3eaea3SApple OSS Distributions } else if (!enable_telemetry) {
322*5e3eaea3SApple OSS Distributions verify_zero_work_interval_data(&start_data, supports_cpi);
323*5e3eaea3SApple OSS Distributions }
324*5e3eaea3SApple OSS Distributions
325*5e3eaea3SApple OSS Distributions thread_barrier();
326*5e3eaea3SApple OSS Distributions contribute_to_work_sum();
327*5e3eaea3SApple OSS Distributions
328*5e3eaea3SApple OSS Distributions duration_sum += finish_work_interval_instance(wi_instance, &finish_data);
329*5e3eaea3SApple OSS Distributions if (enable_telemetry) {
330*5e3eaea3SApple OSS Distributions verify_monotonic_work_interval_data(&finish_data, &start_data, supports_cpi);
331*5e3eaea3SApple OSS Distributions } else {
332*5e3eaea3SApple OSS Distributions verify_zero_work_interval_data(&finish_data, supports_cpi);
333*5e3eaea3SApple OSS Distributions }
334*5e3eaea3SApple OSS Distributions }
335*5e3eaea3SApple OSS Distributions
336*5e3eaea3SApple OSS Distributions ret = work_interval_leave();
337*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "work_interval_leave");
338*5e3eaea3SApple OSS Distributions thread_barrier();
339*5e3eaea3SApple OSS Distributions
340*5e3eaea3SApple OSS Distributions if (enable_telemetry) {
341*5e3eaea3SApple OSS Distributions T_ASSERT_TRUE(true, "Overall wid_external_wakeups: %u\n", finish_data.wid_external_wakeups);
342*5e3eaea3SApple OSS Distributions // Only the wakeups from usleep() are guaranteed to occur
343*5e3eaea3SApple OSS Distributions T_ASSERT_GE(finish_data.wid_total_wakeups, num_iterations, "wid_total_wakeups at least accounts for the usleep() wakeups");
344*5e3eaea3SApple OSS Distributions }
345*5e3eaea3SApple OSS Distributions T_ASSERT_TRUE(true, "Workload survived %u iterations without failures!!! Avg. work interval duration was %llu ns out of a requested %llu ns", num_iterations, duration_sum / num_iterations, interval_nanos);
346*5e3eaea3SApple OSS Distributions }
347*5e3eaea3SApple OSS Distributions
348*5e3eaea3SApple OSS Distributions static const unsigned int DEFAULT_ITERS = 1000;
349*5e3eaea3SApple OSS Distributions static const uint64_t DEFAULT_INTERVAL_NS = 15000000; // 15 ms
350*5e3eaea3SApple OSS Distributions static const uint64_t DEFAULT_THREAD_COUNT = 3;
351*5e3eaea3SApple OSS Distributions
352*5e3eaea3SApple OSS Distributions T_DECL(work_interval_rt_coreaudio_quality_telemetry_data, "receiving accurate telemetry data as a coreaudio work interval",
353*5e3eaea3SApple OSS Distributions T_META_ASROOT(YES), XNU_T_META_SOC_SPECIFIC, T_META_ENABLED(TARGET_CPU_ARM64))
354*5e3eaea3SApple OSS Distributions {
355*5e3eaea3SApple OSS Distributions run_work_interval_data_test(
356*5e3eaea3SApple OSS Distributions DEFAULT_ITERS,
357*5e3eaea3SApple OSS Distributions DEFAULT_INTERVAL_NS,
358*5e3eaea3SApple OSS Distributions DEFAULT_THREAD_COUNT,
359*5e3eaea3SApple OSS Distributions true, // enable_telemetry
360*5e3eaea3SApple OSS Distributions 0); // no added flags
361*5e3eaea3SApple OSS Distributions }
362*5e3eaea3SApple OSS Distributions
363*5e3eaea3SApple OSS Distributions T_DECL(work_interval_rt_coreaudio_telemetry_disabled, "reading telemetry data should see all zeroes if it isn't enabled",
364*5e3eaea3SApple OSS Distributions T_META_ASROOT(YES), XNU_T_META_SOC_SPECIFIC, T_META_ENABLED(TARGET_CPU_ARM64))
365*5e3eaea3SApple OSS Distributions {
366*5e3eaea3SApple OSS Distributions run_work_interval_data_test(
367*5e3eaea3SApple OSS Distributions DEFAULT_ITERS,
368*5e3eaea3SApple OSS Distributions DEFAULT_INTERVAL_NS,
369*5e3eaea3SApple OSS Distributions DEFAULT_THREAD_COUNT,
370*5e3eaea3SApple OSS Distributions false, // enable_telemetry
371*5e3eaea3SApple OSS Distributions 0); // no added flags
372*5e3eaea3SApple OSS Distributions }
373*5e3eaea3SApple OSS Distributions
374*5e3eaea3SApple OSS Distributions T_DECL(work_interval_rt_coreaudio_telemetry_data_many_threads, "work interval telemetry data works with many joined threads",
375*5e3eaea3SApple OSS Distributions T_META_ASROOT(YES), XNU_T_META_SOC_SPECIFIC, T_META_ENABLED(TARGET_CPU_ARM64))
376*5e3eaea3SApple OSS Distributions {
377*5e3eaea3SApple OSS Distributions run_work_interval_data_test(
378*5e3eaea3SApple OSS Distributions DEFAULT_ITERS,
379*5e3eaea3SApple OSS Distributions DEFAULT_INTERVAL_NS,
380*5e3eaea3SApple OSS Distributions 20, // threads
381*5e3eaea3SApple OSS Distributions true, // enable_telemetry
382*5e3eaea3SApple OSS Distributions 0); // no added flags
383*5e3eaea3SApple OSS Distributions }
384*5e3eaea3SApple OSS Distributions
385*5e3eaea3SApple OSS Distributions T_DECL(work_interval_rt_coreaudio_telemetry_supported_with_other_flags, "telemetry supported when the other creation flags used by coreaudio are set",
386*5e3eaea3SApple OSS Distributions T_META_ASROOT(YES), XNU_T_META_SOC_SPECIFIC, T_META_ENABLED(TARGET_CPU_ARM64))
387*5e3eaea3SApple OSS Distributions {
388*5e3eaea3SApple OSS Distributions T_LOG("Coreaudio work interval with auto-join and deferred finish enabled");
389*5e3eaea3SApple OSS Distributions run_work_interval_data_test(
390*5e3eaea3SApple OSS Distributions DEFAULT_ITERS,
391*5e3eaea3SApple OSS Distributions DEFAULT_INTERVAL_NS,
392*5e3eaea3SApple OSS Distributions DEFAULT_THREAD_COUNT, // threads
393*5e3eaea3SApple OSS Distributions true, // enable_telemetry
394*5e3eaea3SApple OSS Distributions WORK_INTERVAL_FLAG_ENABLE_AUTO_JOIN | WORK_INTERVAL_FLAG_ENABLE_DEFERRED_FINISH);
395*5e3eaea3SApple OSS Distributions
396*5e3eaea3SApple OSS Distributions T_LOG("Coreaudio work interval with auto-join, deferred finish, and unrestricted flags enabled");
397*5e3eaea3SApple OSS Distributions run_work_interval_data_test(
398*5e3eaea3SApple OSS Distributions DEFAULT_ITERS,
399*5e3eaea3SApple OSS Distributions DEFAULT_INTERVAL_NS,
400*5e3eaea3SApple OSS Distributions DEFAULT_THREAD_COUNT, // threads
401*5e3eaea3SApple OSS Distributions true, // enable_telemetry
402*5e3eaea3SApple OSS Distributions WORK_INTERVAL_FLAG_ENABLE_AUTO_JOIN | WORK_INTERVAL_FLAG_ENABLE_DEFERRED_FINISH | WORK_INTERVAL_FLAG_UNRESTRICTED);
403*5e3eaea3SApple OSS Distributions }
404