]> git.sur5r.net Git - freertos/blob - FreeRTOS/Demo/RL78_E2Studio_GCC/src/Common-Demo-Tasks/PollQ.c
804ec06c55f84c633e96c1eae85cb349cbc6287d
[freertos] / FreeRTOS / Demo / RL78_E2Studio_GCC / src / Common-Demo-Tasks / PollQ.c
1 /*\r
2  * FreeRTOS Kernel V10.1.0\r
3  * Copyright (C) 2018 Amazon.com, Inc. or its affiliates.  All Rights Reserved.\r
4  *\r
5  * Permission is hereby granted, free of charge, to any person obtaining a copy of\r
6  * this software and associated documentation files (the "Software"), to deal in\r
7  * the Software without restriction, including without limitation the rights to\r
8  * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\r
9  * the Software, and to permit persons to whom the Software is furnished to do so,\r
10  * subject to the following conditions:\r
11  *\r
12  * The above copyright notice and this permission notice shall be included in all\r
13  * copies or substantial portions of the Software.\r
14  *\r
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\r
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\r
17  * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\r
18  * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\r
19  * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\r
20  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\r
21  *\r
22  * http://www.FreeRTOS.org\r
23  * http://aws.amazon.com/freertos\r
24  *\r
25  * 1 tab == 4 spaces!\r
26  */\r
27 \r
28 /*\r
29  * This version of PollQ. c is for use on systems that have limited stack\r
30  * space and no display facilities.  The complete version can be found in\r
31  * the Demo/Common/Full directory.\r
32  *\r
33  * Creates two tasks that communicate over a single queue.  One task acts as a\r
34  * producer, the other a consumer.\r
35  *\r
36  * The producer loops for three iteration, posting an incrementing number onto the\r
37  * queue each cycle.  It then delays for a fixed period before doing exactly the\r
38  * same again.\r
39  *\r
40  * The consumer loops emptying the queue.  Each item removed from the queue is\r
41  * checked to ensure it contains the expected value.  When the queue is empty it\r
42  * blocks for a fixed period, then does the same again.\r
43  *\r
44  * All queue access is performed without blocking.  The consumer completely empties\r
45  * the queue each time it runs so the producer should never find the queue full.\r
46  *\r
47  * An error is flagged if the consumer obtains an unexpected value or the producer\r
48  * find the queue is full.\r
49  */\r
50 \r
51 /*\r
52 Changes from V2.0.0\r
53 \r
54         + Delay periods are now specified using variables and constants of\r
55           TickType_t rather than unsigned long.\r
56 */\r
57 \r
58 #include <stdlib.h>\r
59 \r
60 /* Scheduler include files. */\r
61 #include "FreeRTOS.h"\r
62 #include "task.h"\r
63 #include "queue.h"\r
64 \r
65 /* Demo program include files. */\r
66 #include "PollQ.h"\r
67 \r
68 #define pollqSTACK_SIZE                 configMINIMAL_STACK_SIZE\r
69 #define pollqQUEUE_SIZE                 ( 10 )\r
70 #define pollqPRODUCER_DELAY             ( ( TickType_t ) 200 / portTICK_PERIOD_MS )\r
71 #define pollqCONSUMER_DELAY             ( pollqPRODUCER_DELAY - ( TickType_t ) ( 20 / portTICK_PERIOD_MS ) )\r
72 #define pollqNO_DELAY                   ( ( TickType_t ) 0 )\r
73 #define pollqVALUES_TO_PRODUCE  ( ( signed portBASE_TYPE ) 3 )\r
74 #define pollqINITIAL_VALUE              ( ( signed portBASE_TYPE ) 0 )\r
75 \r
76 /* The task that posts the incrementing number onto the queue. */\r
77 static portTASK_FUNCTION_PROTO( vPolledQueueProducer, pvParameters );\r
78 \r
79 /* The task that empties the queue. */\r
80 static portTASK_FUNCTION_PROTO( vPolledQueueConsumer, pvParameters );\r
81 \r
82 /* Variables that are used to check that the tasks are still running with no\r
83 errors. */\r
84 static volatile signed portBASE_TYPE xPollingConsumerCount = pollqINITIAL_VALUE, xPollingProducerCount = pollqINITIAL_VALUE;\r
85 \r
86 /*-----------------------------------------------------------*/\r
87 \r
88 void vStartPolledQueueTasks( unsigned portBASE_TYPE uxPriority )\r
89 {\r
90 static QueueHandle_t xPolledQueue;\r
91 \r
92         /* Create the queue used by the producer and consumer. */\r
93         xPolledQueue = xQueueCreate( pollqQUEUE_SIZE, ( unsigned portBASE_TYPE ) sizeof( unsigned short ) );\r
94 \r
95         /* vQueueAddToRegistry() adds the queue to the queue registry, if one is\r
96         in use.  The queue registry is provided as a means for kernel aware\r
97         debuggers to locate queues and has no purpose if a kernel aware debugger\r
98         is not being used.  The call to vQueueAddToRegistry() will be removed\r
99         by the pre-processor if configQUEUE_REGISTRY_SIZE is not defined or is\r
100         defined to be less than 1. */\r
101         vQueueAddToRegistry( xPolledQueue, ( signed char * ) "Poll_Test_Queue" );\r
102 \r
103         /* Spawn the producer and consumer. */\r
104         xTaskCreate( vPolledQueueConsumer, "QConsNB", pollqSTACK_SIZE, ( void * ) &xPolledQueue, uxPriority, ( TaskHandle_t * ) NULL );\r
105         xTaskCreate( vPolledQueueProducer, "QProdNB", pollqSTACK_SIZE, ( void * ) &xPolledQueue, uxPriority, ( TaskHandle_t * ) NULL );\r
106 }\r
107 /*-----------------------------------------------------------*/\r
108 \r
109 static portTASK_FUNCTION( vPolledQueueProducer, pvParameters )\r
110 {\r
111 unsigned short usValue = ( unsigned short ) 0;\r
112 signed portBASE_TYPE xError = pdFALSE, xLoop;\r
113 \r
114         for( ;; )\r
115         {\r
116                 for( xLoop = 0; xLoop < pollqVALUES_TO_PRODUCE; xLoop++ )\r
117                 {\r
118                         /* Send an incrementing number on the queue without blocking. */\r
119                         if( xQueueSend( *( ( QueueHandle_t * ) pvParameters ), ( void * ) &usValue, pollqNO_DELAY ) != pdPASS )\r
120                         {\r
121                                 /* We should never find the queue full so if we get here there\r
122                                 has been an error. */\r
123                                 xError = pdTRUE;\r
124                         }\r
125                         else\r
126                         {\r
127                                 if( xError == pdFALSE )\r
128                                 {\r
129                                         /* If an error has ever been recorded we stop incrementing the\r
130                                         check variable. */\r
131                                         portENTER_CRITICAL();\r
132                                                 xPollingProducerCount++;\r
133                                         portEXIT_CRITICAL();\r
134                                 }\r
135 \r
136                                 /* Update the value we are going to post next time around. */\r
137                                 usValue++;\r
138                         }\r
139                 }\r
140 \r
141                 /* Wait before we start posting again to ensure the consumer runs and\r
142                 empties the queue. */\r
143                 vTaskDelay( pollqPRODUCER_DELAY );\r
144         }\r
145 }  /*lint !e818 Function prototype must conform to API. */\r
146 /*-----------------------------------------------------------*/\r
147 \r
148 static portTASK_FUNCTION( vPolledQueueConsumer, pvParameters )\r
149 {\r
150 unsigned short usData, usExpectedValue = ( unsigned short ) 0;\r
151 signed portBASE_TYPE xError = pdFALSE;\r
152 \r
153         for( ;; )\r
154         {\r
155                 /* Loop until the queue is empty. */\r
156                 while( uxQueueMessagesWaiting( *( ( QueueHandle_t * ) pvParameters ) ) )\r
157                 {\r
158                         if( xQueueReceive( *( ( QueueHandle_t * ) pvParameters ), &usData, pollqNO_DELAY ) == pdPASS )\r
159                         {\r
160                                 if( usData != usExpectedValue )\r
161                                 {\r
162                                         /* This is not what we expected to receive so an error has\r
163                                         occurred. */\r
164                                         xError = pdTRUE;\r
165 \r
166                                         /* Catch-up to the value we received so our next expected\r
167                                         value should again be correct. */\r
168                                         usExpectedValue = usData;\r
169                                 }\r
170                                 else\r
171                                 {\r
172                                         if( xError == pdFALSE )\r
173                                         {\r
174                                                 /* Only increment the check variable if no errors have\r
175                                                 occurred. */\r
176                                                 portENTER_CRITICAL();\r
177                                                         xPollingConsumerCount++;\r
178                                                 portEXIT_CRITICAL();\r
179                                         }\r
180                                 }\r
181 \r
182                                 /* Next time round we would expect the number to be one higher. */\r
183                                 usExpectedValue++;\r
184                         }\r
185                 }\r
186 \r
187                 /* Now the queue is empty we block, allowing the producer to place more\r
188                 items in the queue. */\r
189                 vTaskDelay( pollqCONSUMER_DELAY );\r
190         }\r
191 } /*lint !e818 Function prototype must conform to API. */\r
192 /*-----------------------------------------------------------*/\r
193 \r
194 /* This is called to check that all the created tasks are still running with no errors. */\r
195 portBASE_TYPE xArePollingQueuesStillRunning( void )\r
196 {\r
197 portBASE_TYPE xReturn;\r
198 \r
199         /* Check both the consumer and producer poll count to check they have both\r
200         been changed since out last trip round.  We do not need a critical section\r
201         around the check variables as this is called from a higher priority than\r
202         the other tasks that access the same variables. */\r
203         if( ( xPollingConsumerCount == pollqINITIAL_VALUE ) ||\r
204                 ( xPollingProducerCount == pollqINITIAL_VALUE )\r
205           )\r
206         {\r
207                 xReturn = pdFALSE;\r
208         }\r
209         else\r
210         {\r
211                 xReturn = pdTRUE;\r
212         }\r
213 \r
214         /* Set the check variables back down so we know if they have been\r
215         incremented the next time around. */\r
216         xPollingConsumerCount = pollqINITIAL_VALUE;\r
217         xPollingProducerCount = pollqINITIAL_VALUE;\r
218 \r
219         return xReturn;\r
220 }\r