xref: /freebsd-src/contrib/netbsd-tests/lib/libc/sys/t_getcontext.c (revision 14b841d4a8e6c25d964f5e3bbda379a1541a27ec)
157718be8SEnji Cooper /* $NetBSD: t_getcontext.c,v 1.3 2011/07/14 04:59:14 jruoho Exp $ */
257718be8SEnji Cooper 
357718be8SEnji Cooper /*-
457718be8SEnji Cooper  * Copyright (c) 2008 The NetBSD Foundation, Inc.
557718be8SEnji Cooper  * All rights reserved.
657718be8SEnji Cooper  *
757718be8SEnji Cooper  * Redistribution and use in source and binary forms, with or without
857718be8SEnji Cooper  * modification, are permitted provided that the following conditions
957718be8SEnji Cooper  * are met:
1057718be8SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
1157718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
1257718be8SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
1357718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer in the
1457718be8SEnji Cooper  *    documentation and/or other materials provided with the distribution.
1557718be8SEnji Cooper  *
1657718be8SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
1757718be8SEnji Cooper  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
1857718be8SEnji Cooper  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
1957718be8SEnji Cooper  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
2057718be8SEnji Cooper  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
2157718be8SEnji Cooper  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
2257718be8SEnji Cooper  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2357718be8SEnji Cooper  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
2457718be8SEnji Cooper  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
2557718be8SEnji Cooper  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
2657718be8SEnji Cooper  * POSSIBILITY OF SUCH DAMAGE.
2757718be8SEnji Cooper  */
2857718be8SEnji Cooper 
2957718be8SEnji Cooper #include <sys/cdefs.h>
3057718be8SEnji Cooper __COPYRIGHT("@(#) Copyright (c) 2008\
3157718be8SEnji Cooper  The NetBSD Foundation, inc. All rights reserved.");
3257718be8SEnji Cooper __RCSID("$NetBSD: t_getcontext.c,v 1.3 2011/07/14 04:59:14 jruoho Exp $");
3357718be8SEnji Cooper 
3457718be8SEnji Cooper #include <atf-c.h>
3557718be8SEnji Cooper #include <errno.h>
3657718be8SEnji Cooper #include <stdarg.h>
3757718be8SEnji Cooper #include <stdlib.h>
3857718be8SEnji Cooper #include <ucontext.h>
3957718be8SEnji Cooper 
4057718be8SEnji Cooper #define STACKSZ (10*1024)
4157718be8SEnji Cooper #define DEPTH 3
4257718be8SEnji Cooper 
4357718be8SEnji Cooper static int calls;
4457718be8SEnji Cooper 
4557718be8SEnji Cooper static void
run(int n,...)4657718be8SEnji Cooper run(int n, ...)
4757718be8SEnji Cooper {
4857718be8SEnji Cooper 	va_list va;
4957718be8SEnji Cooper 	int i, ia;
5057718be8SEnji Cooper 
5157718be8SEnji Cooper 	ATF_REQUIRE_EQ(n, DEPTH - calls - 1);
5257718be8SEnji Cooper 
5357718be8SEnji Cooper 	va_start(va, n);
547bb9bc9cSEnji Cooper #ifdef __FreeBSD__
5568e709cbSJohn Baldwin #if defined(__amd64__) || defined(__sparc64__)
56521b1af4SJohn Baldwin 	for (i = 0; i < 5; i++) {
57*38012218SRuslan Bukin #elif defined(__aarch64__) || defined(__riscv)
58b5c035f2SAndrew Turner 	for (i = 0; i < 7; i++) {
5944219714SEnji Cooper #else
6044219714SEnji Cooper 	for (i = 0; i < 9; i++) {
617bb9bc9cSEnji Cooper #endif
62521b1af4SJohn Baldwin #else
6357718be8SEnji Cooper 	for (i = 0; i < 9; i++) {
64521b1af4SJohn Baldwin #endif
6557718be8SEnji Cooper 		ia = va_arg(va, int);
6657718be8SEnji Cooper 		ATF_REQUIRE_EQ(i, ia);
6757718be8SEnji Cooper 	}
6857718be8SEnji Cooper 	va_end(va);
6957718be8SEnji Cooper 
7057718be8SEnji Cooper 	calls++;
7157718be8SEnji Cooper }
7257718be8SEnji Cooper 
7357718be8SEnji Cooper ATF_TC(getcontext_err);
7457718be8SEnji Cooper ATF_TC_HEAD(getcontext_err, tc)
7557718be8SEnji Cooper {
7657718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test errors from getcontext(2)");
7757718be8SEnji Cooper }
7857718be8SEnji Cooper 
7957718be8SEnji Cooper ATF_TC_BODY(getcontext_err, tc)
8057718be8SEnji Cooper {
8157718be8SEnji Cooper 
8257718be8SEnji Cooper 	errno = 0;
8357718be8SEnji Cooper 	ATF_REQUIRE_ERRNO(EFAULT, getcontext((void *)-1) == -1);
8457718be8SEnji Cooper }
8557718be8SEnji Cooper 
8657718be8SEnji Cooper ATF_TC(setcontext_err);
8757718be8SEnji Cooper ATF_TC_HEAD(setcontext_err, tc)
8857718be8SEnji Cooper {
8957718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test errors from setcontext(2)");
9057718be8SEnji Cooper }
9157718be8SEnji Cooper 
9257718be8SEnji Cooper ATF_TC_BODY(setcontext_err, tc)
9357718be8SEnji Cooper {
9457718be8SEnji Cooper 
9557718be8SEnji Cooper 	errno = 0;
9657718be8SEnji Cooper 	ATF_REQUIRE_ERRNO(EFAULT, setcontext((void *)-1) == -1);
9757718be8SEnji Cooper }
9857718be8SEnji Cooper 
9957718be8SEnji Cooper ATF_TC(setcontext_link);
10057718be8SEnji Cooper ATF_TC_HEAD(setcontext_link, tc)
10157718be8SEnji Cooper {
10257718be8SEnji Cooper 
10357718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr",
10457718be8SEnji Cooper 	"Checks get/make/setcontext(), context linking via uc_link(), "
10557718be8SEnji Cooper 	    "and argument passing to the new context");
10657718be8SEnji Cooper }
10757718be8SEnji Cooper 
10857718be8SEnji Cooper ATF_TC_BODY(setcontext_link, tc)
10957718be8SEnji Cooper {
11057718be8SEnji Cooper 	ucontext_t uc[DEPTH];
11157718be8SEnji Cooper 	ucontext_t save;
11257718be8SEnji Cooper 	volatile int i = 0; /* avoid longjmp clobbering */
11357718be8SEnji Cooper 
11457718be8SEnji Cooper 	for (i = 0; i < DEPTH; ++i) {
11557718be8SEnji Cooper 		ATF_REQUIRE_EQ(getcontext(&uc[i]), 0);
11657718be8SEnji Cooper 
11757718be8SEnji Cooper 		uc[i].uc_stack.ss_sp = malloc(STACKSZ);
11857718be8SEnji Cooper 		uc[i].uc_stack.ss_size = STACKSZ;
11957718be8SEnji Cooper 		uc[i].uc_link = (i > 0) ? &uc[i - 1] : &save;
12057718be8SEnji Cooper 
1217bb9bc9cSEnji Cooper #ifdef __FreeBSD__
12268e709cbSJohn Baldwin #if defined(__amd64__) || defined(__sparc64__)
12368e709cbSJohn Baldwin 		/*
12468e709cbSJohn Baldwin 		 * FreeBSD/amd64 and FreeBSD/sparc64 only permit up to
12568e709cbSJohn Baldwin 		 * 6 arguments.
12668e709cbSJohn Baldwin 		 */
127521b1af4SJohn Baldwin 		makecontext(&uc[i], (void *)run, 6, i,
128521b1af4SJohn Baldwin 			0, 1, 2, 3, 4);
129*38012218SRuslan Bukin #elif defined(__aarch64__) || defined(__riscv)
13068e709cbSJohn Baldwin 		/*
13168e709cbSJohn Baldwin 		 * FreeBSD/arm64 and FreeBSD/riscv64 only permit up to
13268e709cbSJohn Baldwin 		 * 8 arguments.
13368e709cbSJohn Baldwin 		 */
134b5c035f2SAndrew Turner 		makecontext(&uc[i], (void *)run, 8, i,
135b5c035f2SAndrew Turner 			0, 1, 2, 3, 4, 5, 6);
13662d02a8fSEnji Cooper #else
13762d02a8fSEnji Cooper 		makecontext(&uc[i], (void *)run, 10, i,
13862d02a8fSEnji Cooper 			0, 1, 2, 3, 4, 5, 6, 7, 8);
1397bb9bc9cSEnji Cooper #endif
140521b1af4SJohn Baldwin #else
14157718be8SEnji Cooper 		makecontext(&uc[i], (void *)run, 10, i,
14257718be8SEnji Cooper 			0, 1, 2, 3, 4, 5, 6, 7, 8);
143521b1af4SJohn Baldwin #endif
14457718be8SEnji Cooper 	}
14557718be8SEnji Cooper 
14657718be8SEnji Cooper 	ATF_REQUIRE_EQ(getcontext(&save), 0);
14757718be8SEnji Cooper 
14857718be8SEnji Cooper 	if (calls == 0)
14957718be8SEnji Cooper 		ATF_REQUIRE_EQ(setcontext(&uc[DEPTH-1]), 0);
15057718be8SEnji Cooper }
15157718be8SEnji Cooper 
15257718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
15357718be8SEnji Cooper {
15457718be8SEnji Cooper 
15557718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, getcontext_err);
15657718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, setcontext_err);
15757718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, setcontext_link);
15857718be8SEnji Cooper 
15957718be8SEnji Cooper 	return atf_no_error();
16057718be8SEnji Cooper }
161