| 
									
										
										
										
											2016-05-18 02:20:24 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |  * Copyright 2016-2017 The OpenSSL Project Authors. All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-12-06 20:05:25 +08:00
										 |  |  |  * Licensed under the Apache License 2.0 (the "License").  You may not use | 
					
						
							| 
									
										
										
										
											2016-05-18 02:20:24 +08:00
										 |  |  |  * this file except in compliance with the License.  You can obtain a copy | 
					
						
							|  |  |  |  * in the file LICENSE in the source distribution or at | 
					
						
							|  |  |  |  * https://www.openssl.org/source/license.html
 | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-26 19:40:15 +08:00
										 |  |  | #if defined(_WIN32)
 | 
					
						
							|  |  |  | # include <windows.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  | #include <openssl/crypto.h>
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  | #include "testutil.h"
 | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if !defined(OPENSSL_THREADS) || defined(CRYPTO_TDEBUG)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef unsigned int thread_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int run_thread(thread_t *t, void (*f)(void)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     f(); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wait_for_thread(thread_t thread) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #elif defined(OPENSSL_SYS_WINDOWS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef HANDLE thread_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static DWORD WINAPI thread_run(LPVOID arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void (*f)(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *(void **) (&f) = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     f(); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int run_thread(thread_t *t, void (*f)(void)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     *t = CreateThread(NULL, 0, thread_run, *(void **) &f, 0, NULL); | 
					
						
							|  |  |  |     return *t != NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wait_for_thread(thread_t thread) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return WaitForSingleObject(thread, INFINITE) == 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef pthread_t thread_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void *thread_run(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void (*f)(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *(void **) (&f) = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     f(); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int run_thread(thread_t *t, void (*f)(void)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return pthread_create(t, NULL, thread_run, *(void **) &f) == 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int wait_for_thread(thread_t thread) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return pthread_join(thread, NULL) == 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int test_lock(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CRYPTO_RWLOCK *lock = CRYPTO_THREAD_lock_new(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_true(CRYPTO_THREAD_read_lock(lock)) | 
					
						
							|  |  |  |         || !TEST_true(CRYPTO_THREAD_unlock(lock))) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CRYPTO_THREAD_lock_free(lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CRYPTO_ONCE once_run = CRYPTO_ONCE_STATIC_INIT; | 
					
						
							|  |  |  | static unsigned once_run_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void once_do_run(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     once_run_count++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void once_run_thread_cb(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CRYPTO_THREAD_run_once(&once_run, once_do_run); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int test_once(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     thread_t thread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_true(run_thread(&thread, once_run_thread_cb)) | 
					
						
							|  |  |  |         || !TEST_true(wait_for_thread(thread)) | 
					
						
							|  |  |  |         || !CRYPTO_THREAD_run_once(&once_run, once_do_run) | 
					
						
							|  |  |  |         || !TEST_int_eq(once_run_count, 1)) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CRYPTO_THREAD_LOCAL thread_local_key; | 
					
						
							|  |  |  | static unsigned destructor_run_count = 0; | 
					
						
							|  |  |  | static int thread_local_thread_cb_ok = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void thread_local_destructor(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned *count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (arg == NULL) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     count = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     (*count)++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void thread_local_thread_cb(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void *ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ptr = CRYPTO_THREAD_get_local(&thread_local_key); | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_ptr_null(ptr) | 
					
						
							|  |  |  |         || !TEST_true(CRYPTO_THREAD_set_local(&thread_local_key, | 
					
						
							|  |  |  |                                               &destructor_run_count))) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ptr = CRYPTO_THREAD_get_local(&thread_local_key); | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_ptr_eq(ptr, &destructor_run_count)) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     thread_local_thread_cb_ok = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int test_thread_local(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     thread_t thread; | 
					
						
							|  |  |  |     void *ptr = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_true(CRYPTO_THREAD_init_local(&thread_local_key, | 
					
						
							|  |  |  |                                             thread_local_destructor))) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ptr = CRYPTO_THREAD_get_local(&thread_local_key); | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_ptr_null(ptr) | 
					
						
							|  |  |  |         || !TEST_true(run_thread(&thread, thread_local_thread_cb)) | 
					
						
							|  |  |  |         || !TEST_true(wait_for_thread(thread)) | 
					
						
							|  |  |  |         || !TEST_int_eq(thread_local_thread_cb_ok, 1)) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(OPENSSL_THREADS) && !defined(CRYPTO_TDEBUG)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ptr = CRYPTO_THREAD_get_local(&thread_local_key); | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_ptr_null(ptr)) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # if !defined(OPENSSL_SYS_WINDOWS)
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_int_eq(destructor_run_count, 1)) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | # endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     if (!TEST_true(CRYPTO_THREAD_cleanup_local(&thread_local_key))) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-18 09:48:27 +08:00
										 |  |  | int setup_tests(void) | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-12 12:24:21 +08:00
										 |  |  |     ADD_TEST(test_lock); | 
					
						
							|  |  |  |     ADD_TEST(test_once); | 
					
						
							|  |  |  |     ADD_TEST(test_thread_local); | 
					
						
							| 
									
										
										
										
											2017-07-18 09:48:27 +08:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2015-10-26 00:43:55 +08:00
										 |  |  | } |