1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
|
/*
* Copyright (C) 2019 embedded brains GmbH
* Copyright (C) 2019 Sebastian Huber
*
* COPYRIGHT (c) 1989-2009.
* 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
#define CONFIGURE_INIT
#include "system.h"
const char rtems_test_name[] = "PSXONCE 1";
static pthread_once_t once_a = PTHREAD_ONCE_INIT;
static pthread_once_t once_b = PTHREAD_ONCE_INIT;
static rtems_id master;
static int test_init_routine_call_counter = 0;
static void Test_init_routine( void )
{
puts( "Test_init_routine: invoked" );
++test_init_routine_call_counter;
}
static void routine_b( void )
{
rtems_status_code sc;
rtems_test_assert( test_init_routine_call_counter == 2 );
++test_init_routine_call_counter;
sc = rtems_event_send( master, RTEMS_EVENT_0 );
rtems_test_assert( sc == RTEMS_SUCCESSFUL );
}
static void use_b( rtems_task_argument arg )
{
int status;
(void) arg;
status = pthread_once( &once_b, routine_b );
rtems_test_assert( status == 0 );
rtems_task_exit();
}
static void routine_a( void )
{
rtems_status_code sc;
rtems_id id;
rtems_event_set events;
rtems_test_assert( test_init_routine_call_counter == 1 );
++test_init_routine_call_counter;
master = rtems_task_self();
sc = rtems_task_create(
rtems_build_name( 'T', 'A', 'S', 'K' ),
RTEMS_MINIMUM_PRIORITY,
RTEMS_MINIMUM_STACK_SIZE,
RTEMS_DEFAULT_MODES,
RTEMS_DEFAULT_ATTRIBUTES,
&id
);
rtems_test_assert( sc == RTEMS_SUCCESSFUL );
sc = rtems_task_start( id, use_b, 0 );
rtems_test_assert( sc == RTEMS_SUCCESSFUL );
events = 0;
sc = rtems_event_receive(
RTEMS_EVENT_0,
RTEMS_EVENT_ANY | RTEMS_WAIT,
RTEMS_NO_TIMEOUT,
&events
);
rtems_test_assert( sc == RTEMS_SUCCESSFUL );
rtems_test_assert( events == RTEMS_EVENT_0 );
rtems_test_assert( test_init_routine_call_counter == 3 );
}
rtems_task Init(rtems_task_argument argument)
{
int status;
pthread_once_t once = PTHREAD_ONCE_INIT;
TEST_BEGIN();
puts( "Init: pthread_once - EINVAL (NULL once_control)" );
status = pthread_once( NULL, Test_init_routine );
rtems_test_assert( status == EINVAL );
puts( "Init: pthread_once - EINVAL (NULL init_routine)" );
status = pthread_once( &once, NULL );
rtems_test_assert( status == EINVAL );
puts( "Init: pthread_once - SUCCESSFUL (init_routine executes)" );
status = pthread_once( &once, Test_init_routine );
rtems_test_assert( !status );
printf( "Init: call counter: %d\n", test_init_routine_call_counter );
rtems_test_assert( test_init_routine_call_counter == 1 );
puts( "Init: pthread_once - SUCCESSFUL (init_routine does not execute)" );
status = pthread_once( &once, Test_init_routine );
rtems_test_assert( !status );
printf( "Init: call counter: %d\n", test_init_routine_call_counter );
rtems_test_assert( test_init_routine_call_counter == 1 );
status = pthread_once( &once_a, routine_a );
rtems_test_assert( status == 0 );
rtems_test_assert( test_init_routine_call_counter == 3 );
TEST_END();
rtems_test_exit( 0 );
}
|