241 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**************************************************************************/
 | |
| /*!
 | |
|     @file     osal_freeRTOS.h
 | |
|     @author   hathach (tinyusb.org)
 | |
| 
 | |
|     @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.
 | |
| */
 | |
| /**************************************************************************/
 | |
| 
 | |
| /** \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"
 | |
| 
 | |
| //------------- FreeRTOS Headers -------------//
 | |
| #include "FreeRTOS.h"
 | |
| #include "semphr.h"
 | |
| #include "queue.h"
 | |
| #include "task.h"
 | |
| 
 | |
| #ifdef __cplusplus
 | |
| extern "C" {
 | |
| #endif
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // TICK API
 | |
| //--------------------------------------------------------------------+
 | |
| #define osal_tick_get xTaskGetTickCount
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // TASK API
 | |
| //--------------------------------------------------------------------+
 | |
| #define OSAL_TASK_FUNCTION(task_func) \
 | |
|   void task_func
 | |
| 
 | |
| typedef struct {
 | |
|   char const * name;
 | |
|   pdTASK_CODE code;
 | |
|   unsigned portSHORT stack_depth;
 | |
|   unsigned portBASE_TYPE prio;
 | |
| } osal_task_t;
 | |
| 
 | |
| #define OSAL_TASK_DEF(task_variable, task_name, task_code, task_stack_depth, task_prio) \
 | |
|   osal_task_t task_variable = {\
 | |
|       .name        = task_name        , \
 | |
|       .code        = task_code        , \
 | |
|       .stack_depth = task_stack_depth , \
 | |
|       .prio        = task_prio          \
 | |
|   }
 | |
| 
 | |
| static inline tusb_error_t osal_task_create(osal_task_t *task) ATTR_WARN_UNUSED_RESULT ATTR_ALWAYS_INLINE;
 | |
| static inline tusb_error_t osal_task_create(osal_task_t *task)
 | |
| {
 | |
|   return pdPASS == xTaskCreate(task->code, (signed portCHAR const *) task->name, task->stack_depth, NULL, task->prio, NULL) ?
 | |
|     TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TASK_CREATE_FAILED;
 | |
| }
 | |
| 
 | |
| static inline void osal_task_delay(uint32_t msec) ATTR_ALWAYS_INLINE;
 | |
| static inline void osal_task_delay(uint32_t msec)
 | |
| {
 | |
|   vTaskDelay( (TUSB_CFG_OS_TICKS_PER_SECOND * msec) / 1000 );
 | |
| }
 | |
| 
 | |
| #define OSAL_TASK_LOOP_BEGIN \
 | |
|   while(1) {
 | |
| 
 | |
| #define OSAL_TASK_LOOP_END \
 | |
|   }
 | |
| 
 | |
| //------------- Sub Task -------------//
 | |
| #define OSAL_SUBTASK_BEGIN // TODO refractor move
 | |
| #define OSAL_SUBTASK_END \
 | |
|   return TUSB_ERROR_NONE;
 | |
| 
 | |
| #define SUBTASK_EXIT(error)   return error;
 | |
| 
 | |
| #define OSAL_SUBTASK_INVOKED_AND_WAIT(subtask, status) \
 | |
|   status = subtask
 | |
| 
 | |
| //------------- Sub Task Assert -------------//
 | |
| 
 | |
| #define SUBTASK_ASSERT_STATUS(sts) ASSERT_STATUS(sts)
 | |
| #define SUBTASK_ASSERT(condition)  ASSERT(condition, TUSB_ERROR_OSAL_TASK_FAILED)
 | |
| 
 | |
| #define _SUBTASK_ASSERT_ERROR_HANDLER(error, func_call)\
 | |
|     func_call; return error
 | |
| 
 | |
| #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])
 | |
| 
 | |
| #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")
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Semaphore API
 | |
| //--------------------------------------------------------------------+
 | |
| #define OSAL_SEM_DEF(name)
 | |
| typedef xSemaphoreHandle osal_semaphore_handle_t;
 | |
| 
 | |
| // create FreeRTOS binary semaphore with zero as init value TODO: omit semaphore take from vSemaphoreCreateBinary API, should double checks this
 | |
| #define osal_semaphore_create(x) \
 | |
|   xQueueGenericCreate( ( unsigned portBASE_TYPE ) 1, semSEMAPHORE_QUEUE_ITEM_LENGTH, queueQUEUE_TYPE_BINARY_SEMAPHORE )
 | |
| 
 | |
| // TODO add timeout (with instant return from ISR option) for semaphore post & queue send
 | |
| 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)
 | |
| {
 | |
|   portBASE_TYPE task_waken;
 | |
|   return (xSemaphoreGiveFromISR(sem_hdl, &task_waken) == pdTRUE) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_SEMAPHORE_FAILED;
 | |
| }
 | |
| 
 | |
| 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;
 | |
| }
 | |
| 
 | |
| 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);
 | |
| }
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // MUTEX API (priority inheritance)
 | |
| //--------------------------------------------------------------------+
 | |
| #define OSAL_MUTEX_DEF  OSAL_SEM_DEF
 | |
| typedef xSemaphoreHandle osal_mutex_handle_t;
 | |
| 
 | |
| #define osal_mutex_create(x) \
 | |
|   xSemaphoreCreateMutex()
 | |
| 
 | |
| static inline  tusb_error_t osal_mutex_release(osal_mutex_handle_t const mutex_hdl) ATTR_ALWAYS_INLINE;
 | |
| static inline  tusb_error_t osal_mutex_release(osal_mutex_handle_t const mutex_hdl)
 | |
| {
 | |
|   return (xSemaphoreGive(mutex_hdl) == pdPASS) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_SEMAPHORE_FAILED;
 | |
| }
 | |
| 
 | |
| static inline void osal_mutex_wait(osal_mutex_handle_t const mutex_hdl, uint32_t msec, tusb_error_t *p_error) ATTR_ALWAYS_INLINE;
 | |
| static inline void osal_mutex_wait(osal_mutex_handle_t const mutex_hdl, uint32_t msec, tusb_error_t *p_error)
 | |
| {
 | |
|   (*p_error) = ( xSemaphoreTake(mutex_hdl, osal_tick_from_msec(msec)) == pdPASS ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TIMEOUT;
 | |
| }
 | |
| 
 | |
| static inline void osal_mutex_reset(osal_mutex_handle_t const mutex_hdl) ATTR_ALWAYS_INLINE;
 | |
| static inline void osal_mutex_reset(osal_mutex_handle_t const mutex_hdl)
 | |
| {
 | |
|   xSemaphoreGive(mutex_hdl);
 | |
| }
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // QUEUE API
 | |
| //--------------------------------------------------------------------+
 | |
| typedef struct{
 | |
|   uint8_t const depth;     ///< buffer size
 | |
|   uint8_t const item_size; ///< size of each item
 | |
| } osal_queue_t;
 | |
| 
 | |
| typedef xQueueHandle osal_queue_handle_t;
 | |
| 
 | |
| #define OSAL_QUEUE_DEF(name, queue_depth, type)\
 | |
|   osal_queue_t name = {\
 | |
|       .depth     = queue_depth,\
 | |
|       .item_size = sizeof(type)\
 | |
|   }
 | |
| 
 | |
| #define osal_queue_create(p_queue) \
 | |
|   xQueueCreate((p_queue)->depth, (p_queue)->item_size)
 | |
| 
 | |
| 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)
 | |
| {
 | |
|   (*p_error) = ( xQueueReceive(queue_hdl, p_data, osal_tick_from_msec(msec)) == pdPASS ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_TIMEOUT;
 | |
| }
 | |
| 
 | |
| 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)
 | |
| {
 | |
|   portBASE_TYPE taskWaken;
 | |
|   return ( xQueueSendFromISR(queue_hdl, data, &taskWaken) == pdTRUE ) ? TUSB_ERROR_NONE : TUSB_ERROR_OSAL_QUEUE_FAILED;
 | |
| }
 | |
| 
 | |
| 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);
 | |
| }
 | |
| 
 | |
| #ifdef __cplusplus
 | |
|  }
 | |
| #endif
 | |
| 
 | |
| #endif /* _TUSB_OSAL_FREERTOS_H_ */
 | |
| 
 | |
| /** @} */
 | 
