]> git.ipfire.org Git - thirdparty/openssl.git/blob - test/threadstest.c
Add a test for the new CRYPTO_atomic_* functions
[thirdparty/openssl.git] / test / threadstest.c
1 /*
2 * Copyright 2016-2017 The OpenSSL Project Authors. All Rights Reserved.
3 *
4 * Licensed under the Apache License 2.0 (the "License"). You may not use
5 * this file except in compliance with the License. You can obtain a copy
6 * in the file LICENSE in the source distribution or at
7 * https://www.openssl.org/source/license.html
8 */
9
10 #if defined(_WIN32)
11 # include <windows.h>
12 #endif
13
14 #include <openssl/crypto.h>
15 #include "testutil.h"
16
17 #if !defined(OPENSSL_THREADS) || defined(CRYPTO_TDEBUG)
18
19 typedef unsigned int thread_t;
20
21 static int run_thread(thread_t *t, void (*f)(void))
22 {
23 f();
24 return 1;
25 }
26
27 static int wait_for_thread(thread_t thread)
28 {
29 return 1;
30 }
31
32 #elif defined(OPENSSL_SYS_WINDOWS)
33
34 typedef HANDLE thread_t;
35
36 static DWORD WINAPI thread_run(LPVOID arg)
37 {
38 void (*f)(void);
39
40 *(void **) (&f) = arg;
41
42 f();
43 return 0;
44 }
45
46 static int run_thread(thread_t *t, void (*f)(void))
47 {
48 *t = CreateThread(NULL, 0, thread_run, *(void **) &f, 0, NULL);
49 return *t != NULL;
50 }
51
52 static int wait_for_thread(thread_t thread)
53 {
54 return WaitForSingleObject(thread, INFINITE) == 0;
55 }
56
57 #else
58
59 typedef pthread_t thread_t;
60
61 static void *thread_run(void *arg)
62 {
63 void (*f)(void);
64
65 *(void **) (&f) = arg;
66
67 f();
68 return NULL;
69 }
70
71 static int run_thread(thread_t *t, void (*f)(void))
72 {
73 return pthread_create(t, NULL, thread_run, *(void **) &f) == 0;
74 }
75
76 static int wait_for_thread(thread_t thread)
77 {
78 return pthread_join(thread, NULL) == 0;
79 }
80
81 #endif
82
83 static int test_lock(void)
84 {
85 CRYPTO_RWLOCK *lock = CRYPTO_THREAD_lock_new();
86
87 if (!TEST_true(CRYPTO_THREAD_read_lock(lock))
88 || !TEST_true(CRYPTO_THREAD_unlock(lock)))
89 return 0;
90
91 CRYPTO_THREAD_lock_free(lock);
92
93 return 1;
94 }
95
96 static CRYPTO_ONCE once_run = CRYPTO_ONCE_STATIC_INIT;
97 static unsigned once_run_count = 0;
98
99 static void once_do_run(void)
100 {
101 once_run_count++;
102 }
103
104 static void once_run_thread_cb(void)
105 {
106 CRYPTO_THREAD_run_once(&once_run, once_do_run);
107 }
108
109 static int test_once(void)
110 {
111 thread_t thread;
112
113 if (!TEST_true(run_thread(&thread, once_run_thread_cb))
114 || !TEST_true(wait_for_thread(thread))
115 || !CRYPTO_THREAD_run_once(&once_run, once_do_run)
116 || !TEST_int_eq(once_run_count, 1))
117 return 0;
118 return 1;
119 }
120
121 static CRYPTO_THREAD_LOCAL thread_local_key;
122 static unsigned destructor_run_count = 0;
123 static int thread_local_thread_cb_ok = 0;
124
125 static void thread_local_destructor(void *arg)
126 {
127 unsigned *count;
128
129 if (arg == NULL)
130 return;
131
132 count = arg;
133
134 (*count)++;
135 }
136
137 static void thread_local_thread_cb(void)
138 {
139 void *ptr;
140
141 ptr = CRYPTO_THREAD_get_local(&thread_local_key);
142 if (!TEST_ptr_null(ptr)
143 || !TEST_true(CRYPTO_THREAD_set_local(&thread_local_key,
144 &destructor_run_count)))
145 return;
146
147 ptr = CRYPTO_THREAD_get_local(&thread_local_key);
148 if (!TEST_ptr_eq(ptr, &destructor_run_count))
149 return;
150
151 thread_local_thread_cb_ok = 1;
152 }
153
154 static int test_thread_local(void)
155 {
156 thread_t thread;
157 void *ptr = NULL;
158
159 if (!TEST_true(CRYPTO_THREAD_init_local(&thread_local_key,
160 thread_local_destructor)))
161 return 0;
162
163 ptr = CRYPTO_THREAD_get_local(&thread_local_key);
164 if (!TEST_ptr_null(ptr)
165 || !TEST_true(run_thread(&thread, thread_local_thread_cb))
166 || !TEST_true(wait_for_thread(thread))
167 || !TEST_int_eq(thread_local_thread_cb_ok, 1))
168 return 0;
169
170 #if defined(OPENSSL_THREADS) && !defined(CRYPTO_TDEBUG)
171
172 ptr = CRYPTO_THREAD_get_local(&thread_local_key);
173 if (!TEST_ptr_null(ptr))
174 return 0;
175
176 # if !defined(OPENSSL_SYS_WINDOWS)
177 if (!TEST_int_eq(destructor_run_count, 1))
178 return 0;
179 # endif
180 #endif
181
182 if (!TEST_true(CRYPTO_THREAD_cleanup_local(&thread_local_key)))
183 return 0;
184 return 1;
185 }
186
187 static int test_atomic(void)
188 {
189 int val = 0, ret = 0, testresult = 0;
190 uint64_t val64 = 1, ret64 = 0;
191 CRYPTO_RWLOCK *lock = CRYPTO_THREAD_lock_new();
192
193 if (!TEST_ptr(lock))
194 return 0;
195
196 if (CRYPTO_atomic_add(&val, 1, &ret, NULL)) {
197 /* This succeeds therefore we're on a platform with lockless atomics */
198 if (!TEST_int_eq(val, 1) || !TEST_int_eq(val, ret))
199 goto err;
200 } else {
201 /* This failed therefore we're on a platform without lockless atomics */
202 if (!TEST_int_eq(val, 0) || !TEST_int_eq(val, ret))
203 goto err;
204 }
205 val = 0;
206 ret = 0;
207
208 if (!TEST_true(CRYPTO_atomic_add(&val, 1, &ret, lock)))
209 goto err;
210 if (!TEST_int_eq(val, 1) || !TEST_int_eq(val, ret))
211 goto err;
212
213 if (CRYPTO_atomic_or(&val64, 2, &ret64, NULL)) {
214 /* This succeeds therefore we're on a platform with lockless atomics */
215 if (!TEST_uint_eq((unsigned int)val64, 3)
216 || !TEST_uint_eq((unsigned int)val64, (unsigned int)ret64))
217 goto err;
218 } else {
219 /* This failed therefore we're on a platform without lockless atomics */
220 if (!TEST_uint_eq((unsigned int)val64, 1)
221 || !TEST_int_eq((unsigned int)ret64, 0))
222 goto err;
223 }
224 val64 = 1;
225 ret64 = 0;
226
227 if (!TEST_true(CRYPTO_atomic_or(&val64, 2, &ret64, lock)))
228 goto err;
229
230 if (!TEST_uint_eq((unsigned int)val64, 3)
231 || !TEST_uint_eq((unsigned int)val64, (unsigned int)ret64))
232 goto err;
233
234 ret64 = 0;
235 if (CRYPTO_atomic_load(&val64, &ret64, NULL)) {
236 /* This succeeds therefore we're on a platform with lockless atomics */
237 if (!TEST_uint_eq((unsigned int)val64, 3)
238 || !TEST_uint_eq((unsigned int)val64, (unsigned int)ret64))
239 goto err;
240 } else {
241 /* This failed therefore we're on a platform without lockless atomics */
242 if (!TEST_uint_eq((unsigned int)val64, 3)
243 || !TEST_int_eq((unsigned int)ret64, 0))
244 goto err;
245 }
246
247 ret64 = 0;
248 if (!TEST_true(CRYPTO_atomic_load(&val64, &ret64, lock)))
249 goto err;
250
251 if (!TEST_uint_eq((unsigned int)val64, 3)
252 || !TEST_uint_eq((unsigned int)val64, (unsigned int)ret64))
253 goto err;
254
255 testresult = 1;
256 err:
257
258 CRYPTO_THREAD_lock_free(lock);
259 return testresult;
260 }
261
262 int setup_tests(void)
263 {
264 ADD_TEST(test_lock);
265 ADD_TEST(test_once);
266 ADD_TEST(test_thread_local);
267 ADD_TEST(test_atomic);
268 return 1;
269 }