1*4f1223e8SApple OSS Distributions #include <unistd.h>
2*4f1223e8SApple OSS Distributions #include <stdlib.h>
3*4f1223e8SApple OSS Distributions #include <sys/mman.h>
4*4f1223e8SApple OSS Distributions #include <sys/sysctl.h>
5*4f1223e8SApple OSS Distributions #include <mach/mach.h>
6*4f1223e8SApple OSS Distributions #include <mach/vm_map.h>
7*4f1223e8SApple OSS Distributions #include <darwintest.h>
8*4f1223e8SApple OSS Distributions #include <TargetConditionals.h>
9*4f1223e8SApple OSS Distributions #include <perfcheck_keys.h>
10*4f1223e8SApple OSS Distributions
11*4f1223e8SApple OSS Distributions #include "benchmark/helpers.h"
12*4f1223e8SApple OSS Distributions #include "test_utils.h"
13*4f1223e8SApple OSS Distributions
14*4f1223e8SApple OSS Distributions T_GLOBAL_META(
15*4f1223e8SApple OSS Distributions T_META_NAMESPACE("xnu.vm.perf"),
16*4f1223e8SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
17*4f1223e8SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("VM"),
18*4f1223e8SApple OSS Distributions T_META_CHECK_LEAKS(false),
19*4f1223e8SApple OSS Distributions T_META_TAG_PERF
20*4f1223e8SApple OSS Distributions );
21*4f1223e8SApple OSS Distributions
22*4f1223e8SApple OSS Distributions #ifdef DT_IOSMARK
23*4f1223e8SApple OSS Distributions #define MEMSIZE (1UL<<29) /* 512 MB */
24*4f1223e8SApple OSS Distributions #else
25*4f1223e8SApple OSS Distributions #define MEMSIZE (1UL<<27) /* 128 MB */
26*4f1223e8SApple OSS Distributions #endif
27*4f1223e8SApple OSS Distributions
28*4f1223e8SApple OSS Distributions #define VM_TAG1 100
29*4f1223e8SApple OSS Distributions #define VM_TAG2 101
30*4f1223e8SApple OSS Distributions
31*4f1223e8SApple OSS Distributions enum {
32*4f1223e8SApple OSS Distributions SOFT_FAULT,
33*4f1223e8SApple OSS Distributions ZERO_FILL,
34*4f1223e8SApple OSS Distributions NUM_FAULT_TYPES
35*4f1223e8SApple OSS Distributions };
36*4f1223e8SApple OSS Distributions
37*4f1223e8SApple OSS Distributions enum {
38*4f1223e8SApple OSS Distributions VARIANT_DEFAULT = 1,
39*4f1223e8SApple OSS Distributions VARIANT_SINGLE_REGION,
40*4f1223e8SApple OSS Distributions VARIANT_MULTIPLE_REGIONS,
41*4f1223e8SApple OSS Distributions NUM_MAPPING_VARIANTS
42*4f1223e8SApple OSS Distributions };
43*4f1223e8SApple OSS Distributions
44*4f1223e8SApple OSS Distributions static char *variant_str[] = {
45*4f1223e8SApple OSS Distributions "none",
46*4f1223e8SApple OSS Distributions "default",
47*4f1223e8SApple OSS Distributions "single-region",
48*4f1223e8SApple OSS Distributions "multiple-regions"
49*4f1223e8SApple OSS Distributions };
50*4f1223e8SApple OSS Distributions
51*4f1223e8SApple OSS Distributions
52*4f1223e8SApple OSS Distributions typedef struct {
53*4f1223e8SApple OSS Distributions char *region_addr;
54*4f1223e8SApple OSS Distributions char *shared_region_addr;
55*4f1223e8SApple OSS Distributions size_t region_len;
56*4f1223e8SApple OSS Distributions } memregion_config;
57*4f1223e8SApple OSS Distributions
58*4f1223e8SApple OSS Distributions static memregion_config *memregion_config_per_thread;
59*4f1223e8SApple OSS Distributions
60*4f1223e8SApple OSS Distributions static size_t pgsize;
61*4f1223e8SApple OSS Distributions static int num_threads;
62*4f1223e8SApple OSS Distributions static int ready_thread_count;
63*4f1223e8SApple OSS Distributions static int finished_thread_count;
64*4f1223e8SApple OSS Distributions static dt_stat_time_t runtime;
65*4f1223e8SApple OSS Distributions static pthread_cond_t start_cvar;
66*4f1223e8SApple OSS Distributions static pthread_cond_t threads_ready_cvar;
67*4f1223e8SApple OSS Distributions static pthread_cond_t threads_finished_cvar;
68*4f1223e8SApple OSS Distributions static pthread_mutex_t ready_thread_count_lock;
69*4f1223e8SApple OSS Distributions static pthread_mutex_t finished_thread_count_lock;
70*4f1223e8SApple OSS Distributions
71*4f1223e8SApple OSS Distributions static void map_mem_regions_default(int fault_type, size_t memsize);
72*4f1223e8SApple OSS Distributions static void map_mem_regions_single(int fault_type, size_t memsize);
73*4f1223e8SApple OSS Distributions static void map_mem_regions_multiple(int fault_type, size_t memsize);
74*4f1223e8SApple OSS Distributions static void map_mem_regions(int fault_type, int mapping_variant, size_t memsize);
75*4f1223e8SApple OSS Distributions static void unmap_mem_regions(int mapping_variant, size_t memsize);
76*4f1223e8SApple OSS Distributions static void setup_per_thread_regions(char *memblock, char *memblock_share, int fault_type, size_t memsize);
77*4f1223e8SApple OSS Distributions static void fault_pages(int thread_id);
78*4f1223e8SApple OSS Distributions static void execute_threads(void);
79*4f1223e8SApple OSS Distributions static void *thread_setup(void *arg);
80*4f1223e8SApple OSS Distributions static void run_test(int fault_type, int mapping_variant, size_t memsize);
81*4f1223e8SApple OSS Distributions static void setup_and_run_test(int test, int threads);
82*4f1223e8SApple OSS Distributions
83*4f1223e8SApple OSS Distributions /* Allocates memory using the default mmap behavior. Each VM region created is capped at 128 MB. */
84*4f1223e8SApple OSS Distributions static void
map_mem_regions_default(int fault_type,size_t memsize)85*4f1223e8SApple OSS Distributions map_mem_regions_default(int fault_type, size_t memsize)
86*4f1223e8SApple OSS Distributions {
87*4f1223e8SApple OSS Distributions volatile char val;
88*4f1223e8SApple OSS Distributions vm_prot_t curprot, maxprot;
89*4f1223e8SApple OSS Distributions char *ptr, *memblock, *memblock_share = NULL;
90*4f1223e8SApple OSS Distributions
91*4f1223e8SApple OSS Distributions memblock = (char *)mmap(NULL, memsize, PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE, -1, 0);
92*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_NE((void *)memblock, MAP_FAILED, "mmap");
93*4f1223e8SApple OSS Distributions
94*4f1223e8SApple OSS Distributions if (fault_type == SOFT_FAULT) {
95*4f1223e8SApple OSS Distributions /* Fault in all the pages of the original region. */
96*4f1223e8SApple OSS Distributions for (ptr = memblock; ptr < memblock + memsize; ptr += pgsize) {
97*4f1223e8SApple OSS Distributions val = *ptr;
98*4f1223e8SApple OSS Distributions }
99*4f1223e8SApple OSS Distributions /* Remap the region so that subsequent accesses result in read soft faults. */
100*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_remap(mach_task_self(), (vm_address_t *)&memblock_share,
101*4f1223e8SApple OSS Distributions memsize, 0, VM_FLAGS_ANYWHERE, mach_task_self(), (vm_address_t)memblock, FALSE,
102*4f1223e8SApple OSS Distributions &curprot, &maxprot, VM_INHERIT_DEFAULT), "vm_remap");
103*4f1223e8SApple OSS Distributions }
104*4f1223e8SApple OSS Distributions setup_per_thread_regions(memblock, memblock_share, fault_type, memsize);
105*4f1223e8SApple OSS Distributions }
106*4f1223e8SApple OSS Distributions
107*4f1223e8SApple OSS Distributions /* Creates a single VM region by mapping in a named memory entry. */
108*4f1223e8SApple OSS Distributions static void
map_mem_regions_single(int fault_type,size_t memsize)109*4f1223e8SApple OSS Distributions map_mem_regions_single(int fault_type, size_t memsize)
110*4f1223e8SApple OSS Distributions {
111*4f1223e8SApple OSS Distributions volatile char val;
112*4f1223e8SApple OSS Distributions vm_prot_t curprot, maxprot;
113*4f1223e8SApple OSS Distributions char *ptr, *memblock = NULL, *memblock_share = NULL;
114*4f1223e8SApple OSS Distributions vm_size_t size = memsize;
115*4f1223e8SApple OSS Distributions vm_offset_t addr1 = 0;
116*4f1223e8SApple OSS Distributions mach_port_t mem_handle = MACH_PORT_NULL;
117*4f1223e8SApple OSS Distributions
118*4f1223e8SApple OSS Distributions /* Allocate a region and fault in all the pages. */
119*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_allocate(mach_task_self(), &addr1, size, VM_FLAGS_ANYWHERE), "vm_allocate");
120*4f1223e8SApple OSS Distributions for (ptr = (char *)addr1; ptr < (char *)addr1 + memsize; ptr += pgsize) {
121*4f1223e8SApple OSS Distributions val = *ptr;
122*4f1223e8SApple OSS Distributions }
123*4f1223e8SApple OSS Distributions
124*4f1223e8SApple OSS Distributions /* Create a named memory entry from the region allocated above, and de-allocate said region. */
125*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(mach_make_memory_entry(mach_task_self(), &size, addr1, VM_PROT_ALL | MAP_MEM_NAMED_CREATE,
126*4f1223e8SApple OSS Distributions &mem_handle, MACH_PORT_NULL), "mach_make_memory_entry");
127*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_deallocate(mach_task_self(), addr1, size), "vm_deallocate");
128*4f1223e8SApple OSS Distributions
129*4f1223e8SApple OSS Distributions /* Map in the named entry and deallocate it. */
130*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_map(mach_task_self(), (vm_address_t *)&memblock, size, 0, VM_FLAGS_ANYWHERE, mem_handle, 0,
131*4f1223e8SApple OSS Distributions FALSE, VM_PROT_DEFAULT, VM_PROT_ALL, VM_INHERIT_NONE), "vm_map");
132*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(mach_port_deallocate(mach_task_self(), mem_handle), "mach_port_deallocate");
133*4f1223e8SApple OSS Distributions
134*4f1223e8SApple OSS Distributions if (fault_type == SOFT_FAULT) {
135*4f1223e8SApple OSS Distributions /* Fault in all the pages of the original region. */
136*4f1223e8SApple OSS Distributions for (ptr = memblock; ptr < memblock + memsize; ptr += pgsize) {
137*4f1223e8SApple OSS Distributions val = *ptr;
138*4f1223e8SApple OSS Distributions }
139*4f1223e8SApple OSS Distributions /* Remap the region so that subsequent accesses result in read soft faults. */
140*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_remap(mach_task_self(), (vm_address_t *)&memblock_share,
141*4f1223e8SApple OSS Distributions memsize, 0, VM_FLAGS_ANYWHERE, mach_task_self(), (vm_address_t)memblock, FALSE,
142*4f1223e8SApple OSS Distributions &curprot, &maxprot, VM_INHERIT_DEFAULT), "vm_remap");
143*4f1223e8SApple OSS Distributions }
144*4f1223e8SApple OSS Distributions setup_per_thread_regions(memblock, memblock_share, fault_type, memsize);
145*4f1223e8SApple OSS Distributions }
146*4f1223e8SApple OSS Distributions
147*4f1223e8SApple OSS Distributions /* Allocates a separate VM region for each thread. */
148*4f1223e8SApple OSS Distributions static void
map_mem_regions_multiple(int fault_type,size_t memsize)149*4f1223e8SApple OSS Distributions map_mem_regions_multiple(int fault_type, size_t memsize)
150*4f1223e8SApple OSS Distributions {
151*4f1223e8SApple OSS Distributions int i;
152*4f1223e8SApple OSS Distributions size_t region_len, num_pages;
153*4f1223e8SApple OSS Distributions volatile char val;
154*4f1223e8SApple OSS Distributions char *ptr, *memblock, *memblock_share;
155*4f1223e8SApple OSS Distributions vm_prot_t curprot, maxprot;
156*4f1223e8SApple OSS Distributions
157*4f1223e8SApple OSS Distributions num_pages = memsize / pgsize;
158*4f1223e8SApple OSS Distributions
159*4f1223e8SApple OSS Distributions for (i = 0; i < num_threads; i++) {
160*4f1223e8SApple OSS Distributions memblock = NULL;
161*4f1223e8SApple OSS Distributions
162*4f1223e8SApple OSS Distributions region_len = num_pages / (size_t)num_threads;
163*4f1223e8SApple OSS Distributions if ((size_t)i < num_pages % (size_t)num_threads) {
164*4f1223e8SApple OSS Distributions region_len++;
165*4f1223e8SApple OSS Distributions }
166*4f1223e8SApple OSS Distributions region_len *= pgsize;
167*4f1223e8SApple OSS Distributions
168*4f1223e8SApple OSS Distributions int fd = VM_MAKE_TAG((i % 2)? VM_TAG1 : VM_TAG2);
169*4f1223e8SApple OSS Distributions memblock = (char *)mmap(NULL, region_len, PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE, fd, 0);
170*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_NE((void *)memblock, MAP_FAILED, "mmap");
171*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_addr = memblock;
172*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].shared_region_addr = 0;
173*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_len = region_len;
174*4f1223e8SApple OSS Distributions
175*4f1223e8SApple OSS Distributions if (fault_type == SOFT_FAULT) {
176*4f1223e8SApple OSS Distributions /* Fault in all the pages of the original region. */
177*4f1223e8SApple OSS Distributions for (ptr = memblock; ptr < memblock + region_len; ptr += pgsize) {
178*4f1223e8SApple OSS Distributions val = *ptr;
179*4f1223e8SApple OSS Distributions }
180*4f1223e8SApple OSS Distributions memblock_share = NULL;
181*4f1223e8SApple OSS Distributions /* Remap the region so that subsequent accesses result in read soft faults. */
182*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(vm_remap(mach_task_self(), (vm_address_t *)&memblock_share,
183*4f1223e8SApple OSS Distributions region_len, 0, VM_FLAGS_ANYWHERE, mach_task_self(), (vm_address_t)memblock, FALSE,
184*4f1223e8SApple OSS Distributions &curprot, &maxprot, VM_INHERIT_DEFAULT), "vm_remap");
185*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].shared_region_addr = memblock_share;
186*4f1223e8SApple OSS Distributions }
187*4f1223e8SApple OSS Distributions }
188*4f1223e8SApple OSS Distributions }
189*4f1223e8SApple OSS Distributions
190*4f1223e8SApple OSS Distributions static void
map_mem_regions(int fault_type,int mapping_variant,size_t memsize)191*4f1223e8SApple OSS Distributions map_mem_regions(int fault_type, int mapping_variant, size_t memsize)
192*4f1223e8SApple OSS Distributions {
193*4f1223e8SApple OSS Distributions memregion_config_per_thread = (memregion_config *)malloc(sizeof(*memregion_config_per_thread) * (size_t)num_threads);
194*4f1223e8SApple OSS Distributions switch (mapping_variant) {
195*4f1223e8SApple OSS Distributions case VARIANT_SINGLE_REGION:
196*4f1223e8SApple OSS Distributions map_mem_regions_single(fault_type, memsize);
197*4f1223e8SApple OSS Distributions break;
198*4f1223e8SApple OSS Distributions case VARIANT_MULTIPLE_REGIONS:
199*4f1223e8SApple OSS Distributions map_mem_regions_multiple(fault_type, memsize);
200*4f1223e8SApple OSS Distributions break;
201*4f1223e8SApple OSS Distributions case VARIANT_DEFAULT:
202*4f1223e8SApple OSS Distributions default:
203*4f1223e8SApple OSS Distributions map_mem_regions_default(fault_type, memsize);
204*4f1223e8SApple OSS Distributions }
205*4f1223e8SApple OSS Distributions }
206*4f1223e8SApple OSS Distributions
207*4f1223e8SApple OSS Distributions static void
setup_per_thread_regions(char * memblock,char * memblock_share,int fault_type,size_t memsize)208*4f1223e8SApple OSS Distributions setup_per_thread_regions(char *memblock, char *memblock_share, int fault_type, size_t memsize)
209*4f1223e8SApple OSS Distributions {
210*4f1223e8SApple OSS Distributions int i;
211*4f1223e8SApple OSS Distributions size_t region_len, region_start, num_pages;
212*4f1223e8SApple OSS Distributions
213*4f1223e8SApple OSS Distributions num_pages = memsize / pgsize;
214*4f1223e8SApple OSS Distributions for (i = 0; i < num_threads; i++) {
215*4f1223e8SApple OSS Distributions region_len = num_pages / (size_t)num_threads;
216*4f1223e8SApple OSS Distributions region_start = region_len * (size_t)i;
217*4f1223e8SApple OSS Distributions
218*4f1223e8SApple OSS Distributions if ((size_t)i < num_pages % (size_t)num_threads) {
219*4f1223e8SApple OSS Distributions region_start += (size_t)i;
220*4f1223e8SApple OSS Distributions region_len++;
221*4f1223e8SApple OSS Distributions } else {
222*4f1223e8SApple OSS Distributions region_start += num_pages % (size_t)num_threads;
223*4f1223e8SApple OSS Distributions }
224*4f1223e8SApple OSS Distributions
225*4f1223e8SApple OSS Distributions region_start *= pgsize;
226*4f1223e8SApple OSS Distributions region_len *= pgsize;
227*4f1223e8SApple OSS Distributions
228*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_addr = memblock + region_start;
229*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].shared_region_addr = ((fault_type == SOFT_FAULT) ?
230*4f1223e8SApple OSS Distributions memblock_share + region_start : 0);
231*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_len = region_len;
232*4f1223e8SApple OSS Distributions }
233*4f1223e8SApple OSS Distributions }
234*4f1223e8SApple OSS Distributions
235*4f1223e8SApple OSS Distributions static void
unmap_mem_regions(int mapping_variant,size_t memsize)236*4f1223e8SApple OSS Distributions unmap_mem_regions(int mapping_variant, size_t memsize)
237*4f1223e8SApple OSS Distributions {
238*4f1223e8SApple OSS Distributions if (mapping_variant == VARIANT_MULTIPLE_REGIONS) {
239*4f1223e8SApple OSS Distributions int i;
240*4f1223e8SApple OSS Distributions for (i = 0; i < num_threads; i++) {
241*4f1223e8SApple OSS Distributions if (memregion_config_per_thread[i].shared_region_addr != 0) {
242*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(munmap(memregion_config_per_thread[i].shared_region_addr,
243*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_len), "munmap");
244*4f1223e8SApple OSS Distributions }
245*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(munmap(memregion_config_per_thread[i].region_addr,
246*4f1223e8SApple OSS Distributions memregion_config_per_thread[i].region_len), "munmap");
247*4f1223e8SApple OSS Distributions }
248*4f1223e8SApple OSS Distributions } else {
249*4f1223e8SApple OSS Distributions if (memregion_config_per_thread[0].shared_region_addr != 0) {
250*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(munmap(memregion_config_per_thread[0].shared_region_addr, memsize), "munmap");
251*4f1223e8SApple OSS Distributions }
252*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(munmap(memregion_config_per_thread[0].region_addr, memsize), "munmap");
253*4f1223e8SApple OSS Distributions }
254*4f1223e8SApple OSS Distributions }
255*4f1223e8SApple OSS Distributions
256*4f1223e8SApple OSS Distributions static void
fault_pages(int thread_id)257*4f1223e8SApple OSS Distributions fault_pages(int thread_id)
258*4f1223e8SApple OSS Distributions {
259*4f1223e8SApple OSS Distributions char *ptr, *block;
260*4f1223e8SApple OSS Distributions volatile char val;
261*4f1223e8SApple OSS Distributions
262*4f1223e8SApple OSS Distributions block = memregion_config_per_thread[thread_id].shared_region_addr ?
263*4f1223e8SApple OSS Distributions memregion_config_per_thread[thread_id].shared_region_addr :
264*4f1223e8SApple OSS Distributions memregion_config_per_thread[thread_id].region_addr;
265*4f1223e8SApple OSS Distributions for (ptr = block; ptr < block + memregion_config_per_thread[thread_id].region_len; ptr += pgsize) {
266*4f1223e8SApple OSS Distributions val = *ptr;
267*4f1223e8SApple OSS Distributions }
268*4f1223e8SApple OSS Distributions }
269*4f1223e8SApple OSS Distributions
270*4f1223e8SApple OSS Distributions static void *
thread_setup(void * arg)271*4f1223e8SApple OSS Distributions thread_setup(void *arg)
272*4f1223e8SApple OSS Distributions {
273*4f1223e8SApple OSS Distributions int my_index = *((int *)arg);
274*4f1223e8SApple OSS Distributions
275*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_lock(&ready_thread_count_lock), "pthread_mutex_lock");
276*4f1223e8SApple OSS Distributions ready_thread_count++;
277*4f1223e8SApple OSS Distributions if (ready_thread_count == num_threads) {
278*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_signal(&threads_ready_cvar), "pthread_cond_signal");
279*4f1223e8SApple OSS Distributions }
280*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_wait(&start_cvar, &ready_thread_count_lock), "pthread_cond_wait");
281*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_unlock(&ready_thread_count_lock), "pthread_mutex_unlock");
282*4f1223e8SApple OSS Distributions
283*4f1223e8SApple OSS Distributions fault_pages(my_index);
284*4f1223e8SApple OSS Distributions
285*4f1223e8SApple OSS Distributions /* Up the finished count */
286*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_lock(&finished_thread_count_lock), "pthread_mutex_lock");
287*4f1223e8SApple OSS Distributions finished_thread_count++;
288*4f1223e8SApple OSS Distributions if (finished_thread_count == num_threads) {
289*4f1223e8SApple OSS Distributions /* All the threads are done. Wake up the main thread */
290*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_signal(&threads_finished_cvar), "pthread_cond_signal");
291*4f1223e8SApple OSS Distributions }
292*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_unlock(&finished_thread_count_lock), "pthread_mutex_unlock");
293*4f1223e8SApple OSS Distributions return NULL;
294*4f1223e8SApple OSS Distributions }
295*4f1223e8SApple OSS Distributions
296*4f1223e8SApple OSS Distributions static void
execute_threads(void)297*4f1223e8SApple OSS Distributions execute_threads(void)
298*4f1223e8SApple OSS Distributions {
299*4f1223e8SApple OSS Distributions int thread_index, thread_retval;
300*4f1223e8SApple OSS Distributions int *thread_indices;
301*4f1223e8SApple OSS Distributions void *thread_retval_ptr = &thread_retval;
302*4f1223e8SApple OSS Distributions pthread_t* threads;
303*4f1223e8SApple OSS Distributions
304*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_init(&threads_ready_cvar, NULL), "pthread_cond_init");
305*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_init(&start_cvar, NULL), "pthread_cond_init");
306*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_init(&ready_thread_count_lock, NULL), "pthread_mutex_init");
307*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_init(&threads_finished_cvar, NULL), "pthread_cond_init");
308*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_init(&finished_thread_count_lock, NULL), "pthread_mutex_init");
309*4f1223e8SApple OSS Distributions ready_thread_count = 0;
310*4f1223e8SApple OSS Distributions finished_thread_count = 0;
311*4f1223e8SApple OSS Distributions
312*4f1223e8SApple OSS Distributions threads = (pthread_t *)malloc(sizeof(*threads) * (size_t)num_threads);
313*4f1223e8SApple OSS Distributions thread_indices = (int *)malloc(sizeof(*thread_indices) * (size_t)num_threads);
314*4f1223e8SApple OSS Distributions for (thread_index = 0; thread_index < num_threads; thread_index++) {
315*4f1223e8SApple OSS Distributions thread_indices[thread_index] = thread_index;
316*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_create(&threads[thread_index], NULL,
317*4f1223e8SApple OSS Distributions thread_setup, (void *)&thread_indices[thread_index]), "pthread_create");
318*4f1223e8SApple OSS Distributions }
319*4f1223e8SApple OSS Distributions
320*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_lock(&ready_thread_count_lock), "pthread_mutex_lock");
321*4f1223e8SApple OSS Distributions while (ready_thread_count != num_threads) {
322*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_wait(&threads_ready_cvar, &ready_thread_count_lock),
323*4f1223e8SApple OSS Distributions "pthread_cond_wait");
324*4f1223e8SApple OSS Distributions }
325*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_unlock(&ready_thread_count_lock), "pthread_mutex_unlock");
326*4f1223e8SApple OSS Distributions
327*4f1223e8SApple OSS Distributions T_STAT_MEASURE(runtime) {
328*4f1223e8SApple OSS Distributions /* Ungate the threads */
329*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_broadcast(&start_cvar), "pthread_cond_broadcast");
330*4f1223e8SApple OSS Distributions /* Wait for the threads to finish */
331*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_mutex_lock(&finished_thread_count_lock), "pthread_mutex_lock");
332*4f1223e8SApple OSS Distributions while (finished_thread_count != num_threads) {
333*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_cond_wait(&threads_finished_cvar, &finished_thread_count_lock), "pthread_cond_wait");
334*4f1223e8SApple OSS Distributions }
335*4f1223e8SApple OSS Distributions };
336*4f1223e8SApple OSS Distributions
337*4f1223e8SApple OSS Distributions /* Join the threads */
338*4f1223e8SApple OSS Distributions for (thread_index = 0; thread_index < num_threads; thread_index++) {
339*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(pthread_join(threads[thread_index], &thread_retval_ptr),
340*4f1223e8SApple OSS Distributions "pthread_join");
341*4f1223e8SApple OSS Distributions }
342*4f1223e8SApple OSS Distributions
343*4f1223e8SApple OSS Distributions free(threads);
344*4f1223e8SApple OSS Distributions free(thread_indices);
345*4f1223e8SApple OSS Distributions }
346*4f1223e8SApple OSS Distributions
347*4f1223e8SApple OSS Distributions static void
run_test(int fault_type,int mapping_variant,size_t memsize)348*4f1223e8SApple OSS Distributions run_test(int fault_type, int mapping_variant, size_t memsize)
349*4f1223e8SApple OSS Distributions {
350*4f1223e8SApple OSS Distributions char metric_str[32];
351*4f1223e8SApple OSS Distributions size_t num_pages;
352*4f1223e8SApple OSS Distributions size_t sysctl_size = sizeof(pgsize);
353*4f1223e8SApple OSS Distributions int ret = sysctlbyname("vm.pagesize", &pgsize, &sysctl_size, NULL, 0);
354*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "sysctl vm.pagesize failed");
355*4f1223e8SApple OSS Distributions
356*4f1223e8SApple OSS Distributions num_pages = memsize / pgsize;
357*4f1223e8SApple OSS Distributions
358*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_LT(fault_type, NUM_FAULT_TYPES, "invalid test type");
359*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_LT(mapping_variant, NUM_MAPPING_VARIANTS, "invalid mapping variant");
360*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_GT(num_threads, 0, "num_threads <= 0");
361*4f1223e8SApple OSS Distributions T_QUIET; T_ASSERT_GT((int)num_pages / num_threads, 0, "num_pages/num_threads <= 0");
362*4f1223e8SApple OSS Distributions
363*4f1223e8SApple OSS Distributions T_LOG("No. of cpus: %d", get_ncpu());
364*4f1223e8SApple OSS Distributions T_LOG("No. of threads: %d", num_threads);
365*4f1223e8SApple OSS Distributions T_LOG("No. of pages: %ld", num_pages);
366*4f1223e8SApple OSS Distributions T_LOG("Pagesize: %ld", pgsize);
367*4f1223e8SApple OSS Distributions T_LOG("Allocation size: %ld MB", memsize / (1024 * 1024));
368*4f1223e8SApple OSS Distributions T_LOG("Mapping variant: %s", variant_str[mapping_variant]);
369*4f1223e8SApple OSS Distributions
370*4f1223e8SApple OSS Distributions snprintf(metric_str, 32, "Runtime-%s", variant_str[mapping_variant]);
371*4f1223e8SApple OSS Distributions runtime = dt_stat_time_create(metric_str);
372*4f1223e8SApple OSS Distributions
373*4f1223e8SApple OSS Distributions while (!dt_stat_stable(runtime)) {
374*4f1223e8SApple OSS Distributions map_mem_regions(fault_type, mapping_variant, memsize);
375*4f1223e8SApple OSS Distributions execute_threads();
376*4f1223e8SApple OSS Distributions unmap_mem_regions(mapping_variant, memsize);
377*4f1223e8SApple OSS Distributions }
378*4f1223e8SApple OSS Distributions
379*4f1223e8SApple OSS Distributions dt_stat_finalize(runtime);
380*4f1223e8SApple OSS Distributions T_LOG("Throughput-%s (MB/s): %lf\n\n", variant_str[mapping_variant], (double)memsize / (1024 * 1024) / dt_stat_mean((dt_stat_t)runtime));
381*4f1223e8SApple OSS Distributions }
382*4f1223e8SApple OSS Distributions
383*4f1223e8SApple OSS Distributions static void
setup_and_run_test(int fault_type,int threads)384*4f1223e8SApple OSS Distributions setup_and_run_test(int fault_type, int threads)
385*4f1223e8SApple OSS Distributions {
386*4f1223e8SApple OSS Distributions int i, mapping_variant;
387*4f1223e8SApple OSS Distributions size_t memsize;
388*4f1223e8SApple OSS Distributions char *e;
389*4f1223e8SApple OSS Distributions
390*4f1223e8SApple OSS Distributions mapping_variant = VARIANT_DEFAULT;
391*4f1223e8SApple OSS Distributions memsize = MEMSIZE;
392*4f1223e8SApple OSS Distributions num_threads = threads;
393*4f1223e8SApple OSS Distributions
394*4f1223e8SApple OSS Distributions if ((e = getenv("NTHREADS"))) {
395*4f1223e8SApple OSS Distributions if (threads == 1) {
396*4f1223e8SApple OSS Distributions T_SKIP("Custom environment variables specified. Skipping single threaded version.");
397*4f1223e8SApple OSS Distributions }
398*4f1223e8SApple OSS Distributions num_threads = (int)strtol(e, NULL, 0);
399*4f1223e8SApple OSS Distributions }
400*4f1223e8SApple OSS Distributions
401*4f1223e8SApple OSS Distributions if ((e = getenv("MEMSIZEMB"))) {
402*4f1223e8SApple OSS Distributions memsize = (size_t)strtol(e, NULL, 0) * 1024 * 1024;
403*4f1223e8SApple OSS Distributions }
404*4f1223e8SApple OSS Distributions
405*4f1223e8SApple OSS Distributions if ((e = getenv("VARIANT"))) {
406*4f1223e8SApple OSS Distributions mapping_variant = (int)strtol(e, NULL, 0);
407*4f1223e8SApple OSS Distributions run_test(fault_type, mapping_variant, memsize);
408*4f1223e8SApple OSS Distributions } else {
409*4f1223e8SApple OSS Distributions for (i = VARIANT_DEFAULT; i < NUM_MAPPING_VARIANTS; i++) {
410*4f1223e8SApple OSS Distributions run_test(fault_type, i, memsize);
411*4f1223e8SApple OSS Distributions }
412*4f1223e8SApple OSS Distributions }
413*4f1223e8SApple OSS Distributions
414*4f1223e8SApple OSS Distributions T_END;
415*4f1223e8SApple OSS Distributions }
416*4f1223e8SApple OSS Distributions
417*4f1223e8SApple OSS Distributions T_DECL(read_soft_fault,
418*4f1223e8SApple OSS Distributions "Read soft faults (single thread)", T_META_TAG_VM_NOT_ELIGIBLE)
419*4f1223e8SApple OSS Distributions {
420*4f1223e8SApple OSS Distributions setup_and_run_test(SOFT_FAULT, 1);
421*4f1223e8SApple OSS Distributions }
422*4f1223e8SApple OSS Distributions
423*4f1223e8SApple OSS Distributions T_DECL(read_soft_fault_multithreaded,
424*4f1223e8SApple OSS Distributions "Read soft faults (multi-threaded)", T_META_TAG_VM_NOT_ELIGIBLE)
425*4f1223e8SApple OSS Distributions {
426*4f1223e8SApple OSS Distributions char *e;
427*4f1223e8SApple OSS Distributions int nthreads;
428*4f1223e8SApple OSS Distributions
429*4f1223e8SApple OSS Distributions /* iOSMark passes in the no. of threads via an env. variable */
430*4f1223e8SApple OSS Distributions if ((e = getenv("DT_STAT_NTHREADS"))) {
431*4f1223e8SApple OSS Distributions nthreads = (int)strtol(e, NULL, 0);
432*4f1223e8SApple OSS Distributions } else {
433*4f1223e8SApple OSS Distributions nthreads = get_ncpu();
434*4f1223e8SApple OSS Distributions if (nthreads == 1) {
435*4f1223e8SApple OSS Distributions T_SKIP("Skipping multi-threaded test on single core device.");
436*4f1223e8SApple OSS Distributions }
437*4f1223e8SApple OSS Distributions }
438*4f1223e8SApple OSS Distributions setup_and_run_test(SOFT_FAULT, nthreads);
439*4f1223e8SApple OSS Distributions }
440*4f1223e8SApple OSS Distributions
441*4f1223e8SApple OSS Distributions T_DECL(zero_fill_fault,
442*4f1223e8SApple OSS Distributions "Zero fill faults (single thread)", T_META_TAG_VM_NOT_ELIGIBLE)
443*4f1223e8SApple OSS Distributions {
444*4f1223e8SApple OSS Distributions setup_and_run_test(ZERO_FILL, 1);
445*4f1223e8SApple OSS Distributions }
446*4f1223e8SApple OSS Distributions
447*4f1223e8SApple OSS Distributions T_DECL(zero_fill_fault_multithreaded,
448*4f1223e8SApple OSS Distributions "Zero fill faults (multi-threaded)",
449*4f1223e8SApple OSS Distributions XNU_T_META_SOC_SPECIFIC, T_META_TAG_VM_NOT_ELIGIBLE)
450*4f1223e8SApple OSS Distributions {
451*4f1223e8SApple OSS Distributions char *e;
452*4f1223e8SApple OSS Distributions int nthreads;
453*4f1223e8SApple OSS Distributions
454*4f1223e8SApple OSS Distributions /* iOSMark passes in the no. of threads via an env. variable */
455*4f1223e8SApple OSS Distributions if ((e = getenv("DT_STAT_NTHREADS"))) {
456*4f1223e8SApple OSS Distributions nthreads = (int)strtol(e, NULL, 0);
457*4f1223e8SApple OSS Distributions } else {
458*4f1223e8SApple OSS Distributions nthreads = get_ncpu();
459*4f1223e8SApple OSS Distributions if (nthreads == 1) {
460*4f1223e8SApple OSS Distributions T_SKIP("Skipping multi-threaded test on single core device.");
461*4f1223e8SApple OSS Distributions }
462*4f1223e8SApple OSS Distributions }
463*4f1223e8SApple OSS Distributions setup_and_run_test(ZERO_FILL, nthreads);
464*4f1223e8SApple OSS Distributions }
465