mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-30 20:51:41 +00:00 
			
		
		
		
	 565eee12a0
			
		
	
	565eee12a0
	
	
	
		
			
			Closes: https://github.com/espressif/esp-idf/issues/944 Closes: https://github.com/espressif/esp-idf/issues/3931 Closes: WIFI-1019
		
			
				
	
	
		
			556 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			556 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2001-2003 Swedish Institute of Computer Science.
 | |
|  * 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. The name of the author may not be used to endorse or promote products
 | |
|  *    derived from this software without specific prior written permission.
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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 lwIP TCP/IP stack.
 | |
|  *
 | |
|  * Author: Adam Dunkels <adam@sics.se>
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /* lwIP includes. */
 | |
| 
 | |
| #include <pthread.h>
 | |
| #include "lwip/debug.h"
 | |
| #include "lwip/def.h"
 | |
| #include "lwip/sys.h"
 | |
| #include "lwip/mem.h"
 | |
| #include "arch/sys_arch.h"
 | |
| #include "lwip/stats.h"
 | |
| #include "esp_log.h"
 | |
| #include "esp_compiler.h"
 | |
| 
 | |
| static const char* TAG = "lwip_arch";
 | |
| 
 | |
| static sys_mutex_t g_lwip_protect_mutex = NULL;
 | |
| 
 | |
| static pthread_key_t sys_thread_sem_key;
 | |
| static void sys_thread_sem_free(void* data);
 | |
| 
 | |
| #if !LWIP_COMPAT_MUTEX
 | |
| 
 | |
| /**
 | |
|  * @brief Create a new mutex
 | |
|  *
 | |
|  * @param pxMutex pointer of the mutex to create
 | |
|  * @return ERR_OK on success, ERR_MEM when out of memory
 | |
|  */
 | |
| err_t
 | |
| sys_mutex_new(sys_mutex_t *pxMutex)
 | |
