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