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