2018-10-08 12:20:49 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*
|
|
|
|
* FreeRTOS to RIOT-OS adaption module for source code compatibility
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef DOXYGEN
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "esp_common.h"
|
|
|
|
#include "irq_arch.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "mutex.h"
|
|
|
|
#include "rmutex.h"
|
|
|
|
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In FreeRTOS different types of semaphores, mutexes and queues are all
|
|
|
|
* mapped to a single generic queue type. With all these different types,
|
|
|
|
* single functions for send, receive, give and take are then used. To be
|
|
|
|
* able to dsitinguish between these different types in RIOT, we need typed
|
|
|
|
* objects.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
uint8_t type; /* type of the mutex, MUST be the first element */
|
|
|
|
mutex_t mutex; /* the mutex */
|
|
|
|
} _mutex_t;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
uint8_t type; /* type of the mutex, MUST be the first element */
|
|
|
|
rmutex_t rmutex; /* the mutex */
|
|
|
|
} _rmutex_t;
|
|
|
|
|
|
|
|
SemaphoreHandle_t xSemaphoreCreateMutex(void)
|
|
|
|
{
|
|
|
|
_mutex_t* _tmp = (_mutex_t*)malloc (sizeof(_mutex_t));
|
|
|
|
_tmp->type = queueQUEUE_TYPE_MUTEX;
|
|
|
|
mutex_init(&_tmp->mutex);
|
|
|
|
|
|
|
|
DEBUG("%s mutex=%p\n", __func__, _tmp);
|
|
|
|
return _tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vSemaphoreDelete( SemaphoreHandle_t xSemaphore )
|
|
|
|
{
|
|
|
|
DEBUG("%s mutex=%p\n", __func__, xSemaphore);
|
|
|
|
|
|
|
|
CHECK_PARAM(xSemaphore != NULL);
|
|
|
|
free(xSemaphore);
|
|
|
|
}
|
|
|
|
|
|
|
|
BaseType_t xSemaphoreGive (SemaphoreHandle_t xSemaphore)
|
|
|
|
{
|
|
|
|
DEBUG("%s mutex=%p\n", __func__, xSemaphore);
|
|
|
|
|
|
|
|
CHECK_PARAM_RET(xSemaphore != NULL, pdFALSE);
|
|
|
|
|
|
|
|
uint8_t type = ((_mutex_t*)xSemaphore)->type;
|
|
|
|
mutex_t* mutex= &((_mutex_t*)xSemaphore)->mutex;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case queueQUEUE_TYPE_MUTEX:
|
|
|
|
mutex_unlock(mutex);
|
|
|
|
break;
|
|
|
|
case queueQUEUE_TYPE_RECURSIVE_MUTEX:
|
|
|
|
return xSemaphoreGiveRecursive (xSemaphore);
|
|
|
|
default:
|
|
|
|
return xQueueGenericSend(xSemaphore, NULL, 0, queueSEND_TO_BACK);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pdTRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BaseType_t xSemaphoreTake (SemaphoreHandle_t xSemaphore,
|
|
|
|
TickType_t xTicksToWait)
|
|
|
|
{
|
|
|
|
DEBUG("%s mutex=%p wait=%u\n", __func__, xSemaphore, xTicksToWait);
|
|
|
|
|
|
|
|
CHECK_PARAM_RET(xSemaphore != NULL, pdFALSE);
|
|
|
|
|
|
|
|
uint8_t type = ((_mutex_t*)xSemaphore)->type;
|
|
|
|
mutex_t* mutex= &((_mutex_t*)xSemaphore)->mutex;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case queueQUEUE_TYPE_MUTEX:
|
|
|
|
{
|
|
|
|
if (xTicksToWait == 0) {
|
2019-03-20 17:37:15 +01:00
|
|
|
return (mutex_trylock(mutex) == 1) ? pdPASS : pdFAIL;
|
2018-10-08 12:20:49 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
mutex_lock(mutex);
|
|
|
|
/* TODO timeout handling */
|
|
|
|
return pdTRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case queueQUEUE_TYPE_RECURSIVE_MUTEX:
|
|
|
|
return xSemaphoreTakeRecursive (xSemaphore, xTicksToWait);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return xQueueGenericReceive(xSemaphore, NULL, xTicksToWait, pdFALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SemaphoreHandle_t xSemaphoreCreateRecursiveMutex(void)
|
|
|
|
{
|
|
|
|
_rmutex_t* _tmp = (_rmutex_t*)malloc (sizeof(_rmutex_t));
|
|
|
|
_tmp->type = queueQUEUE_TYPE_RECURSIVE_MUTEX;
|
|
|
|
rmutex_init(&_tmp->rmutex);
|
|
|
|
|
|
|
|
DEBUG("%s rmutex=%p\n", __func__, _tmp);
|
|
|
|
|
|
|
|
return _tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
BaseType_t xSemaphoreGiveRecursive (SemaphoreHandle_t xSemaphore)
|
|
|
|
{
|
|
|
|
DEBUG("%s rmutex=%p\n", __func__, xSemaphore);
|
|
|
|
|
|
|
|
CHECK_PARAM_RET(xSemaphore != NULL, pdFALSE);
|
|
|
|
CHECK_PARAM_RET(((_rmutex_t*)xSemaphore)->type ==
|
|
|
|
queueQUEUE_TYPE_RECURSIVE_MUTEX, pdFALSE);
|
|
|
|
|
|
|
|
rmutex_unlock(&((_rmutex_t*)xSemaphore)->rmutex);
|
|
|
|
return pdTRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BaseType_t xSemaphoreTakeRecursive (SemaphoreHandle_t xSemaphore,
|
|
|
|
TickType_t xTicksToWait)
|
|
|
|
{
|
|
|
|
DEBUG("%s rmutex=%p wait=%u\n", __func__, xSemaphore, xTicksToWait);
|
|
|
|
|
|
|
|
CHECK_PARAM_RET(xSemaphore != NULL, pdFALSE);
|
|
|
|
CHECK_PARAM_RET(((_rmutex_t*)xSemaphore)->type ==
|
|
|
|
queueQUEUE_TYPE_RECURSIVE_MUTEX, pdFALSE);
|
|
|
|
|
|
|
|
BaseType_t ret = pdTRUE;
|
|
|
|
rmutex_t* rmutex = &((_rmutex_t*)xSemaphore)->rmutex;
|
|
|
|
|
|
|
|
if (xTicksToWait == 0) {
|
2019-03-20 17:37:15 +01:00
|
|
|
ret = (rmutex_trylock(rmutex) == 1) ? pdPASS : pdFAIL;
|
2018-10-08 12:20:49 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
rmutex_lock(&((_rmutex_t*)xSemaphore)->rmutex);
|
|
|
|
/* TODO timeout handling */
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vPortCPUAcquireMutex(portMUX_TYPE *mux)
|
|
|
|
{
|
|
|
|
DEBUG("%s pid=%d prio=%d mux=%p\n", __func__,
|
|
|
|
thread_getpid(), sched_threads[thread_getpid()]->priority, mux);
|
|
|
|
critical_enter();
|
|
|
|
mutex_lock(mux); /* lock the mutex with interrupts disabled */
|
|
|
|
critical_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
void vPortCPUReleaseMutex(portMUX_TYPE *mux)
|
|
|
|
{
|
|
|
|
DEBUG("%s pid=%d prio=%d mux=%p\n", __func__,
|
|
|
|
thread_getpid(), sched_threads[thread_getpid()]->priority, mux);
|
|
|
|
critical_enter();
|
|
|
|
mutex_unlock(mux); /* unlock the mutex with interrupts disabled */
|
|
|
|
critical_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* DOXYGEN */
|