Thread Synchronization¶
The TI-RTOS7 kernel provides several modules for synchronizing tasks such as Semaphore, Event, and Queue. The following sections discuss these common TI-RTOS7 primitives.
Semaphores¶
Semaphores are commonly used for task synchronization and mutual exclusions
throughout TI-RTOS7 applications. Figure 14. shows the semaphore
functionality. Semaphores can be counting semaphores or binary semaphores.
Counting semaphores keep track of the number of times the semaphore is posted
with Semaphore_post()
. When a group of resources are shared between tasks,
this function is useful. Such tasks might call Semaphore_pend()
to see if a
resource is available before using one. Binary semaphores can have only two
states: available (count = 1) and unavailable (count = 0). Binary semaphores
can be used to share a single resource between tasks or for a basic-signaling
mechanism where the semaphore can be posted multiple times. Binary semaphores
do not keep track of the count; they track only whether the semaphore has been
posted.
Initializing a Semaphore¶
The following code depicts how a semaphore is initialized in TI-RTOS7. Semaphores can be created and constructed as explained in Creating vs. Constructing.
See Listing 1. on how to create a Semaphore.
See Listing 2. on how to construct a Semaphore.
Pending on a Semaphore¶
Semaphore_pend()
is a blocking function call. This call may only be called
from within a Task context. A task calling this function will allow lower
priority tasks to execute, if they are ready to run. A task calling
Semaphore_pend()
will block if its counter is 0, otherwise it will
decrement the counter by one. The task will remain blocked until another thread
calls Semaphore_post()
or if the supplied system tick timeout has occurred;
whichever comes first. By reading the return value of Semaphore_pend()
it
is possible to distinguish if a semaphore was posted or if it timed out.
1bool isSuccessful;
2uint32_t timeout = 1000 * (1000/Clock_tickPeriod);
3
4/* Pend (approximately) up to 1 second */
5isSuccessful = Semaphore_pend(sem, timeoutInTicks);
6
7if (isSuccessful)
8{
9 System_printf("Semaphore was posted");
10}
11else
12{
13 System_printf("Semaphore timed out");
14}
The following example shows the minimum resolution of the TI-RTOS clock tick period as configured via SysConfig in the TI-RTOS configuration panel.
Note
The default TI-RTOS kernel tick period is 1 millisecond. For CC13xx or CC26xx
devices, this is reconfigured by SysConfig inside TI-RTOS
→
Core Kernel
→ Clock
, where
Clock Tick Period in microseconds
is set to 10.
Posting a Semaphore¶
Posting a semaphore is accomplished via a call to Semaphore_post()
. A task
that is pending on a posted semaphore will transition from a blocked state to
a ready state. If no higher priority thread is ready to run, it will allow
the previously pending task to execute. If no task is pending on the semaphore,
a call to Semaphore_post()
will increment its counter. Binary semaphores
have a maximum count of 1.
1Semaphore_post(sem);
Event¶
Semaphores themselves provide rudimentary synchronization between threads. There are cases just the Semaphore itself is enough to understand on what process needs to be triggered. Often however, a specific causes for the synchronization need to be passed across threads as well. To help accomplish this, one can utilize the TI-RTOS7 Event module.
Events are similar to Semaphores in a sense that each instance of an Event object actually contains a Semaphore. The added advantage of using Events lie in the fact that tasks can be notified of specific events in a thread-safe manner.
Initializing an Event¶
Creating and constructing Events follow the same guidelines as explained in Creating vs. Constructing. Shown in Listing 5. is an example on how to construct an Event instance.
1Event_Handle event;
2Event_Params eventParams;
3Event_Struct structEvent; /* Memory allocated at build time */
4
5Event_Params_init(&eventParams);
6Event_construct(&structEvent, 0, &eventParams);
7
8/* It's optional to store the handle */
9event = Event_handle(&structEvent);
Pending on an Event¶
Similar to Semaphore_pend()
, a Task thread would typically block on an
Event_pend()
until an event is posted via an Event_post()
or if the
specified timeout expired. Shown in Listing 6. is a
snippet of a task pending on any of the 3 sample event IDs shown below.
BIOS_WAIT_FOREVER
is used to prevent a timeout from occurring. As a result,
Event_pend()
will have one or more events posted in the returned bit-masked
value.
Each event returned from Event_pend()
has been automatically cleared within
the event instance in a thread-safe manner. Therefore, it is only necessary to
keep a local copy of posted events. For full details on how to use
Event_pend()
, see the TI-RTOS7 Kernel (SYS/BIOS) User’s Guide.
1#define START_ADVERTISING_EVT Event_Id_00
2#define START_CONN_UPDATE_EVT Event_Id_01
3#define CONN_PARAM_TIMEOUT_EVT Event_Id_02
4
5void TaskFxn(..)
6{
7 /* Local copy of events that have been posted */
8 uint32_t events;
9
10 while(1)
11 {
12 /* Wait for an event to be posted */
13 events = Event_pend(event,
14 Event_Id_NONE,
15 START_ADVERTISING_EVT |
16 START_CONN_UPDATE_EVT |
17 CONN_PARAM_TIMEOUT_EVT,
18 BIOS_WAIT_FOREVER);
19
20 if (events & START_ADVERTISING_EVT)
21 {
22 /* Process this event */
23 }
24
25 if (events & START_CONN_UPDATE_EVT)
26 {
27 /* Process this event */
28 }
29
30 if (events & CONN_PARAM_TIMEOUT_EVT)
31 {
32 /* Process this event */
33 }
34 }
35}
The following example shows the minimum resolution of the TI-RTOS clock tick period as configured via SysConfig in the TI-RTOS configuration panel.
Note
The default TI-RTOS kernel tick period is 1 millisecond. For CC13xx or CC26xx
devices, this is reconfigured by SysConfig inside TI-RTOS
→
Core Kernel
→ Clock
, where
Clock Tick Period in microseconds
is set to 10.
Posting an Event¶
Events may be posted from any TI-RTOS7 kernel contexts and is simply done by
calling Event_post()
of the Event instance and the Event ID.
Listing 7. shows how a high priority thread such as a Swi
could post a specific event.
1#define START_ADVERTISING_EVT Event_Id_00
2#define START_CONN_UPDATE_EVT Event_Id_01
3#define CONN_PARAM_TIMEOUT_EVT Event_Id_02
4
5void SwiFxn(UArg arg)
6{
7 Event_post(event, START_ADVERTISING_EVT);
8}