xref: /freebsd-src/contrib/netbsd-tests/lib/libc/sys/t_swapcontext.c (revision 9268022b74279434ed6300244e3f977e56a8ceb5)
1*57718be8SEnji Cooper /* $NetBSD: t_swapcontext.c,v 1.3 2013/05/05 10:28:11 skrll Exp $ */
2*57718be8SEnji Cooper 
3*57718be8SEnji Cooper /*
4*57718be8SEnji Cooper  * Copyright (c) 2012 Emmanuel Dreyfus. All rights reserved.
5*57718be8SEnji Cooper  *
6*57718be8SEnji Cooper  * Redistribution and use in source and binary forms, with or without
7*57718be8SEnji Cooper  * modification, are permitted provided that the following conditions
8*57718be8SEnji Cooper  * are met:
9*57718be8SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
10*57718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
11*57718be8SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
12*57718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer in the
13*57718be8SEnji Cooper  *    documentation and/or other materials provided with the distribution.
14*57718be8SEnji Cooper  *
15*57718be8SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
16*57718be8SEnji Cooper  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
17*57718be8SEnji Cooper  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18*57718be8SEnji Cooper  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
19*57718be8SEnji Cooper  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
20*57718be8SEnji Cooper  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
21*57718be8SEnji Cooper  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22*57718be8SEnji Cooper  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23*57718be8SEnji Cooper  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
24*57718be8SEnji Cooper  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25*57718be8SEnji Cooper  * POSSIBILITY OF SUCH DAMAGE.
26*57718be8SEnji Cooper  */
27*57718be8SEnji Cooper 
28*57718be8SEnji Cooper #include <sys/cdefs.h>
29*57718be8SEnji Cooper __RCSID("$NetBSD");
30*57718be8SEnji Cooper 
31*57718be8SEnji Cooper #include <ucontext.h>
32*57718be8SEnji Cooper #include <stdio.h>
33*57718be8SEnji Cooper #include <stdlib.h>
34*57718be8SEnji Cooper #include <lwp.h>
35*57718be8SEnji Cooper 
36*57718be8SEnji Cooper #include <atf-c.h>
37*57718be8SEnji Cooper 
38*57718be8SEnji Cooper #define STACKSIZE 65536
39*57718be8SEnji Cooper 
40*57718be8SEnji Cooper char stack[STACKSIZE];
41*57718be8SEnji Cooper ucontext_t nctx;
42*57718be8SEnji Cooper ucontext_t octx;
43*57718be8SEnji Cooper void *otls;
44*57718be8SEnji Cooper void *ntls;
45*57718be8SEnji Cooper int val1, val2;
46*57718be8SEnji Cooper int alter_tlsbase;
47*57718be8SEnji Cooper 
48*57718be8SEnji Cooper /* ARGSUSED0 */
49*57718be8SEnji Cooper static void
swapfunc(void * arg)50*57718be8SEnji Cooper swapfunc(void *arg)
51*57718be8SEnji Cooper {
52*57718be8SEnji Cooper 	ntls = _lwp_getprivate();
53*57718be8SEnji Cooper 	printf("after swapcontext TLS pointer = %p\n", ntls);
54*57718be8SEnji Cooper 
55*57718be8SEnji Cooper 	if (alter_tlsbase) {
56*57718be8SEnji Cooper 		ATF_REQUIRE_EQ(ntls, &val1);
57*57718be8SEnji Cooper 		printf("TLS pointer modified by swapcontext()\n");
58*57718be8SEnji Cooper 	} else {
59*57718be8SEnji Cooper 		ATF_REQUIRE_EQ(ntls, &val2);
60*57718be8SEnji Cooper 		printf("TLS pointer left untouched by swapcontext()\n");
61*57718be8SEnji Cooper 	}
62*57718be8SEnji Cooper 
63*57718be8SEnji Cooper 	/* Go back in main */
64*57718be8SEnji Cooper 	ATF_REQUIRE(swapcontext(&nctx, &octx));
65*57718be8SEnji Cooper 
66*57718be8SEnji Cooper 	/* NOTREACHED */
67*57718be8SEnji Cooper 	return;
68*57718be8SEnji Cooper }
69*57718be8SEnji Cooper 
70*57718be8SEnji Cooper static void
mainfunc(void)71*57718be8SEnji Cooper mainfunc(void)
72*57718be8SEnji Cooper {
73*57718be8SEnji Cooper 	printf("Testing if swapcontext() alters TLS pointer if _UC_TLSBASE "
74*57718be8SEnji Cooper 	       "is %s\n", (alter_tlsbase) ? "left set" : "cleared");
75*57718be8SEnji Cooper 
76*57718be8SEnji Cooper 	_lwp_setprivate(&val1);
77*57718be8SEnji Cooper 	printf("before swapcontext TLS pointer = %p\n", &val1);
78*57718be8SEnji Cooper 
79*57718be8SEnji Cooper 	ATF_REQUIRE(getcontext(&nctx) == 0);
80*57718be8SEnji Cooper 
81*57718be8SEnji Cooper 	nctx.uc_stack.ss_sp = stack;
82*57718be8SEnji Cooper 	nctx.uc_stack.ss_size = sizeof(stack);
83*57718be8SEnji Cooper 
84*57718be8SEnji Cooper #ifndef _UC_TLSBASE
85*57718be8SEnji Cooper 	ATF_REQUIRE_MSG(0, "_UC_TLSBASE is not defined");
86*57718be8SEnji Cooper #else /* _UC_TLSBASE */
87*57718be8SEnji Cooper 	ATF_REQUIRE(nctx.uc_flags & _UC_TLSBASE);
88*57718be8SEnji Cooper 	if (!alter_tlsbase)
89*57718be8SEnji Cooper 		nctx.uc_flags &= ~_UC_TLSBASE;
90*57718be8SEnji Cooper #endif /* _UC_TLSBASE */
91*57718be8SEnji Cooper 
92*57718be8SEnji Cooper 	makecontext(&nctx, swapfunc, 0);
93*57718be8SEnji Cooper 
94*57718be8SEnji Cooper 	_lwp_setprivate(&val2);
95*57718be8SEnji Cooper 	otls = _lwp_getprivate();
96*57718be8SEnji Cooper 	printf("before swapcontext TLS pointer = %p\n", otls);
97*57718be8SEnji Cooper 	ATF_REQUIRE(swapcontext(&octx, &nctx) == 0);
98*57718be8SEnji Cooper 
99*57718be8SEnji Cooper 	printf("Test completed\n");
100*57718be8SEnji Cooper }
101*57718be8SEnji Cooper 
102*57718be8SEnji Cooper 
103*57718be8SEnji Cooper ATF_TC(swapcontext1);
ATF_TC_HEAD(swapcontext1,tc)104*57718be8SEnji Cooper ATF_TC_HEAD(swapcontext1, tc)
105*57718be8SEnji Cooper {
106*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Testing if swapcontext() can let "
107*57718be8SEnji Cooper 	    "TLS pointer untouched");
108*57718be8SEnji Cooper }
ATF_TC_BODY(swapcontext1,tc)109*57718be8SEnji Cooper ATF_TC_BODY(swapcontext1, tc)
110*57718be8SEnji Cooper {
111*57718be8SEnji Cooper 	alter_tlsbase = 0;
112*57718be8SEnji Cooper 	mainfunc();
113*57718be8SEnji Cooper }
114*57718be8SEnji Cooper 
115*57718be8SEnji Cooper ATF_TC(swapcontext2);
ATF_TC_HEAD(swapcontext2,tc)116*57718be8SEnji Cooper ATF_TC_HEAD(swapcontext2, tc)
117*57718be8SEnji Cooper {
118*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Testing if swapcontext() can "
119*57718be8SEnji Cooper 	    "modify TLS pointer");
120*57718be8SEnji Cooper }
ATF_TC_BODY(swapcontext2,tc)121*57718be8SEnji Cooper ATF_TC_BODY(swapcontext2, tc)
122*57718be8SEnji Cooper {
123*57718be8SEnji Cooper 	alter_tlsbase = 1;
124*57718be8SEnji Cooper 	mainfunc();
125*57718be8SEnji Cooper }
126*57718be8SEnji Cooper 
ATF_TP_ADD_TCS(tp)127*57718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
128*57718be8SEnji Cooper {
129*57718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, swapcontext1);
130*57718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, swapcontext2);
131*57718be8SEnji Cooper 
132*57718be8SEnji Cooper 	return atf_no_error();
133*57718be8SEnji Cooper }
134