GM6000 Digital Heater Controller Branch: main
SDX-1330
croutine.h
1/*
2 * FreeRTOS Kernel V10.0.0
3 * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a copy of
6 * this software and associated documentation files (the "Software"), to deal in
7 * the Software without restriction, including without limitation the rights to
8 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9 * the Software, and to permit persons to whom the Software is furnished to do so,
10 * subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice shall be included in all
13 * copies or substantial portions of the Software. If you wish to use our Amazon
14 * FreeRTOS name, please do so in a fair use way that does not cause confusion.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
18 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
19 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
20 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
21 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22 *
23 * http://www.FreeRTOS.org
24 * http://aws.amazon.com/freertos
25 *
26 * 1 tab == 4 spaces!
27 */
28
29#ifndef CO_ROUTINE_H
30#define CO_ROUTINE_H
31
32#ifndef INC_FREERTOS_H
33#error "include FreeRTOS.h must appear in source files before include croutine.h"
34#endif
35
36#include "list.h"
37
38#ifdef __cplusplus
39extern "C" {
40#endif
41
42/* Used to hide the implementation of the co-routine control block. The
43control block structure however has to be included in the header due to
44the macro implementation of the co-routine functionality. */
45typedef void *CoRoutineHandle_t;
46
47/* Defines the prototype to which co-routine functions must conform. */
48typedef void (*crCOROUTINE_CODE)(CoRoutineHandle_t, UBaseType_t);
49
51 crCOROUTINE_CODE pxCoRoutineFunction;
52 ListItem_t xGenericListItem; /*< List item used to place the CRCB in ready and blocked queues. */
53 ListItem_t xEventListItem; /*< List item used to place the CRCB in event lists. */
54 UBaseType_t uxPriority; /*< The priority of the co-routine in relation to other co-routines. */
55 UBaseType_t uxIndex; /*< Used to distinguish between co-routines when multiple co-routines use the same co-routine
56 function. */
57 uint16_t uxState; /*< Used internally by the co-routine implementation. */
58} CRCB_t; /* Co-routine control block. Note must be identical in size down to uxPriority with TCB_t. */
59
60/**
61 * croutine. h
62 *<pre>
63 BaseType_t xCoRoutineCreate(
64 crCOROUTINE_CODE pxCoRoutineCode,
65 UBaseType_t uxPriority,
66 UBaseType_t uxIndex
67 );</pre>
68 *
69 * Create a new co-routine and add it to the list of co-routines that are
70 * ready to run.
71 *
72 * @param pxCoRoutineCode Pointer to the co-routine function. Co-routine
73 * functions require special syntax - see the co-routine section of the WEB
74 * documentation for more information.
75 *
76 * @param uxPriority The priority with respect to other co-routines at which
77 * the co-routine will run.
78 *
79 * @param uxIndex Used to distinguish between different co-routines that
80 * execute the same function. See the example below and the co-routine section
81 * of the WEB documentation for further information.
82 *
83 * @return pdPASS if the co-routine was successfully created and added to a ready
84 * list, otherwise an error code defined with ProjDefs.h.
85 *
86 * Example usage:
87 <pre>
88 // Co-routine to be created.
89 void vFlashCoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
90 {
91 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
92 // This may not be necessary for const variables.
93 static const char cLedToFlash[ 2 ] = { 5, 6 };
94 static const TickType_t uxFlashRates[ 2 ] = { 200, 400 };
95
96 // Must start every co-routine with a call to crSTART();
97 crSTART( xHandle );
98
99 for( ;; )
100 {
101 // This co-routine just delays for a fixed period, then toggles
102 // an LED. Two co-routines are created using this function, so
103 // the uxIndex parameter is used to tell the co-routine which
104 // LED to flash and how int32_t to delay. This assumes xQueue has
105 // already been created.
106 vParTestToggleLED( cLedToFlash[ uxIndex ] );
107 crDELAY( xHandle, uxFlashRates[ uxIndex ] );
108 }
109
110 // Must end every co-routine with a call to crEND();
111 crEND();
112 }
113
114 // Function that creates two co-routines.
115 void vOtherFunction( void )
116 {
117 uint8_t ucParameterToPass;
118 TaskHandle_t xHandle;
119
120 // Create two co-routines at priority 0. The first is given index 0
121 // so (from the code above) toggles LED 5 every 200 ticks. The second
122 // is given index 1 so toggles LED 6 every 400 ticks.
123 for( uxIndex = 0; uxIndex < 2; uxIndex++ )
124 {
125 xCoRoutineCreate( vFlashCoRoutine, 0, uxIndex );
126 }
127 }
128 </pre>
129 * \defgroup xCoRoutineCreate xCoRoutineCreate
130 * \ingroup Tasks
131 */
132BaseType_t xCoRoutineCreate(crCOROUTINE_CODE pxCoRoutineCode, UBaseType_t uxPriority, UBaseType_t uxIndex);
133
134/**
135 * croutine. h
136 *<pre>
137 void vCoRoutineSchedule( void );</pre>
138 *
139 * Run a co-routine.
140 *
141 * vCoRoutineSchedule() executes the highest priority co-routine that is able
142 * to run. The co-routine will execute until it either blocks, yields or is
143 * preempted by a task. Co-routines execute cooperatively so one
144 * co-routine cannot be preempted by another, but can be preempted by a task.
145 *
146 * If an application comprises of both tasks and co-routines then
147 * vCoRoutineSchedule should be called from the idle task (in an idle task
148 * hook).
149 *
150 * Example usage:
151 <pre>
152 // This idle task hook will schedule a co-routine each time it is called.
153 // The rest of the idle task will execute between co-routine calls.
154 void vApplicationIdleHook( void )
155 {
156 vCoRoutineSchedule();
157 }
158
159 // Alternatively, if you do not require any other part of the idle task to
160 // execute, the idle task hook can call vCoRoutineScheduler() within an
161 // infinite loop.
162 void vApplicationIdleHook( void )
163 {
164 for( ;; )
165 {
166 vCoRoutineSchedule();
167 }
168 }
169 </pre>
170 * \defgroup vCoRoutineSchedule vCoRoutineSchedule
171 * \ingroup Tasks
172 */
173void vCoRoutineSchedule(void);
174
175/**
176 * croutine. h
177 * <pre>
178 crSTART( CoRoutineHandle_t xHandle );</pre>
179 *
180 * This macro MUST always be called at the start of a co-routine function.
181 *
182 * Example usage:
183 <pre>
184 // Co-routine to be created.
185 void vACoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
186 {
187 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
188 static int32_t ulAVariable;
189
190 // Must start every co-routine with a call to crSTART();
191 crSTART( xHandle );
192
193 for( ;; )
194 {
195 // Co-routine functionality goes here.
196 }
197
198 // Must end every co-routine with a call to crEND();
199 crEND();
200 }</pre>
201 * \defgroup crSTART crSTART
202 * \ingroup Tasks
203 */
204#define crSTART(pxCRCB) \
205 switch (((CRCB_t *)(pxCRCB))->uxState) { \
206 case 0:
207
208/**
209 * croutine. h
210 * <pre>
211 crEND();</pre>
212 *
213 * This macro MUST always be called at the end of a co-routine function.
214 *
215 * Example usage:
216 <pre>
217 // Co-routine to be created.
218 void vACoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
219 {
220 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
221 static int32_t ulAVariable;
222
223 // Must start every co-routine with a call to crSTART();
224 crSTART( xHandle );
225
226 for( ;; )
227 {
228 // Co-routine functionality goes here.
229 }
230
231 // Must end every co-routine with a call to crEND();
232 crEND();
233 }</pre>
234 * \defgroup crSTART crSTART
235 * \ingroup Tasks
236 */
237#define crEND() }
238
239/*
240 * These macros are intended for internal use by the co-routine implementation
241 * only. The macros should not be used directly by application writers.
242 */
243#define crSET_STATE0(xHandle) \
244 ((CRCB_t *)(xHandle))->uxState = (__LINE__ * 2); \
245 return; \
246 case (__LINE__ * 2):
247#define crSET_STATE1(xHandle) \
248 ((CRCB_t *)(xHandle))->uxState = ((__LINE__ * 2) + 1); \
249 return; \
250 case ((__LINE__ * 2) + 1):
251
252/**
253 * croutine. h
254 *<pre>
255 crDELAY( CoRoutineHandle_t xHandle, TickType_t xTicksToDelay );</pre>
256 *
257 * Delay a co-routine for a fixed period of time.
258 *
259 * crDELAY can only be called from the co-routine function itself - not
260 * from within a function called by the co-routine function. This is because
261 * co-routines do not maintain their own stack.
262 *
263 * @param xHandle The handle of the co-routine to delay. This is the xHandle
264 * parameter of the co-routine function.
265 *
266 * @param xTickToDelay The number of ticks that the co-routine should delay
267 * for. The actual amount of time this equates to is defined by
268 * configTICK_RATE_HZ (set in FreeRTOSConfig.h). The constant portTICK_PERIOD_MS
269 * can be used to convert ticks to milliseconds.
270 *
271 * Example usage:
272 <pre>
273 // Co-routine to be created.
274 void vACoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
275 {
276 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
277 // This may not be necessary for const variables.
278 // We are to delay for 200ms.
279 static const xTickType xDelayTime = 200 / portTICK_PERIOD_MS;
280
281 // Must start every co-routine with a call to crSTART();
282 crSTART( xHandle );
283
284 for( ;; )
285 {
286 // Delay for 200ms.
287 crDELAY( xHandle, xDelayTime );
288
289 // Do something here.
290 }
291
292 // Must end every co-routine with a call to crEND();
293 crEND();
294 }</pre>
295 * \defgroup crDELAY crDELAY
296 * \ingroup Tasks
297 */
298#define crDELAY(xHandle, xTicksToDelay) \
299 if ((xTicksToDelay) > 0) { \
300 vCoRoutineAddToDelayedList((xTicksToDelay), NULL); \
301 } \
302 crSET_STATE0((xHandle));
303
304/**
305 * <pre>
306 crQUEUE_SEND(
307 CoRoutineHandle_t xHandle,
308 QueueHandle_t pxQueue,
309 void *pvItemToQueue,
310 TickType_t xTicksToWait,
311 BaseType_t *pxResult
312 )</pre>
313 *
314 * The macro's crQUEUE_SEND() and crQUEUE_RECEIVE() are the co-routine
315 * equivalent to the xQueueSend() and xQueueReceive() functions used by tasks.
316 *
317 * crQUEUE_SEND and crQUEUE_RECEIVE can only be used from a co-routine whereas
318 * xQueueSend() and xQueueReceive() can only be used from tasks.
319 *
320 * crQUEUE_SEND can only be called from the co-routine function itself - not
321 * from within a function called by the co-routine function. This is because
322 * co-routines do not maintain their own stack.
323 *
324 * See the co-routine section of the WEB documentation for information on
325 * passing data between tasks and co-routines and between ISR's and
326 * co-routines.
327 *
328 * @param xHandle The handle of the calling co-routine. This is the xHandle
329 * parameter of the co-routine function.
330 *
331 * @param pxQueue The handle of the queue on which the data will be posted.
332 * The handle is obtained as the return value when the queue is created using
333 * the xQueueCreate() API function.
334 *
335 * @param pvItemToQueue A pointer to the data being posted onto the queue.
336 * The number of bytes of each queued item is specified when the queue is
337 * created. This number of bytes is copied from pvItemToQueue into the queue
338 * itself.
339 *
340 * @param xTickToDelay The number of ticks that the co-routine should block
341 * to wait for space to become available on the queue, should space not be
342 * available immediately. The actual amount of time this equates to is defined
343 * by configTICK_RATE_HZ (set in FreeRTOSConfig.h). The constant
344 * portTICK_PERIOD_MS can be used to convert ticks to milliseconds (see example
345 * below).
346 *
347 * @param pxResult The variable pointed to by pxResult will be set to pdPASS if
348 * data was successfully posted onto the queue, otherwise it will be set to an
349 * error defined within ProjDefs.h.
350 *
351 * Example usage:
352 <pre>
353 // Co-routine function that blocks for a fixed period then posts a number onto
354 // a queue.
355 static void prvCoRoutineFlashTask( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
356 {
357 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
358 static BaseType_t xNumberToPost = 0;
359 static BaseType_t xResult;
360
361 // Co-routines must begin with a call to crSTART().
362 crSTART( xHandle );
363
364 for( ;; )
365 {
366 // This assumes the queue has already been created.
367 crQUEUE_SEND( xHandle, xCoRoutineQueue, &xNumberToPost, NO_DELAY, &xResult );
368
369 if( xResult != pdPASS )
370 {
371 // The message was not posted!
372 }
373
374 // Increment the number to be posted onto the queue.
375 xNumberToPost++;
376
377 // Delay for 100 ticks.
378 crDELAY( xHandle, 100 );
379 }
380
381 // Co-routines must end with a call to crEND().
382 crEND();
383 }</pre>
384 * \defgroup crQUEUE_SEND crQUEUE_SEND
385 * \ingroup Tasks
386 */
387#define crQUEUE_SEND(xHandle, pxQueue, pvItemToQueue, xTicksToWait, pxResult) \
388 { \
389 *(pxResult) = xQueueCRSend((pxQueue), (pvItemToQueue), (xTicksToWait)); \
390 if (*(pxResult) == errQUEUE_BLOCKED) { \
391 crSET_STATE0((xHandle)); \
392 *pxResult = xQueueCRSend((pxQueue), (pvItemToQueue), 0); \
393 } \
394 if (*pxResult == errQUEUE_YIELD) { \
395 crSET_STATE1((xHandle)); \
396 *pxResult = pdPASS; \
397 } \
398 }
399
400/**
401 * croutine. h
402 * <pre>
403 crQUEUE_RECEIVE(
404 CoRoutineHandle_t xHandle,
405 QueueHandle_t pxQueue,
406 void *pvBuffer,
407 TickType_t xTicksToWait,
408 BaseType_t *pxResult
409 )</pre>
410 *
411 * The macro's crQUEUE_SEND() and crQUEUE_RECEIVE() are the co-routine
412 * equivalent to the xQueueSend() and xQueueReceive() functions used by tasks.
413 *
414 * crQUEUE_SEND and crQUEUE_RECEIVE can only be used from a co-routine whereas
415 * xQueueSend() and xQueueReceive() can only be used from tasks.
416 *
417 * crQUEUE_RECEIVE can only be called from the co-routine function itself - not
418 * from within a function called by the co-routine function. This is because
419 * co-routines do not maintain their own stack.
420 *
421 * See the co-routine section of the WEB documentation for information on
422 * passing data between tasks and co-routines and between ISR's and
423 * co-routines.
424 *
425 * @param xHandle The handle of the calling co-routine. This is the xHandle
426 * parameter of the co-routine function.
427 *
428 * @param pxQueue The handle of the queue from which the data will be received.
429 * The handle is obtained as the return value when the queue is created using
430 * the xQueueCreate() API function.
431 *
432 * @param pvBuffer The buffer into which the received item is to be copied.
433 * The number of bytes of each queued item is specified when the queue is
434 * created. This number of bytes is copied into pvBuffer.
435 *
436 * @param xTickToDelay The number of ticks that the co-routine should block
437 * to wait for data to become available from the queue, should data not be
438 * available immediately. The actual amount of time this equates to is defined
439 * by configTICK_RATE_HZ (set in FreeRTOSConfig.h). The constant
440 * portTICK_PERIOD_MS can be used to convert ticks to milliseconds (see the
441 * crQUEUE_SEND example).
442 *
443 * @param pxResult The variable pointed to by pxResult will be set to pdPASS if
444 * data was successfully retrieved from the queue, otherwise it will be set to
445 * an error code as defined within ProjDefs.h.
446 *
447 * Example usage:
448 <pre>
449 // A co-routine receives the number of an LED to flash from a queue. It
450 // blocks on the queue until the number is received.
451 static void prvCoRoutineFlashWorkTask( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
452 {
453 // Variables in co-routines must be declared static if they must maintain value across a blocking call.
454 static BaseType_t xResult;
455 static UBaseType_t uxLEDToFlash;
456
457 // All co-routines must start with a call to crSTART().
458 crSTART( xHandle );
459
460 for( ;; )
461 {
462 // Wait for data to become available on the queue.
463 crQUEUE_RECEIVE( xHandle, xCoRoutineQueue, &uxLEDToFlash, portMAX_DELAY, &xResult );
464
465 if( xResult == pdPASS )
466 {
467 // We received the LED to flash - flash it!
468 vParTestToggleLED( uxLEDToFlash );
469 }
470 }
471
472 crEND();
473 }</pre>
474 * \defgroup crQUEUE_RECEIVE crQUEUE_RECEIVE
475 * \ingroup Tasks
476 */
477#define crQUEUE_RECEIVE(xHandle, pxQueue, pvBuffer, xTicksToWait, pxResult) \
478 { \
479 *(pxResult) = xQueueCRReceive((pxQueue), (pvBuffer), (xTicksToWait)); \
480 if (*(pxResult) == errQUEUE_BLOCKED) { \
481 crSET_STATE0((xHandle)); \
482 *(pxResult) = xQueueCRReceive((pxQueue), (pvBuffer), 0); \
483 } \
484 if (*(pxResult) == errQUEUE_YIELD) { \
485 crSET_STATE1((xHandle)); \
486 *(pxResult) = pdPASS; \
487 } \
488 }
489
490/**
491 * croutine. h
492 * <pre>
493 crQUEUE_SEND_FROM_ISR(
494 QueueHandle_t pxQueue,
495 void *pvItemToQueue,
496 BaseType_t xCoRoutinePreviouslyWoken
497 )</pre>
498 *
499 * The macro's crQUEUE_SEND_FROM_ISR() and crQUEUE_RECEIVE_FROM_ISR() are the
500 * co-routine equivalent to the xQueueSendFromISR() and xQueueReceiveFromISR()
501 * functions used by tasks.
502 *
503 * crQUEUE_SEND_FROM_ISR() and crQUEUE_RECEIVE_FROM_ISR() can only be used to
504 * pass data between a co-routine and and ISR, whereas xQueueSendFromISR() and
505 * xQueueReceiveFromISR() can only be used to pass data between a task and and
506 * ISR.
507 *
508 * crQUEUE_SEND_FROM_ISR can only be called from an ISR to send data to a queue
509 * that is being used from within a co-routine.
510 *
511 * See the co-routine section of the WEB documentation for information on
512 * passing data between tasks and co-routines and between ISR's and
513 * co-routines.
514 *
515 * @param xQueue The handle to the queue on which the item is to be posted.
516 *
517 * @param pvItemToQueue A pointer to the item that is to be placed on the
518 * queue. The size of the items the queue will hold was defined when the
519 * queue was created, so this many bytes will be copied from pvItemToQueue
520 * into the queue storage area.
521 *
522 * @param xCoRoutinePreviouslyWoken This is included so an ISR can post onto
523 * the same queue multiple times from a single interrupt. The first call
524 * should always pass in pdFALSE. Subsequent calls should pass in
525 * the value returned from the previous call.
526 *
527 * @return pdTRUE if a co-routine was woken by posting onto the queue. This is
528 * used by the ISR to determine if a context switch may be required following
529 * the ISR.
530 *
531 * Example usage:
532 <pre>
533 // A co-routine that blocks on a queue waiting for characters to be received.
534 static void vReceivingCoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
535 {
536 char cRxedChar;
537 BaseType_t xResult;
538
539 // All co-routines must start with a call to crSTART().
540 crSTART( xHandle );
541
542 for( ;; )
543 {
544 // Wait for data to become available on the queue. This assumes the
545 // queue xCommsRxQueue has already been created!
546 crQUEUE_RECEIVE( xHandle, xCommsRxQueue, &uxLEDToFlash, portMAX_DELAY, &xResult );
547
548 // Was a character received?
549 if( xResult == pdPASS )
550 {
551 // Process the character here.
552 }
553 }
554
555 // All co-routines must end with a call to crEND().
556 crEND();
557 }
558
559 // An ISR that uses a queue to send characters received on a serial port to
560 // a co-routine.
561 void vUART_ISR( void )
562 {
563 char cRxedChar;
564 BaseType_t xCRWokenByPost = pdFALSE;
565
566 // We loop around reading characters until there are none left in the UART.
567 while( UART_RX_REG_NOT_EMPTY() )
568 {
569 // Obtain the character from the UART.
570 cRxedChar = UART_RX_REG;
571
572 // Post the character onto a queue. xCRWokenByPost will be pdFALSE
573 // the first time around the loop. If the post causes a co-routine
574 // to be woken (unblocked) then xCRWokenByPost will be set to pdTRUE.
575 // In this manner we can ensure that if more than one co-routine is
576 // blocked on the queue only one is woken by this ISR no matter how
577 // many characters are posted to the queue.
578 xCRWokenByPost = crQUEUE_SEND_FROM_ISR( xCommsRxQueue, &cRxedChar, xCRWokenByPost );
579 }
580 }</pre>
581 * \defgroup crQUEUE_SEND_FROM_ISR crQUEUE_SEND_FROM_ISR
582 * \ingroup Tasks
583 */
584#define crQUEUE_SEND_FROM_ISR(pxQueue, pvItemToQueue, xCoRoutinePreviouslyWoken) \
585 xQueueCRSendFromISR((pxQueue), (pvItemToQueue), (xCoRoutinePreviouslyWoken))
586
587/**
588 * croutine. h
589 * <pre>
590 crQUEUE_SEND_FROM_ISR(
591 QueueHandle_t pxQueue,
592 void *pvBuffer,
593 BaseType_t * pxCoRoutineWoken
594 )</pre>
595 *
596 * The macro's crQUEUE_SEND_FROM_ISR() and crQUEUE_RECEIVE_FROM_ISR() are the
597 * co-routine equivalent to the xQueueSendFromISR() and xQueueReceiveFromISR()
598 * functions used by tasks.
599 *
600 * crQUEUE_SEND_FROM_ISR() and crQUEUE_RECEIVE_FROM_ISR() can only be used to
601 * pass data between a co-routine and and ISR, whereas xQueueSendFromISR() and
602 * xQueueReceiveFromISR() can only be used to pass data between a task and and
603 * ISR.
604 *
605 * crQUEUE_RECEIVE_FROM_ISR can only be called from an ISR to receive data
606 * from a queue that is being used from within a co-routine (a co-routine
607 * posted to the queue).
608 *
609 * See the co-routine section of the WEB documentation for information on
610 * passing data between tasks and co-routines and between ISR's and
611 * co-routines.
612 *
613 * @param xQueue The handle to the queue on which the item is to be posted.
614 *
615 * @param pvBuffer A pointer to a buffer into which the received item will be
616 * placed. The size of the items the queue will hold was defined when the
617 * queue was created, so this many bytes will be copied from the queue into
618 * pvBuffer.
619 *
620 * @param pxCoRoutineWoken A co-routine may be blocked waiting for space to become
621 * available on the queue. If crQUEUE_RECEIVE_FROM_ISR causes such a
622 * co-routine to unblock *pxCoRoutineWoken will get set to pdTRUE, otherwise
623 * *pxCoRoutineWoken will remain unchanged.
624 *
625 * @return pdTRUE an item was successfully received from the queue, otherwise
626 * pdFALSE.
627 *
628 * Example usage:
629 <pre>
630 // A co-routine that posts a character to a queue then blocks for a fixed
631 // period. The character is incremented each time.
632 static void vSendingCoRoutine( CoRoutineHandle_t xHandle, UBaseType_t uxIndex )
633 {
634 // cChar holds its value while this co-routine is blocked and must therefore
635 // be declared static.
636 static char cCharToTx = 'a';
637 BaseType_t xResult;
638
639 // All co-routines must start with a call to crSTART().
640 crSTART( xHandle );
641
642 for( ;; )
643 {
644 // Send the next character to the queue.
645 crQUEUE_SEND( xHandle, xCoRoutineQueue, &cCharToTx, NO_DELAY, &xResult );
646
647 if( xResult == pdPASS )
648 {
649 // The character was successfully posted to the queue.
650 }
651 else
652 {
653 // Could not post the character to the queue.
654 }
655
656 // Enable the UART Tx interrupt to cause an interrupt in this
657 // hypothetical UART. The interrupt will obtain the character
658 // from the queue and send it.
659 ENABLE_RX_INTERRUPT();
660
661 // Increment to the next character then block for a fixed period.
662 // cCharToTx will maintain its value across the delay as it is
663 // declared static.
664 cCharToTx++;
665 if( cCharToTx > 'x' )
666 {
667 cCharToTx = 'a';
668 }
669 crDELAY( 100 );
670 }
671
672 // All co-routines must end with a call to crEND().
673 crEND();
674 }
675
676 // An ISR that uses a queue to receive characters to send on a UART.
677 void vUART_ISR( void )
678 {
679 char cCharToTx;
680 BaseType_t xCRWokenByPost = pdFALSE;
681
682 while( UART_TX_REG_EMPTY() )
683 {
684 // Are there any characters in the queue waiting to be sent?
685 // xCRWokenByPost will automatically be set to pdTRUE if a co-routine
686 // is woken by the post - ensuring that only a single co-routine is
687 // woken no matter how many times we go around this loop.
688 if( crQUEUE_RECEIVE_FROM_ISR( pxQueue, &cCharToTx, &xCRWokenByPost ) )
689 {
690 SEND_CHARACTER( cCharToTx );
691 }
692 }
693 }</pre>
694 * \defgroup crQUEUE_RECEIVE_FROM_ISR crQUEUE_RECEIVE_FROM_ISR
695 * \ingroup Tasks
696 */
697#define crQUEUE_RECEIVE_FROM_ISR(pxQueue, pvBuffer, pxCoRoutineWoken) \
698 xQueueCRReceiveFromISR((pxQueue), (pvBuffer), (pxCoRoutineWoken))
699
700/*
701 * This function is intended for internal use by the co-routine macros only.
702 * The macro nature of the co-routine implementation requires that the
703 * prototype appears here. The function should not be used by application
704 * writers.
705 *
706 * Removes the current co-routine from its ready list and places it in the
707 * appropriate delayed list.
708 */
709void vCoRoutineAddToDelayedList(TickType_t xTicksToDelay, List_t *pxEventList);
710
711/*
712 * This function is intended for internal use by the queue implementation only.
713 * The function should not be used by application writers.
714 *
715 * Removes the highest priority co-routine from the event list and places it in
716 * the pending ready list.
717 */
718BaseType_t xCoRoutineRemoveFromEventList(const List_t *pxEventList);
719
720#ifdef __cplusplus
721}
722#endif
723
724#endif /* CO_ROUTINE_H */
Definition croutine.h:50
Definition list.h:145
Definition list.h:174