1*8d741a5dSApple OSS Distributions #include <stdatomic.h>
2*8d741a5dSApple OSS Distributions #include <sys/kern_sysctl.h>
3*8d741a5dSApple OSS Distributions
4*8d741a5dSApple OSS Distributions #include <darwintest_utils.h>
5*8d741a5dSApple OSS Distributions #include <darwintest.h>
6*8d741a5dSApple OSS Distributions
7*8d741a5dSApple OSS Distributions #include "counter/common.h"
8*8d741a5dSApple OSS Distributions #include "test_utils.h"
9*8d741a5dSApple OSS Distributions
10*8d741a5dSApple OSS Distributions static unsigned int ncpu(void);
11*8d741a5dSApple OSS Distributions
12*8d741a5dSApple OSS Distributions static uint64_t
sysctl_read(const char * name)13*8d741a5dSApple OSS Distributions sysctl_read(const char *name)
14*8d741a5dSApple OSS Distributions {
15*8d741a5dSApple OSS Distributions int result;
16*8d741a5dSApple OSS Distributions uint64_t value;
17*8d741a5dSApple OSS Distributions size_t size = sizeof(value);
18*8d741a5dSApple OSS Distributions result = sysctlbyname(name, &value, &size, NULL, 0);
19*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(result, "Read from %s", name);
20*8d741a5dSApple OSS Distributions return value;
21*8d741a5dSApple OSS Distributions }
22*8d741a5dSApple OSS Distributions
23*8d741a5dSApple OSS Distributions static void
sysctl_write(const char * name,int64_t amount)24*8d741a5dSApple OSS Distributions sysctl_write(const char* name, int64_t amount)
25*8d741a5dSApple OSS Distributions {
26*8d741a5dSApple OSS Distributions kern_return_t result;
27*8d741a5dSApple OSS Distributions result = sysctlbyname(name, NULL, NULL, &amount, sizeof(int64_t));
28*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(result, "Write to %s", name);
29*8d741a5dSApple OSS Distributions }
30*8d741a5dSApple OSS Distributions
31*8d741a5dSApple OSS Distributions static void
scalable_counter_add(int64_t amount)32*8d741a5dSApple OSS Distributions scalable_counter_add(int64_t amount)
33*8d741a5dSApple OSS Distributions {
34*8d741a5dSApple OSS Distributions sysctl_write("kern.scalable_counter_test_add", amount);
35*8d741a5dSApple OSS Distributions }
36*8d741a5dSApple OSS Distributions
37*8d741a5dSApple OSS Distributions static void
static_scalable_counter_add(int64_t amount)38*8d741a5dSApple OSS Distributions static_scalable_counter_add(int64_t amount)
39*8d741a5dSApple OSS Distributions {
40*8d741a5dSApple OSS Distributions sysctl_write("kern.static_scalable_counter_test_add", amount);
41*8d741a5dSApple OSS Distributions }
42*8d741a5dSApple OSS Distributions
43*8d741a5dSApple OSS Distributions static int64_t
scalable_counter_load(void)44*8d741a5dSApple OSS Distributions scalable_counter_load(void)
45*8d741a5dSApple OSS Distributions {
46*8d741a5dSApple OSS Distributions return (int64_t) sysctl_read("kern.scalable_counter_test_load");
47*8d741a5dSApple OSS Distributions }
48*8d741a5dSApple OSS Distributions
49*8d741a5dSApple OSS Distributions static int64_t
static_scalable_counter_load(void)50*8d741a5dSApple OSS Distributions static_scalable_counter_load(void)
51*8d741a5dSApple OSS Distributions {
52*8d741a5dSApple OSS Distributions return (int64_t) sysctl_read("kern.static_scalable_counter_test_load");
53*8d741a5dSApple OSS Distributions }
54*8d741a5dSApple OSS Distributions
55*8d741a5dSApple OSS Distributions /*
56*8d741a5dSApple OSS Distributions * A background thread that bangs on the percpu counter and then exits.
57*8d741a5dSApple OSS Distributions * @param num_iterations How many times to bang on the counter. Each iteration makes the counter
58*8d741a5dSApple OSS Distributions * bigger by 100.
59*8d741a5dSApple OSS Distributions */
60*8d741a5dSApple OSS Distributions static void*
background_scalable_counter_thread(void * num_iterations_ptr)61*8d741a5dSApple OSS Distributions background_scalable_counter_thread(void* num_iterations_ptr)
62*8d741a5dSApple OSS Distributions {
63*8d741a5dSApple OSS Distributions int64_t i, num_iterations;
64*8d741a5dSApple OSS Distributions num_iterations = (int64_t)(num_iterations_ptr);
65*8d741a5dSApple OSS Distributions for (i = 0; i < num_iterations; i++) {
66*8d741a5dSApple OSS Distributions scalable_counter_add(-25);
67*8d741a5dSApple OSS Distributions scalable_counter_add(75);
68*8d741a5dSApple OSS Distributions scalable_counter_add(-100);
69*8d741a5dSApple OSS Distributions scalable_counter_add(150);
70*8d741a5dSApple OSS Distributions }
71*8d741a5dSApple OSS Distributions atomic_thread_fence(memory_order_release);
72*8d741a5dSApple OSS Distributions return 0;
73*8d741a5dSApple OSS Distributions }
74*8d741a5dSApple OSS Distributions
75*8d741a5dSApple OSS Distributions static
76*8d741a5dSApple OSS Distributions void
darwin_test_fini_scalable_counter_test()77*8d741a5dSApple OSS Distributions darwin_test_fini_scalable_counter_test()
78*8d741a5dSApple OSS Distributions {
79*8d741a5dSApple OSS Distributions int ret = fini_scalable_counter_test();
80*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "fini_scalable_counter_test");
81*8d741a5dSApple OSS Distributions }
82*8d741a5dSApple OSS Distributions
83*8d741a5dSApple OSS Distributions static
84*8d741a5dSApple OSS Distributions void
darwin_test_setup(void)85*8d741a5dSApple OSS Distributions darwin_test_setup(void)
86*8d741a5dSApple OSS Distributions {
87*8d741a5dSApple OSS Distributions T_SETUPBEGIN;
88*8d741a5dSApple OSS Distributions int dev_kernel = is_development_kernel();
89*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(dev_kernel, "sysctlbyname kern.development");
90*8d741a5dSApple OSS Distributions if (is_development_kernel() != 1) {
91*8d741a5dSApple OSS Distributions T_SKIP("Skipping test on non development kernel.");
92*8d741a5dSApple OSS Distributions }
93*8d741a5dSApple OSS Distributions init_scalable_counter_test();
94*8d741a5dSApple OSS Distributions T_SETUPEND;
95*8d741a5dSApple OSS Distributions T_ATEND(darwin_test_fini_scalable_counter_test);
96*8d741a5dSApple OSS Distributions }
97*8d741a5dSApple OSS Distributions
98*8d741a5dSApple OSS Distributions T_DECL(test_scalable_counters_single_threaded, "Test single threaded operations on scalable_counters", T_META_ASROOT(true), T_META_TAG_VM_PREFERRED)
99*8d741a5dSApple OSS Distributions {
100*8d741a5dSApple OSS Distributions static int64_t kNumIterations = 100, i, expected_value = 0;
101*8d741a5dSApple OSS Distributions darwin_test_setup();
102*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(scalable_counter_load(), 0LL, "Counter starts at zero");
103*8d741a5dSApple OSS Distributions
104*8d741a5dSApple OSS Distributions /* Simple add, subtract, and read */
105*8d741a5dSApple OSS Distributions scalable_counter_add(1);
106*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(scalable_counter_load(), 1LL, "0 + 1 == 1");
107*8d741a5dSApple OSS Distributions scalable_counter_add(-1);
108*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(scalable_counter_load(), 0LL, "1 - 1 == 0");
109*8d741a5dSApple OSS Distributions for (i = 0; i < kNumIterations; i++) {
110*8d741a5dSApple OSS Distributions scalable_counter_add(i);
111*8d741a5dSApple OSS Distributions expected_value += i;
112*8d741a5dSApple OSS Distributions }
113*8d741a5dSApple OSS Distributions for (i = 0; i < kNumIterations / 2; i++) {
114*8d741a5dSApple OSS Distributions scalable_counter_add(-i);
115*8d741a5dSApple OSS Distributions expected_value -= i;
116*8d741a5dSApple OSS Distributions }
117*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(scalable_counter_load(), expected_value, "Counter value is correct.");
118*8d741a5dSApple OSS Distributions T_END;
119*8d741a5dSApple OSS Distributions }
120*8d741a5dSApple OSS Distributions
121*8d741a5dSApple OSS Distributions T_DECL(test_static_counter, "Test staticly declared counter", T_META_ASROOT(true), T_META_TAG_VM_PREFERRED)
122*8d741a5dSApple OSS Distributions {
123*8d741a5dSApple OSS Distributions static size_t kNumIterations = 100;
124*8d741a5dSApple OSS Distributions int64_t start_value;
125*8d741a5dSApple OSS Distributions darwin_test_setup();
126*8d741a5dSApple OSS Distributions start_value = static_scalable_counter_load();
127*8d741a5dSApple OSS Distributions for (size_t i = 0; i < kNumIterations; i++) {
128*8d741a5dSApple OSS Distributions static_scalable_counter_add(1);
129*8d741a5dSApple OSS Distributions }
130*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(static_scalable_counter_load(), (long long) kNumIterations + start_value, "Counter value is correct");
131*8d741a5dSApple OSS Distributions T_END;
132*8d741a5dSApple OSS Distributions }
133*8d741a5dSApple OSS Distributions
134*8d741a5dSApple OSS Distributions T_DECL(test_scalable_counters_multithreaded, "Test multi-threaded operations on scalable_counters", T_META_ASROOT(true), T_META_TAG_VM_PREFERRED)
135*8d741a5dSApple OSS Distributions {
136*8d741a5dSApple OSS Distributions unsigned int kNumThreads = ncpu() * 5;
137*8d741a5dSApple OSS Distributions int ret;
138*8d741a5dSApple OSS Distributions int64_t i;
139*8d741a5dSApple OSS Distributions pthread_attr_t pthread_attr;
140*8d741a5dSApple OSS Distributions pthread_t *threads;
141*8d741a5dSApple OSS Distributions
142*8d741a5dSApple OSS Distributions darwin_test_setup();
143*8d741a5dSApple OSS Distributions
144*8d741a5dSApple OSS Distributions threads = malloc(sizeof(pthread_t) * kNumThreads);
145*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(threads, "Out of memory");
146*8d741a5dSApple OSS Distributions
147*8d741a5dSApple OSS Distributions ret = pthread_attr_init(&pthread_attr);
148*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_attr_init");
149*8d741a5dSApple OSS Distributions
150*8d741a5dSApple OSS Distributions int64_t expected_value = 0;
151*8d741a5dSApple OSS Distributions for (i = 0; i < kNumThreads; i++) {
152*8d741a5dSApple OSS Distributions ret = pthread_create(&threads[i], &pthread_attr, background_scalable_counter_thread, (void*)(i));
153*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_create");
154*8d741a5dSApple OSS Distributions expected_value += 100 * i;
155*8d741a5dSApple OSS Distributions }
156*8d741a5dSApple OSS Distributions
157*8d741a5dSApple OSS Distributions for (i = 0; i < kNumThreads; i++) {
158*8d741a5dSApple OSS Distributions void *exit_code;
159*8d741a5dSApple OSS Distributions ret = pthread_join(threads[i], &exit_code);
160*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_join");
161*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_EQ((ptrdiff_t) exit_code, (ptrdiff_t) 0, "Background thread exited sucessfully.");
162*8d741a5dSApple OSS Distributions }
163*8d741a5dSApple OSS Distributions atomic_thread_fence(memory_order_acquire);
164*8d741a5dSApple OSS Distributions
165*8d741a5dSApple OSS Distributions T_QUIET; T_EXPECT_EQ(scalable_counter_load(), expected_value, "Counter value is correct.");
166*8d741a5dSApple OSS Distributions
167*8d741a5dSApple OSS Distributions ret = pthread_attr_destroy(&pthread_attr);
168*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "pthread_attr_destroy");
169*8d741a5dSApple OSS Distributions free(threads);
170*8d741a5dSApple OSS Distributions }
171*8d741a5dSApple OSS Distributions
172*8d741a5dSApple OSS Distributions static unsigned int
ncpu()173*8d741a5dSApple OSS Distributions ncpu()
174*8d741a5dSApple OSS Distributions {
175*8d741a5dSApple OSS Distributions kern_return_t result;
176*8d741a5dSApple OSS Distributions int ncpu;
177*8d741a5dSApple OSS Distributions size_t size = sizeof(ncpu);
178*8d741a5dSApple OSS Distributions result = sysctlbyname("hw.ncpu", &ncpu, &size, NULL, 0);
179*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(result, "hw.npu");
180*8d741a5dSApple OSS Distributions return (unsigned int) ncpu;
181*8d741a5dSApple OSS Distributions }
182