xref: /xnu-8792.81.2/tools/tests/perf_index/test_fault_helper.c (revision 19c3b8c28c31cb8130e034cfb5df6bf9ba342d90)
1*19c3b8c2SApple OSS Distributions #include "test_fault_helper.h"
2*19c3b8c2SApple OSS Distributions #include "fail.h"
3*19c3b8c2SApple OSS Distributions #include <sys/mman.h>
4*19c3b8c2SApple OSS Distributions #include <stdlib.h>
5*19c3b8c2SApple OSS Distributions #include <unistd.h>
6*19c3b8c2SApple OSS Distributions #include <assert.h>
7*19c3b8c2SApple OSS Distributions #include <TargetConditionals.h>
8*19c3b8c2SApple OSS Distributions 
9*19c3b8c2SApple OSS Distributions #if (TARGET_OS_IPHONE && !TARGET_OS_SIMULATOR)
10*19c3b8c2SApple OSS Distributions #define MEMSIZE (1L<<28)
11*19c3b8c2SApple OSS Distributions #else
12*19c3b8c2SApple OSS Distributions #define MEMSIZE (1L<<30)
13*19c3b8c2SApple OSS Distributions #endif
14*19c3b8c2SApple OSS Distributions 
15*19c3b8c2SApple OSS Distributions static char* memblock;
16*19c3b8c2SApple OSS Distributions 
17*19c3b8c2SApple OSS Distributions int
test_fault_setup()18*19c3b8c2SApple OSS Distributions test_fault_setup()
19*19c3b8c2SApple OSS Distributions {
20*19c3b8c2SApple OSS Distributions 	char *ptr;
21*19c3b8c2SApple OSS Distributions 	int pgsz = getpagesize();
22*19c3b8c2SApple OSS Distributions 	int retval;
23*19c3b8c2SApple OSS Distributions 
24*19c3b8c2SApple OSS Distributions 	memblock = (char *)mmap(NULL, MEMSIZE, PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE, -1, 0);
25*19c3b8c2SApple OSS Distributions 	VERIFY(memblock != MAP_FAILED, "mmap failed");
26*19c3b8c2SApple OSS Distributions 
27*19c3b8c2SApple OSS Distributions 	/* make sure memory is paged */
28*19c3b8c2SApple OSS Distributions 	for (ptr = memblock; ptr < memblock + MEMSIZE; ptr += pgsz) {
29*19c3b8c2SApple OSS Distributions 		*ptr = 1;
30*19c3b8c2SApple OSS Distributions 	}
31*19c3b8c2SApple OSS Distributions 
32*19c3b8c2SApple OSS Distributions 	/* set to read only, then back to read write so it faults on first write */
33*19c3b8c2SApple OSS Distributions 	retval = mprotect(memblock, MEMSIZE, PROT_READ);
34*19c3b8c2SApple OSS Distributions 	VERIFY(retval == 0, "mprotect failed");
35*19c3b8c2SApple OSS Distributions 
36*19c3b8c2SApple OSS Distributions 	retval = mprotect(memblock, MEMSIZE, PROT_READ | PROT_WRITE);
37*19c3b8c2SApple OSS Distributions 	VERIFY(retval == 0, "mprotect failed");
38*19c3b8c2SApple OSS Distributions 
39*19c3b8c2SApple OSS Distributions 	return PERFINDEX_SUCCESS;
40*19c3b8c2SApple OSS Distributions }
41*19c3b8c2SApple OSS Distributions 
42*19c3b8c2SApple OSS Distributions int
test_fault_helper(int thread_id,int num_threads,long long length,testtype_t testtype)43*19c3b8c2SApple OSS Distributions test_fault_helper(int thread_id, int num_threads, long long length, testtype_t testtype)
44*19c3b8c2SApple OSS Distributions {
45*19c3b8c2SApple OSS Distributions 	char *ptr;
46*19c3b8c2SApple OSS Distributions 	int pgsz = getpagesize();
47*19c3b8c2SApple OSS Distributions 	int retval;
48*19c3b8c2SApple OSS Distributions 
49*19c3b8c2SApple OSS Distributions 	long long num_pages = MEMSIZE / pgsz;
50*19c3b8c2SApple OSS Distributions 	long long region_len = num_pages / num_threads;
51*19c3b8c2SApple OSS Distributions 	long long region_start = region_len * thread_id;
52*19c3b8c2SApple OSS Distributions 	long long region_end;
53*19c3b8c2SApple OSS Distributions 
54*19c3b8c2SApple OSS Distributions 	if (thread_id < num_pages % num_threads) {
55*19c3b8c2SApple OSS Distributions 		region_start += thread_id;
56*19c3b8c2SApple OSS Distributions 		region_len++;
57*19c3b8c2SApple OSS Distributions 	} else {
58*19c3b8c2SApple OSS Distributions 		region_start += num_pages % num_threads;
59*19c3b8c2SApple OSS Distributions 	}
60*19c3b8c2SApple OSS Distributions 
61*19c3b8c2SApple OSS Distributions 	region_start *= pgsz;
62*19c3b8c2SApple OSS Distributions 	region_len *= pgsz;
63*19c3b8c2SApple OSS Distributions 	region_end = region_start + region_len;
64*19c3b8c2SApple OSS Distributions 
65*19c3b8c2SApple OSS Distributions 	long long left = length;
66*19c3b8c2SApple OSS Distributions 
67*19c3b8c2SApple OSS Distributions 	while (1) {
68*19c3b8c2SApple OSS Distributions 		for (ptr = memblock + region_start; ptr < memblock + region_end; ptr += pgsz) {
69*19c3b8c2SApple OSS Distributions 			*ptr = 1;
70*19c3b8c2SApple OSS Distributions 			left--;
71*19c3b8c2SApple OSS Distributions 			if (left == 0) {
72*19c3b8c2SApple OSS Distributions 				break;
73*19c3b8c2SApple OSS Distributions 			}
74*19c3b8c2SApple OSS Distributions 		}
75*19c3b8c2SApple OSS Distributions 
76*19c3b8c2SApple OSS Distributions 		if (left == 0) {
77*19c3b8c2SApple OSS Distributions 			break;
78*19c3b8c2SApple OSS Distributions 		}
79*19c3b8c2SApple OSS Distributions 
80*19c3b8c2SApple OSS Distributions 		if (testtype == TESTFAULT) {
81*19c3b8c2SApple OSS Distributions 			retval = mprotect(memblock + region_start, region_len, PROT_READ) == 0;
82*19c3b8c2SApple OSS Distributions 			VERIFY(retval == 0, "mprotect failed");
83*19c3b8c2SApple OSS Distributions 			retval = mprotect(memblock + region_start, region_len, PROT_READ | PROT_WRITE) == 0;
84*19c3b8c2SApple OSS Distributions 			VERIFY(retval == 0, "mprotect failed");
85*19c3b8c2SApple OSS Distributions 		} else if (testtype == TESTZFOD) {
86*19c3b8c2SApple OSS Distributions 			retval = munmap(memblock + region_start, region_len) == 0;
87*19c3b8c2SApple OSS Distributions 			VERIFY(retval == 0, "munmap failed");
88*19c3b8c2SApple OSS Distributions 			ptr = mmap(memblock + region_start, region_len, PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE | MAP_FIXED, -1, 0);
89*19c3b8c2SApple OSS Distributions 			VERIFY(ptr != 0, "mmap failed");
90*19c3b8c2SApple OSS Distributions 		}
91*19c3b8c2SApple OSS Distributions 	}
92*19c3b8c2SApple OSS Distributions 	return PERFINDEX_SUCCESS;
93*19c3b8c2SApple OSS Distributions }
94