1*57718be8SEnji Cooper /* $NetBSD: t_getenv_thread.c,v 1.2 2012/03/15 02:02:23 joerg Exp $ */ 2*57718be8SEnji Cooper 3*57718be8SEnji Cooper /*- 4*57718be8SEnji Cooper * Copyright (c) 2010 The NetBSD Foundation, Inc. 5*57718be8SEnji Cooper * All rights reserved. 6*57718be8SEnji Cooper * 7*57718be8SEnji Cooper * This code is derived from software contributed to The NetBSD Foundation 8*57718be8SEnji Cooper * by Matthias Scheler. 9*57718be8SEnji Cooper * 10*57718be8SEnji Cooper * Redistribution and use in source and binary forms, with or without 11*57718be8SEnji Cooper * modification, are permitted provided that the following conditions 12*57718be8SEnji Cooper * are met: 13*57718be8SEnji Cooper * 1. Redistributions of source code must retain the above copyright 14*57718be8SEnji Cooper * notice, this list of conditions and the following disclaimer. 15*57718be8SEnji Cooper * 2. Redistributions in binary form must reproduce the above copyright 16*57718be8SEnji Cooper * notice, this list of conditions and the following disclaimer in the 17*57718be8SEnji Cooper * documentation and/or other materials provided with the distribution. 18*57718be8SEnji Cooper * 19*57718be8SEnji Cooper * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20*57718be8SEnji Cooper * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21*57718be8SEnji Cooper * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22*57718be8SEnji Cooper * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23*57718be8SEnji Cooper * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24*57718be8SEnji Cooper * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25*57718be8SEnji Cooper * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26*57718be8SEnji Cooper * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27*57718be8SEnji Cooper * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28*57718be8SEnji Cooper * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29*57718be8SEnji Cooper * POSSIBILITY OF SUCH DAMAGE. 30*57718be8SEnji Cooper */ 31*57718be8SEnji Cooper #include <sys/cdefs.h> 32*57718be8SEnji Cooper __RCSID("$NetBSD: t_getenv_thread.c,v 1.2 2012/03/15 02:02:23 joerg Exp $"); 33*57718be8SEnji Cooper 34*57718be8SEnji Cooper #include <atf-c.h> 35*57718be8SEnji Cooper #include <errno.h> 36*57718be8SEnji Cooper #include <pthread.h> 37*57718be8SEnji Cooper #include <stdio.h> 38*57718be8SEnji Cooper #include <stdlib.h> 39*57718be8SEnji Cooper #include <string.h> 40*57718be8SEnji Cooper #include <time.h> 41*57718be8SEnji Cooper 42*57718be8SEnji Cooper #define THREADED_NUM_THREADS 8 43*57718be8SEnji Cooper #define THREADED_NUM_VARS 16 44*57718be8SEnji Cooper #define THREADED_VAR_NAME "THREADED%zu" 45*57718be8SEnji Cooper #define THREADED_RUN_TIME 10 46*57718be8SEnji Cooper 47*57718be8SEnji Cooper static void *thread_getenv_r(void *); 48*57718be8SEnji Cooper static void *thread_putenv(void *); 49*57718be8SEnji Cooper static void *thread_setenv(void *); 50*57718be8SEnji Cooper static void *thread_unsetenv(void *); 51*57718be8SEnji Cooper 52*57718be8SEnji Cooper static void * 53*57718be8SEnji Cooper thread_getenv_r(void *arg) 54*57718be8SEnji Cooper { 55*57718be8SEnji Cooper time_t endtime; 56*57718be8SEnji Cooper 57*57718be8SEnji Cooper endtime = *(time_t *)arg; 58*57718be8SEnji Cooper do { 59*57718be8SEnji Cooper size_t i; 60*57718be8SEnji Cooper char name[32], value[128]; 61*57718be8SEnji Cooper 62*57718be8SEnji Cooper i = lrand48() % THREADED_NUM_VARS; 63*57718be8SEnji Cooper (void)snprintf(name, sizeof(name), THREADED_VAR_NAME, i); 64*57718be8SEnji Cooper 65*57718be8SEnji Cooper if (getenv_r(name, value, sizeof(value)) == -1) { 66*57718be8SEnji Cooper ATF_CHECK(errno == ENOENT); 67*57718be8SEnji Cooper } 68*57718be8SEnji Cooper } while (time(NULL) < endtime); 69*57718be8SEnji Cooper 70*57718be8SEnji Cooper return NULL; 71*57718be8SEnji Cooper } 72*57718be8SEnji Cooper 73*57718be8SEnji Cooper 74*57718be8SEnji Cooper static void * 75*57718be8SEnji Cooper thread_putenv(void *arg) 76*57718be8SEnji Cooper { 77*57718be8SEnji Cooper time_t endtime; 78*57718be8SEnji Cooper size_t i; 79*57718be8SEnji Cooper static char vars[THREADED_NUM_VARS][128]; 80*57718be8SEnji Cooper 81*57718be8SEnji Cooper for (i = 0; i < THREADED_NUM_VARS; i++) { 82*57718be8SEnji Cooper (void)snprintf(vars[i], sizeof(vars[i]), 83*57718be8SEnji Cooper THREADED_VAR_NAME "=putenv %ld", i, lrand48()); 84*57718be8SEnji Cooper } 85*57718be8SEnji Cooper 86*57718be8SEnji Cooper endtime = *(time_t *)arg; 87*57718be8SEnji Cooper do { 88*57718be8SEnji Cooper char name[128]; 89*57718be8SEnji Cooper 90*57718be8SEnji Cooper i = lrand48() % THREADED_NUM_VARS; 91*57718be8SEnji Cooper (void)strlcpy(name, vars[i], sizeof(name)); 92*57718be8SEnji Cooper *strchr(name, '=') = '\0'; 93*57718be8SEnji Cooper 94*57718be8SEnji Cooper ATF_CHECK(unsetenv(name) != -1); 95*57718be8SEnji Cooper ATF_CHECK(putenv(vars[i]) != -1); 96*57718be8SEnji Cooper } while (time(NULL) < endtime); 97*57718be8SEnji Cooper 98*57718be8SEnji Cooper return NULL; 99*57718be8SEnji Cooper } 100*57718be8SEnji Cooper 101*57718be8SEnji Cooper static void * 102*57718be8SEnji Cooper thread_setenv(void *arg) 103*57718be8SEnji Cooper { 104*57718be8SEnji Cooper time_t endtime; 105*57718be8SEnji Cooper 106*57718be8SEnji Cooper endtime = *(time_t *)arg; 107*57718be8SEnji Cooper do { 108*57718be8SEnji Cooper size_t i; 109*57718be8SEnji Cooper char name[32], value[64]; 110*57718be8SEnji Cooper 111*57718be8SEnji Cooper i = lrand48() % THREADED_NUM_VARS; 112*57718be8SEnji Cooper (void)snprintf(name, sizeof(name), THREADED_VAR_NAME, i); 113*57718be8SEnji Cooper (void)snprintf(value, sizeof(value), "setenv %ld", lrand48()); 114*57718be8SEnji Cooper 115*57718be8SEnji Cooper ATF_CHECK(setenv(name, value, 1) != -1); 116*57718be8SEnji Cooper } while (time(NULL) < endtime); 117*57718be8SEnji Cooper 118*57718be8SEnji Cooper return NULL; 119*57718be8SEnji Cooper } 120*57718be8SEnji Cooper 121*57718be8SEnji Cooper static void * 122*57718be8SEnji Cooper thread_unsetenv(void *arg) 123*57718be8SEnji Cooper { 124*57718be8SEnji Cooper time_t endtime; 125*57718be8SEnji Cooper 126*57718be8SEnji Cooper endtime = *(time_t *)arg; 127*57718be8SEnji Cooper do { 128*57718be8SEnji Cooper size_t i; 129*57718be8SEnji Cooper char name[32]; 130*57718be8SEnji Cooper 131*57718be8SEnji Cooper i = lrand48() % THREADED_NUM_VARS; 132*57718be8SEnji Cooper (void)snprintf(name, sizeof(name), THREADED_VAR_NAME, i); 133*57718be8SEnji Cooper 134*57718be8SEnji Cooper ATF_CHECK(unsetenv(name) != -1); 135*57718be8SEnji Cooper } while (time(NULL) < endtime); 136*57718be8SEnji Cooper 137*57718be8SEnji Cooper return NULL; 138*57718be8SEnji Cooper } 139*57718be8SEnji Cooper 140*57718be8SEnji Cooper ATF_TC(getenv_r_thread); 141*57718be8SEnji Cooper ATF_TC_HEAD(getenv_r_thread, tc) 142*57718be8SEnji Cooper { 143*57718be8SEnji Cooper 144*57718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test getenv_r(3) with threads"); 145*57718be8SEnji Cooper atf_tc_set_md_var(tc, "timeout", "%d", THREADED_RUN_TIME + 5); 146*57718be8SEnji Cooper } 147*57718be8SEnji Cooper 148*57718be8SEnji Cooper ATF_TC_BODY(getenv_r_thread, tc) 149*57718be8SEnji Cooper { 150*57718be8SEnji Cooper pthread_t threads[THREADED_NUM_THREADS]; 151*57718be8SEnji Cooper time_t endtime; 152*57718be8SEnji Cooper size_t i, j; 153*57718be8SEnji Cooper 154*57718be8SEnji Cooper endtime = time(NULL) + THREADED_RUN_TIME; 155*57718be8SEnji Cooper 156*57718be8SEnji Cooper for (i = j = 0; j < 2; j++) { 157*57718be8SEnji Cooper 158*57718be8SEnji Cooper ATF_CHECK(pthread_create(&threads[i++], NULL, thread_getenv_r, 159*57718be8SEnji Cooper &endtime) == 0); 160*57718be8SEnji Cooper } 161*57718be8SEnji Cooper 162*57718be8SEnji Cooper for (j = 0; j < i; j++) 163*57718be8SEnji Cooper ATF_CHECK(pthread_join(threads[j], NULL) == 0); 164*57718be8SEnji Cooper } 165*57718be8SEnji Cooper 166*57718be8SEnji Cooper ATF_TC(putenv_thread); 167*57718be8SEnji Cooper ATF_TC_HEAD(putenv_thread, tc) 168*57718be8SEnji Cooper { 169*57718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test concurrent access by putenv(3)"); 170*57718be8SEnji Cooper atf_tc_set_md_var(tc, "timeout", "%d", THREADED_RUN_TIME + 5); 171*57718be8SEnji Cooper } 172*57718be8SEnji Cooper 173*57718be8SEnji Cooper ATF_TC_BODY(putenv_thread, tc) 174*57718be8SEnji Cooper { 175*57718be8SEnji Cooper pthread_t threads[THREADED_NUM_THREADS]; 176*57718be8SEnji Cooper time_t endtime; 177*57718be8SEnji Cooper size_t i, j; 178*57718be8SEnji Cooper 179*57718be8SEnji Cooper endtime = time(NULL) + THREADED_RUN_TIME; 180*57718be8SEnji Cooper 181*57718be8SEnji Cooper for (i = j = 0; j < 2; j++) { 182*57718be8SEnji Cooper 183*57718be8SEnji Cooper ATF_CHECK(pthread_create(&threads[i++], NULL, thread_putenv, 184*57718be8SEnji Cooper &endtime) == 0); 185*57718be8SEnji Cooper } 186*57718be8SEnji Cooper 187*57718be8SEnji Cooper for (j = 0; j < i; j++) 188*57718be8SEnji Cooper ATF_CHECK(pthread_join(threads[j], NULL) == 0); 189*57718be8SEnji Cooper } 190*57718be8SEnji Cooper 191*57718be8SEnji Cooper ATF_TC(setenv_thread); 192*57718be8SEnji Cooper ATF_TC_HEAD(setenv_thread, tc) 193*57718be8SEnji Cooper { 194*57718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test concurrent access by setenv(3)"); 195*57718be8SEnji Cooper atf_tc_set_md_var(tc, "timeout", "%d", THREADED_RUN_TIME + 5); 196*57718be8SEnji Cooper } 197*57718be8SEnji Cooper 198*57718be8SEnji Cooper ATF_TC_BODY(setenv_thread, tc) 199*57718be8SEnji Cooper { 200*57718be8SEnji Cooper pthread_t threads[THREADED_NUM_THREADS]; 201*57718be8SEnji Cooper time_t endtime; 202*57718be8SEnji Cooper size_t i, j; 203*57718be8SEnji Cooper 204*57718be8SEnji Cooper endtime = time(NULL) + THREADED_RUN_TIME; 205*57718be8SEnji Cooper 206*57718be8SEnji Cooper for (i = j = 0; j < 2; j++) { 207*57718be8SEnji Cooper 208*57718be8SEnji Cooper ATF_CHECK(pthread_create(&threads[i++], NULL, thread_setenv, 209*57718be8SEnji Cooper &endtime) == 0); 210*57718be8SEnji Cooper } 211*57718be8SEnji Cooper 212*57718be8SEnji Cooper for (j = 0; j < i; j++) 213*57718be8SEnji Cooper ATF_CHECK(pthread_join(threads[j], NULL) == 0); 214*57718be8SEnji Cooper } 215*57718be8SEnji Cooper 216*57718be8SEnji Cooper ATF_TC(unsetenv_thread); 217*57718be8SEnji Cooper ATF_TC_HEAD(unsetenv_thread, tc) 218*57718be8SEnji Cooper { 219*57718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test unsetenv(3) with threads"); 220*57718be8SEnji Cooper atf_tc_set_md_var(tc, "timeout", "%d", THREADED_RUN_TIME + 5); 221*57718be8SEnji Cooper } 222*57718be8SEnji Cooper 223*57718be8SEnji Cooper ATF_TC_BODY(unsetenv_thread, tc) 224*57718be8SEnji Cooper { 225*57718be8SEnji Cooper pthread_t threads[THREADED_NUM_THREADS]; 226*57718be8SEnji Cooper time_t endtime; 227*57718be8SEnji Cooper size_t i, j; 228*57718be8SEnji Cooper 229*57718be8SEnji Cooper endtime = time(NULL) + THREADED_RUN_TIME; 230*57718be8SEnji Cooper 231*57718be8SEnji Cooper for (i = j = 0; j < 2; j++) { 232*57718be8SEnji Cooper 233*57718be8SEnji Cooper ATF_CHECK(pthread_create(&threads[i++], NULL, thread_unsetenv, 234*57718be8SEnji Cooper &endtime) == 0); 235*57718be8SEnji Cooper } 236*57718be8SEnji Cooper 237*57718be8SEnji Cooper for (j = 0; j < i; j++) 238*57718be8SEnji Cooper ATF_CHECK(pthread_join(threads[j], NULL) == 0); 239*57718be8SEnji Cooper } 240*57718be8SEnji Cooper 241*57718be8SEnji Cooper ATF_TP_ADD_TCS(tp) 242*57718be8SEnji Cooper { 243*57718be8SEnji Cooper 244*57718be8SEnji Cooper ATF_TP_ADD_TC(tp, getenv_r_thread); 245*57718be8SEnji Cooper ATF_TP_ADD_TC(tp, putenv_thread); 246*57718be8SEnji Cooper ATF_TP_ADD_TC(tp, setenv_thread); 247*57718be8SEnji Cooper ATF_TP_ADD_TC(tp, unsetenv_thread); 248*57718be8SEnji Cooper 249*57718be8SEnji Cooper return atf_no_error(); 250*57718be8SEnji Cooper } 251