Queues¶
The TI-RTOS Queue module provides a thread-safe unidirectional message passing module operating in a first in, first out (FIFO) basis. Queues are commonly used to allow high priority threads to pass messages to lower priority tasks for deferred processing; therefore allowing low priority tasks to block until necessary to run.
In Figure 19. a queue is configured for unidirectional communication from task A to task B. Task A “puts” messages into the queue and task B “gets” messages from the queue.
Functional Example “””””””””””””””””” Figure 20. and Figure 21. illustrate how a queue is used to enqueue a button press message from a Hwi to a Swi to be post-processed within a task context.
With interrupts enabled, a pin interrupt can occur asynchronously within a
Hwi context. To keep interrupts as short as possible, the work
associated to the interrupt is deferred to tasks for processing. In the
some example applications, pin interrupts are abstracted
via the Board Key module. This module notifies registered functions via a
Swi callback. In this case, Application_keyChangeHandler
is the registered callback function.
Step 1 in Figure 20. shows the callback to
Application_keyChangeHandler
when a key is pressed. This event is
placed into the application’s queue for processing.
1void Application_keyChangeHandler(uint8 keys)
2{
3 Application_enqueueMsg(APP_KEY_CHANGE_EVT, keys, NULL);
4}
Step 2 in Figure 20. shows how this key press is enqueued for the application task. Here, memory is allocated so the message can be added to the queue.
1static uint8_t Application_enqueueMsg(uint8_t event, uint8_t state, uint8_t *pData)
2{
3 qEvt_t *pMsg = malloc(sizeof(qEvt_t));
4
5 // Create dynamic pointer to message.
6 if (pMsg)
7 {
8 pMsg->hdr.event = event;
9 pMsg->hdr.state = state;
10 pMsg->pData = pData;
11
12 // Enqueue the message.
13 return Queue_put(appMsgQueue,pMsg->_elem);
14 }
15
16 return (false);
17}
Step 3 in Figure 21., the application is unblocked by the posted event where it proceeds to check if messages have been placed in the queue for processing.
1// If RTOS queue is not empty, process app message
2if (events & APP_QUEUE_EVT)
3{
4 scEvt_t *pMsg;
5 while (pMsg = (scEvt_t *)Queue_get(appMsgQueue))
6 {
7 // Process message
8 Application_processAppMsg(pMsg);
9
10 // Free the space from the message
11 free(pMsg);
12 }
13}
Step 4 in Figure 21., the application takes the dequeued message and processes it.
1static void Application_processAppMsg(sbcEvt_t *pMsg)
2{
3 switch (pMsg->hdr.event)
4 {
5 case APP_KEY_CHANGE_EVT:
6 Application_handleKeys(pMsg->hdr.state);
7 break;
8 //...
9 }
10}
Step 5 in Figure 21., the application can now free the memory allocated in Step 2.