xref: /freebsd-src/contrib/netbsd-tests/lib/libc/stdlib/t_getenv_thread.c (revision 9268022b74279434ed6300244e3f977e56a8ceb5)
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 *
thread_getenv_r(void * arg)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 *
thread_putenv(void * arg)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 *
thread_setenv(void * arg)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 *
thread_unsetenv(void * arg)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);
ATF_TC_HEAD(getenv_r_thread,tc)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 
ATF_TC_BODY(getenv_r_thread,tc)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);
ATF_TC_HEAD(putenv_thread,tc)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 
ATF_TC_BODY(putenv_thread,tc)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);
ATF_TC_HEAD(setenv_thread,tc)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 
ATF_TC_BODY(setenv_thread,tc)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);
ATF_TC_HEAD(unsetenv_thread,tc)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 
ATF_TC_BODY(unsetenv_thread,tc)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 
ATF_TP_ADD_TCS(tp)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