xref: /freebsd-src/contrib/netbsd-tests/lib/libc/sys/t_getcontext.c (revision 4421971444c1e90ecaf338bcc43b596911b2a472)
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
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__
557bb9bc9cSEnji Cooper #if defined(__amd64__)
56521b1af4SJohn Baldwin 	for (i = 0; i < 5; i++) {
577bb9bc9cSEnji Cooper #elif defined(__aarch64__)
58b5c035f2SAndrew Turner 	for (i = 0; i < 7; i++) {
597bb9bc9cSEnji Cooper #elif defined(__mips__)
6061521662SRuslan Bukin 	for (i = 0; i < 5; i++) {
61*44219714SEnji Cooper #else
62*44219714SEnji Cooper 	for (i = 0; i < 9; i++) {
637bb9bc9cSEnji Cooper #endif
64521b1af4SJohn Baldwin #else
6557718be8SEnji Cooper 	for (i = 0; i < 9; i++) {
66521b1af4SJohn Baldwin #endif
6757718be8SEnji Cooper 		ia = va_arg(va, int);
6857718be8SEnji Cooper 		ATF_REQUIRE_EQ(i, ia);
6957718be8SEnji Cooper 	}
7057718be8SEnji Cooper 	va_end(va);
7157718be8SEnji Cooper 
7257718be8SEnji Cooper 	calls++;
7357718be8SEnji Cooper }
7457718be8SEnji Cooper 
7557718be8SEnji Cooper ATF_TC(getcontext_err);
7657718be8SEnji Cooper ATF_TC_HEAD(getcontext_err, tc)
7757718be8SEnji Cooper {
7857718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test errors from getcontext(2)");
7957718be8SEnji Cooper }
8057718be8SEnji Cooper 
8157718be8SEnji Cooper ATF_TC_BODY(getcontext_err, tc)
8257718be8SEnji Cooper {
8357718be8SEnji Cooper 
8457718be8SEnji Cooper 	errno = 0;
8557718be8SEnji Cooper 	ATF_REQUIRE_ERRNO(EFAULT, getcontext((void *)-1) == -1);
8657718be8SEnji Cooper }
8757718be8SEnji Cooper 
8857718be8SEnji Cooper ATF_TC(setcontext_err);
8957718be8SEnji Cooper ATF_TC_HEAD(setcontext_err, tc)
9057718be8SEnji Cooper {
9157718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test errors from setcontext(2)");
9257718be8SEnji Cooper }
9357718be8SEnji Cooper 
9457718be8SEnji Cooper ATF_TC_BODY(setcontext_err, tc)
9557718be8SEnji Cooper {
9657718be8SEnji Cooper 
9757718be8SEnji Cooper 	errno = 0;
9857718be8SEnji Cooper 	ATF_REQUIRE_ERRNO(EFAULT, setcontext((void *)-1) == -1);
9957718be8SEnji Cooper }
10057718be8SEnji Cooper 
10157718be8SEnji Cooper ATF_TC(setcontext_link);
10257718be8SEnji Cooper ATF_TC_HEAD(setcontext_link, tc)
10357718be8SEnji Cooper {
10457718be8SEnji Cooper 
10557718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr",
10657718be8SEnji Cooper 	"Checks get/make/setcontext(), context linking via uc_link(), "
10757718be8SEnji Cooper 	    "and argument passing to the new context");
10857718be8SEnji Cooper }
10957718be8SEnji Cooper 
11057718be8SEnji Cooper ATF_TC_BODY(setcontext_link, tc)
11157718be8SEnji Cooper {
11257718be8SEnji Cooper 	ucontext_t uc[DEPTH];
11357718be8SEnji Cooper 	ucontext_t save;
11457718be8SEnji Cooper 	volatile int i = 0; /* avoid longjmp clobbering */
11557718be8SEnji Cooper 
11657718be8SEnji Cooper 	for (i = 0; i < DEPTH; ++i) {
11757718be8SEnji Cooper 		ATF_REQUIRE_EQ(getcontext(&uc[i]), 0);
11857718be8SEnji Cooper 
11957718be8SEnji Cooper 		uc[i].uc_stack.ss_sp = malloc(STACKSZ);
12057718be8SEnji Cooper 		uc[i].uc_stack.ss_size = STACKSZ;
12157718be8SEnji Cooper 		uc[i].uc_link = (i > 0) ? &uc[i - 1] : &save;
12257718be8SEnji Cooper 
1237bb9bc9cSEnji Cooper #ifdef __FreeBSD__
1247bb9bc9cSEnji Cooper #if defined(__amd64__)
125521b1af4SJohn Baldwin 		/* FreeBSD/amd64 only permits up to 6 arguments. */
126521b1af4SJohn Baldwin 		makecontext(&uc[i], (void *)run, 6, i,
127521b1af4SJohn Baldwin 			0, 1, 2, 3, 4);
1287bb9bc9cSEnji Cooper #elif defined(__aarch64__)
129b5c035f2SAndrew Turner 		/* FreeBSD/arm64 only permits up to 8 arguments. */
130b5c035f2SAndrew Turner 		makecontext(&uc[i], (void *)run, 8, i,
131b5c035f2SAndrew Turner 			0, 1, 2, 3, 4, 5, 6);
1327bb9bc9cSEnji Cooper #elif defined(__mips__)
13361521662SRuslan Bukin 		/* FreeBSD/mips only permits up to 6 arguments. */
13461521662SRuslan Bukin 		makecontext(&uc[i], (void *)run, 6, i,
13561521662SRuslan Bukin 			0, 1, 2, 3, 4);
1367bb9bc9cSEnji Cooper #endif
137521b1af4SJohn Baldwin #else
13857718be8SEnji Cooper 		makecontext(&uc[i], (void *)run, 10, i,
13957718be8SEnji Cooper 			0, 1, 2, 3, 4, 5, 6, 7, 8);
140521b1af4SJohn Baldwin #endif
14157718be8SEnji Cooper 	}
14257718be8SEnji Cooper 
14357718be8SEnji Cooper 	ATF_REQUIRE_EQ(getcontext(&save), 0);
14457718be8SEnji Cooper 
14557718be8SEnji Cooper 	if (calls == 0)
14657718be8SEnji Cooper 		ATF_REQUIRE_EQ(setcontext(&uc[DEPTH-1]), 0);
14757718be8SEnji Cooper }
14857718be8SEnji Cooper 
14957718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
15057718be8SEnji Cooper {
15157718be8SEnji Cooper 
15257718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, getcontext_err);
15357718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, setcontext_err);
15457718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, setcontext_link);
15557718be8SEnji Cooper 
15657718be8SEnji Cooper 	return atf_no_error();
15757718be8SEnji Cooper }
158