summaryrefslogtreecommitdiffstats
path: root/testsuites/psxtests/psxonce01/init.c
blob: 3b50c945ab645c72f164556c2cebc9919c5cfb71 (plain) (blame)
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 );
}