ugfx/src/gos/gos_freertos.c

170 lines
3.7 KiB
C
Raw Normal View History

/*
* This file is subject to the terms of the GFX License. If a copy of
* the license was not distributed with this file, you can obtain one at:
*
* http://ugfx.org/license.html
*/
#include "../../gfx.h"
#include <string.h>
2014-02-09 19:21:08 +00:00
#if GFX_USE_OS_FREERTOS
2014-02-01 16:30:02 +00:00
#if INCLUDE_vTaskDelay != 1
#error "GOS: INCLUDE_vTaskDelay must be defined in FreeRTOSConfig.h"
#endif
#if configUSE_MUTEXES != 1
#error "GOS: configUSE_MUTEXES must be defined in FreeRTOSConfig.h"
#endif
#if configUSE_COUNTING_SEMAPHORES != 1
#error "GOS: configUSE_COUNTING_SEMAPHORES must be defined in FreeRTOSConfig.h"
#endif
#if !GFX_OS_NO_INIT && INCLUDE_xTaskGetSchedulerState != 1 && configUSE_TIMERS != 1
#error "GOS: Either INCLUDE_xTaskGetSchedulerState or configUSE_TIMERS must be defined in FreeRTOSConfig.h"
#endif
#if !GFX_OS_NO_INIT && !GFX_OS_CALL_UGFXMAIN
#error "GOS: Either GFX_OS_NO_INIT or GFX_OS_CALL_UGFXMAIN must be defined for FreeRTOS"
#endif
void _gosInit(void)
{
#if GFX_OS_NO_INIT && !GFX_OS_INIT_NO_WARNING
#warning "GOS: Operating System initialization has been turned off. Make sure you call vTaskStartScheduler()."
#endif
}
#if !GFX_OS_NO_INIT && GFX_OS_CALL_UGFXMAIN
2017-03-04 07:01:02 +00:00
extern void uGFXMain(void);
static DECLARE_THREAD_FUNCTION(startUGFX_FreeRTOS, p) {
(void) p;
uGFXMain();
}
#endif
void _gosPostInit(void)
{
#if !GFX_OS_NO_INIT && GFX_OS_CALL_UGFXMAIN
if (xTaskGetSchedulerState() == taskSCHEDULER_NOT_STARTED) {
2017-03-04 07:01:02 +00:00
gfxThreadCreate(0, GFX_OS_UGFXMAIN_STACKSIZE, NORMAL_PRIORITY, startUGFX_FreeRTOS, 0);
vTaskStartScheduler();
gfxHalt("Unable to start FreeRTOS scheduler. Out of memory?");
}
#endif
}
void _gosDeinit(void)
{
#if !GFX_OS_NO_INIT
vTaskDelete(0);
#endif
}
void* gfxRealloc(void *ptr, size_t oldsz, size_t newsz)
{
void *np;
if (newsz <= oldsz)
return ptr;
np = gfxAlloc(newsz);
if (!np)
return 0;
if (oldsz) {
memcpy(np, ptr, oldsz);
2017-03-01 00:46:14 +00:00
gfxFree(ptr);
}
return np;
}
void gfxSleepMilliseconds(delaytime_t ms)
{
2017-03-01 00:46:14 +00:00
vTaskDelay(gfxMillisecondsToTicks(ms));
}
void gfxSleepMicroseconds(delaytime_t ms)
{
2017-03-01 00:46:14 +00:00
// delay milli seconds - microsecond resolution delay is not supported in FreeRTOS
vTaskDelay(gfxMillisecondsToTicks(ms/1000));
// vUsDelay(ms%1000);
}
2017-03-01 00:46:14 +00:00
void gfxMutexInit(gfxMutex *pmutex)
{
2017-03-01 00:46:14 +00:00
*pmutex = xSemaphoreCreateMutex();
2014-05-10 16:26:44 +00:00
#if GFX_FREERTOS_USE_TRACE
2017-03-01 00:46:14 +00:00
vTraceSetMutexName(*pmutex,"uGFXMutex");
2014-05-10 16:26:44 +00:00
#endif
}
void gfxSemInit(gfxSem* psem, semcount_t val, semcount_t limit)
{
if (val > limit)
val = limit;
2017-03-01 00:46:14 +00:00
*psem = xSemaphoreCreateCounting(limit,val);
2014-05-10 16:13:51 +00:00
#if GFX_FREERTOS_USE_TRACE
2017-03-01 00:46:14 +00:00
vTraceSetSemaphoreName(*psem, "uGFXSema");
2014-05-10 16:13:51 +00:00
#endif
}
bool_t gfxSemWait(gfxSem* psem, delaytime_t ms)
{
2017-03-01 00:46:14 +00:00
if (xSemaphoreTake(*psem, gfxMillisecondsToTicks(ms)) == pdPASS)
return TRUE;
return FALSE;
}
2014-03-13 21:39:02 +00:00
bool_t gfxSemWaitI(gfxSem* psem)
{
portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;
2017-03-01 00:46:14 +00:00
if (xSemaphoreTakeFromISR(*psem, &xHigherPriorityTaskWoken) == pdTRUE)
2014-03-13 21:39:02 +00:00
return TRUE;
return FALSE;
}
void gfxSemSignal(gfxSem* psem)
{
2017-03-01 00:46:14 +00:00
xSemaphoreGive(*psem);
taskYIELD();
}
void gfxSemSignalI(gfxSem* psem)
{
portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;
2017-03-01 00:46:14 +00:00
xSemaphoreGiveFromISR(*psem,&xHigherPriorityTaskWoken);
}
gfxThreadHandle gfxThreadCreate(void *stackarea, size_t stacksz, threadpriority_t prio, DECLARE_THREAD_FUNCTION((*fn),p), void *param)
{
2017-03-01 00:46:14 +00:00
gfxThreadHandle task;
(void) stackarea;
if (stacksz < configMINIMAL_STACK_SIZE)
stacksz = configMINIMAL_STACK_SIZE;
2017-03-01 00:46:14 +00:00
task = 0;
if (xTaskCreate(fn, "uGFX_TASK", stacksz, param, prio, &task) != pdPASS)
return 0;
return task;
}
#if INCLUDE_eTaskGetState == 1
threadreturn_t gfxThreadWait(gfxThreadHandle thread) {
while (eTaskGetState(thread) != eDeleted)
gfxYield();
}
#endif
#endif /* GFX_USE_OS_FREERTOS */