xref: /freebsd/tests/sys/vm/stack/stack_mprotect_exec_test.c (revision f8bc606aad35daf9853e3198f14d4a7baba36fcb)
18920c5f2SDmitry Chagin /*-
28920c5f2SDmitry Chagin  * Copyright (c) 2023 Dmitry Chagin <dchagin@FreeBSD.org>
38920c5f2SDmitry Chagin  *
48920c5f2SDmitry Chagin  * SPDX-License-Identifier: BSD-2-Clause
58920c5f2SDmitry Chagin  *
68920c5f2SDmitry Chagin  * PR:	272585
78920c5f2SDmitry Chagin  * Test provided by John F. Carr
88920c5f2SDmitry Chagin  */
98920c5f2SDmitry Chagin 
108920c5f2SDmitry Chagin #include <sys/systm.h>
118920c5f2SDmitry Chagin #include <sys/mman.h>
128920c5f2SDmitry Chagin #include <vm/vm_param.h>
138920c5f2SDmitry Chagin 
148920c5f2SDmitry Chagin #include <atf-c.h>
15*f8bc606aSDmitry Chagin #include <signal.h>
168920c5f2SDmitry Chagin #include <unistd.h>
178920c5f2SDmitry Chagin 
18*f8bc606aSDmitry Chagin static void
sigsegv_handler(int sig __unused)19*f8bc606aSDmitry Chagin sigsegv_handler(int sig __unused)
20*f8bc606aSDmitry Chagin {
21*f8bc606aSDmitry Chagin 
22*f8bc606aSDmitry Chagin 	atf_tc_fail("Invalid stack protection mode after grows");
23*f8bc606aSDmitry Chagin }
248920c5f2SDmitry Chagin 
258920c5f2SDmitry Chagin ATF_TC_WITHOUT_HEAD(mprotect_exec_test);
ATF_TC_BODY(mprotect_exec_test,tc)268920c5f2SDmitry Chagin ATF_TC_BODY(mprotect_exec_test, tc)
278920c5f2SDmitry Chagin {
288920c5f2SDmitry Chagin 	long pagesize;
298920c5f2SDmitry Chagin 	char *addr, *guard;
308920c5f2SDmitry Chagin 	size_t alloc_size;
318920c5f2SDmitry Chagin 
32*f8bc606aSDmitry Chagin 	signal(SIGSEGV, sigsegv_handler);
33*f8bc606aSDmitry Chagin 
348920c5f2SDmitry Chagin 	pagesize = sysconf(_SC_PAGESIZE);
358920c5f2SDmitry Chagin 	ATF_REQUIRE(pagesize > 0);
368920c5f2SDmitry Chagin 
378920c5f2SDmitry Chagin 	alloc_size = SGROWSIZ * 2;
388920c5f2SDmitry Chagin 	addr = mmap(NULL, alloc_size, PROT_READ | PROT_WRITE,
398920c5f2SDmitry Chagin 	    MAP_STACK | MAP_PRIVATE | MAP_ANON, -1, 0);
408920c5f2SDmitry Chagin 	ATF_REQUIRE(addr != MAP_FAILED);
418920c5f2SDmitry Chagin 
428920c5f2SDmitry Chagin 	/*
438920c5f2SDmitry Chagin 	 * Change prot of the last page in the mmaped stack area.
448920c5f2SDmitry Chagin 	 */
458920c5f2SDmitry Chagin 	guard = addr + alloc_size - SGROWSIZ;
468920c5f2SDmitry Chagin 	ATF_REQUIRE(mprotect(guard, pagesize, PROT_NONE) == 0);
478920c5f2SDmitry Chagin 
488920c5f2SDmitry Chagin 	((volatile char *)guard)[-1];
498920c5f2SDmitry Chagin }
508920c5f2SDmitry Chagin 
ATF_TP_ADD_TCS(tp)518920c5f2SDmitry Chagin ATF_TP_ADD_TCS(tp)
528920c5f2SDmitry Chagin {
538920c5f2SDmitry Chagin 
548920c5f2SDmitry Chagin 	ATF_TP_ADD_TC(tp, mprotect_exec_test);
558920c5f2SDmitry Chagin 
568920c5f2SDmitry Chagin 	return (atf_no_error());
578920c5f2SDmitry Chagin }
58