xref: /freebsd/tests/sys/posixshm/posixshm_test.c (revision 45dd2eaac379e5576f745380260470204c49beac)
1 /*-
2  * Copyright (c) 2006 Robert N. M. Watson
3  * All rights reserved.
4  *
5  * Copyright (c) 2021 The FreeBSD Foundation
6  *
7  * Portions of this software were developed by Ka Ho Ng
8  * under sponsorship from the FreeBSD Foundation.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29  * SUCH DAMAGE.
30  */
31 
32 #include <sys/cdefs.h>
33 __FBSDID("$FreeBSD$");
34 
35 #include <sys/param.h>
36 #include <sys/ioctl.h>
37 #include <sys/mman.h>
38 #include <sys/resource.h>
39 #include <sys/stat.h>
40 #include <sys/syscall.h>
41 #include <sys/sysctl.h>
42 #include <sys/wait.h>
43 
44 #include <ctype.h>
45 #include <errno.h>
46 #include <fcntl.h>
47 #include <signal.h>
48 #include <stdio.h>
49 #include <stdlib.h>
50 #include <string.h>
51 #include <unistd.h>
52 
53 #include <atf-c.h>
54 
55 #define	TEST_PATH_LEN	256
56 static char test_path[TEST_PATH_LEN];
57 static char test_path2[TEST_PATH_LEN];
58 static unsigned int test_path_idx = 0;
59 
60 static void
61 gen_a_test_path(char *path)
62 {
63 	snprintf(path, TEST_PATH_LEN, "/%s/tmp.XXXXXX%d",
64 	    getenv("TMPDIR") == NULL ? "/tmp" : getenv("TMPDIR"),
65 	    test_path_idx);
66 
67 	test_path_idx++;
68 
69 	ATF_REQUIRE_MSG(mkstemp(path) != -1,
70 	    "mkstemp failed; errno=%d", errno);
71 	ATF_REQUIRE_MSG(unlink(path) == 0,
72 	    "unlink failed; errno=%d", errno);
73 }
74 
75 static void
76 gen_test_path(void)
77 {
78 	gen_a_test_path(test_path);
79 }
80 
81 static void
82 gen_test_path2(void)
83 {
84 	gen_a_test_path(test_path2);
85 }
86 
87 /*
88  * Attempt a shm_open() that should fail with an expected error of 'error'.
89  */
90 static void
91 shm_open_should_fail(const char *path, int flags, mode_t mode, int error)
92 {
93 	int fd;
94 
95 	fd = shm_open(path, flags, mode);
96 	ATF_CHECK_MSG(fd == -1, "shm_open didn't fail");
97 	ATF_CHECK_MSG(error == errno,
98 	    "shm_open didn't fail with expected errno; errno=%d; expected "
99 	    "errno=%d", errno, error);
100 }
101 
102 /*
103  * Attempt a shm_unlink() that should fail with an expected error of 'error'.
104  */
105 static void
106 shm_unlink_should_fail(const char *path, int error)
107 {
108 
109 	ATF_CHECK_MSG(shm_unlink(path) == -1, "shm_unlink didn't fail");
110 	ATF_CHECK_MSG(error == errno,
111 	    "shm_unlink didn't fail with expected errno; errno=%d; expected "
112 	    "errno=%d", errno, error);
113 }
114 
115 /*
116  * Open the test object and write a value to the first byte.  Returns valid fd
117  * on success and -1 on failure.
118  */
119 static int
120 scribble_object(const char *path, char value)
121 {
122 	char *page;
123 	int fd, pagesize;
124 
125 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
126 
127 	fd = shm_open(path, O_CREAT|O_EXCL|O_RDWR, 0777);
128 	if (fd < 0 && errno == EEXIST) {
129 		if (shm_unlink(test_path) < 0)
130 			atf_tc_fail("shm_unlink");
131 		fd = shm_open(test_path, O_CREAT | O_EXCL | O_RDWR, 0777);
132 	}
133 	if (fd < 0)
134 		atf_tc_fail("shm_open failed; errno=%d", errno);
135 	if (ftruncate(fd, pagesize) < 0)
136 		atf_tc_fail("ftruncate failed; errno=%d", errno);
137 
138 	page = mmap(0, pagesize, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
139 	if (page == MAP_FAILED)
140 		atf_tc_fail("mmap failed; errno=%d", errno);
141 
142 	page[0] = value;
143 	ATF_REQUIRE_MSG(munmap(page, pagesize) == 0, "munmap failed; errno=%d",
144 	    errno);
145 
146 	return (fd);
147 }
148 
149 /*
150  * Fail the test case if the 'path' does not refer to an shm whose first byte
151  * is equal to expected_value
152  */
153 static void
154 verify_object(const char *path, char expected_value)
155 {
156 	int fd;
157 	int pagesize;
158 	char *page;
159 
160 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
161 
162 	fd = shm_open(path, O_RDONLY, 0777);
163 	if (fd < 0)
164 		atf_tc_fail("shm_open failed in verify_object; errno=%d, path=%s",
165 		    errno, path);
166 
167 	page = mmap(0, pagesize, PROT_READ, MAP_SHARED, fd, 0);
168 	if (page == MAP_FAILED)
169 		atf_tc_fail("mmap(1)");
170 	if (page[0] != expected_value)
171 		atf_tc_fail("Renamed object has incorrect value; has"
172 		    "%d (0x%x, '%c'), expected %d (0x%x, '%c')\n",
173 		    page[0], page[0], isprint(page[0]) ? page[0] : ' ',
174 		    expected_value, expected_value,
175 		    isprint(expected_value) ? expected_value : ' ');
176 	ATF_REQUIRE_MSG(munmap(page, pagesize) == 0, "munmap failed; errno=%d",
177 	    errno);
178 	close(fd);
179 }
180 
181 static off_t shm_max_pages = 32;
182 static const char byte_to_fill = 0x5f;
183 
184 static int
185 shm_fill(int fd, off_t offset, off_t len)
186 {
187 	int error;
188 	size_t blen, page_size;
189 	char *buf;
190 
191 	error = 0;
192 	page_size = getpagesize();
193 	buf = malloc(page_size);
194 	if (buf == NULL)
195 		return (1);
196 
197 	while (len > 0) {
198 		blen = len < (off_t)page_size ? len : page_size;
199 		memset(buf, byte_to_fill, blen);
200 		if (pwrite(fd, buf, blen, offset) != (ssize_t)blen) {
201 			error = 1;
202 			break;
203 		}
204 		len -= blen;
205 		offset += blen;
206 	}
207 
208 	free(buf);
209 	return (error);
210 }
211 
212 static int
213 check_content_dealloc(int fd, off_t hole_start, off_t hole_len, off_t shm_sz)
214 {
215 	int error;
216 	size_t blen, page_size;
217 	off_t offset, resid;
218 	struct stat statbuf;
219 	char *buf, *sblk;
220 
221 	error = 0;
222 	page_size = getpagesize();
223 	buf = malloc(page_size * 2);
224 	if (buf == NULL)
225 		return (1);
226 	sblk = buf + page_size;
227 
228 	memset(sblk, 0, page_size);
229 
230 	if ((uint64_t)hole_start + hole_len > (uint64_t)shm_sz)
231 		hole_len = shm_sz - hole_start;
232 
233 	/*
234 	 * Check hole is zeroed.
235 	 */
236 	offset = hole_start;
237 	resid = hole_len;
238 	while (resid > 0) {
239 		blen = resid < (off_t)page_size ? resid : page_size;
240 		if (pread(fd, buf, blen, offset) != (ssize_t)blen) {
241 			error = 1;
242 			break;
243 		}
244 		if (memcmp(buf, sblk, blen) != 0) {
245 			error = 1;
246 			break;
247 		}
248 		resid -= blen;
249 		offset += blen;
250 	}
251 
252 	memset(sblk, byte_to_fill, page_size);
253 
254 	/*
255 	 * Check file region before hole is zeroed.
256 	 */
257 	offset = 0;
258 	resid = hole_start;
259 	while (resid > 0) {
260 		blen = resid < (off_t)page_size ? resid : page_size;
261 		if (pread(fd, buf, blen, offset) != (ssize_t)blen) {
262 			error = 1;
263 			break;
264 		}
265 		if (memcmp(buf, sblk, blen) != 0) {
266 			error = 1;
267 			break;
268 		}
269 		resid -= blen;
270 		offset += blen;
271 	}
272 
273 	/*
274 	 * Check file region after hole is zeroed.
275 	 */
276 	offset = hole_start + hole_len;
277 	resid = shm_sz - offset;
278 	while (resid > 0) {
279 		blen = resid < (off_t)page_size ? resid : page_size;
280 		if (pread(fd, buf, blen, offset) != (ssize_t)blen) {
281 			error = 1;
282 			break;
283 		}
284 		if (memcmp(buf, sblk, blen) != 0) {
285 			error = 1;
286 			break;
287 		}
288 		resid -= blen;
289 		offset += blen;
290 	}
291 
292 	/*
293 	 * Check file size matches with expected file size.
294 	 */
295 	if (fstat(fd, &statbuf) == -1)
296 		error = -1;
297 	if (statbuf.st_size != shm_sz)
298 		error = -1;
299 
300 	free(buf);
301 	return (error);
302 }
303 
304 ATF_TC_WITHOUT_HEAD(remap_object);
305 ATF_TC_BODY(remap_object, tc)
306 {
307 	char *page;
308 	int fd, pagesize;
309 
310 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
311 
312 	gen_test_path();
313 	fd = scribble_object(test_path, '1');
314 
315 	page = mmap(0, pagesize, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
316 	if (page == MAP_FAILED)
317 		atf_tc_fail("mmap(2) failed; errno=%d", errno);
318 
319 	if (page[0] != '1')
320 		atf_tc_fail("missing data ('%c' != '1')", page[0]);
321 
322 	close(fd);
323 	ATF_REQUIRE_MSG(munmap(page, pagesize) == 0, "munmap failed; errno=%d",
324 	    errno);
325 
326 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1,
327 	    "shm_unlink failed; errno=%d", errno);
328 }
329 
330 ATF_TC_WITHOUT_HEAD(rename_from_anon);
331 ATF_TC_BODY(rename_from_anon, tc)
332 {
333 	int rc;
334 
335 	gen_test_path();
336 	rc = shm_rename(SHM_ANON, test_path, 0);
337 	if (rc != -1)
338 		atf_tc_fail("shm_rename from SHM_ANON succeeded unexpectedly");
339 }
340 
341 ATF_TC_WITHOUT_HEAD(rename_bad_path_pointer);
342 ATF_TC_BODY(rename_bad_path_pointer, tc)
343 {
344 	const char *bad_path;
345 	int rc;
346 
347 	bad_path = (const char *)0x1;
348 
349 	gen_test_path();
350 	rc = shm_rename(test_path, bad_path, 0);
351 	if (rc != -1)
352 		atf_tc_fail("shm_rename of nonexisting shm succeeded unexpectedly");
353 
354 	rc = shm_rename(bad_path, test_path, 0);
355 	if (rc != -1)
356 		atf_tc_fail("shm_rename of nonexisting shm succeeded unexpectedly");
357 }
358 
359 ATF_TC_WITHOUT_HEAD(rename_from_nonexisting);
360 ATF_TC_BODY(rename_from_nonexisting, tc)
361 {
362 	int rc;
363 
364 	gen_test_path();
365 	gen_test_path2();
366 	rc = shm_rename(test_path, test_path2, 0);
367 	if (rc != -1)
368 		atf_tc_fail("shm_rename of nonexisting shm succeeded unexpectedly");
369 
370 	if (errno != ENOENT)
371 		atf_tc_fail("Expected ENOENT to rename of nonexistent shm; got %d",
372 		    errno);
373 }
374 
375 ATF_TC_WITHOUT_HEAD(rename_to_anon);
376 ATF_TC_BODY(rename_to_anon, tc)
377 {
378 	int rc;
379 
380 	gen_test_path();
381 	rc = shm_rename(test_path, SHM_ANON, 0);
382 	if (rc != -1)
383 		atf_tc_fail("shm_rename to SHM_ANON succeeded unexpectedly");
384 }
385 
386 ATF_TC_WITHOUT_HEAD(rename_to_replace);
387 ATF_TC_BODY(rename_to_replace, tc)
388 {
389 	char expected_value;
390 	int fd;
391 	int fd2;
392 
393 	// Some contents we can verify later
394 	expected_value = 'g';
395 
396 	gen_test_path();
397 	fd = scribble_object(test_path, expected_value);
398 	close(fd);
399 
400 	// Give the other some different value so we can detect success
401 	gen_test_path2();
402 	fd2 = scribble_object(test_path2, 'h');
403 	close(fd2);
404 
405 	ATF_REQUIRE_MSG(shm_rename(test_path, test_path2, 0) == 0,
406 	    "shm_rename failed; errno=%d", errno);
407 
408 	// Read back renamed; verify contents
409 	verify_object(test_path2, expected_value);
410 }
411 
412 ATF_TC_WITHOUT_HEAD(rename_to_noreplace);
413 ATF_TC_BODY(rename_to_noreplace, tc)
414 {
415 	char expected_value_from;
416 	char expected_value_to;
417 	int fd_from;
418 	int fd_to;
419 	int rc;
420 
421 	// Some contents we can verify later
422 	expected_value_from = 'g';
423 	gen_test_path();
424 	fd_from = scribble_object(test_path, expected_value_from);
425 	close(fd_from);
426 
427 	// Give the other some different value so we can detect success
428 	expected_value_to = 'h';
429 	gen_test_path2();
430 	fd_to = scribble_object(test_path2, expected_value_to);
431 	close(fd_to);
432 
433 	rc = shm_rename(test_path, test_path2, SHM_RENAME_NOREPLACE);
434 	ATF_REQUIRE_MSG((rc == -1) && (errno == EEXIST),
435 	    "shm_rename didn't fail as expected; errno: %d; return: %d", errno,
436 	    rc);
437 
438 	// Read back renamed; verify contents
439 	verify_object(test_path2, expected_value_to);
440 }
441 
442 ATF_TC_WITHOUT_HEAD(rename_to_exchange);
443 ATF_TC_BODY(rename_to_exchange, tc)
444 {
445 	char expected_value_from;
446 	char expected_value_to;
447 	int fd_from;
448 	int fd_to;
449 
450 	// Some contents we can verify later
451 	expected_value_from = 'g';
452 	gen_test_path();
453 	fd_from = scribble_object(test_path, expected_value_from);
454 	close(fd_from);
455 
456 	// Give the other some different value so we can detect success
457 	expected_value_to = 'h';
458 	gen_test_path2();
459 	fd_to = scribble_object(test_path2, expected_value_to);
460 	close(fd_to);
461 
462 	ATF_REQUIRE_MSG(shm_rename(test_path, test_path2,
463 	    SHM_RENAME_EXCHANGE) == 0,
464 	    "shm_rename failed; errno=%d", errno);
465 
466 	// Read back renamed; verify contents
467 	verify_object(test_path, expected_value_to);
468 	verify_object(test_path2, expected_value_from);
469 }
470 
471 ATF_TC_WITHOUT_HEAD(rename_to_exchange_nonexisting);
472 ATF_TC_BODY(rename_to_exchange_nonexisting, tc)
473 {
474 	char expected_value_from;
475 	int fd_from;
476 
477 	// Some contents we can verify later
478 	expected_value_from = 'g';
479 	gen_test_path();
480 	fd_from = scribble_object(test_path, expected_value_from);
481 	close(fd_from);
482 
483 	gen_test_path2();
484 
485 	ATF_REQUIRE_MSG(shm_rename(test_path, test_path2,
486 	    SHM_RENAME_EXCHANGE) == 0,
487 	    "shm_rename failed; errno=%d", errno);
488 
489 	// Read back renamed; verify contents
490 	verify_object(test_path2, expected_value_from);
491 }
492 
493 ATF_TC_WITHOUT_HEAD(rename_to_self);
494 ATF_TC_BODY(rename_to_self, tc)
495 {
496 	int fd;
497 	char expected_value;
498 
499 	expected_value = 't';
500 
501 	gen_test_path();
502 	fd = scribble_object(test_path, expected_value);
503 	close(fd);
504 
505 	ATF_REQUIRE_MSG(shm_rename(test_path, test_path, 0) == 0,
506 	    "shm_rename failed; errno=%d", errno);
507 
508 	verify_object(test_path, expected_value);
509 }
510 
511 ATF_TC_WITHOUT_HEAD(rename_bad_flag);
512 ATF_TC_BODY(rename_bad_flag, tc)
513 {
514 	int fd;
515 	int rc;
516 
517 	/* Make sure we don't fail out due to ENOENT */
518 	gen_test_path();
519 	gen_test_path2();
520 	fd = scribble_object(test_path, 'd');
521 	close(fd);
522 	fd = scribble_object(test_path2, 'd');
523 	close(fd);
524 
525 	/*
526 	 * Note: if we end up with enough flags that we use all the bits,
527 	 * then remove this test completely.
528 	 */
529 	rc = shm_rename(test_path, test_path2, INT_MIN);
530 	ATF_REQUIRE_MSG((rc == -1) && (errno == EINVAL),
531 	    "shm_rename should have failed with EINVAL; got: return=%d, "
532 	    "errno=%d", rc, errno);
533 }
534 
535 ATF_TC_WITHOUT_HEAD(reopen_object);
536 ATF_TC_BODY(reopen_object, tc)
537 {
538 	char *page;
539 	int fd, pagesize;
540 
541 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
542 
543 	gen_test_path();
544 	fd = scribble_object(test_path, '1');
545 	close(fd);
546 
547 	fd = shm_open(test_path, O_RDONLY, 0777);
548 	if (fd < 0)
549 		atf_tc_fail("shm_open(2) failed; errno=%d", errno);
550 
551 	page = mmap(0, pagesize, PROT_READ, MAP_SHARED, fd, 0);
552 	if (page == MAP_FAILED)
553 		atf_tc_fail("mmap(2) failed; errno=%d", errno);
554 
555 	if (page[0] != '1')
556 		atf_tc_fail("missing data ('%c' != '1')", page[0]);
557 
558 	ATF_REQUIRE_MSG(munmap(page, pagesize) == 0, "munmap failed; errno=%d",
559 	    errno);
560 	close(fd);
561 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1,
562 	    "shm_unlink failed; errno=%d", errno);
563 }
564 
565 ATF_TC_WITHOUT_HEAD(readonly_mmap_write);
566 ATF_TC_BODY(readonly_mmap_write, tc)
567 {
568 	char *page;
569 	int fd, pagesize;
570 
571 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
572 
573 	gen_test_path();
574 
575 	fd = shm_open(test_path, O_RDONLY | O_CREAT, 0777);
576 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
577 
578 	/* PROT_WRITE should fail with EACCES. */
579 	page = mmap(0, pagesize, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
580 	if (page != MAP_FAILED)
581 		atf_tc_fail("mmap(PROT_WRITE) succeeded unexpectedly");
582 
583 	if (errno != EACCES)
584 		atf_tc_fail("mmap(PROT_WRITE) didn't fail with EACCES; "
585 		    "errno=%d", errno);
586 
587 	close(fd);
588 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1,
589 	    "shm_unlink failed; errno=%d", errno);
590 }
591 
592 ATF_TC_WITHOUT_HEAD(open_after_link);
593 ATF_TC_BODY(open_after_link, tc)
594 {
595 	int fd;
596 
597 	gen_test_path();
598 
599 	fd = shm_open(test_path, O_RDONLY | O_CREAT, 0777);
600 	ATF_REQUIRE_MSG(fd >= 0, "shm_open(1) failed; errno=%d", errno);
601 	close(fd);
602 
603 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1, "shm_unlink failed: %d",
604 	    errno);
605 
606 	shm_open_should_fail(test_path, O_RDONLY, 0777, ENOENT);
607 }
608 
609 ATF_TC_WITHOUT_HEAD(open_invalid_path);
610 ATF_TC_BODY(open_invalid_path, tc)
611 {
612 
613 	shm_open_should_fail("blah", O_RDONLY, 0777, EINVAL);
614 }
615 
616 ATF_TC_WITHOUT_HEAD(open_write_only);
617 ATF_TC_BODY(open_write_only, tc)
618 {
619 
620 	gen_test_path();
621 
622 	shm_open_should_fail(test_path, O_WRONLY, 0777, EINVAL);
623 }
624 
625 ATF_TC_WITHOUT_HEAD(open_extra_flags);
626 ATF_TC_BODY(open_extra_flags, tc)
627 {
628 
629 	gen_test_path();
630 
631 	shm_open_should_fail(test_path, O_RDONLY | O_DIRECT, 0777, EINVAL);
632 }
633 
634 ATF_TC_WITHOUT_HEAD(open_anon);
635 ATF_TC_BODY(open_anon, tc)
636 {
637 	int fd;
638 
639 	fd = shm_open(SHM_ANON, O_RDWR, 0777);
640 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
641 	close(fd);
642 }
643 
644 ATF_TC_WITHOUT_HEAD(open_anon_readonly);
645 ATF_TC_BODY(open_anon_readonly, tc)
646 {
647 
648 	shm_open_should_fail(SHM_ANON, O_RDONLY, 0777, EINVAL);
649 }
650 
651 ATF_TC_WITHOUT_HEAD(open_bad_path_pointer);
652 ATF_TC_BODY(open_bad_path_pointer, tc)
653 {
654 
655 	shm_open_should_fail((char *)1024, O_RDONLY, 0777, EFAULT);
656 }
657 
658 ATF_TC_WITHOUT_HEAD(open_path_too_long);
659 ATF_TC_BODY(open_path_too_long, tc)
660 {
661 	char *page;
662 
663 	page = malloc(MAXPATHLEN + 1);
664 	memset(page, 'a', MAXPATHLEN);
665 	page[MAXPATHLEN] = '\0';
666 	shm_open_should_fail(page, O_RDONLY, 0777, ENAMETOOLONG);
667 	free(page);
668 }
669 
670 ATF_TC_WITHOUT_HEAD(open_nonexisting_object);
671 ATF_TC_BODY(open_nonexisting_object, tc)
672 {
673 
674 	shm_open_should_fail("/notreallythere", O_RDONLY, 0777, ENOENT);
675 }
676 
677 ATF_TC_WITHOUT_HEAD(open_create_existing_object);
678 ATF_TC_BODY(open_create_existing_object, tc)
679 {
680 	int fd;
681 
682 	gen_test_path();
683 
684 	fd = shm_open(test_path, O_RDONLY|O_CREAT, 0777);
685 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
686 	close(fd);
687 
688 	shm_open_should_fail(test_path, O_RDONLY|O_CREAT|O_EXCL,
689 	    0777, EEXIST);
690 
691 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1,
692 	    "shm_unlink failed; errno=%d", errno);
693 }
694 
695 ATF_TC_WITHOUT_HEAD(trunc_resets_object);
696 ATF_TC_BODY(trunc_resets_object, tc)
697 {
698 	struct stat sb;
699 	int fd;
700 
701 	gen_test_path();
702 
703 	/* Create object and set size to 1024. */
704 	fd = shm_open(test_path, O_RDWR | O_CREAT, 0777);
705 	ATF_REQUIRE_MSG(fd >= 0, "shm_open(1) failed; errno=%d", errno);
706 	ATF_REQUIRE_MSG(ftruncate(fd, 1024) != -1,
707 	    "ftruncate failed; errno=%d", errno);
708 	ATF_REQUIRE_MSG(fstat(fd, &sb) != -1,
709 	    "fstat(1) failed; errno=%d", errno);
710 	ATF_REQUIRE_MSG(sb.st_size == 1024, "size %d != 1024", (int)sb.st_size);
711 	close(fd);
712 
713 	/* Open with O_TRUNC which should reset size to 0. */
714 	fd = shm_open(test_path, O_RDWR | O_TRUNC, 0777);
715 	ATF_REQUIRE_MSG(fd >= 0, "shm_open(2) failed; errno=%d", errno);
716 	ATF_REQUIRE_MSG(fstat(fd, &sb) != -1,
717 	    "fstat(2) failed; errno=%d", errno);
718 	ATF_REQUIRE_MSG(sb.st_size == 0,
719 	    "size was not 0 after truncation: %d", (int)sb.st_size);
720 	close(fd);
721 	ATF_REQUIRE_MSG(shm_unlink(test_path) != -1,
722 	    "shm_unlink failed; errno=%d", errno);
723 }
724 
725 ATF_TC_WITHOUT_HEAD(unlink_bad_path_pointer);
726 ATF_TC_BODY(unlink_bad_path_pointer, tc)
727 {
728 
729 	shm_unlink_should_fail((char *)1024, EFAULT);
730 }
731 
732 ATF_TC_WITHOUT_HEAD(unlink_path_too_long);
733 ATF_TC_BODY(unlink_path_too_long, tc)
734 {
735 	char *page;
736 
737 	page = malloc(MAXPATHLEN + 1);
738 	memset(page, 'a', MAXPATHLEN);
739 	page[MAXPATHLEN] = '\0';
740 	shm_unlink_should_fail(page, ENAMETOOLONG);
741 	free(page);
742 }
743 
744 ATF_TC_WITHOUT_HEAD(object_resize);
745 ATF_TC_BODY(object_resize, tc)
746 {
747 	pid_t pid;
748 	struct stat sb;
749 	char *page;
750 	int fd, pagesize, status;
751 
752 	ATF_REQUIRE(0 < (pagesize = getpagesize()));
753 
754 	/* Start off with a size of a single page. */
755 	fd = shm_open(SHM_ANON, O_CREAT|O_RDWR, 0777);
756 	if (fd < 0)
757 		atf_tc_fail("shm_open failed; errno=%d", errno);
758 
759 	if (ftruncate(fd, pagesize) < 0)
760 		atf_tc_fail("ftruncate(1) failed; errno=%d", errno);
761 
762 	if (fstat(fd, &sb) < 0)
763 		atf_tc_fail("fstat(1) failed; errno=%d", errno);
764 
765 	if (sb.st_size != pagesize)
766 		atf_tc_fail("first resize failed (%d != %d)",
767 		    (int)sb.st_size, pagesize);
768 
769 	/* Write a '1' to the first byte. */
770 	page = mmap(0, pagesize, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
771 	if (page == MAP_FAILED)
772 		atf_tc_fail("mmap(1)");
773 
774 	page[0] = '1';
775 
776 	ATF_REQUIRE_MSG(munmap(page, pagesize) == 0, "munmap failed; errno=%d",
777 	    errno);
778 
779 	/* Grow the object to 2 pages. */
780 	if (ftruncate(fd, pagesize * 2) < 0)
781 		atf_tc_fail("ftruncate(2) failed; errno=%d", errno);
782 
783 	if (fstat(fd, &sb) < 0)
784 		atf_tc_fail("fstat(2) failed; errno=%d", errno);
785 
786 	if (sb.st_size != pagesize * 2)
787 		atf_tc_fail("second resize failed (%d != %d)",
788 		    (int)sb.st_size, pagesize * 2);
789 
790 	/* Check for '1' at the first byte. */
791 	page = mmap(0, pagesize * 2, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
792 	if (page == MAP_FAILED)
793 		atf_tc_fail("mmap(2) failed; errno=%d", errno);
794 
795 	if (page[0] != '1')
796 		atf_tc_fail("'%c' != '1'", page[0]);
797 
798 	/* Write a '2' at the start of the second page. */
799 	page[pagesize] = '2';
800 
801 	/* Shrink the object back to 1 page. */
802 	if (ftruncate(fd, pagesize) < 0)
803 		atf_tc_fail("ftruncate(3) failed; errno=%d", errno);
804 
805 	if (fstat(fd, &sb) < 0)
806 		atf_tc_fail("fstat(3) failed; errno=%d", errno);
807 
808 	if (sb.st_size != pagesize)
809 		atf_tc_fail("third resize failed (%d != %d)",
810 		    (int)sb.st_size, pagesize);
811 
812 	/*
813 	 * Fork a child process to make sure the second page is no
814 	 * longer valid.
815 	 */
816 	pid = fork();
817 	if (pid == -1)
818 		atf_tc_fail("fork failed; errno=%d", errno);
819 
820 	if (pid == 0) {
821 		struct rlimit lim;
822 		char c;
823 
824 		/* Don't generate a core dump. */
825 		ATF_REQUIRE(getrlimit(RLIMIT_CORE, &lim) == 0);
826 		lim.rlim_cur = 0;
827 		ATF_REQUIRE(setrlimit(RLIMIT_CORE, &lim) == 0);
828 
829 		/*
830 		 * The previous ftruncate(2) shrunk the backing object
831 		 * so that this address is no longer valid, so reading
832 		 * from it should trigger a SIGBUS.
833 		 */
834 		c = page[pagesize];
835 		fprintf(stderr, "child: page 1: '%c'\n", c);
836 		exit(0);
837 	}
838 
839 	if (wait(&status) < 0)
840 		atf_tc_fail("wait failed; errno=%d", errno);
841 
842 	if (!WIFSIGNALED(status) || WTERMSIG(status) != SIGBUS)
843 		atf_tc_fail("child terminated with status %x", status);
844 
845 	/* Grow the object back to 2 pages. */
846 	if (ftruncate(fd, pagesize * 2) < 0)
847 		atf_tc_fail("ftruncate(2) failed; errno=%d", errno);
848 
849 	if (fstat(fd, &sb) < 0)
850 		atf_tc_fail("fstat(2) failed; errno=%d", errno);
851 
852 	if (sb.st_size != pagesize * 2)
853 		atf_tc_fail("fourth resize failed (%d != %d)",
854 		    (int)sb.st_size, pagesize);
855 
856 	/*
857 	 * Note that the mapping at 'page' for the second page is
858 	 * still valid, and now that the shm object has been grown
859 	 * back up to 2 pages, there is now memory backing this page
860 	 * so the read will work.  However, the data should be zero
861 	 * rather than '2' as the old data was thrown away when the
862 	 * object was shrunk and the new pages when an object are
863 	 * grown are zero-filled.
864 	 */
865 	if (page[pagesize] != 0)
866 		atf_tc_fail("invalid data at %d: %x != 0",
867 		    pagesize, (int)page[pagesize]);
868 
869 	close(fd);
870 }
871 
872 /* Signal handler which does nothing. */
873 static void
874 ignoreit(int sig __unused)
875 {
876 	;
877 }
878 
879 ATF_TC_WITHOUT_HEAD(shm_functionality_across_fork);
880 ATF_TC_BODY(shm_functionality_across_fork, tc)
881 {
882 	char *cp, c;
883 	int error, desc, rv;
884 	long scval;
885 	sigset_t ss;
886 	struct sigaction sa;
887 	void *region;
888 	size_t i, psize;
889 
890 #ifndef _POSIX_SHARED_MEMORY_OBJECTS
891 	printf("_POSIX_SHARED_MEMORY_OBJECTS is undefined\n");
892 #else
893 	printf("_POSIX_SHARED_MEMORY_OBJECTS is defined as %ld\n",
894 	       (long)_POSIX_SHARED_MEMORY_OBJECTS - 0);
895 	if (_POSIX_SHARED_MEMORY_OBJECTS - 0 == -1)
896 		printf("***Indicates this feature may be unsupported!\n");
897 #endif
898 	errno = 0;
899 	scval = sysconf(_SC_SHARED_MEMORY_OBJECTS);
900 	if (scval == -1 && errno != 0) {
901 		atf_tc_fail("sysconf(_SC_SHARED_MEMORY_OBJECTS) failed; "
902 		    "errno=%d", errno);
903 	} else {
904 		printf("sysconf(_SC_SHARED_MEMORY_OBJECTS) returns %ld\n",
905 		       scval);
906 		if (scval == -1)
907 			printf("***Indicates this feature is unsupported!\n");
908 	}
909 
910 	errno = 0;
911 	scval = sysconf(_SC_PAGESIZE);
912 	if (scval == -1 && errno != 0) {
913 		atf_tc_fail("sysconf(_SC_PAGESIZE) failed; errno=%d", errno);
914 	} else if (scval <= 0) {
915 		fprintf(stderr, "bogus return from sysconf(_SC_PAGESIZE): %ld",
916 		    scval);
917 		psize = 4096;
918 	} else {
919 		printf("sysconf(_SC_PAGESIZE) returns %ld\n", scval);
920 		psize = scval;
921 	}
922 
923 	gen_test_path();
924 	desc = shm_open(test_path, O_EXCL | O_CREAT | O_RDWR, 0600);
925 
926 	ATF_REQUIRE_MSG(desc >= 0, "shm_open failed; errno=%d", errno);
927 	ATF_REQUIRE_MSG(shm_unlink(test_path) == 0,
928 	    "shm_unlink failed; errno=%d", errno);
929 	ATF_REQUIRE_MSG(ftruncate(desc, (off_t)psize) != -1,
930 	    "ftruncate failed; errno=%d", errno);
931 
932 	region = mmap(NULL, psize, PROT_READ | PROT_WRITE, MAP_SHARED, desc, 0);
933 	ATF_REQUIRE_MSG(region != MAP_FAILED, "mmap failed; errno=%d", errno);
934 	memset(region, '\377', psize);
935 
936 	sa.sa_flags = 0;
937 	sa.sa_handler = ignoreit;
938 	sigemptyset(&sa.sa_mask);
939 	ATF_REQUIRE_MSG(sigaction(SIGUSR1, &sa, (struct sigaction *)0) == 0,
940 	    "sigaction failed; errno=%d", errno);
941 
942 	sigemptyset(&ss);
943 	sigaddset(&ss, SIGUSR1);
944 	ATF_REQUIRE_MSG(sigprocmask(SIG_BLOCK, &ss, (sigset_t *)0) == 0,
945 	    "sigprocmask failed; errno=%d", errno);
946 
947 	rv = fork();
948 	ATF_REQUIRE_MSG(rv != -1, "fork failed; errno=%d", errno);
949 	if (rv == 0) {
950 		sigemptyset(&ss);
951 		sigsuspend(&ss);
952 
953 		for (cp = region; cp < (char *)region + psize; cp++) {
954 			if (*cp != '\151')
955 				_exit(1);
956 		}
957 		if (lseek(desc, 0, SEEK_SET) == -1)
958 			_exit(1);
959 		for (i = 0; i < psize; i++) {
960 			error = read(desc, &c, 1);
961 			if (c != '\151')
962 				_exit(1);
963 		}
964 		_exit(0);
965 	} else {
966 		int status;
967 
968 		memset(region, '\151', psize - 2);
969 		error = pwrite(desc, region, 2, psize - 2);
970 		if (error != 2) {
971 			if (error >= 0)
972 				atf_tc_fail("short write; %d bytes written",
973 				    error);
974 			else
975 				atf_tc_fail("shmfd write");
976 		}
977 		kill(rv, SIGUSR1);
978 		waitpid(rv, &status, 0);
979 
980 		if (WIFEXITED(status) && WEXITSTATUS(status) == 0) {
981 			printf("Functionality test successful\n");
982 		} else if (WIFEXITED(status)) {
983 			atf_tc_fail("Child process exited with status %d",
984 			    WEXITSTATUS(status));
985 		} else {
986 			atf_tc_fail("Child process terminated with %s",
987 			    strsignal(WTERMSIG(status)));
988 		}
989 	}
990 
991 	ATF_REQUIRE_MSG(munmap(region, psize) == 0, "munmap failed; errno=%d",
992 	    errno);
993 	shm_unlink(test_path);
994 }
995 
996 ATF_TC_WITHOUT_HEAD(cloexec);
997 ATF_TC_BODY(cloexec, tc)
998 {
999 	int fd;
1000 
1001 	gen_test_path();
1002 
1003 	/* shm_open(2) is required to set FD_CLOEXEC */
1004 	fd = shm_open(SHM_ANON, O_RDWR, 0777);
1005 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1006 	ATF_REQUIRE((fcntl(fd, F_GETFD) & FD_CLOEXEC) != 0);
1007 	close(fd);
1008 
1009 	/* Also make sure that named shm is correct */
1010 	fd = shm_open(test_path, O_CREAT | O_RDWR, 0600);
1011 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1012 	ATF_REQUIRE((fcntl(fd, F_GETFD) & FD_CLOEXEC) != 0);
1013 	close(fd);
1014 }
1015 
1016 ATF_TC_WITHOUT_HEAD(mode);
1017 ATF_TC_BODY(mode, tc)
1018 {
1019 	struct stat st;
1020 	int fd;
1021 	mode_t restore_mask;
1022 
1023 	gen_test_path();
1024 
1025 	/* Remove inhibitions from umask */
1026 	restore_mask = umask(0);
1027 	fd = shm_open(test_path, O_CREAT | O_RDWR, 0600);
1028 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1029 	ATF_REQUIRE(fstat(fd, &st) == 0);
1030 	ATF_REQUIRE((st.st_mode & ACCESSPERMS) == 0600);
1031 	close(fd);
1032 	ATF_REQUIRE(shm_unlink(test_path) == 0);
1033 
1034 	fd = shm_open(test_path, O_CREAT | O_RDWR, 0660);
1035 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1036 	ATF_REQUIRE(fstat(fd, &st) == 0);
1037 	ATF_REQUIRE((st.st_mode & ACCESSPERMS) == 0660);
1038 	close(fd);
1039 	ATF_REQUIRE(shm_unlink(test_path) == 0);
1040 
1041 	fd = shm_open(test_path, O_CREAT | O_RDWR, 0666);
1042 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1043 	ATF_REQUIRE(fstat(fd, &st) == 0);
1044 	ATF_REQUIRE((st.st_mode & ACCESSPERMS) == 0666);
1045 	close(fd);
1046 	ATF_REQUIRE(shm_unlink(test_path) == 0);
1047 
1048 	umask(restore_mask);
1049 }
1050 
1051 ATF_TC_WITHOUT_HEAD(fallocate);
1052 ATF_TC_BODY(fallocate, tc)
1053 {
1054 	struct stat st;
1055 	int error, fd, sz;
1056 
1057 	/*
1058 	 * Primitive test case for posix_fallocate with shmd.  Effectively
1059 	 * expected to work like a smarter ftruncate that will grow the region
1060 	 * as needed in a race-free way.
1061 	 */
1062 	fd = shm_open(SHM_ANON, O_RDWR, 0666);
1063 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno=%d", errno);
1064 	/* Set the initial size. */
1065 	sz = 32;
1066 	ATF_REQUIRE(ftruncate(fd, sz) == 0);
1067 
1068 	/* Now grow it. */
1069 	error = 0;
1070 	sz *= 2;
1071 	ATF_REQUIRE_MSG((error = posix_fallocate(fd, 0, sz)) == 0,
1072 	    "posix_fallocate failed; error=%d", error);
1073 	ATF_REQUIRE(fstat(fd, &st) == 0);
1074 	ATF_REQUIRE(st.st_size == sz);
1075 	/* Attempt to shrink it; should succeed, but not change the size. */
1076 	ATF_REQUIRE_MSG((error = posix_fallocate(fd, 0, sz / 2)) == 0,
1077 	    "posix_fallocate failed; error=%d", error);
1078 	ATF_REQUIRE(fstat(fd, &st) == 0);
1079 	ATF_REQUIRE(st.st_size == sz);
1080 	/* Grow it using an offset of sz and len of sz. */
1081 	ATF_REQUIRE_MSG((error = posix_fallocate(fd, sz, sz)) == 0,
1082 	    "posix_fallocate failed; error=%d", error);
1083 	ATF_REQUIRE(fstat(fd, &st) == 0);
1084 	ATF_REQUIRE(st.st_size == (sz * 2));
1085 
1086 	close(fd);
1087 }
1088 
1089 ATF_TC_WITHOUT_HEAD(fspacectl);
1090 ATF_TC_BODY(fspacectl, tc)
1091 {
1092 	struct spacectl_range range;
1093 	off_t offset, length, shm_sz;
1094 	size_t page_size;
1095 	int fd, error;
1096 
1097 	page_size = getpagesize();
1098 	shm_sz = shm_max_pages * page_size;
1099 
1100 	fd = shm_open("/testtest", O_RDWR|O_CREAT, 0666);
1101 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; errno:%d", errno);
1102 	ATF_REQUIRE_MSG((error = posix_fallocate(fd, 0, shm_sz)) == 0,
1103 	    "posix_fallocate failed; error=%d", error);
1104 
1105 	/* Aligned fspacectl(fd, SPACECTL_DEALLOC, ...) */
1106 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1107 	range.r_offset = offset = page_size;
1108 	range.r_len = length = ((shm_max_pages - 1) * page_size) -
1109 	    range.r_offset;
1110 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1111 	    "Aligned fspacectl failed; errno=%d", errno);
1112 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1113 	    "Aligned fspacectl content checking failed");
1114 
1115 	/* Unaligned fspacectl(fd, SPACECTL_DEALLOC, ...) */
1116 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1117 	range.r_offset = offset = page_size / 2;
1118 	range.r_len = length = ((shm_max_pages - 1) * page_size) +
1119 	    (page_size / 2) - offset;
1120 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1121 	    "Unaligned fspacectl failed; errno=%d", errno);
1122 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1123 	    "Unaligned fspacectl content checking failed");
1124 
1125 	/* Aligned fspacectl(fd, SPACECTL_DEALLOC, ...) to OFF_MAX */
1126 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1127 	range.r_offset = offset = page_size;
1128 	range.r_len = length = OFF_MAX - offset;
1129 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1130 	    "Aligned fspacectl to OFF_MAX failed; errno=%d", errno);
1131 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1132 	    "Aligned fspacectl to OFF_MAX content checking failed");
1133 
1134 	/* Unaligned fspacectl(fd, SPACECTL_DEALLOC, ...) to OFF_MAX */
1135 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1136 	range.r_offset = offset = page_size / 2;
1137 	range.r_len = length = OFF_MAX - offset;
1138 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1139 	    "Unaligned fspacectl to OFF_MAX failed; errno=%d", errno);
1140 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1141 	    "Unaligned fspacectl to OFF_MAX content checking failed");
1142 
1143 	/* Aligned fspacectl(fd, SPACECTL_DEALLOC, ...) past shm_sz */
1144 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1145 	range.r_offset = offset = page_size;
1146 	range.r_len = length = ((shm_max_pages + 1) * page_size) - offset;
1147 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1148 	    "Aligned fspacectl past shm_sz failed; errno=%d", errno);
1149 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1150 	    "Aligned fspacectl past shm_sz content checking failed");
1151 
1152 	/* Unaligned fspacectl(fd, SPACECTL_DEALLOC, ...) past shm_sz */
1153 	ATF_REQUIRE(shm_fill(fd, 0, shm_sz) == 0);
1154 	range.r_offset = offset = page_size / 2;
1155 	range.r_len = length = ((shm_max_pages + 1) * page_size) - offset;
1156 	ATF_CHECK_MSG(fspacectl(fd, SPACECTL_DEALLOC, &range, 0, &range) == 0,
1157 	    "Unaligned fspacectl past shm_sz failed; errno=%d", errno);
1158 	ATF_CHECK_MSG(check_content_dealloc(fd, offset, length, shm_sz) == 0,
1159 	    "Unaligned fspacectl past shm_sz content checking failed");
1160 
1161 	ATF_REQUIRE(close(fd) == 0);
1162 }
1163 
1164 static int
1165 shm_open_large(int psind, int policy, size_t sz)
1166 {
1167 	int error, fd;
1168 
1169 	fd = shm_create_largepage(SHM_ANON, O_CREAT | O_RDWR, psind, policy, 0);
1170 	if (fd < 0 && errno == ENOTTY)
1171 		atf_tc_skip("no large page support");
1172 	ATF_REQUIRE_MSG(fd >= 0, "shm_create_largepage failed; errno=%d", errno);
1173 
1174 	error = ftruncate(fd, sz);
1175 	if (error != 0 && errno == ENOMEM)
1176 		/*
1177 		 * The test system might not have enough memory to accommodate
1178 		 * the request.
1179 		 */
1180 		atf_tc_skip("failed to allocate %zu-byte superpage", sz);
1181 	ATF_REQUIRE_MSG(error == 0, "ftruncate failed; errno=%d", errno);
1182 
1183 	return (fd);
1184 }
1185 
1186 static int
1187 pagesizes(size_t ps[MAXPAGESIZES])
1188 {
1189 	int pscnt;
1190 
1191 	pscnt = getpagesizes(ps, MAXPAGESIZES);
1192 	ATF_REQUIRE_MSG(pscnt != -1, "getpagesizes failed; errno=%d", errno);
1193 	ATF_REQUIRE_MSG(ps[0] != 0, "psind 0 is %zu", ps[0]);
1194 	ATF_REQUIRE_MSG(pscnt <= MAXPAGESIZES, "invalid pscnt %d", pscnt);
1195 	if (pscnt == 1)
1196 		atf_tc_skip("no large page support");
1197 	return (pscnt);
1198 }
1199 
1200 ATF_TC_WITHOUT_HEAD(largepage_basic);
1201 ATF_TC_BODY(largepage_basic, tc)
1202 {
1203 	char *zeroes;
1204 	char *addr, *vec;
1205 	size_t ps[MAXPAGESIZES];
1206 	int error, fd, pscnt;
1207 
1208 	pscnt = pagesizes(ps);
1209 	zeroes = calloc(1, ps[0]);
1210 	ATF_REQUIRE(zeroes != NULL);
1211 	for (int i = 1; i < pscnt; i++) {
1212 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1213 
1214 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1215 		    0);
1216 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1217 		    "mmap(%zu bytes) failed; errno=%d", ps[i], errno);
1218 		ATF_REQUIRE_MSG(((uintptr_t)addr & (ps[i] - 1)) == 0,
1219 		    "mmap(%zu bytes) returned unaligned mapping; addr=%p",
1220 		    ps[i], addr);
1221 
1222 		/* Force a page fault. */
1223 		*(volatile char *)addr = 0;
1224 
1225 		vec = malloc(ps[i] / ps[0]);
1226 		ATF_REQUIRE(vec != NULL);
1227 		error = mincore(addr, ps[i], vec);
1228 		ATF_REQUIRE_MSG(error == 0, "mincore failed; errno=%d", errno);
1229 
1230 		/* Verify that all pages in the run are mapped. */
1231 		for (size_t p = 0; p < ps[i] / ps[0]; p++) {
1232 			ATF_REQUIRE_MSG((vec[p] & MINCORE_INCORE) != 0,
1233 			    "page %zu is not mapped", p);
1234 			ATF_REQUIRE_MSG((vec[p] & MINCORE_PSIND(i)) != 0,
1235 			    "page %zu is not in a %zu-byte superpage",
1236 			    p, ps[i]);
1237 		}
1238 
1239 		/* Validate zeroing. */
1240 		for (size_t p = 0; p < ps[i] / ps[0]; p++) {
1241 			ATF_REQUIRE_MSG(memcmp(addr + p * ps[0], zeroes,
1242 			    ps[0]) == 0, "page %zu miscompare", p);
1243 		}
1244 
1245 		free(vec);
1246 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1247 		ATF_REQUIRE(close(fd) == 0);
1248 	}
1249 
1250 	free(zeroes);
1251 }
1252 
1253 extern int __sys_shm_open2(const char *, int, mode_t, int, const char *);
1254 
1255 ATF_TC_WITHOUT_HEAD(largepage_config);
1256 ATF_TC_BODY(largepage_config, tc)
1257 {
1258 	struct shm_largepage_conf lpc;
1259 	char *addr, *buf;
1260 	size_t ps[MAXPAGESIZES + 1]; /* silence warnings if MAXPAGESIZES == 1 */
1261 	int error, fd;
1262 
1263 	(void)pagesizes(ps);
1264 
1265 	fd = shm_open(SHM_ANON, O_CREAT | O_RDWR, 0);
1266 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; error=%d", errno);
1267 
1268 	/*
1269 	 * Configure a large page policy for an object created without
1270 	 * SHM_LARGEPAGE.
1271 	 */
1272 	lpc.psind = 1;
1273 	lpc.alloc_policy = SHM_LARGEPAGE_ALLOC_DEFAULT;
1274 	error = ioctl(fd, FIOSSHMLPGCNF, &lpc);
1275 	ATF_REQUIRE(error != 0);
1276 	ATF_REQUIRE_MSG(errno == ENOTTY, "ioctl(FIOSSHMLPGCNF) returned %d",
1277 	    errno);
1278 	ATF_REQUIRE(close(fd) == 0);
1279 
1280 	/*
1281 	 * Create a largepage object and try to use it without actually
1282 	 * configuring anything.
1283 	 */
1284 	fd = __sys_shm_open2(SHM_ANON, O_CREAT | O_RDWR, 0, SHM_LARGEPAGE,
1285 	    NULL);
1286 	if (fd < 0 && errno == ENOTTY)
1287 		atf_tc_skip("no large page support");
1288 	ATF_REQUIRE_MSG(fd >= 0, "shm_open2 failed; error=%d", errno);
1289 
1290 	error = ftruncate(fd, ps[1]);
1291 	ATF_REQUIRE(error != 0);
1292 	ATF_REQUIRE_MSG(errno == EINVAL, "ftruncate returned %d", errno);
1293 
1294 	addr = mmap(NULL, ps[1], PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
1295 	ATF_REQUIRE(addr == MAP_FAILED);
1296 	ATF_REQUIRE_MSG(errno == EINVAL, "mmap returned %d", errno);
1297 	addr = mmap(NULL, 0, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
1298 	ATF_REQUIRE(addr == MAP_FAILED);
1299 	ATF_REQUIRE_MSG(errno == EINVAL, "mmap returned %d", errno);
1300 
1301 	buf = calloc(1, ps[0]);
1302 	ATF_REQUIRE(buf != NULL);
1303 	ATF_REQUIRE(write(fd, buf, ps[0]) == -1);
1304 	ATF_REQUIRE_MSG(errno == EINVAL, "write returned %d", errno);
1305 	free(buf);
1306 	buf = calloc(1, ps[1]);
1307 	ATF_REQUIRE(buf != NULL);
1308 	ATF_REQUIRE(write(fd, buf, ps[1]) == -1);
1309 	ATF_REQUIRE_MSG(errno == EINVAL, "write returned %d", errno);
1310 	free(buf);
1311 
1312 	error = posix_fallocate(fd, 0, ps[0]);
1313 	ATF_REQUIRE_MSG(error == EINVAL, "posix_fallocate returned %d", error);
1314 
1315 	ATF_REQUIRE(close(fd) == 0);
1316 }
1317 
1318 ATF_TC_WITHOUT_HEAD(largepage_mmap);
1319 ATF_TC_BODY(largepage_mmap, tc)
1320 {
1321 	char *addr, *addr1, *vec;
1322 	size_t ps[MAXPAGESIZES];
1323 	int fd, pscnt;
1324 
1325 	pscnt = pagesizes(ps);
1326 	for (int i = 1; i < pscnt; i++) {
1327 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1328 
1329 		/* For mincore(). */
1330 		vec = malloc(ps[i]);
1331 		ATF_REQUIRE(vec != NULL);
1332 
1333 		/*
1334 		 * Wrong mapping size.
1335 		 */
1336 		addr = mmap(NULL, ps[i - 1], PROT_READ | PROT_WRITE, MAP_SHARED,
1337 		    fd, 0);
1338 		ATF_REQUIRE_MSG(addr == MAP_FAILED,
1339 		    "mmap(%zu bytes) succeeded", ps[i - 1]);
1340 		ATF_REQUIRE_MSG(errno == EINVAL,
1341 		    "mmap(%zu bytes) failed; error=%d", ps[i - 1], errno);
1342 
1343 		/*
1344 		 * Fixed mappings.
1345 		 */
1346 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1347 		    0);
1348 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1349 		    "mmap(%zu bytes) failed; errno=%d", ps[i], errno);
1350 		ATF_REQUIRE_MSG(((uintptr_t)addr & (ps[i] - 1)) == 0,
1351 		    "mmap(%zu bytes) returned unaligned mapping; addr=%p",
1352 		    ps[i], addr);
1353 
1354 		/* Try mapping a small page with anonymous memory. */
1355 		addr1 = mmap(addr, ps[i - 1], PROT_READ | PROT_WRITE,
1356 		    MAP_PRIVATE | MAP_ANON | MAP_FIXED, -1, 0);
1357 		ATF_REQUIRE_MSG(addr1 == MAP_FAILED,
1358 		    "anon mmap(%zu bytes) succeeded", ps[i - 1]);
1359 		ATF_REQUIRE_MSG(errno == EINVAL, "mmap returned %d", errno);
1360 
1361 		/* Check MAP_EXCL when creating a second largepage mapping. */
1362 		addr1 = mmap(addr, ps[i], PROT_READ | PROT_WRITE,
1363 		    MAP_SHARED | MAP_FIXED | MAP_EXCL, fd, 0);
1364 		ATF_REQUIRE_MSG(addr1 == MAP_FAILED,
1365 		    "mmap(%zu bytes) succeeded", ps[i]);
1366 		/* XXX wrong errno */
1367 		ATF_REQUIRE_MSG(errno == ENOSPC, "mmap returned %d", errno);
1368 
1369 		/* Overwrite a largepage mapping with a lagepage mapping. */
1370 		addr1 = mmap(addr, ps[i], PROT_READ | PROT_WRITE,
1371 		    MAP_SHARED | MAP_FIXED, fd, 0);
1372 		ATF_REQUIRE_MSG(addr1 != MAP_FAILED,
1373 		    "mmap(%zu bytes) failed; errno=%d", ps[i], errno);
1374 		ATF_REQUIRE_MSG(addr == addr1,
1375 		    "mmap(%zu bytes) moved from %p to %p", ps[i], addr, addr1);
1376 
1377 		ATF_REQUIRE(munmap(addr, ps[i] == 0));
1378 
1379 		/* Clobber an anonymous mapping with a superpage. */
1380 		addr1 = mmap(NULL, ps[0], PROT_READ | PROT_WRITE,
1381 		    MAP_ANON | MAP_PRIVATE | MAP_ALIGNED(ffsl(ps[i]) - 1), -1,
1382 		    0);
1383 		ATF_REQUIRE_MSG(addr1 != MAP_FAILED,
1384 		    "mmap failed; error=%d", errno);
1385 		*(volatile char *)addr1 = '\0';
1386 		addr = mmap(addr1, ps[i], PROT_READ | PROT_WRITE,
1387 		    MAP_SHARED | MAP_FIXED, fd, 0);
1388 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1389 		    "mmap failed; error=%d", errno);
1390 		ATF_REQUIRE_MSG(addr == addr1,
1391 		    "mmap disobeyed MAP_FIXED, %p %p", addr, addr1);
1392 		*(volatile char *)addr = 0; /* fault */
1393 		ATF_REQUIRE(mincore(addr, ps[i], vec) == 0);
1394 		for (size_t p = 0; p < ps[i] / ps[0]; p++) {
1395 			ATF_REQUIRE_MSG((vec[p] & MINCORE_INCORE) != 0,
1396 			    "page %zu is not resident", p);
1397 			ATF_REQUIRE_MSG((vec[p] & MINCORE_PSIND(i)) != 0,
1398 			    "page %zu is not resident", p);
1399 		}
1400 
1401 		/*
1402 		 * Copy-on-write mappings are not permitted.
1403 		 */
1404 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_PRIVATE,
1405 		    fd, 0);
1406 		ATF_REQUIRE_MSG(addr == MAP_FAILED,
1407 		    "mmap(%zu bytes) succeeded", ps[i]);
1408 
1409 		ATF_REQUIRE(close(fd) == 0);
1410 	}
1411 }
1412 
1413 ATF_TC_WITHOUT_HEAD(largepage_munmap);
1414 ATF_TC_BODY(largepage_munmap, tc)
1415 {
1416 	char *addr;
1417 	size_t ps[MAXPAGESIZES], ps1;
1418 	int fd, pscnt;
1419 
1420 	pscnt = pagesizes(ps);
1421 	for (int i = 1; i < pscnt; i++) {
1422 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1423 		ps1 = ps[i - 1];
1424 
1425 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1426 		    0);
1427 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1428 		    "mmap(%zu bytes) failed; errno=%d", ps[i], errno);
1429 
1430 		/* Try several unaligned munmap() requests. */
1431 		ATF_REQUIRE(munmap(addr, ps1) != 0);
1432 		ATF_REQUIRE_MSG(errno == EINVAL,
1433 		    "unexpected error %d from munmap", errno);
1434 		ATF_REQUIRE(munmap(addr, ps[i] - ps1));
1435 		ATF_REQUIRE_MSG(errno == EINVAL,
1436 		    "unexpected error %d from munmap", errno);
1437 		ATF_REQUIRE(munmap(addr + ps1, ps1) != 0);
1438 		ATF_REQUIRE_MSG(errno == EINVAL,
1439 		    "unexpected error %d from munmap", errno);
1440 		ATF_REQUIRE(munmap(addr, 0));
1441 		ATF_REQUIRE_MSG(errno == EINVAL,
1442 		    "unexpected error %d from munmap", errno);
1443 
1444 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1445 		ATF_REQUIRE(close(fd) == 0);
1446 	}
1447 }
1448 
1449 static void
1450 largepage_madvise(char *addr, size_t sz, int advice, int error)
1451 {
1452 	if (error == 0) {
1453 		ATF_REQUIRE_MSG(madvise(addr, sz, advice) == 0,
1454 		    "madvise(%zu, %d) failed; error=%d", sz, advice, errno);
1455 	} else {
1456 		ATF_REQUIRE_MSG(madvise(addr, sz, advice) != 0,
1457 		    "madvise(%zu, %d) succeeded", sz, advice);
1458 		ATF_REQUIRE_MSG(errno == error,
1459 		    "unexpected error %d from madvise(%zu, %d)",
1460 		    errno, sz, advice);
1461 	}
1462 }
1463 
1464 ATF_TC_WITHOUT_HEAD(largepage_madvise);
1465 ATF_TC_BODY(largepage_madvise, tc)
1466 {
1467 	char *addr;
1468 	size_t ps[MAXPAGESIZES];
1469 	int fd, pscnt;
1470 
1471 	pscnt = pagesizes(ps);
1472 	for (int i = 1; i < pscnt; i++) {
1473 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1474 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1475 		    0);
1476 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1477 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1478 
1479 		/* Advice that requires clipping. */
1480 		largepage_madvise(addr, ps[0], MADV_NORMAL, EINVAL);
1481 		largepage_madvise(addr, ps[i], MADV_NORMAL, 0);
1482 		largepage_madvise(addr, ps[0], MADV_RANDOM, EINVAL);
1483 		largepage_madvise(addr, ps[i], MADV_RANDOM, 0);
1484 		largepage_madvise(addr, ps[0], MADV_SEQUENTIAL, EINVAL);
1485 		largepage_madvise(addr, ps[i], MADV_SEQUENTIAL, 0);
1486 		largepage_madvise(addr, ps[0], MADV_NOSYNC, EINVAL);
1487 		largepage_madvise(addr, ps[i], MADV_NOSYNC, 0);
1488 		largepage_madvise(addr, ps[0], MADV_AUTOSYNC, EINVAL);
1489 		largepage_madvise(addr, ps[i], MADV_AUTOSYNC, 0);
1490 		largepage_madvise(addr, ps[0], MADV_CORE, EINVAL);
1491 		largepage_madvise(addr, ps[i], MADV_CORE, 0);
1492 		largepage_madvise(addr, ps[0], MADV_NOCORE, EINVAL);
1493 		largepage_madvise(addr, ps[i], MADV_NOCORE, 0);
1494 
1495 		/* Advice that does not result in clipping. */
1496 		largepage_madvise(addr, ps[0], MADV_DONTNEED, 0);
1497 		largepage_madvise(addr, ps[i], MADV_DONTNEED, 0);
1498 		largepage_madvise(addr, ps[0], MADV_WILLNEED, 0);
1499 		largepage_madvise(addr, ps[i], MADV_WILLNEED, 0);
1500 		largepage_madvise(addr, ps[0], MADV_FREE, 0);
1501 		largepage_madvise(addr, ps[i], MADV_FREE, 0);
1502 
1503 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1504 		ATF_REQUIRE(close(fd) == 0);
1505 	}
1506 }
1507 
1508 ATF_TC(largepage_mlock);
1509 ATF_TC_HEAD(largepage_mlock, tc)
1510 {
1511 	/* Needed to set rlimit. */
1512 	atf_tc_set_md_var(tc, "require.user", "root");
1513 }
1514 ATF_TC_BODY(largepage_mlock, tc)
1515 {
1516 	struct rlimit rl;
1517 	char *addr;
1518 	size_t ps[MAXPAGESIZES], sz;
1519 	u_long max_wired, wired;
1520 	int fd, error, pscnt;
1521 
1522 	rl.rlim_cur = rl.rlim_max = RLIM_INFINITY;
1523 	ATF_REQUIRE_MSG(setrlimit(RLIMIT_MEMLOCK, &rl) == 0,
1524 	    "setrlimit failed; error=%d", errno);
1525 
1526 	sz = sizeof(max_wired);
1527 	error = sysctlbyname("vm.max_user_wired", &max_wired, &sz, NULL, 0);
1528 	ATF_REQUIRE_MSG(error == 0,
1529 	    "sysctlbyname(vm.max_user_wired) failed; error=%d", errno);
1530 
1531 	sz = sizeof(wired);
1532 	error = sysctlbyname("vm.stats.vm.v_user_wire_count", &wired, &sz, NULL,
1533 	    0);
1534 	ATF_REQUIRE_MSG(error == 0,
1535 	    "sysctlbyname(vm.stats.vm.v_user_wire_count) failed; error=%d",
1536 	    errno);
1537 
1538 	pscnt = pagesizes(ps);
1539 	for (int i = 1; i < pscnt; i++) {
1540 		if (ps[i] / ps[0] > max_wired - wired) {
1541 			/* Cannot wire past the limit. */
1542 			atf_tc_skip("test would exceed wiring limit");
1543 		}
1544 
1545 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1546 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1547 		    0);
1548 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1549 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1550 
1551 		ATF_REQUIRE(mlock(addr, ps[0]) != 0);
1552 		ATF_REQUIRE_MSG(errno == EINVAL,
1553 		    "unexpected error %d from mlock(%zu bytes)", errno, ps[i]);
1554 		ATF_REQUIRE(mlock(addr, ps[i] - ps[0]) != 0);
1555 		ATF_REQUIRE_MSG(errno == EINVAL,
1556 		    "unexpected error %d from mlock(%zu bytes)", errno, ps[i]);
1557 
1558 		ATF_REQUIRE_MSG(mlock(addr, ps[i]) == 0,
1559 		    "mlock failed; error=%d", errno);
1560 
1561 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1562 
1563 		ATF_REQUIRE(mlockall(MCL_FUTURE) == 0);
1564 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1565 		    0);
1566 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1567 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1568 
1569 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1570 		ATF_REQUIRE(close(fd) == 0);
1571 	}
1572 }
1573 
1574 ATF_TC_WITHOUT_HEAD(largepage_msync);
1575 ATF_TC_BODY(largepage_msync, tc)
1576 {
1577 	char *addr;
1578 	size_t ps[MAXPAGESIZES];
1579 	int fd, pscnt;
1580 
1581 	pscnt = pagesizes(ps);
1582 	for (int i = 1; i < pscnt; i++) {
1583 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1584 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1585 		    0);
1586 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1587 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1588 
1589 		memset(addr, 0, ps[i]);
1590 
1591 		/*
1592 		 * "Sync" requests are no-ops for SHM objects, so small
1593 		 * PAGE_SIZE-sized requests succeed.
1594 		 */
1595 		ATF_REQUIRE_MSG(msync(addr, ps[0], MS_ASYNC) == 0,
1596 		    "msync(MS_ASYNC) failed; error=%d", errno);
1597 		ATF_REQUIRE_MSG(msync(addr, ps[i], MS_ASYNC) == 0,
1598 		    "msync(MS_ASYNC) failed; error=%d", errno);
1599 		ATF_REQUIRE_MSG(msync(addr, ps[0], MS_SYNC) == 0,
1600 		    "msync(MS_SYNC) failed; error=%d", errno);
1601 		ATF_REQUIRE_MSG(msync(addr, ps[i], MS_SYNC) == 0,
1602 		    "msync(MS_SYNC) failed; error=%d", errno);
1603 
1604 		ATF_REQUIRE_MSG(msync(addr, ps[0], MS_INVALIDATE) != 0,
1605 		    "msync(MS_INVALIDATE) succeeded");
1606 		/* XXX wrong errno */
1607 		ATF_REQUIRE_MSG(errno == EBUSY,
1608 		    "unexpected error %d from msync(MS_INVALIDATE)", errno);
1609 		ATF_REQUIRE_MSG(msync(addr, ps[i], MS_INVALIDATE) == 0,
1610 		    "msync(MS_INVALIDATE) failed; error=%d", errno);
1611 		memset(addr, 0, ps[i]);
1612 
1613 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1614 		ATF_REQUIRE(close(fd) == 0);
1615 	}
1616 }
1617 
1618 static void
1619 largepage_protect(char *addr, size_t sz, int prot, int error)
1620 {
1621 	if (error == 0) {
1622 		ATF_REQUIRE_MSG(mprotect(addr, sz, prot) == 0,
1623 		    "mprotect(%zu, %x) failed; error=%d", sz, prot, errno);
1624 	} else {
1625 		ATF_REQUIRE_MSG(mprotect(addr, sz, prot) != 0,
1626 		    "mprotect(%zu, %x) succeeded", sz, prot);
1627 		ATF_REQUIRE_MSG(errno == error,
1628 		    "unexpected error %d from mprotect(%zu, %x)",
1629 		    errno, sz, prot);
1630 	}
1631 }
1632 
1633 ATF_TC_WITHOUT_HEAD(largepage_mprotect);
1634 ATF_TC_BODY(largepage_mprotect, tc)
1635 {
1636 	char *addr, *addr1;
1637 	size_t ps[MAXPAGESIZES];
1638 	int fd, pscnt;
1639 
1640 	pscnt = pagesizes(ps);
1641 	for (int i = 1; i < pscnt; i++) {
1642 		/*
1643 		 * Reserve a contiguous region in the address space to avoid
1644 		 * spurious failures in the face of ASLR.
1645 		 */
1646 		addr = mmap(NULL, ps[i] * 2, PROT_NONE,
1647 		    MAP_ANON | MAP_ALIGNED(ffsl(ps[i]) - 1), -1, 0);
1648 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1649 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1650 		ATF_REQUIRE(munmap(addr, ps[i] * 2) == 0);
1651 
1652 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1653 		addr = mmap(addr, ps[i], PROT_READ | PROT_WRITE,
1654 		    MAP_SHARED | MAP_FIXED, fd, 0);
1655 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1656 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1657 
1658 		/*
1659 		 * These should be no-ops from the pmap perspective since the
1660 		 * page is not yet entered into the pmap.
1661 		 */
1662 		largepage_protect(addr, ps[0], PROT_READ, EINVAL);
1663 		largepage_protect(addr, ps[i], PROT_READ, 0);
1664 		largepage_protect(addr, ps[0], PROT_NONE, EINVAL);
1665 		largepage_protect(addr, ps[i], PROT_NONE, 0);
1666 		largepage_protect(addr, ps[0],
1667 		    PROT_READ | PROT_WRITE | PROT_EXEC, EINVAL);
1668 		largepage_protect(addr, ps[i],
1669 		    PROT_READ | PROT_WRITE | PROT_EXEC, 0);
1670 
1671 		/* Trigger creation of a mapping and try again. */
1672 		*(volatile char *)addr = 0;
1673 		largepage_protect(addr, ps[0], PROT_READ, EINVAL);
1674 		largepage_protect(addr, ps[i], PROT_READ, 0);
1675 		largepage_protect(addr, ps[0], PROT_NONE, EINVAL);
1676 		largepage_protect(addr, ps[i], PROT_NONE, 0);
1677 		largepage_protect(addr, ps[0],
1678 		    PROT_READ | PROT_WRITE | PROT_EXEC, EINVAL);
1679 		largepage_protect(addr, ps[i],
1680 		    PROT_READ | PROT_WRITE | PROT_EXEC, 0);
1681 
1682 		memset(addr, 0, ps[i]);
1683 
1684 		/* Map two contiguous large pages and merge map entries. */
1685 		addr1 = mmap(addr + ps[i], ps[i], PROT_READ | PROT_WRITE,
1686 		    MAP_SHARED | MAP_FIXED | MAP_EXCL, fd, 0);
1687 		ATF_REQUIRE_MSG(addr1 != MAP_FAILED,
1688 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1689 
1690 		largepage_protect(addr1 - ps[0], ps[0] * 2,
1691 		    PROT_READ | PROT_WRITE, EINVAL);
1692 		largepage_protect(addr, ps[i] * 2, PROT_READ | PROT_WRITE, 0);
1693 
1694 		memset(addr, 0, ps[i] * 2);
1695 
1696 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1697 		ATF_REQUIRE(munmap(addr1, ps[i]) == 0);
1698 		ATF_REQUIRE(close(fd) == 0);
1699 	}
1700 }
1701 
1702 ATF_TC_WITHOUT_HEAD(largepage_minherit);
1703 ATF_TC_BODY(largepage_minherit, tc)
1704 {
1705 	char *addr;
1706 	size_t ps[MAXPAGESIZES];
1707 	pid_t child;
1708 	int fd, pscnt, status;
1709 
1710 	pscnt = pagesizes(ps);
1711 	for (int i = 1; i < pscnt; i++) {
1712 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1713 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1714 		    0);
1715 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1716 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1717 
1718 		ATF_REQUIRE(minherit(addr, ps[0], INHERIT_SHARE) != 0);
1719 
1720 		ATF_REQUIRE_MSG(minherit(addr, ps[i], INHERIT_SHARE) == 0,
1721 		    "minherit(%zu bytes) failed; error=%d", ps[i], errno);
1722 		child = fork();
1723 		ATF_REQUIRE_MSG(child != -1, "fork failed; error=%d", errno);
1724 		if (child == 0) {
1725 			char v;
1726 
1727 			*(volatile char *)addr = 0;
1728 			if (mincore(addr, ps[0], &v) != 0)
1729 				_exit(1);
1730 			if ((v & MINCORE_PSIND(i)) == 0)
1731 				_exit(2);
1732 			_exit(0);
1733 		}
1734 		ATF_REQUIRE_MSG(waitpid(child, &status, 0) == child,
1735 		    "waitpid failed; error=%d", errno);
1736 		ATF_REQUIRE_MSG(WIFEXITED(status),
1737 		    "child was killed by signal %d", WTERMSIG(status));
1738 		ATF_REQUIRE_MSG(WEXITSTATUS(status) == 0,
1739 		    "child exited with status %d", WEXITSTATUS(status));
1740 
1741 		ATF_REQUIRE_MSG(minherit(addr, ps[i], INHERIT_NONE) == 0,
1742 		    "minherit(%zu bytes) failed; error=%d", ps[i], errno);
1743 		child = fork();
1744 		ATF_REQUIRE_MSG(child != -1, "fork failed; error=%d", errno);
1745 		if (child == 0) {
1746 			char v;
1747 
1748 			if (mincore(addr, ps[0], &v) == 0)
1749 				_exit(1);
1750 			_exit(0);
1751 		}
1752 		ATF_REQUIRE_MSG(waitpid(child, &status, 0) == child,
1753 		    "waitpid failed; error=%d", errno);
1754 		ATF_REQUIRE_MSG(WIFEXITED(status),
1755 		    "child was killed by signal %d", WTERMSIG(status));
1756 		ATF_REQUIRE_MSG(WEXITSTATUS(status) == 0,
1757 		    "child exited with status %d", WEXITSTATUS(status));
1758 
1759 		/* Copy-on-write is not supported for static large pages. */
1760 		ATF_REQUIRE_MSG(minherit(addr, ps[i], INHERIT_COPY) != 0,
1761 		    "minherit(%zu bytes) succeeded", ps[i]);
1762 
1763 		ATF_REQUIRE_MSG(minherit(addr, ps[i], INHERIT_ZERO) == 0,
1764 		    "minherit(%zu bytes) failed; error=%d", ps[i], errno);
1765 		child = fork();
1766 		ATF_REQUIRE_MSG(child != -1, "fork failed; error=%d", errno);
1767 		if (child == 0) {
1768 			char v;
1769 
1770 			*(volatile char *)addr = 0;
1771 			if (mincore(addr, ps[0], &v) != 0)
1772 				_exit(1);
1773 			if ((v & MINCORE_SUPER) != 0)
1774 				_exit(2);
1775 			_exit(0);
1776 		}
1777 		ATF_REQUIRE_MSG(waitpid(child, &status, 0) == child,
1778 		    "waitpid failed; error=%d", errno);
1779 		ATF_REQUIRE_MSG(WIFEXITED(status),
1780 		    "child was killed by signal %d", WTERMSIG(status));
1781 		ATF_REQUIRE_MSG(WEXITSTATUS(status) == 0,
1782 		    "child exited with status %d", WEXITSTATUS(status));
1783 
1784 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1785 		ATF_REQUIRE(close(fd) == 0);
1786 	}
1787 }
1788 
1789 ATF_TC_WITHOUT_HEAD(largepage_pipe);
1790 ATF_TC_BODY(largepage_pipe, tc)
1791 {
1792 	size_t ps[MAXPAGESIZES];
1793 	char *addr;
1794 	ssize_t len;
1795 	int fd, pfd[2], pscnt, status;
1796 	pid_t child;
1797 
1798 	pscnt = pagesizes(ps);
1799 
1800 	for (int i = 1; i < pscnt; i++) {
1801 		fd = shm_open_large(i, SHM_LARGEPAGE_ALLOC_DEFAULT, ps[i]);
1802 		addr = mmap(NULL, ps[i], PROT_READ | PROT_WRITE, MAP_SHARED, fd,
1803 		    0);
1804 		ATF_REQUIRE_MSG(addr != MAP_FAILED,
1805 		    "mmap(%zu bytes) failed; error=%d", ps[i], errno);
1806 
1807 		/* Trigger creation of a mapping. */
1808 		*(volatile char *)addr = '\0';
1809 
1810 		ATF_REQUIRE(pipe(pfd) == 0);
1811 		child = fork();
1812 		ATF_REQUIRE_MSG(child != -1, "fork() failed; error=%d", errno);
1813 		if (child == 0) {
1814 			char buf[BUFSIZ];
1815 			ssize_t resid;
1816 
1817 			(void)close(pfd[0]);
1818 			for (resid = (size_t)ps[i]; resid > 0; resid -= len) {
1819 				len = read(pfd[1], buf, sizeof(buf));
1820 				if (len < 0)
1821 					_exit(1);
1822 			}
1823 			_exit(0);
1824 		}
1825 		ATF_REQUIRE(close(pfd[1]) == 0);
1826 		len = write(pfd[0], addr, ps[i]);
1827 		ATF_REQUIRE_MSG(len >= 0, "write() failed; error=%d", errno);
1828 		ATF_REQUIRE_MSG(len == (ssize_t)ps[i],
1829 		    "short write; len=%zd", len);
1830 		ATF_REQUIRE(close(pfd[0]) == 0);
1831 
1832 		ATF_REQUIRE_MSG(waitpid(child, &status, 0) == child,
1833 		    "waitpid() failed; error=%d", errno);
1834 		ATF_REQUIRE_MSG(WIFEXITED(status),
1835 		    "child was killed by signal %d", WTERMSIG(status));
1836 		ATF_REQUIRE_MSG(WEXITSTATUS(status) == 0,
1837 		    "child exited with status %d", WEXITSTATUS(status));
1838 
1839 		ATF_REQUIRE(munmap(addr, ps[i]) == 0);
1840 		ATF_REQUIRE(close(fd) == 0);
1841 	}
1842 }
1843 
1844 ATF_TC_WITHOUT_HEAD(largepage_reopen);
1845 ATF_TC_BODY(largepage_reopen, tc)
1846 {
1847 	char *addr, *vec;
1848 	size_t ps[MAXPAGESIZES];
1849 	int fd, psind;
1850 
1851 	(void)pagesizes(ps);
1852 	psind = 1;
1853 
1854 	gen_test_path();
1855 	fd = shm_create_largepage(test_path, O_CREAT | O_RDWR, psind,
1856 	    SHM_LARGEPAGE_ALLOC_DEFAULT, 0600);
1857 	if (fd < 0 && errno == ENOTTY)
1858 		atf_tc_skip("no large page support");
1859 	ATF_REQUIRE_MSG(fd >= 0, "shm_create_largepage failed; error=%d", errno);
1860 
1861 	ATF_REQUIRE_MSG(ftruncate(fd, ps[psind]) == 0,
1862 	    "ftruncate failed; error=%d", errno);
1863 
1864 	ATF_REQUIRE_MSG(close(fd) == 0, "close failed; error=%d", errno);
1865 
1866 	fd = shm_open(test_path, O_RDWR, 0);
1867 	ATF_REQUIRE_MSG(fd >= 0, "shm_open failed; error=%d", errno);
1868 
1869 	addr = mmap(NULL, ps[psind], PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
1870 	ATF_REQUIRE_MSG(addr != MAP_FAILED, "mmap failed; error=%d", errno);
1871 
1872 	/* Trigger a fault and mapping creation. */
1873 	*(volatile char *)addr = 0;
1874 
1875 	vec = malloc(ps[psind] / ps[0]);
1876 	ATF_REQUIRE(vec != NULL);
1877 	ATF_REQUIRE_MSG(mincore(addr, ps[psind], vec) == 0,
1878 	    "mincore failed; error=%d", errno);
1879 	ATF_REQUIRE_MSG((vec[0] & MINCORE_PSIND(psind)) != 0,
1880 	    "page not mapped into a %zu-byte superpage", ps[psind]);
1881 
1882 	ATF_REQUIRE_MSG(shm_unlink(test_path) == 0,
1883 	    "shm_unlink failed; errno=%d", errno);
1884 	ATF_REQUIRE_MSG(close(fd) == 0,
1885 	    "close failed; errno=%d", errno);
1886 }
1887 
1888 ATF_TP_ADD_TCS(tp)
1889 {
1890 
1891 	ATF_TP_ADD_TC(tp, remap_object);
1892 	ATF_TP_ADD_TC(tp, rename_from_anon);
1893 	ATF_TP_ADD_TC(tp, rename_bad_path_pointer);
1894 	ATF_TP_ADD_TC(tp, rename_from_nonexisting);
1895 	ATF_TP_ADD_TC(tp, rename_to_anon);
1896 	ATF_TP_ADD_TC(tp, rename_to_replace);
1897 	ATF_TP_ADD_TC(tp, rename_to_noreplace);
1898 	ATF_TP_ADD_TC(tp, rename_to_exchange);
1899 	ATF_TP_ADD_TC(tp, rename_to_exchange_nonexisting);
1900 	ATF_TP_ADD_TC(tp, rename_to_self);
1901 	ATF_TP_ADD_TC(tp, rename_bad_flag);
1902 	ATF_TP_ADD_TC(tp, reopen_object);
1903 	ATF_TP_ADD_TC(tp, readonly_mmap_write);
1904 	ATF_TP_ADD_TC(tp, open_after_link);
1905 	ATF_TP_ADD_TC(tp, open_invalid_path);
1906 	ATF_TP_ADD_TC(tp, open_write_only);
1907 	ATF_TP_ADD_TC(tp, open_extra_flags);
1908 	ATF_TP_ADD_TC(tp, open_anon);
1909 	ATF_TP_ADD_TC(tp, open_anon_readonly);
1910 	ATF_TP_ADD_TC(tp, open_bad_path_pointer);
1911 	ATF_TP_ADD_TC(tp, open_path_too_long);
1912 	ATF_TP_ADD_TC(tp, open_nonexisting_object);
1913 	ATF_TP_ADD_TC(tp, open_create_existing_object);
1914 	ATF_TP_ADD_TC(tp, shm_functionality_across_fork);
1915 	ATF_TP_ADD_TC(tp, trunc_resets_object);
1916 	ATF_TP_ADD_TC(tp, unlink_bad_path_pointer);
1917 	ATF_TP_ADD_TC(tp, unlink_path_too_long);
1918 	ATF_TP_ADD_TC(tp, object_resize);
1919 	ATF_TP_ADD_TC(tp, cloexec);
1920 	ATF_TP_ADD_TC(tp, mode);
1921 	ATF_TP_ADD_TC(tp, fallocate);
1922 	ATF_TP_ADD_TC(tp, fspacectl);
1923 	ATF_TP_ADD_TC(tp, largepage_basic);
1924 	ATF_TP_ADD_TC(tp, largepage_config);
1925 	ATF_TP_ADD_TC(tp, largepage_mmap);
1926 	ATF_TP_ADD_TC(tp, largepage_munmap);
1927 	ATF_TP_ADD_TC(tp, largepage_madvise);
1928 	ATF_TP_ADD_TC(tp, largepage_mlock);
1929 	ATF_TP_ADD_TC(tp, largepage_msync);
1930 	ATF_TP_ADD_TC(tp, largepage_mprotect);
1931 	ATF_TP_ADD_TC(tp, largepage_minherit);
1932 	ATF_TP_ADD_TC(tp, largepage_pipe);
1933 	ATF_TP_ADD_TC(tp, largepage_reopen);
1934 
1935 	return (atf_no_error());
1936 }
1937