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