| 
									
										
										
										
											2013-05-06 12:50:19 +07:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*!
 | 
					
						
							|  |  |  |     @file     osal_freeRTOS.h | 
					
						
							|  |  |  |     @author   hathach (tinyusb.org) | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 12:50:19 +07:00
										 |  |  |     @section LICENSE | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Software License Agreement (BSD License) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Copyright (c) 2013, hathach (tinyusb.org) | 
					
						
							|  |  |  |     All rights reserved. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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. | 
					
						
							|  |  |  |     3. Neither the name of the copyright holders nor the | 
					
						
							|  |  |  |     names of its contributors may be used to endorse or promote products | 
					
						
							|  |  |  |     derived from this software without specific prior written permission. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''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 HOLDER 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This file is part of the tinyusb stack. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  *  \brief TBD | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  \note TBD | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \ingroup TBD
 | 
					
						
							|  |  |  |  *  \defgroup TBD | 
					
						
							|  |  |  |  *  \brief TBD | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  @{ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef _TUSB_OSAL_FREERTOS_H_
 | 
					
						
							|  |  |  | #define _TUSB_OSAL_FREERTOS_H_
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "osal_common.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //------------- FreeRTOS Headers -------------//
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | #include "FreeRTOS.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | #include "semphr.h"
 | 
					
						
							|  |  |  | #include "queue.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #include "task.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-08 12:12:10 +07:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // TICK API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | #define osal_tick_get xTaskGetTickCount
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // TASK API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define OSAL_TASK_FUNCTION(task_func) \
 | 
					
						
							|  |  |  |   void task_func | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2013-04-25 11:00:56 +07:00
										 |  |  |   char const * name; | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  |   pdTASK_CODE code; | 
					
						
							|  |  |  |   unsigned portSHORT stack_depth; | 
					
						
							|  |  |  |   unsigned portBASE_TYPE prio; | 
					
						
							|  |  |  | } osal_task_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 16:41:00 +07:00
										 |  |  | #define OSAL_TASK_DEF(task_variable, task_name, task_code, task_stack_depth, task_prio) \
 | 
					
						
							|  |  |  |   osal_task_t task_variable = {\ | 
					
						
							|  |  |  |       .name        = task_name        , \ | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  |       .code        = task_code        , \ | 
					
						
							|  |  |  |       .stack_depth = task_stack_depth , \ | 
					
						
							|  |  |  |       .prio        = task_prio          \ | 
					
						
							| 
									
										
										
										
											2013-04-25 11:55:24 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 11:55:24 +07:00
										 |  |  | static inline tusb_error_t osal_task_create(osal_task_t *task) ATTR_WARN_UNUSED_RESULT ATTR_ALWAYS_INLINE; | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | static inline tusb_error_t osal_task_create(osal_task_t *task) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-25 11:00:56 +07:00
										 |  |  |   return pdPASS == xTaskCreate(task->code, (signed portCHAR const *) task->name, task->stack_depth, NULL, task->prio, NULL) ? | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  |     TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TASK_CREATE_FAILED; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-04-23 15:35:27 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-25 16:41:00 +07:00
										 |  |  | static inline void osal_task_delay(uint32_t msec) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void osal_task_delay(uint32_t msec) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-25 17:48:55 +07:00
										 |  |  |   vTaskDelay( (TUSB_CFG_OS_TICKS_PER_SECOND * msec) / 1000 ); | 
					
						
							| 
									
										
										
										
											2013-04-25 16:41:00 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | #define OSAL_TASK_LOOP_BEGIN \
 | 
					
						
							|  |  |  |   while(1) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define OSAL_TASK_LOOP_END \
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | //------------- Sub Task -------------//
 | 
					
						
							| 
									
										
										
										
											2013-04-23 15:35:27 +07:00
										 |  |  | #define OSAL_SUBTASK_BEGIN // TODO refractor move
 | 
					
						
							| 
									
										
										
										
											2013-04-25 11:00:56 +07:00
										 |  |  | #define OSAL_SUBTASK_END \
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define OSAL_SUBTASK_INVOKED_AND_WAIT(subtask, status) \
 | 
					
						
							|  |  |  |   status = subtask | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | //------------- Sub Task Assert -------------//
 | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define SUBTASK_ASSERT_STATUS(sts) ASSERT_STATUS(sts)
 | 
					
						
							|  |  |  | #define SUBTASK_ASSERT(condition)  ASSERT(condition, TUSB_ERROR_OSAL_TASK_FAILED)
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define _SUBTASK_ASSERT_ERROR_HANDLER(error, func_call)\
 | 
					
						
							|  |  |  |     func_call; return error | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define SUBTASK_ASSERT_STATUS_WITH_HANDLER(sts, func_call) \
 | 
					
						
							|  |  |  |     ASSERT_DEFINE_WITH_HANDLER(_SUBTASK_ASSERT_ERROR_HANDLER, func_call, tusb_error_t status = (tusb_error_t)(sts),\ | 
					
						
							|  |  |  |                                TUSB_ERROR_NONE == status, status, "%s", TUSB_ErrorStr[status]) | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | #define SUBTASK_ASSERT_WITH_HANDLER(condition, func_call) \
 | 
					
						
							|  |  |  |     ASSERT_DEFINE_WITH_HANDLER(_SUBTASK_ASSERT_ERROR_HANDLER, func_call, ,\ | 
					
						
							|  |  |  |                                condition, TUSB_ERROR_OSAL_TASK_FAILED, "%s", "evaluated to false") | 
					
						
							| 
									
										
										
										
											2013-04-21 00:28:25 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Semaphore API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | #define OSAL_SEM_DEF(name)
 | 
					
						
							|  |  |  | typedef xSemaphoreHandle osal_semaphore_handle_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | // create FreeRTOS binary semaphore with zero as init value TODO: omit semaphore take from vSemaphoreCreateBinary API, should double checks this
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | #define osal_semaphore_create(x) \
 | 
					
						
							|  |  |  |   xQueueGenericCreate( ( unsigned portBASE_TYPE ) 1, semSEMAPHORE_QUEUE_ITEM_LENGTH, queueQUEUE_TYPE_BINARY_SEMAPHORE ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | // TODO add timeout (with instant return from ISR option) for semaphore post & queue send
 | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | static inline  tusb_error_t osal_semaphore_post(osal_semaphore_handle_t const sem_hdl) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline  tusb_error_t osal_semaphore_post(osal_semaphore_handle_t const sem_hdl) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  |   portBASE_TYPE task_waken; | 
					
						
							|  |  |  |   return (xSemaphoreGiveFromISR(sem_hdl, &task_waken) == pdTRUE) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_SEMAPHORE_FAILED; | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void osal_semaphore_wait(osal_semaphore_handle_t const sem_hdl, uint32_t msec, tusb_error_t *p_error) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void osal_semaphore_wait(osal_semaphore_handle_t const sem_hdl, uint32_t msec, tusb_error_t *p_error) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (*p_error) = ( xSemaphoreTake(sem_hdl, osal_tick_from_msec(msec)) == pdPASS ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TIMEOUT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | static inline void osal_semaphore_reset(osal_semaphore_handle_t const sem_hdl) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void osal_semaphore_reset(osal_semaphore_handle_t const sem_hdl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   portBASE_TYPE task_waken; | 
					
						
							|  |  |  |   xSemaphoreTakeFromISR(sem_hdl, &task_waken); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // QUEUE API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | typedef struct{ | 
					
						
							| 
									
										
										
										
											2013-04-27 16:40:18 +07:00
										 |  |  |   uint8_t const depth;     ///< buffer size
 | 
					
						
							|  |  |  |   uint8_t const item_size; ///< size of each item
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | } osal_queue_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef xQueueHandle osal_queue_handle_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define OSAL_QUEUE_DEF(name, queue_depth, type)\
 | 
					
						
							|  |  |  |   osal_queue_t name = {\ | 
					
						
							| 
									
										
										
										
											2013-04-27 16:40:18 +07:00
										 |  |  |       .depth     = queue_depth,\ | 
					
						
							|  |  |  |       .item_size = sizeof(type)\ | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 17:13:26 +07:00
										 |  |  | #define osal_queue_create(p_queue) \
 | 
					
						
							| 
									
										
										
										
											2013-04-27 16:40:18 +07:00
										 |  |  |   xQueueCreate((p_queue)->depth, (p_queue)->item_size) | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | static inline void osal_queue_receive (osal_queue_handle_t const queue_hdl, void *p_data, uint32_t msec, tusb_error_t *p_error) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void osal_queue_receive (osal_queue_handle_t const queue_hdl, void *p_data, uint32_t msec, tusb_error_t *p_error) | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   (*p_error) = ( xQueueReceive(queue_hdl, p_data, osal_tick_from_msec(msec)) == pdPASS ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TIMEOUT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-27 00:24:15 +07:00
										 |  |  | static inline tusb_error_t osal_queue_send(osal_queue_handle_t const queue_hdl, void const * data) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline tusb_error_t osal_queue_send(osal_queue_handle_t const queue_hdl, void const * data) | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   portBASE_TYPE taskWaken; | 
					
						
							| 
									
										
										
										
											2013-04-10 01:13:31 +07:00
										 |  |  |   return ( xQueueSendFromISR(queue_hdl, data, &taskWaken) == pdTRUE ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_QUEUE_FAILED; | 
					
						
							| 
									
										
										
										
											2013-02-04 18:05:22 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-24 17:53:43 +07:00
										 |  |  | static inline void osal_queue_flush(osal_queue_handle_t const queue_hdl) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void osal_queue_flush(osal_queue_handle_t const queue_hdl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   xQueueReset(queue_hdl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  |  } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* _TUSB_OSAL_FREERTOS_H_ */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** @} */ |