xref: /xnu-8792.81.2/tests/memcmp_zero.c (revision 19c3b8c28c31cb8130e034cfb5df6bf9ba342d90)
1*19c3b8c2SApple OSS Distributions #include <darwintest.h>
2*19c3b8c2SApple OSS Distributions #include <stdio.h>
3*19c3b8c2SApple OSS Distributions #include <stdlib.h>
4*19c3b8c2SApple OSS Distributions #include <string.h>
5*19c3b8c2SApple OSS Distributions #include <sys/mman.h>
6*19c3b8c2SApple OSS Distributions #include <mach/machine/vm_param.h>
7*19c3b8c2SApple OSS Distributions 
8*19c3b8c2SApple OSS Distributions static inline unsigned char *
get_guarded_page(void)9*19c3b8c2SApple OSS Distributions get_guarded_page(void)
10*19c3b8c2SApple OSS Distributions {
11*19c3b8c2SApple OSS Distributions 	unsigned char *p = mmap(NULL, 3 * PAGE_SIZE, PROT_NONE, MAP_SHARED | MAP_ANON, 0, 0);
12*19c3b8c2SApple OSS Distributions 	p += PAGE_SIZE;
13*19c3b8c2SApple OSS Distributions 	mprotect(p, PAGE_SIZE, PROT_READ | PROT_WRITE);
14*19c3b8c2SApple OSS Distributions 	return p;
15*19c3b8c2SApple OSS Distributions }
16*19c3b8c2SApple OSS Distributions 
17*19c3b8c2SApple OSS Distributions static inline void
free_guarded_page(unsigned char * p)18*19c3b8c2SApple OSS Distributions free_guarded_page(unsigned char *p)
19*19c3b8c2SApple OSS Distributions {
20*19c3b8c2SApple OSS Distributions 	munmap(p - PAGE_SIZE, 3 * PAGE_SIZE);
21*19c3b8c2SApple OSS Distributions }
22*19c3b8c2SApple OSS Distributions 
23*19c3b8c2SApple OSS Distributions /* memcmp_zero_ptr_aligned() checks string s of n bytes contains all zeros.
24*19c3b8c2SApple OSS Distributions  * Address and size of the string s must be pointer-aligned.
25*19c3b8c2SApple OSS Distributions  * Return 0 if true, 1 otherwise. Also return 0 if n is 0.
26*19c3b8c2SApple OSS Distributions  */
27*19c3b8c2SApple OSS Distributions extern int
28*19c3b8c2SApple OSS Distributions memcmp_zero_ptr_aligned(const void *s, size_t n);
29*19c3b8c2SApple OSS Distributions 
30*19c3b8c2SApple OSS Distributions T_DECL(memcmp_zero, "memcmp_zero")
31*19c3b8c2SApple OSS Distributions {
32*19c3b8c2SApple OSS Distributions 	// the assembly version is for the kernel and doesn't support arm64_32
33*19c3b8c2SApple OSS Distributions #if defined(__arm64__) && __LP64__
34*19c3b8c2SApple OSS Distributions 	unsigned char *buffer = get_guarded_page();
35*19c3b8c2SApple OSS Distributions 	unsigned char *right = buffer + PAGE_SIZE - 512;
36*19c3b8c2SApple OSS Distributions 	const int ptr_size = sizeof(buffer);
37*19c3b8c2SApple OSS Distributions 
38*19c3b8c2SApple OSS Distributions 	for (size_t i = 0; i < 256; i += ptr_size) {
39*19c3b8c2SApple OSS Distributions 		for (size_t j = i; j < 256; ++j) {
40*19c3b8c2SApple OSS Distributions 			for (size_t k = 0; k < 256; ++k) {
41*19c3b8c2SApple OSS Distributions 				if (k < i) {
42*19c3b8c2SApple OSS Distributions 					buffer[k] = (unsigned char)rand();
43*19c3b8c2SApple OSS Distributions 				} else if (k < j) {
44*19c3b8c2SApple OSS Distributions 					buffer[k] = '\0';
45*19c3b8c2SApple OSS Distributions 				} else if (k == j) {
46*19c3b8c2SApple OSS Distributions 					do {
47*19c3b8c2SApple OSS Distributions 						buffer[k] = (unsigned char)rand();
48*19c3b8c2SApple OSS Distributions 					} while (!buffer[k]);
49*19c3b8c2SApple OSS Distributions 				} else {
50*19c3b8c2SApple OSS Distributions 					buffer[k] = '\0';
51*19c3b8c2SApple OSS Distributions 				}
52*19c3b8c2SApple OSS Distributions 			}
53*19c3b8c2SApple OSS Distributions 			for (size_t m = 0; m < 128; m += ptr_size) {
54*19c3b8c2SApple OSS Distributions 				int result = memcmp_zero_ptr_aligned(&buffer[i], m);
55*19c3b8c2SApple OSS Distributions 				int ref = j - i < m ? 1 : 0;
56*19c3b8c2SApple OSS Distributions 				T_QUIET; T_ASSERT_EQ(result, ref, "expected %d, saw %d\n"
57*19c3b8c2SApple OSS Distributions 				    "memcmp_zero_ptr_aligned(buf[%zd], %zd)\n",
58*19c3b8c2SApple OSS Distributions 				    ref, result, i, m);
59*19c3b8c2SApple OSS Distributions 			}
60*19c3b8c2SApple OSS Distributions 
61*19c3b8c2SApple OSS Distributions 
62*19c3b8c2SApple OSS Distributions 			for (size_t k = 0; k < 256; ++k) {
63*19c3b8c2SApple OSS Distributions 				if (k < i) {
64*19c3b8c2SApple OSS Distributions 					right[k] = (unsigned char)rand();
65*19c3b8c2SApple OSS Distributions 				} else if (k < j) {
66*19c3b8c2SApple OSS Distributions 					right[k] = '\0';
67*19c3b8c2SApple OSS Distributions 				} else if (k == j) {
68*19c3b8c2SApple OSS Distributions 					do {
69*19c3b8c2SApple OSS Distributions 						right[k] = (unsigned char)rand();
70*19c3b8c2SApple OSS Distributions 					} while (!right[k]);
71*19c3b8c2SApple OSS Distributions 				} else {
72*19c3b8c2SApple OSS Distributions 					right[k] = '\0';
73*19c3b8c2SApple OSS Distributions 				}
74*19c3b8c2SApple OSS Distributions 			}
75*19c3b8c2SApple OSS Distributions 			for (size_t m = 0; m < 256; m += ptr_size) {
76*19c3b8c2SApple OSS Distributions 				int result = memcmp_zero_ptr_aligned(&right[i], m);
77*19c3b8c2SApple OSS Distributions 				int ref = j - i < m ? 1 : 0;
78*19c3b8c2SApple OSS Distributions 				T_QUIET; T_ASSERT_EQ(result, ref, "expected %d, saw %d\n"
79*19c3b8c2SApple OSS Distributions 				    "memcmp_zero_ptr_aligned(buf[%zd], %zd)\n",
80*19c3b8c2SApple OSS Distributions 				    ref, result, i, m);
81*19c3b8c2SApple OSS Distributions 			}
82*19c3b8c2SApple OSS Distributions 		}
83*19c3b8c2SApple OSS Distributions 	}
84*19c3b8c2SApple OSS Distributions 
85*19c3b8c2SApple OSS Distributions 	T_PASS("success");
86*19c3b8c2SApple OSS Distributions 
87*19c3b8c2SApple OSS Distributions 	free_guarded_page(buffer);
88*19c3b8c2SApple OSS Distributions #else
89*19c3b8c2SApple OSS Distributions 	T_SKIP("no optimized version to test");
90*19c3b8c2SApple OSS Distributions #endif
91*19c3b8c2SApple OSS Distributions }
92