/* * Copyright (c) 2012 Zhongwei Yao. * COPYRIGHT (c) 1989-2014. * On-Line Applications Research Corporation (OAR). * * The license and distribution terms for this file may be * found in the file LICENSE in this distribution or at * http://www.rtems.org/license/LICENSE. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include "tmacros.h" #include "pmacros.h" const char rtems_test_name[] = "PSXKEY 6"; static int Data_array[4] = {1, 2, 3, 4}; static pthread_key_t key1, key2, key3; static rtems_id Thread_Master; static int Key3_Destructor_Counter; static void Wake_Up_Master(void) { rtems_status_code rc; rc = rtems_event_transient_send( Thread_Master ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); } static void Wait_For_Worker(void) { rtems_status_code rc; rc = rtems_event_transient_receive( RTEMS_WAIT, RTEMS_NO_TIMEOUT ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); } static rtems_task Test_Thread1( rtems_task_argument argument ) { int sc; int *value; struct timespec delay_request; puts( "Test_Thread 1 - key1 pthread_setspecific - OK" ); sc = pthread_setspecific( key1, &Data_array[0] ); rtems_test_assert( !sc ); puts( "Test_Thread 1 - key2 pthread_setspecific - OK" ); sc = pthread_setspecific( key2, &Data_array[1] ); rtems_test_assert( !sc ); puts( "Test_Thread 1 - sleep - let thread2 run - OK" ); delay_request.tv_sec = 0; delay_request.tv_nsec = 4 * 100000000; sc = nanosleep( &delay_request, NULL ); rtems_test_assert( !sc ); puts( "Test_Thread 1 - key1 pthread_getspecific - OK" ); value = pthread_getspecific( key1 ); rtems_test_assert( *value == Data_array[0] ); puts( "Test_Thread 1 - key2 pthread_getspecific - OK" ); value = pthread_getspecific( key2 ); rtems_test_assert( *value == Data_array[1] ); rtems_task_delete( RTEMS_SELF ); } static rtems_task Test_Thread2( rtems_task_argument argument ) { int sc; int *value; puts( "Test_Thread 2 - key1 pthread_setspecific - OK" ); sc = pthread_setspecific( key1, &Data_array[2] ); rtems_test_assert( !sc ); puts( "Test_Thread 2 - key2 pthread_setspecific - OK" ); sc = pthread_setspecific( key2, &Data_array[3] ); rtems_test_assert( !sc ); puts( "Test_Thread 2 - key1 pthread_getspecific - OK" ); value = pthread_getspecific( key1 ); rtems_test_assert( *value == Data_array[2] ); puts( "Test_Thread 2 - key2 pthread_getspecific - OK" ); value = pthread_getspecific( key2 ); rtems_test_assert( *value == Data_array[3] ); rtems_task_delete( RTEMS_SELF ); } static void Key3_Destructor( void *value ) { rtems_test_assert( value == &Thread_Master ); ++Key3_Destructor_Counter; } static rtems_task Test_Thread3( rtems_task_argument argument ) { int sc; void *value; puts( "Test_Thread 3 - key3 pthread_getspecific - OK" ); value = pthread_getspecific( key3 ); rtems_test_assert( value == NULL ); puts( "Test_Thread 3 - key3 pthread_setspecific - OK" ); sc = pthread_setspecific( key3, &Thread_Master ); rtems_test_assert( sc == 0 ); if ( argument == 0 ) { puts( "Test_Thread 3 - restart self - OK" ); rtems_task_restart( RTEMS_SELF, 1 ); } else if ( argument == 1 ) { Wake_Up_Master(); rtems_task_delete( RTEMS_SELF ); } rtems_test_assert( false ); } static rtems_task Init( rtems_task_argument ignored ) { rtems_id thread1; rtems_id thread2; rtems_id thread3; rtems_status_code rc; int sc; struct timespec delay_request; TEST_BEGIN(); Thread_Master = rtems_task_self(); puts( "Init - pthread key1 create - OK" ); sc = pthread_key_create( &key1, NULL ); rtems_test_assert( !sc ); puts( "Init - pthread key2 create - OK" ); sc = pthread_key_create( &key2, NULL ); rtems_test_assert( !sc ); puts( "Init - thread1 create - OK" ); rc = rtems_task_create( rtems_build_name( 'T', 'E', 'S', 'T' ), 1, RTEMS_MINIMUM_STACK_SIZE, RTEMS_DEFAULT_MODES, RTEMS_DEFAULT_ATTRIBUTES, &thread1 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); rc = rtems_task_start( thread1, Test_Thread1, 0 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); puts( "Init - thread2 create - OK" ); rc = rtems_task_create( rtems_build_name( 'T', 'E', 'S', 'T' ), 1, RTEMS_MINIMUM_STACK_SIZE, RTEMS_DEFAULT_MODES, RTEMS_DEFAULT_ATTRIBUTES, &thread2 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); rc = rtems_task_start( thread2, Test_Thread2, 0 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); puts( "Init - sleep - let thread run - OK" ); delay_request.tv_sec = 0; delay_request.tv_nsec = 8 * 100000000; sc = nanosleep( &delay_request, NULL ); rtems_test_assert( !sc ); puts( "Init - pthread key1 delete - OK" ); sc = pthread_key_delete( key1 ); rtems_test_assert( sc == 0 ); puts( "Init - pthread key2 delete - OK" ); sc = pthread_key_delete( key2 ); rtems_test_assert( sc == 0 ); puts( "Init - pthread key3 create - OK" ); sc = pthread_key_create( &key3, Key3_Destructor ); rtems_test_assert( sc == 0 ); puts( "Init - thread3 create - OK" ); rc = rtems_task_create( rtems_build_name( 'R', 'E', 'S', 'T' ), 1, RTEMS_MINIMUM_STACK_SIZE, RTEMS_DEFAULT_MODES, RTEMS_DEFAULT_ATTRIBUTES, &thread3 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); puts( "Init - thread3 start - OK" ); rc = rtems_task_start( thread3, Test_Thread3, 0 ); rtems_test_assert( rc == RTEMS_SUCCESSFUL ); Wait_For_Worker(); rtems_test_assert( Key3_Destructor_Counter == 2 ); puts( "Init - pthread key3 delete - OK" ); sc = pthread_key_delete( key3 ); rtems_test_assert( sc == 0 ); TEST_END(); rtems_test_exit(0); } /* configuration information */ #define CONFIGURE_APPLICATION_NEEDS_SIMPLE_CONSOLE_DRIVER #define CONFIGURE_APPLICATION_NEEDS_CLOCK_DRIVER #define CONFIGURE_INITIAL_EXTENSIONS RTEMS_TEST_INITIAL_EXTENSION #define CONFIGURE_MAXIMUM_TASKS 3 #define CONFIGURE_MAXIMUM_POSIX_KEYS 2 #define CONFIGURE_RTEMS_INIT_TASKS_TABLE #define CONFIGURE_INIT #include /* global variables */