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 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 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); 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 } 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); 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 } 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 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