source: rtems/testsuites/smptests/smpthreadlife01/init.c @ f95fa387

5
Last change on this file since f95fa387 was f95fa387, checked in by Sebastian Huber <sebastian.huber@…>, on 02/01/17 at 11:11:33

Remove CONFIGURE_SMP_APPLICATION

Enable the SMP support if CONFIGURE_SMP_MAXIMUM_PROCESSORS > 1.

Update #2893.

  • Property mode set to 100644
File size: 11.6 KB
Line 
1/*
2 * Copyright (c) 2014 embedded brains GmbH.  All rights reserved.
3 *
4 *  embedded brains GmbH
5 *  Dornierstr. 4
6 *  82178 Puchheim
7 *  Germany
8 *  <rtems@embedded-brains.de>
9 *
10 * The license and distribution terms for this file may be
11 * found in the file LICENSE in this distribution or at
12 * http://www.rtems.org/license/LICENSE.
13 */
14
15#ifdef HAVE_CONFIG_H
16  #include "config.h"
17#endif
18
19#include "tmacros.h"
20
21#include <rtems.h>
22#include <rtems/counter.h>
23#include <rtems/libcsupport.h>
24#include <rtems/score/profiling.h>
25#include <rtems/score/smpbarrier.h>
26#include <rtems/score/threadimpl.h>
27
28const char rtems_test_name[] = "SMPTHREADLIFE 1";
29
30#define CPU_COUNT 2
31
32typedef struct {
33  volatile rtems_task_argument main_arg;
34  volatile rtems_task_argument worker_arg;
35  volatile bool terminated;
36  SMP_barrier_Control barrier;
37  SMP_barrier_State main_barrier_state;
38  SMP_barrier_State worker_barrier_state;
39  Thread_Control *delay_switch_for_executing;
40  rtems_id worker_id;
41} test_context;
42
43static test_context test_instance = {
44  .barrier = SMP_BARRIER_CONTROL_INITIALIZER,
45  .main_barrier_state = SMP_BARRIER_STATE_INITIALIZER,
46  .worker_barrier_state = SMP_BARRIER_STATE_INITIALIZER
47};
48
49static void barrier(test_context *ctx, SMP_barrier_State *state)
50{
51  _SMP_barrier_Wait(&ctx->barrier, state, CPU_COUNT);
52}
53
54static void restart_extension(
55  Thread_Control *executing,
56  Thread_Control *restarted
57)
58{
59  rtems_test_assert(executing == restarted);
60}
61
62static void delete_extension(
63  Thread_Control *executing,
64  Thread_Control *deleted
65)
66{
67  rtems_test_assert(executing != deleted);
68}
69
70static void terminate_extension(Thread_Control *executing)
71{
72  test_context *ctx = &test_instance;
73
74  ctx->terminated = true;
75}
76
77static void switch_extension(Thread_Control *executing, Thread_Control *heir)
78{
79  test_context *ctx = &test_instance;
80
81  if (ctx->delay_switch_for_executing == executing) {
82    ctx->delay_switch_for_executing = NULL;
83
84    /* (A) */
85    barrier(ctx, &ctx->worker_barrier_state);
86
87    rtems_counter_delay_nanoseconds(100000000);
88
89    /* Avoid bad profiling statisitics */
90    _Profiling_Thread_dispatch_disable( _Per_CPU_Get(), 0 );
91  }
92}
93
94static void worker_task(rtems_task_argument arg)
95{
96  test_context *ctx = &test_instance;
97
98  rtems_test_assert(arg == ctx->main_arg);
99
100  ctx->worker_arg = arg;
101
102  /* (B) */
103  barrier(ctx, &ctx->worker_barrier_state);
104
105  while (true) {
106    /* Do nothing */
107  }
108}
109
110static void test_restart(void)
111{
112  test_context *ctx = &test_instance;
113  rtems_status_code sc;
114  rtems_id id;
115  rtems_task_argument arg;
116  rtems_resource_snapshot snapshot;
117
118  rtems_resource_snapshot_take(&snapshot);
119
120  sc = rtems_task_create(
121    rtems_build_name('W', 'O', 'R', 'K'),
122    1,
123    RTEMS_MINIMUM_STACK_SIZE,
124    RTEMS_DEFAULT_MODES,
125    RTEMS_DEFAULT_ATTRIBUTES,
126    &id
127  );
128  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
129
130  sc = rtems_task_start(id, worker_task, 0);
131  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
132
133  /* (B) */
134  barrier(ctx, &ctx->main_barrier_state);
135
136  for (arg = 1; arg < 23; ++arg) {
137    ctx->main_arg = arg;
138    ctx->worker_arg = 0;
139
140    sc = rtems_task_restart(id, arg);
141    rtems_test_assert(sc == RTEMS_SUCCESSFUL);
142
143    /* (B) */
144    barrier(ctx, &ctx->main_barrier_state);
145
146    rtems_test_assert(ctx->worker_arg == arg);
147  }
148
149  sc = rtems_task_delete(id);
150  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
151
152  rtems_test_assert(rtems_resource_snapshot_check(&snapshot));
153}
154
155static void test_delete(void)
156{
157  test_context *ctx = &test_instance;
158  rtems_status_code sc;
159  rtems_id id;
160  rtems_task_argument arg;
161  rtems_resource_snapshot snapshot;
162
163  rtems_resource_snapshot_take(&snapshot);
164
165  for (arg = 31; arg < 57; ++arg) {
166    ctx->main_arg = arg;
167    ctx->worker_arg = 0;
168    ctx->terminated = false;
169
170    sc = rtems_task_create(
171      rtems_build_name('W', 'O', 'R', 'K'),
172      1,
173      RTEMS_MINIMUM_STACK_SIZE,
174      RTEMS_DEFAULT_MODES,
175      RTEMS_DEFAULT_ATTRIBUTES,
176      &id
177    );
178    rtems_test_assert(sc == RTEMS_SUCCESSFUL);
179
180    sc = rtems_task_start(id, worker_task, arg);
181    rtems_test_assert(sc == RTEMS_SUCCESSFUL);
182
183    /* (B) */
184    barrier(ctx, &ctx->main_barrier_state);
185
186    rtems_test_assert(ctx->worker_arg == arg);
187    rtems_test_assert(!ctx->terminated);
188
189    sc = rtems_task_delete(id);
190    rtems_test_assert(sc == RTEMS_SUCCESSFUL);
191
192    rtems_test_assert(ctx->terminated);
193
194    rtems_test_assert(rtems_resource_snapshot_check(&snapshot));
195  }
196}
197
198static void delay_ipi_task(rtems_task_argument variant)
199{
200  test_context *ctx = &test_instance;
201  ISR_Level level;
202
203  _ISR_Local_disable(level);
204
205  /* (C) */
206  barrier(ctx, &ctx->worker_barrier_state);
207
208  /*
209   * Interrupts are disabled, so the inter-processor interrupt deleting us will
210   * be delayed a bit.
211   */
212  rtems_counter_delay_nanoseconds(100000000);
213
214  if (variant != 0) {
215    _Thread_Dispatch_disable();
216  }
217
218  _ISR_Local_enable(level);
219
220  /*
221   * We get deleted as a side effect of enabling the thread life protection or
222   * later if we enable the thread dispatching.
223   */
224  _Thread_Set_life_protection( THREAD_LIFE_PROTECTED );
225
226  if (variant != 0) {
227    _Thread_Dispatch_enable( _Per_CPU_Get() );
228  }
229
230  rtems_test_assert(0);
231}
232
233static void test_set_life_protection(rtems_task_argument variant)
234{
235  test_context *ctx = &test_instance;
236  rtems_status_code sc;
237  rtems_id id;
238  rtems_resource_snapshot snapshot;
239
240  rtems_resource_snapshot_take(&snapshot);
241
242  sc = rtems_task_create(
243    rtems_build_name('D', 'E', 'L', 'Y'),
244    1,
245    RTEMS_MINIMUM_STACK_SIZE,
246    RTEMS_DEFAULT_MODES,
247    RTEMS_DEFAULT_ATTRIBUTES,
248    &id
249  );
250  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
251
252  sc = rtems_task_start(id, delay_ipi_task, variant);
253  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
254
255  /* (C) */
256  barrier(ctx, &ctx->main_barrier_state);
257
258  sc = rtems_task_delete(id);
259  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
260
261  rtems_test_assert(rtems_resource_snapshot_check(&snapshot));
262}
263
264static void delay_switch_task(rtems_task_argument arg)
265{
266  test_context *ctx = &test_instance;
267  rtems_status_code sc;
268
269  ctx->delay_switch_for_executing = _Thread_Get_executing();
270
271  /* (D) */
272  barrier(ctx, &ctx->worker_barrier_state);
273
274  sc = rtems_task_delete(RTEMS_SELF);
275  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
276}
277
278static void test_wait_for_execution_stop(void)
279{
280  test_context *ctx = &test_instance;
281  rtems_status_code sc;
282  rtems_id id;
283  rtems_resource_snapshot snapshot;
284
285  rtems_resource_snapshot_take(&snapshot);
286
287  sc = rtems_task_create(
288    rtems_build_name('S', 'W', 'I', 'T'),
289    1,
290    RTEMS_MINIMUM_STACK_SIZE,
291    RTEMS_DEFAULT_MODES,
292    RTEMS_DEFAULT_ATTRIBUTES,
293    &id
294  );
295  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
296
297  sc = rtems_task_start(id, delay_switch_task, 0);
298  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
299
300  /* (D) */
301  barrier(ctx, &ctx->main_barrier_state);
302
303  /* (A) */
304  barrier(ctx, &ctx->main_barrier_state);
305
306  sc = rtems_task_create(
307    rtems_build_name('W', 'A', 'I', 'T'),
308    1,
309    RTEMS_MINIMUM_STACK_SIZE,
310    RTEMS_DEFAULT_MODES,
311    RTEMS_DEFAULT_ATTRIBUTES,
312    &id
313  );
314  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
315
316  sc = rtems_task_delete(id);
317  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
318
319  rtems_test_assert(rtems_resource_snapshot_check(&snapshot));
320}
321
322typedef enum {
323  TEST_OP_SUSPEND,
324  TEST_OP_EVENT,
325  TEST_OP_EVENT_SYSTEM
326} test_op;
327
328static void op_begin_suspend(void)
329{
330  rtems_status_code sc;
331
332  sc = rtems_task_suspend(RTEMS_SELF);
333  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
334}
335
336static void op_begin_event(void)
337{
338  rtems_status_code sc;
339  rtems_event_set events;
340
341  events = 0;
342  sc = rtems_event_receive(
343    RTEMS_EVENT_0,
344    RTEMS_EVENT_ALL | RTEMS_WAIT,
345    RTEMS_NO_TIMEOUT,
346    &events
347  );
348  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
349  rtems_test_assert(events == RTEMS_EVENT_0);
350}
351
352static void op_begin_event_system(void)
353{
354  rtems_status_code sc;
355  rtems_event_set events;
356
357  events = 0;
358  sc = rtems_event_system_receive(
359    RTEMS_EVENT_0,
360    RTEMS_EVENT_ALL | RTEMS_WAIT,
361    RTEMS_NO_TIMEOUT,
362    &events
363  );
364  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
365  rtems_test_assert(events == RTEMS_EVENT_0);
366}
367
368static void (*const test_ops_begin[])(void) = {
369  op_begin_suspend,
370  op_begin_event,
371  op_begin_event_system
372};
373
374static void op_end_suspend(test_context *ctx)
375{
376  rtems_status_code sc;
377
378  sc = rtems_task_resume(ctx->worker_id);
379  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
380}
381
382static void op_end_event(test_context *ctx)
383{
384  rtems_status_code sc;
385
386  sc = rtems_event_send(ctx->worker_id, RTEMS_EVENT_0);
387  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
388}
389
390static void op_end_event_system(test_context *ctx)
391{
392  rtems_status_code sc;
393
394  sc = rtems_event_system_send(ctx->worker_id, RTEMS_EVENT_0);
395  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
396}
397
398static void (*const test_ops_end[])(test_context *) = {
399  op_end_suspend,
400  op_end_event,
401  op_end_event_system
402};
403
404static void op_worker_task(rtems_task_argument arg)
405{
406  test_context *ctx = &test_instance;
407  test_op op = arg;
408  Thread_Life_state previous_life_state;
409
410  previous_life_state = _Thread_Set_life_protection(THREAD_LIFE_PROTECTED);
411
412  /* (E) */
413  barrier(ctx, &ctx->worker_barrier_state);
414
415  /* (F) */
416  barrier(ctx, &ctx->worker_barrier_state);
417
418  (*test_ops_begin[op])();
419
420  _Thread_Set_life_protection(previous_life_state);
421  rtems_test_assert(0);
422}
423
424static void help_task(rtems_task_argument arg)
425{
426  test_context *ctx = &test_instance;
427  test_op op = arg;
428
429  /* (F) */
430  barrier(ctx, &ctx->main_barrier_state);
431
432  rtems_counter_delay_nanoseconds(100000000);
433
434  (*test_ops_end[op])(ctx);
435
436  rtems_task_suspend(RTEMS_SELF);
437  rtems_test_assert(0);
438}
439
440static void test_operation_with_delete_in_progress(test_op op)
441{
442  test_context *ctx = &test_instance;
443  rtems_status_code sc;
444  rtems_id help_id;
445  rtems_resource_snapshot snapshot;
446
447  rtems_resource_snapshot_take(&snapshot);
448
449  sc = rtems_task_create(
450    rtems_build_name('W', 'O', 'R', 'K'),
451    1,
452    RTEMS_MINIMUM_STACK_SIZE,
453    RTEMS_DEFAULT_MODES,
454    RTEMS_DEFAULT_ATTRIBUTES,
455    &ctx->worker_id
456  );
457  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
458
459  sc = rtems_task_start(ctx->worker_id, op_worker_task, op);
460  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
461
462  /* (E) */
463  barrier(ctx, &ctx->main_barrier_state);
464
465  sc = rtems_task_create(
466    rtems_build_name('H', 'E', 'L', 'P'),
467    2,
468    RTEMS_MINIMUM_STACK_SIZE,
469    RTEMS_DEFAULT_MODES,
470    RTEMS_DEFAULT_ATTRIBUTES,
471    &help_id
472  );
473  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
474
475  sc = rtems_task_start(help_id, help_task, op);
476  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
477
478  sc = rtems_task_delete(ctx->worker_id);
479  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
480
481  sc = rtems_task_delete(help_id);
482  rtems_test_assert(sc == RTEMS_SUCCESSFUL);
483
484  rtems_test_assert(rtems_resource_snapshot_check(&snapshot));
485}
486
487static void Init(rtems_task_argument arg)
488{
489  TEST_BEGIN();
490
491  if (rtems_get_processor_count() >= CPU_COUNT) {
492    test_restart();
493    test_delete();
494    test_set_life_protection(0);
495    test_set_life_protection(1);
496    test_wait_for_execution_stop();
497    test_operation_with_delete_in_progress(TEST_OP_SUSPEND);
498    test_operation_with_delete_in_progress(TEST_OP_EVENT);
499    test_operation_with_delete_in_progress(TEST_OP_EVENT_SYSTEM);
500  }
501
502  TEST_END();
503  rtems_test_exit(0);
504}
505
506#define CONFIGURE_APPLICATION_DOES_NOT_NEED_CLOCK_DRIVER
507#define CONFIGURE_APPLICATION_NEEDS_CONSOLE_DRIVER
508
509#define CONFIGURE_SMP_MAXIMUM_PROCESSORS CPU_COUNT
510
511#define CONFIGURE_MAXIMUM_TASKS (CPU_COUNT + 1)
512
513#define CONFIGURE_INITIAL_EXTENSIONS \
514  { \
515    .thread_restart = restart_extension, \
516    .thread_delete = delete_extension, \
517    .thread_terminate = terminate_extension, \
518    .thread_switch = switch_extension \
519  }, \
520  RTEMS_TEST_INITIAL_EXTENSION
521
522#define CONFIGURE_RTEMS_INIT_TASKS_TABLE
523
524#define CONFIGURE_INIT
525
526#include <rtems/confdefs.h>
Note: See TracBrowser for help on using the repository browser.