| {
 | |
|   *pxMutex = xSemaphoreCreateMutex();
 | |
|   if (*pxMutex == NULL) {
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sys_mutex_new: out of mem\r\n"));
 | |
|     return ERR_MEM;
 | |
|   }
 | |
| 
 | |
|   LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sys_mutex_new: m=%p\n", *pxMutex));
 | |
| 
 | |
|   return ERR_OK;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Lock a mutex
 | |
|  *
 | |
|  * @param pxMutex pointer of mutex to lock
 | |
|  */
 | |
| void
 | |
| sys_mutex_lock(sys_mutex_t *pxMutex)
 | |
| {
 | |
|   BaseType_t ret = xSemaphoreTake(*pxMutex, portMAX_DELAY);
 | |
| 
 | |
|   LWIP_ASSERT("failed to take the mutex", ret == pdTRUE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Unlock a mutex
 | |
|  *
 | |
|  * @param pxMutex pointer of mutex to unlock
 | |
|  */
 | |
| void
 | |
| sys_mutex_unlock(sys_mutex_t *pxMutex)
 | |
| {
 | |
|   BaseType_t ret = xSemaphoreGive(*pxMutex);
 | |
| 
 | |
|   LWIP_ASSERT("failed to give the mutex", ret == pdTRUE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Delete a mutex
 | |
|  *
 | |
|  * @param pxMutex pointer of mutex to delete
 | |
|  */
 | |
| void
 | |
| sys_mutex_free(sys_mutex_t *pxMutex)
 | |
| {
 | |
|   LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sys_mutex_free: m=%p\n", *pxMutex));
 | |
|   vSemaphoreDelete(*pxMutex);
 | |
|   *pxMutex = NULL;
 | |
| }
 | |
| 
 | |
| #endif /* !LWIP_COMPAT_MUTEX */
 | |
| 
 | |
| /**
 | |
|  * @brief Creates a new semaphore
 | |
|  *
 | |
|  * @param sem pointer of the semaphore
 | |
|  * @param count initial state of the semaphore
 | |
|  * @return err_t
 | |
|  */
 | |
| err_t
 | |
| sys_sem_new(sys_sem_t *sem, u8_t count)
 | |
| {
 | |
|   LWIP_ASSERT("initial_count invalid (neither 0 nor 1)",
 | |
|              (count == 0) || (count == 1));
 | |
| 
 | |
|   *sem = xSemaphoreCreateBinary();
 | |
|   if (*sem == NULL) {
 | |
|       LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sys_sem_new: out of mem\r\n"));
 | |
|       return ERR_MEM;
 | |
|   }
 | |
| 
 | |
|   if (count == 1) {
 | |
|       BaseType_t ret = xSemaphoreGive(*sem);
 | |
|       LWIP_ASSERT("sys_sem_new: initial give failed", ret == pdTRUE);
 | |
|   }
 | |
| 
 | |
|   return ERR_OK;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Signals a semaphore
 | |
|  *
 | |
|  * @param sem pointer of the semaphore
 | |
|  */
 | |
| void
 | |
| sys_sem_signal(sys_sem_t *sem)
 | |
| {
 | |
|   BaseType_t ret = xSemaphoreGive(*sem);
 | |
|   /* queue full is OK, this is a signal only... */
 | |
|   LWIP_ASSERT("sys_sem_signal: sane return value",
 | |
|              (ret == pdTRUE) || (ret == errQUEUE_FULL));
 | |
| }
 | |
| 
 | |
| /*-----------------------------------------------------------------------------------*/
 | |
| // Signals a semaphore (from ISR)
 | |
| int
 | |
| sys_sem_signal_isr(sys_sem_t *sem)
 | |
| {
 | |
|     BaseType_t woken = pdFALSE;
 | |
|     xSemaphoreGiveFromISR(*sem, &woken);
 | |
|     return woken == pdTRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Wait for a semaphore to be signaled
 | |
|  *
 | |
|  * @param sem pointer of the semaphore
 | |
|  * @param timeout if zero, will wait infinitely, or will wait for milliseconds specify by this argument
 | |
|  * @return SYS_ARCH_TIMEOUT when timeout, 0 otherwise
 | |
|  */
 | |
| u32_t
 | |
| sys_arch_sem_wait(sys_sem_t *sem, u32_t timeout)
 | |
| {
 | |
|   BaseType_t ret;
 | |
| 
 | |
|   if (!timeout) {
 | |
|     /* wait infinite */
 | |
|     ret = xSemaphoreTake(*sem, portMAX_DELAY);
 | |
|     LWIP_ASSERT("taking semaphore failed", ret == pdTRUE);
 | |
|   } else {
 | |
|     TickType_t timeout_ticks = timeout / portTICK_RATE_MS;
 | |
|     ret = xSemaphoreTake(*sem, timeout_ticks);
 | |
|     if (ret == errQUEUE_EMPTY) {
 | |
|       /* timed out */
 | |
|       return SYS_ARCH_TIMEOUT;
 | |
|     }
 | |
|     LWIP_ASSERT("taking semaphore failed", ret == pdTRUE);
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Delete a semaphore
 | |
|  *
 | |
|  * @param sem pointer of the semaphore to delete
 | |
|  */
 | |
| void
 | |
| sys_sem_free(sys_sem_t *sem)
 | |
| {
 | |
|   vSemaphoreDelete(*sem);
 | |
|   *sem = NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Create an empty mailbox.
 | |
|  *
 | |
|  * @param mbox pointer of the mailbox
 | |
|  * @param size size of the mailbox
 | |
|  * @return ERR_OK on success, ERR_MEM when out of memory
 | |
|  */
 | |
| err_t
 | |
| sys_mbox_new(sys_mbox_t *mbox, int size)
 | |
| {
 | |
|   *mbox = mem_malloc(sizeof(struct sys_mbox_s));
 | |
|   if (*mbox == NULL){
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("fail to new *mbox\n"));
 | |
|     return ERR_MEM;
 | |
|   }
 | |
| 
 | |
|   (*mbox)->os_mbox = xQueueCreate(size, sizeof(void *));
 | |
| 
 | |
|   if ((*mbox)->os_mbox == NULL) {
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("fail to new (*mbox)->os_mbox\n"));
 | |
|     free(*mbox);
 | |
|     return ERR_MEM;
 | |
|   }
 | |
| 
 | |
| #if ESP_THREAD_SAFE
 | |
|   (*mbox)->owner = NULL;
 | |
| #endif
 | |
| 
 | |
|   LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("new *mbox ok mbox=%p os_mbox=%p\n", *mbox, (*mbox)->os_mbox));
 | |
|   return ERR_OK;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Send message to mailbox
 | |
|  *
 | |
|  * @param mbox pointer of the mailbox
 | |
|  * @param msg pointer of the message to send
 | |
|  */
 | |
| void
 | |
| sys_mbox_post(sys_mbox_t *mbox, void *msg)
 | |
| {
 | |
|   BaseType_t ret = xQueueSendToBack((*mbox)->os_mbox, &msg, portMAX_DELAY);
 | |
|   LWIP_ASSERT("mbox post failed", ret == pdTRUE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Try to post a message to mailbox
 | |
|  *
 | |
|  * @param mbox pointer of the mailbox
 | |
|  * @param msg pointer of the message to send
 | |
|  * @return ERR_OK on success, ERR_MEM when mailbox is full
 | |
|  */
 | |
| err_t
 | |
| sys_mbox_trypost(sys_mbox_t *mbox, void *msg)
 | |
| {
 | |
|   err_t xReturn;
 | |
| 
 | |
|   if (xQueueSend((*mbox)->os_mbox, &msg, 0) == pdTRUE) {
 | |
|     xReturn = ERR_OK;
 | |
|   } else {
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("trypost mbox=%p fail\n", (*mbox)->os_mbox));
 | |
|     xReturn = ERR_MEM;
 | |
|   }
 | |
| 
 | |
|   return xReturn;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Try to post a message to mailbox from ISR
 | |
|  *
 | |
|  * @param mbox pointer of the mailbox
 | |
|  * @param msg pointer of the message to send
 | |
|  * @return  ERR_OK on success
 | |
|  *          ERR_MEM when mailbox is full
 | |
|  *          ERR_NEED_SCHED when high priority task wakes up
 | |
|  */
 | |
| err_t
 | |
| sys_mbox_trypost_fromisr(sys_mbox_t *mbox, void *msg)
 | |
| {
 | |
|   BaseType_t ret;
 | |
|   BaseType_t xHigherPriorityTaskWoken = pdFALSE;
 | |
| 
 | |
|   ret = xQueueSendFromISR((*mbox)->os_mbox, &msg, &xHigherPriorityTaskWoken);
 | |
|   if (ret == pdTRUE) {
 | |
|     if (xHigherPriorityTaskWoken == pdTRUE) {
 | |
|       return ERR_NEED_SCHED;
 | |
|     }
 | |
|     return ERR_OK;
 | |
|   } else {
 | |
|     LWIP_ASSERT("mbox trypost failed", ret == errQUEUE_FULL);
 | |
|     return ERR_MEM;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Fetch message from mailbox
 | |
|  *
 | |
|  * @param mbox pointer of mailbox
 | |
|  * @param msg pointer of the received message, could be NULL to indicate the message should be dropped
 | |
|  * @param timeout if zero, will wait infinitely; or will wait milliseconds specify by this argument
 | |
|  * @return SYS_ARCH_TIMEOUT when timeout, 0 otherwise
 | |
|  */
 | |
| u32_t
 | |
| sys_arch_mbox_fetch(sys_mbox_t *mbox, void **msg, u32_t timeout)
 | |
| {
 | |
|   BaseType_t ret;
 | |
|   void *msg_dummy;
 | |
| 
 | |
|   if (msg == NULL) {
 | |
|     msg = &msg_dummy;
 | |
|   }
 | |
| 
 | |
|   if (timeout == 0) {
 | |
|     /* wait infinite */
 | |
|     ret = xQueueReceive((*mbox)->os_mbox, &(*msg), portMAX_DELAY);
 | |
|     LWIP_ASSERT("mbox fetch failed", ret == pdTRUE);
 | |
|   } else {
 | |
|     TickType_t timeout_ticks = timeout / portTICK_RATE_MS;
 | |
|     ret = xQueueReceive((*mbox)->os_mbox, &(*msg), timeout_ticks);
 | |
|     if (ret == errQUEUE_EMPTY) {
 | |
|       /* timed out */
 | |
|       *msg = NULL;
 | |
|       return SYS_ARCH_TIMEOUT;
 | |
|     }
 | |
|     LWIP_ASSERT("mbox fetch failed", ret == pdTRUE);
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief try to fetch message from mailbox
 | |
|  *
 | |
|  * @param mbox pointer of mailbox
 | |
|  * @param msg pointer of the received message
 | |
|  * @return SYS_MBOX_EMPTY if mailbox is empty, 1 otherwise
 | |
|  */
 | |
| u32_t
 | |
| sys_arch_mbox_tryfetch(sys_mbox_t *mbox, void **msg)
 | |
| {
 | |
|   BaseType_t ret;
 | |
|   void *msg_dummy;
 | |
| 
 | |
|   if (msg == NULL) {
 | |
|     msg = &msg_dummy;
 | |
|   }
 | |
|   ret = xQueueReceive((*mbox)->os_mbox, &(*msg), 0);
 | |
|   if (ret == errQUEUE_EMPTY) {
 | |
|     *msg = NULL;
 | |
|     return SYS_MBOX_EMPTY;
 | |
|   }
 | |
|   LWIP_ASSERT("mbox fetch failed", ret == pdTRUE);
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| void
 | |
| sys_mbox_set_owner(sys_mbox_t *mbox, void* owner)
 | |
| {
 | |
|   if (mbox && *mbox) {
 | |
|     (*mbox)->owner = owner;
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("set mbox=%p owner=%p", *mbox, owner));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Delete a mailbox
 | |
|  *
 | |
|  * @param mbox pointer of the mailbox to delete
 | |
|  */
 | |
| void
 | |
| sys_mbox_free(sys_mbox_t *mbox)
 | |
| {
 | |
|   if ((NULL == mbox) || (NULL == *mbox)) {
 | |
|     return;
 | |
|   }
 | |
|   UBaseType_t msgs_waiting = uxQueueMessagesWaiting((*mbox)->os_mbox);
 | |
|   LWIP_ASSERT("mbox quence not empty", msgs_waiting == 0);
 | |
| 
 | |
|   vQueueDelete((*mbox)->os_mbox);
 | |
|   free(*mbox);
 | |
|   *mbox = NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Create a new thread
 | |
|  *
 | |
|  * @param name thread name
 | |
|  * @param thread thread function
 | |
|  * @param arg thread arguments
 | |
|  * @param stacksize stacksize of the thread
 | |
|  * @param prio priority of the thread
 | |
|  * @return thread ID
 | |
|  */
 | |
| sys_thread_t
 | |
| sys_thread_new(const char *name, lwip_thread_fn thread, void *arg, int stacksize, int prio)
 | |
| {
 | |
|   TaskHandle_t rtos_task;
 | |
|   BaseType_t ret;
 | |
| 
 | |
|   /* LwIP's lwip_thread_fn matches FreeRTOS' TaskFunction_t, so we can pass the
 | |
|      thread function without adaption here. */
 | |
|   ret = xTaskCreatePinnedToCore(thread, name, stacksize, arg, prio, &rtos_task,
 | |
|           CONFIG_LWIP_TCPIP_TASK_AFFINITY);
 | |
| 
 | |
|   if (ret != pdTRUE) {
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   return (sys_thread_t)rtos_task;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Initialize the sys_arch layer
 | |
|  *
 | |
|  */
 | |
| void
 | |
| sys_init(void)
 | |
| {
 | |
|   if (!g_lwip_protect_mutex) {
 | |
|     if (ERR_OK != sys_mutex_new(&g_lwip_protect_mutex)) {
 | |
|       ESP_LOGE(TAG, "sys_init: failed to init lwip protect mutex\n");
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Create the pthreads key for the per-thread semaphore storage
 | |
|   pthread_key_create(&sys_thread_sem_key, sys_thread_sem_free);
 | |
| 
 | |
|   esp_vfs_lwip_sockets_register();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Get system ticks
 | |
|  *
 | |
|  * @return system tick counts
 | |
|  */
 | |
| u32_t
 | |
| sys_jiffies(void)
 | |
| {
 | |
|   return xTaskGetTickCount();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Get current time, in miliseconds
 | |
|  *
 | |
|  * @return current time
 | |
|  */
 | |
| u32_t
 | |
| sys_now(void)
 | |
| {
 | |
|   return xTaskGetTickCount() * portTICK_PERIOD_MS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Protect critical region
 | |
|  *
 | |
|  * @note This function is only called during very short critical regions.
 | |
|  *
 | |
|  * @return previous protection level
 | |
|  */
 | |
| sys_prot_t
 | |
| sys_arch_protect(void)
 | |
| {
 | |
|   if (unlikely(!g_lwip_protect_mutex)) {
 | |
|     sys_mutex_new(&g_lwip_protect_mutex);
 | |
|   }
 | |
|   sys_mutex_lock(&g_lwip_protect_mutex);
 | |
|   return (sys_prot_t) 1;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Unprotect critical region
 | |
|  *
 | |
|  * @param pval protection level
 | |
|  */
 | |
| void
 | |
| sys_arch_unprotect(sys_prot_t pval)
 | |
| {
 | |
|   LWIP_UNUSED_ARG(pval);
 | |
|   sys_mutex_unlock(&g_lwip_protect_mutex);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * get per thread semaphore
 | |
|  */
 | |
| sys_sem_t*
 | |
| sys_thread_sem_get(void)
 | |
| {
 | |
|   sys_sem_t *sem = pthread_getspecific(sys_thread_sem_key);
 | |
| 
 | |
|   if (!sem) {
 | |
|       sem = sys_thread_sem_init();
 | |
|   }
 | |
|   LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sem_get s=%p\n", sem));
 | |
|   return sem;
 | |
| }
 | |
| 
 | |
| static void
 | |
| sys_thread_sem_free(void* data) // destructor for TLS semaphore
 | |
| {
 | |
|   sys_sem_t *sem = (sys_sem_t*)(data);
 | |
| 
 | |
|   if (sem && *sem){
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sem del, sem=%p\n", *sem));
 | |
|     vSemaphoreDelete(*sem);
 | |
|   }
 | |
| 
 | |
|   if (sem) {
 | |
|     LWIP_DEBUGF(ESP_THREAD_SAFE_DEBUG, ("sem pointer del, sem_p=%p\n", sem));
 | |
|     free(sem);
 | |
|   }
 | |
| }
 | |
| 
 | |
| sys_sem_t*
 | |
| sys_thread_sem_init(void)
 | |
| {
 | |
|   sys_sem_t *sem = (sys_sem_t*)mem_malloc(sizeof(sys_sem_t*));
 | |
| 
 | |
|   if (!sem){
 | |
|     ESP_LOGE(TAG, "thread_sem_init: out of memory");
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   *sem = xSemaphoreCreateBinary();
 | |
|   if (!(*sem)){
 | |
|     free(sem);
 | |
|     ESP_LOGE(TAG, "thread_sem_init: out of memory");
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   pthread_setspecific(sys_thread_sem_key, sem);
 | |
|   return sem;
 | |
| }
 | |
| 
 | |
| void
 | |
| sys_thread_sem_deinit(void)
 | |
| {
 | |
|   sys_sem_t *sem = pthread_getspecific(sys_thread_sem_key);
 | |
|   if (sem != NULL) {
 | |
|     sys_thread_sem_free(sem);
 | |
|     pthread_setspecific(sys_thread_sem_key, NULL);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| sys_delay_ms(uint32_t ms)
 | |
| {
 | |
|   vTaskDelay(ms / portTICK_PERIOD_MS);
 | |
| }
 |