summaryrefslogtreecommitdiff
path: root/testsuites/isvv/03_semaphores_multi/semaphores_multi.c
diff options
context:
space:
mode:
Diffstat (limited to 'testsuites/isvv/03_semaphores_multi/semaphores_multi.c')
-rw-r--r--testsuites/isvv/03_semaphores_multi/semaphores_multi.c244
1 files changed, 244 insertions, 0 deletions
diff --git a/testsuites/isvv/03_semaphores_multi/semaphores_multi.c b/testsuites/isvv/03_semaphores_multi/semaphores_multi.c
new file mode 100644
index 0000000000..b46fce2358
--- /dev/null
+++ b/testsuites/isvv/03_semaphores_multi/semaphores_multi.c
@@ -0,0 +1,244 @@
+/* SPDX-License-Identifier: BSD-2-Clause */
+
+/*
+ * Copyright (C) 2022 Critical Software SA
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <rtems.h>
+#include <rtems/bspIo.h>
+#include "../shared/utils.h"
+#include "../shared/isvv_rtems_aux.h"
+
+/**
+ *
+ * @brief Demonstrate the use of semaphores to protect access to objects.
+ *
+ * For the multi-core calculation, the Mandelbrot set is divided into tiles which are distributed
+ * between the calculation tasks; when a task completes a tile it starts on the next available tile.
+ * There is one task for each core.
+ *
+ * A binary semaphore is used to protect the variable containing the next tile information against
+ * simultaneous modification by multiple tasks.
+ */
+
+#define MAX_TLS_SIZE RTEMS_ALIGN_UP(64, RTEMS_TASK_STORAGE_ALIGNMENT)
+
+#define TASK_ATTRIBUTES RTEMS_FLOATING_POINT
+
+#define TASK_STORAGE_SIZE \
+ RTEMS_TASK_STORAGE_SIZE( \
+ MAX_TLS_SIZE + RTEMS_MINIMUM_STACK_SIZE, \
+ TASK_ATTRIBUTES)
+
+#define ITOA_STR_SIZE (8 * sizeof(int) + 1)
+
+// test specific global vars
+#define TASK_COUNT TEST_PROCESSORS
+
+rtems_event_set event_send[4] = {RTEMS_EVENT_1,
+ RTEMS_EVENT_2,
+ RTEMS_EVENT_3,
+ RTEMS_EVENT_4
+ };
+
+uint8_t count_process[64];
+
+typedef struct {
+ rtems_id mutex_id;
+ rtems_id main_task;
+ uint8_t ntiles;
+ uint8_t next_tile;
+ rtems_id task_id[TASK_COUNT];
+} test_context;
+
+typedef test_context Context;
+
+RTEMS_ALIGNED(RTEMS_TASK_STORAGE_ALIGNMENT)
+static char calc_task_storage[TASK_COUNT][TASK_STORAGE_SIZE];
+
+static void calc_task_function(rtems_task_argument arg) {
+ Context *ctx;
+
+ ctx = (Context *)arg;
+
+ uint8_t tile;
+ rtems_id local_id = TaskSelfId();
+ uint8_t task_idx = 255;
+
+ for (int i = 0; i < TASK_COUNT; i++) {
+ if (ctx->task_id[i] == local_id) {
+ task_idx = i;
+ break;
+ }
+ }
+
+ ObtainMutex(ctx->mutex_id);
+ tile = ctx->next_tile;
+ ctx->next_tile++;
+ ReleaseMutex(ctx->mutex_id);
+
+ while (tile <= ctx->ntiles) {
+ count_process[tile-1]++;
+ mandelbrot_tile(tile, ctx->ntiles);
+
+ ObtainMutex(ctx->mutex_id);
+ tile = ctx->next_tile;
+ ctx->next_tile++;
+ ReleaseMutex(ctx->mutex_id);
+ }
+
+ SendEvents(ctx->main_task, event_send[task_idx]);
+ SuspendSelf();
+}
+
+static void Init(rtems_task_argument arg) {
+ (void)arg;
+ test_context ctx;
+ uint32_t start_time, end_time, elapsed_time;
+ char ch;
+ char str[ITOA_STR_SIZE];
+ rtems_event_set received = 0;
+ rtems_event_set total_events = 0;
+ bool correctly_processed = true;
+
+ ctx.main_task = rtems_task_self();
+ ctx.mutex_id = CreateMutex(rtems_build_name('S', 'E', 'M', '1'));
+ ctx.ntiles = 64;
+ ctx.next_tile = 1;
+
+ for (uint8_t i = 0; i < ctx.ntiles; i++) {
+ count_process[i] = 0;
+ }
+
+ start_time = rtems_clock_get_ticks_since_boot();
+ rtems_task_config calc_task_config = {
+ .initial_priority = PRIO_VERY_HIGH,
+ .storage_size = TASK_STORAGE_SIZE,
+ .maximum_thread_local_storage_size =
+ MAX_TLS_SIZE,
+ .initial_modes = RTEMS_DEFAULT_MODES,
+ .attributes = TASK_ATTRIBUTES
+ };
+
+ ObtainMutex(ctx.mutex_id);
+
+ for (uint32_t i = 0; i < TASK_COUNT; i++) {
+ ch = '0' + i;
+
+ calc_task_config.name = rtems_build_name('R', 'U', 'N', ch);
+ calc_task_config.storage_area = &calc_task_storage[i][0];
+
+ ctx.task_id[i] = DoCreateTask(calc_task_config);
+ StartTask(ctx.task_id[i], calc_task_function, &ctx);
+ total_events += event_send[i];
+ }
+ ReleaseMutex(ctx.mutex_id);
+
+ while (received != total_events)
+ {
+ received |= ReceiveAllEvents(total_events);
+ }
+
+ end_time = rtems_clock_get_ticks_since_boot();
+
+ elapsed_time = end_time - start_time;
+
+ print_string("\n");
+ print_string("Multicore Elapsed Time -");
+ print_string(itoa(elapsed_time, &str[0], 10));
+ print_string("\n");
+
+ for (uint8_t i = 0; i < ctx.ntiles; i++)
+ {
+ if (count_process[i] != 1)
+ {
+ correctly_processed = false;
+ break;
+ }
+ }
+
+ if (correctly_processed)
+ {
+ print_string("Each tile only processed once: true\n");
+ }
+ else
+ {
+ print_string("Each tile only processed once: false\n");
+ }
+
+ print_test_results();
+
+ for (uint32_t i = 0; i < TASK_COUNT; i++)
+ {
+ DeleteTask(ctx.task_id[i]);
+ }
+
+ rtems_fatal(RTEMS_FATAL_SOURCE_EXIT, 0);
+}
+
+#define CONFIGURE_APPLICATION_NEEDS_CLOCK_DRIVER
+
+#define CONFIGURE_MAXIMUM_PROCESSORS 4
+
+#define CONFIGURE_MAXIMUM_SEMAPHORES 1
+
+#define CONFIGURE_MAXIMUM_BARRIERS 1
+
+#define CONFIGURE_MAXIMUM_TASKS (TEST_PROCESSORS + 1)
+
+#define CONFIGURE_SCHEDULER_EDF_SMP
+
+#define CONFIGURE_MINIMUM_TASK_STACK_SIZE \
+ RTEMS_MINIMUM_STACK_SIZE + CPU_STACK_ALIGNMENT
+
+#define CONFIGURE_EXTRA_TASK_STACKS RTEMS_MINIMUM_STACK_SIZE
+
+#define CONFIGURE_INIT_TASK_CONSTRUCT_STORAGE_SIZE 2 * TASK_STORAGE_SIZE
+
+#define CONFIGURE_MINIMUM_TASKS_WITH_USER_PROVIDED_STORAGE \
+ CONFIGURE_MAXIMUM_TASKS
+
+#define CONFIGURE_MICROSECONDS_PER_TICK 1000
+
+#define CONFIGURE_MAXIMUM_FILE_DESCRIPTORS 0
+
+#define CONFIGURE_DISABLE_NEWLIB_REENTRANCY
+
+#define CONFIGURE_APPLICATION_DISABLE_FILESYSTEM
+
+#define CONFIGURE_MAXIMUM_THREAD_LOCAL_STORAGE_SIZE MAX_TLS_SIZE
+
+#define CONFIGURE_RTEMS_INIT_TASKS_TABLE
+
+#define CONFIGURE_INIT_TASK_ATTRIBUTES (RTEMS_SYSTEM_TASK | TASK_ATTRIBUTES)
+
+#define CONFIGURE_INIT_TASK_INITIAL_MODES RTEMS_DEFAULT_MODES
+
+#define CONFIGURE_INIT
+
+#include <rtems/confdefs.h>