1 /*
2  * Copyright 2016-2017 The OpenSSL Project Authors. All Rights Reserved.
3  *
4  * Licensed under the OpenSSL license (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 
run_thread(thread_t * t,void (* f)(void))21 static int run_thread(thread_t *t, void (*f)(void))
22 {
23     f();
24     return 1;
25 }
26 
wait_for_thread(thread_t thread)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 
thread_run(LPVOID arg)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 
run_thread(thread_t * t,void (* f)(void))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 
wait_for_thread(thread_t thread)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 
thread_run(void * arg)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 
run_thread(thread_t * t,void (* f)(void))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 
wait_for_thread(thread_t thread)76 static int wait_for_thread(thread_t thread)
77 {
78     return pthread_join(thread, NULL) == 0;
79 }
80 
81 #endif
82 
test_lock(void)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 
once_do_run(void)99 static void once_do_run(void)
100 {
101     once_run_count++;
102 }
103 
once_run_thread_cb(void)104 static void once_run_thread_cb(void)
105 {
106     CRYPTO_THREAD_run_once(&once_run, once_do_run);
107 }
108 
test_once(void)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 
thread_local_destructor(void * arg)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 
thread_local_thread_cb(void)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 
test_thread_local(void)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 
setup_tests(void)187 int setup_tests(void)
188 {
189     ADD_TEST(test_lock);
190     ADD_TEST(test_once);
191     ADD_TEST(test_thread_local);
192     return 1;
193 }
194