mirror of
https://github.com/hathach/tinyusb.git
synced 2025-01-24 05:42:57 +08:00
1963 lines
61 KiB
C
1963 lines
61 KiB
C
/*----------------------------------------------------------------------------
|
|
* RL-ARM - RTX
|
|
*----------------------------------------------------------------------------
|
|
* Name: rt_CMSIS.c
|
|
* Purpose: CMSIS RTOS API
|
|
* Rev.: V4.70
|
|
*----------------------------------------------------------------------------
|
|
*
|
|
* Copyright (c) 1999-2009 KEIL, 2009-2013 ARM Germany GmbH
|
|
* All rights reserved.
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
* - Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* - 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.
|
|
* - Neither the name of ARM 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 AND CONTRIBUTORS "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 COPYRIGHT HOLDERS AND CONTRIBUTORS 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.
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
#define __CMSIS_GENERIC
|
|
|
|
#if defined (__CORTEX_M4) || defined (__CORTEX_M4F)
|
|
#include "core_cm4.h"
|
|
#elif defined (__CORTEX_M3)
|
|
#include "core_cm3.h"
|
|
#elif defined (__CORTEX_M0)
|
|
#include "core_cm0.h"
|
|
#else
|
|
#error "Missing __CORTEX_Mx definition"
|
|
#endif
|
|
|
|
#include "rt_TypeDef.h"
|
|
#include "RTX_Config.h"
|
|
#include "rt_System.h"
|
|
#include "rt_Task.h"
|
|
#include "rt_Event.h"
|
|
#include "rt_List.h"
|
|
#include "rt_Time.h"
|
|
#include "rt_Mutex.h"
|
|
#include "rt_Semaphore.h"
|
|
#include "rt_Mailbox.h"
|
|
#include "rt_MemBox.h"
|
|
#include "rt_Memory.h"
|
|
#include "rt_HAL_CM.h"
|
|
|
|
#define os_thread_cb OS_TCB
|
|
|
|
#include "cmsis_os.h"
|
|
|
|
#if (osFeature_Signals != 16)
|
|
#error Invalid "osFeature_Signals" value!
|
|
#endif
|
|
#if (osFeature_Semaphore > 65535)
|
|
#error Invalid "osFeature_Semaphore" value!
|
|
#endif
|
|
#if (osFeature_Wait != 0)
|
|
#error osWait not supported!
|
|
#endif
|
|
|
|
|
|
// ==== Enumeration, structures, defines ====
|
|
|
|
// Service Calls defines
|
|
|
|
#if defined (__CC_ARM) /* ARM Compiler */
|
|
|
|
#define __NO_RETURN __declspec(noreturn)
|
|
|
|
#define osEvent_type osEvent
|
|
#define osEvent_ret_status ret
|
|
#define osEvent_ret_value ret
|
|
#define osEvent_ret_msg ret
|
|
#define osEvent_ret_mail ret
|
|
|
|
#define osCallback_type osCallback
|
|
#define osCallback_ret ret
|
|
|
|
#define SVC_0_1(f,t,...) \
|
|
__svc_indirect(0) t _##f (t(*)()); \
|
|
t f (void); \
|
|
__attribute__((always_inline)) \
|
|
static __inline t __##f (void) { \
|
|
return _##f(f); \
|
|
}
|
|
|
|
#define SVC_1_1(f,t,t1,...) \
|
|
__svc_indirect(0) t _##f (t(*)(t1),t1); \
|
|
t f (t1 a1); \
|
|
__attribute__((always_inline)) \
|
|
static __inline t __##f (t1 a1) { \
|
|
return _##f(f,a1); \
|
|
}
|
|
|
|
#define SVC_2_1(f,t,t1,t2,...) \
|
|
__svc_indirect(0) t _##f (t(*)(t1,t2),t1,t2); \
|
|
t f (t1 a1, t2 a2); \
|
|
__attribute__((always_inline)) \
|
|
static __inline t __##f (t1 a1, t2 a2) { \
|
|
return _##f(f,a1,a2); \
|
|
}
|
|
|
|
#define SVC_3_1(f,t,t1,t2,t3,...) \
|
|
__svc_indirect(0) t _##f (t(*)(t1,t2,t3),t1,t2,t3); \
|
|
t f (t1 a1, t2 a2, t3 a3); \
|
|
__attribute__((always_inline)) \
|
|
static __inline t __##f (t1 a1, t2 a2, t3 a3) { \
|
|
return _##f(f,a1,a2,a3); \
|
|
}
|
|
|
|
#define SVC_4_1(f,t,t1,t2,t3,t4,...) \
|
|
__svc_indirect(0) t _##f (t(*)(t1,t2,t3,t4),t1,t2,t3,t4); \
|
|
t f (t1 a1, t2 a2, t3 a3, t4 a4); \
|
|
__attribute__((always_inline)) \
|
|
static __inline t __##f (t1 a1, t2 a2, t3 a3, t4 a4) { \
|
|
return _##f(f,a1,a2,a3,a4); \
|
|
}
|
|
|
|
#define SVC_1_2 SVC_1_1
|
|
#define SVC_1_3 SVC_1_1
|
|
#define SVC_2_3 SVC_2_1
|
|
|
|
#elif defined (__GNUC__) /* GNU Compiler */
|
|
|
|
#define __NO_RETURN __attribute__((noreturn))
|
|
|
|
typedef uint32_t __attribute__((vector_size(8))) ret64;
|
|
typedef uint32_t __attribute__((vector_size(16))) ret128;
|
|
|
|
#define RET_pointer __r0
|
|
#define RET_int32_t __r0
|
|
#define RET_uint32_t __r0
|
|
#define RET_osStatus __r0
|
|
#define RET_osPriority __r0
|
|
#define RET_osEvent {(osStatus)__r0, {(uint32_t)__r1}, {(void *)__r2}}
|
|
#define RET_osCallback {(void *)__r0, (void *)__r1}
|
|
|
|
#define osEvent_type ret128
|
|
#define osEvent_ret_status (ret128){ret.status}
|
|
#define osEvent_ret_value (ret128){ret.status, ret.value.v}
|
|
#define osEvent_ret_msg (ret128){ret.status, ret.value.v, (uint32_t)ret.def.message_id}
|
|
#define osEvent_ret_mail (ret128){ret.status, ret.value.v, (uint32_t)ret.def.mail_id}
|
|
|
|
#define osCallback_type ret64
|
|
#define osCallback_ret (ret64) {(uint32_t)ret.fp, (uint32_t)ret.arg}
|
|
|
|
#define SVC_ArgN(n) \
|
|
register int __r##n __asm("r"#n);
|
|
|
|
#define SVC_ArgR(n,t,a) \
|
|
register t __r##n __asm("r"#n) = a;
|
|
|
|
#define SVC_Arg0() \
|
|
SVC_ArgN(0) \
|
|
SVC_ArgN(1) \
|
|
SVC_ArgN(2) \
|
|
SVC_ArgN(3)
|
|
|
|
#define SVC_Arg1(t1) \
|
|
SVC_ArgR(0,t1,a1) \
|
|
SVC_ArgN(1) \
|
|
SVC_ArgN(2) \
|
|
SVC_ArgN(3)
|
|
|
|
#define SVC_Arg2(t1,t2) \
|
|
SVC_ArgR(0,t1,a1) \
|
|
SVC_ArgR(1,t2,a2) \
|
|
SVC_ArgN(2) \
|
|
SVC_ArgN(3)
|
|
|
|
#define SVC_Arg3(t1,t2,t3) \
|
|
SVC_ArgR(0,t1,a1) \
|
|
SVC_ArgR(1,t2,a2) \
|
|
SVC_ArgR(2,t3,a3) \
|
|
SVC_ArgN(3)
|
|
|
|
#define SVC_Arg4(t1,t2,t3,t4) \
|
|
SVC_ArgR(0,t1,a1) \
|
|
SVC_ArgR(1,t2,a2) \
|
|
SVC_ArgR(2,t3,a3) \
|
|
SVC_ArgR(3,t4,a4)
|
|
|
|
#if (defined (__CORTEX_M0))
|
|
#define SVC_Call(f) \
|
|
__asm volatile \
|
|
( \
|
|
"ldr r7,="#f"\n\t" \
|
|
"mov r12,r7\n\t" \
|
|
"svc 0" \
|
|
: "=r" (__r0), "=r" (__r1), "=r" (__r2), "=r" (__r3) \
|
|
: "r" (__r0), "r" (__r1), "r" (__r2), "r" (__r3) \
|
|
: "r7", "r12", "lr", "cc" \
|
|
);
|
|
#else
|
|
#define SVC_Call(f) \
|
|
__asm volatile \
|
|
( \
|
|
"ldr r12,="#f"\n\t" \
|
|
"svc 0" \
|
|
: "=r" (__r0), "=r" (__r1), "=r" (__r2), "=r" (__r3) \
|
|
: "r" (__r0), "r" (__r1), "r" (__r2), "r" (__r3) \
|
|
: "r12", "lr", "cc" \
|
|
);
|
|
#endif
|
|
|
|
#define SVC_0_1(f,t,rv) \
|
|
__attribute__((always_inline)) \
|
|
static inline t __##f (void) { \
|
|
SVC_Arg0(); \
|
|
SVC_Call(f); \
|
|
return (t) rv; \
|
|
}
|
|
|
|
#define SVC_1_1(f,t,t1,rv) \
|
|
__attribute__((always_inline)) \
|
|
static inline t __##f (t1 a1) { \
|
|
SVC_Arg1(t1); \
|
|
SVC_Call(f); \
|
|
return (t) rv; \
|
|
}
|
|
|
|
#define SVC_2_1(f,t,t1,t2,rv) \
|
|
__attribute__((always_inline)) \
|
|
static inline t __##f (t1 a1, t2 a2) { \
|
|
SVC_Arg2(t1,t2); \
|
|
SVC_Call(f); \
|
|
return (t) rv; \
|
|
}
|
|
|
|
#define SVC_3_1(f,t,t1,t2,t3,rv) \
|
|
__attribute__((always_inline)) \
|
|
static inline t __##f (t1 a1, t2 a2, t3 a3) { \
|
|
SVC_Arg3(t1,t2,t3); \
|
|
SVC_Call(f); \
|
|
return (t) rv; \
|
|
}
|
|
|
|
#define SVC_4_1(f,t,t1,t2,t3,t4,rv) \
|
|
__attribute__((always_inline)) \
|
|
static inline t __##f (t1 a1, t2 a2, t3 a3, t4 a4) { \
|
|
SVC_Arg4(t1,t2,t3,t4); \
|
|
SVC_Call(f); \
|
|
return (t) rv; \
|
|
}
|
|
|
|
#define SVC_1_2 SVC_1_1
|
|
#define SVC_1_3 SVC_1_1
|
|
#define SVC_2_3 SVC_2_1
|
|
|
|
#elif defined (__ICCARM__) /* IAR Compiler */
|
|
|
|
#define __NO_RETURN __noreturn
|
|
|
|
#define RET_osEvent "=r"(ret.status), "=r"(ret.value), "=r"(ret.def)
|
|
#define RET_osCallback "=r"(ret.fp), "=r"(ret.arg)
|
|
|
|
#define osEvent_type osEvent
|
|
#define osEvent_ret_status ret
|
|
#define osEvent_ret_value ret
|
|
#define osEvent_ret_msg ret
|
|
#define osEvent_ret_mail ret
|
|
|
|
#define osCallback_type uint64_t
|
|
#define osCallback_ret ((uint64_t)ret.fp | ((uint64_t)ret.arg)<<32)
|
|
|
|
#define SVC_Setup(f) \
|
|
__asm( \
|
|
"mov r12,%0\n" \
|
|
:: "r"(&f): "r12" \
|
|
);
|
|
|
|
#define SVC_Ret3() \
|
|
__asm( \
|
|
"ldr r0,[sp,#0]\n" \
|
|
"ldr r1,[sp,#4]\n" \
|
|
"ldr r2,[sp,#8]\n" \
|
|
);
|
|
|
|
#define SVC_0_1(f,t,...) \
|
|
t f (void); \
|
|
_Pragma("swi_number=0") __swi t _##f (void); \
|
|
static inline t __##f (void) { \
|
|
SVC_Setup(f); \
|
|
return _##f(); \
|
|
}
|
|
|
|
#define SVC_1_1(f,t,t1,...) \
|
|
t f (t1 a1); \
|
|
_Pragma("swi_number=0") __swi t _##f (t1 a1); \
|
|
static inline t __##f (t1 a1) { \
|
|
SVC_Setup(f); \
|
|
return _##f(a1); \
|
|
}
|
|
|
|
#define SVC_2_1(f,t,t1,t2,...) \
|
|
t f (t1 a1, t2 a2); \
|
|
_Pragma("swi_number=0") __swi t _##f (t1 a1, t2 a2); \
|
|
static inline t __##f (t1 a1, t2 a2) { \
|
|
SVC_Setup(f); \
|
|
return _##f(a1,a2); \
|
|
}
|
|
|
|
#define SVC_3_1(f,t,t1,t2,t3,...) \
|
|
t f (t1 a1, t2 a2, t3 a3); \
|
|
_Pragma("swi_number=0") __swi t _##f (t1 a1, t2 a2, t3 a3); \
|
|
static inline t __##f (t1 a1, t2 a2, t3 a3) { \
|
|
SVC_Setup(f); \
|
|
return _##f(a1,a2,a3); \
|
|
}
|
|
|
|
#define SVC_4_1(f,t,t1,t2,t3,t4,...) \
|
|
t f (t1 a1, t2 a2, t3 a3, t4 a4); \
|
|
_Pragma("swi_number=0") __swi t _##f (t1 a1, t2 a2, t3 a3, t4 a4); \
|
|
static inline t __##f (t1 a1, t2 a2, t3 a3, t4 a4) { \
|
|
SVC_Setup(f); \
|
|
return _##f(a1,a2,a3,a4); \
|
|
}
|
|
|
|
#define SVC_1_2(f,t,t1,rr) \
|
|
uint64_t f (t1 a1); \
|
|
_Pragma("swi_number=0") __swi uint64_t _##f (t1 a1); \
|
|
static inline t __##f (t1 a1) { \
|
|
t ret; \
|
|
SVC_Setup(f); \
|
|
_##f(a1); \
|
|
__asm("" : rr : :); \
|
|
return ret; \
|
|
}
|
|
|
|
#define SVC_1_3(f,t,t1,rr) \
|
|
t f (t1 a1); \
|
|
void f##_ (t1 a1) { \
|
|
f(a1); \
|
|
SVC_Ret3(); \
|
|
} \
|
|
_Pragma("swi_number=0") __swi void _##f (t1 a1); \
|
|
static inline t __##f (t1 a1) { \
|
|
t ret; \
|
|
SVC_Setup(f##_); \
|
|
_##f(a1); \
|
|
__asm("" : rr : :); \
|
|
return ret; \
|
|
}
|
|
|
|
#define SVC_2_3(f,t,t1,t2,rr) \
|
|
t f (t1 a1, t2 a2); \
|
|
void f##_ (t1 a1, t2 a2) { \
|
|
f(a1,a2); \
|
|
SVC_Ret3(); \
|
|
} \
|
|
_Pragma("swi_number=0") __swi void _##f (t1 a1, t2 a2); \
|
|
static inline t __##f (t1 a1, t2 a2) { \
|
|
t ret; \
|
|
SVC_Setup(f##_); \
|
|
_##f(a1,a2); \
|
|
__asm("" : rr : :); \
|
|
return ret; \
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
// Callback structure
|
|
typedef struct {
|
|
void *fp; // Function pointer
|
|
void *arg; // Function argument
|
|
} osCallback;
|
|
|
|
|
|
// OS Section definitions
|
|
#ifdef OS_SECTIONS_LINK_INFO
|
|
extern const uint32_t os_section_id$$Base;
|
|
extern const uint32_t os_section_id$$Limit;
|
|
#endif
|
|
|
|
// OS Stack Memory for Threads definitions
|
|
extern uint64_t os_stack_mem[];
|
|
extern const uint32_t os_stack_sz;
|
|
|
|
// OS Timers external resources
|
|
extern const osThreadDef_t os_thread_def_osTimerThread;
|
|
extern osThreadId osThreadId_osTimerThread;
|
|
extern const osMessageQDef_t os_messageQ_def_osTimerMessageQ;
|
|
extern osMessageQId osMessageQId_osTimerMessageQ;
|
|
|
|
|
|
// ==== Helper Functions ====
|
|
|
|
/// Convert timeout in millisec to system ticks
|
|
static uint32_t rt_ms2tick (uint32_t millisec) {
|
|
uint32_t tick;
|
|
|
|
if (millisec == osWaitForever) return 0xFFFF; // Indefinite timeout
|
|
if (millisec > 4000000) return 0xFFFE; // Max ticks supported
|
|
|
|
tick = ((1000 * millisec) + os_clockrate - 1) / os_clockrate;
|
|
if (tick > 0xFFFE) return 0xFFFE;
|
|
|
|
return tick;
|
|
}
|
|
|
|
/// Convert Thread ID to TCB pointer
|
|
static P_TCB rt_tid2ptcb (osThreadId thread_id) {
|
|
P_TCB ptcb;
|
|
|
|
if (thread_id == NULL) return NULL;
|
|
|
|
if ((uint32_t)thread_id & 3) return NULL;
|
|
|
|
#ifdef OS_SECTIONS_LINK_INFO
|
|
if ((os_section_id$$Base != 0) && (os_section_id$$Limit != 0)) {
|
|
if (thread_id < (osThreadId)os_section_id$$Base) return NULL;
|
|
if (thread_id >= (osThreadId)os_section_id$$Limit) return NULL;
|
|
}
|
|
#endif
|
|
|
|
ptcb = thread_id;
|
|
|
|
if (ptcb->cb_type != TCB) return NULL;
|
|
|
|
return ptcb;
|
|
}
|
|
|
|
/// Convert ID pointer to Object pointer
|
|
static void *rt_id2obj (void *id) {
|
|
|
|
if ((uint32_t)id & 3) return NULL;
|
|
|
|
#ifdef OS_SECTIONS_LINK_INFO
|
|
if ((os_section_id$$Base != 0) && (os_section_id$$Limit != 0)) {
|
|
if (id < (void *)os_section_id$$Base) return NULL;
|
|
if (id >= (void *)os_section_id$$Limit) return NULL;
|
|
}
|
|
#endif
|
|
|
|
return id;
|
|
}
|
|
|
|
|
|
// ==== Kernel Control ====
|
|
|
|
uint8_t os_initialized; // Kernel Initialized flag
|
|
uint8_t os_running; // Kernel Running flag
|
|
|
|
// Kernel Control Service Calls declarations
|
|
SVC_0_1(svcKernelInitialize, osStatus, RET_osStatus)
|
|
SVC_0_1(svcKernelStart, osStatus, RET_osStatus)
|
|
SVC_0_1(svcKernelRunning, int32_t, RET_int32_t)
|
|
SVC_0_1(svcKernelSysTick, uint32_t, RET_uint32_t)
|
|
|
|
static void sysThreadError (osStatus status);
|
|
osThreadId svcThreadCreate (const osThreadDef_t *thread_def, void *argument);
|
|
osMessageQId svcMessageCreate (const osMessageQDef_t *queue_def, osThreadId thread_id);
|
|
|
|
// Kernel Control Service Calls
|
|
|
|
/// Initialize the RTOS Kernel for creating objects
|
|
osStatus svcKernelInitialize (void) {
|
|
int ret;
|
|
|
|
if (!os_initialized) {
|
|
|
|
// Init Thread Stack Memory (must be 8-byte aligned)
|
|
if ((uint32_t)os_stack_mem & 7) return osErrorNoMemory;
|
|
ret = rt_init_mem(os_stack_mem, os_stack_sz);
|
|
if (ret != 0) return osErrorNoMemory;
|
|
|
|
rt_sys_init(); // RTX System Initialization
|
|
}
|
|
|
|
os_tsk.run->prio = 255; // Highest priority
|
|
|
|
if (!os_initialized) {
|
|
// Create OS Timers resources (Message Queue & Thread)
|
|
osMessageQId_osTimerMessageQ = svcMessageCreate (&os_messageQ_def_osTimerMessageQ, NULL);
|
|
osThreadId_osTimerThread = svcThreadCreate(&os_thread_def_osTimerThread, NULL);
|
|
}
|
|
|
|
sysThreadError(osOK);
|
|
|
|
os_initialized = 1;
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Start the RTOS Kernel
|
|
osStatus svcKernelStart (void) {
|
|
|
|
if (os_running) return osOK;
|
|
|
|
rt_tsk_prio(0, 0); // Lowest priority
|
|
__set_PSP(os_tsk.run->tsk_stack + 8*4); // New context
|
|
os_tsk.run = NULL; // Force context switch
|
|
|
|
rt_sys_start();
|
|
|
|
os_running = 1;
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Check if the RTOS kernel is already started
|
|
int32_t svcKernelRunning (void) {
|
|
return os_running;
|
|
}
|
|
|
|
/// Get the RTOS kernel system timer counter
|
|
uint32_t svcKernelSysTick (void) {
|
|
uint32_t tick, tick0;
|
|
|
|
tick = os_tick_val();
|
|
if (os_tick_ovf()) {
|
|
tick0 = os_tick_val();
|
|
if (tick0 < tick) tick = tick0;
|
|
tick += (os_trv + 1) * (os_time + 1);
|
|
} else {
|
|
tick += (os_trv + 1) * os_time;
|
|
}
|
|
|
|
return tick;
|
|
}
|
|
|
|
// Kernel Control Public API
|
|
|
|
/// Initialize the RTOS Kernel for creating objects
|
|
osStatus osKernelInitialize (void) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
if ((__get_CONTROL() & 1) == 0) { // Privileged mode
|
|
return svcKernelInitialize();
|
|
} else {
|
|
return __svcKernelInitialize();
|
|
}
|
|
}
|
|
|
|
/// Start the RTOS Kernel
|
|
osStatus osKernelStart (void) {
|
|
uint32_t stack[8];
|
|
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
switch (__get_CONTROL() & 0x03) {
|
|
case 0x00: // Privileged Thread mode & MSP
|
|
__set_PSP((uint32_t)(stack + 8)); // Initial PSP
|
|
if (os_flags & 1) {
|
|
__set_CONTROL(0x02); // Set Privileged Thread mode & PSP
|
|
} else {
|
|
__set_CONTROL(0x03); // Set Unprivileged Thread mode & PSP
|
|
}
|
|
__DSB();
|
|
__ISB();
|
|
break;
|
|
case 0x01: // Unprivileged Thread mode & MSP
|
|
return osErrorOS;
|
|
case 0x02: // Privileged Thread mode & PSP
|
|
if ((os_flags & 1) == 0) { // Unprivileged Thread mode requested
|
|
__set_CONTROL(0x03); // Set Unprivileged Thread mode & PSP
|
|
__DSB();
|
|
__ISB();
|
|
}
|
|
break;
|
|
case 0x03: // Unprivileged Thread mode & PSP
|
|
if (os_flags & 1) return osErrorOS; // Privileged Thread mode requested
|
|
break;
|
|
}
|
|
return __svcKernelStart();
|
|
}
|
|
|
|
/// Check if the RTOS kernel is already started
|
|
int32_t osKernelRunning (void) {
|
|
if ((__get_IPSR() != 0) || ((__get_CONTROL() & 1) == 0)) {
|
|
// in ISR or Privileged
|
|
return os_running;
|
|
} else {
|
|
return __svcKernelRunning();
|
|
}
|
|
}
|
|
|
|
/// Get the RTOS kernel system timer counter
|
|
uint32_t osKernelSysTick (void) {
|
|
if (__get_IPSR() != 0) return 0; // Not allowed in ISR
|
|
return __svcKernelSysTick();
|
|
}
|
|
|
|
|
|
// ==== Thread Management ====
|
|
|
|
/// Set Thread Error (for Create functions which return IDs)
|
|
static void sysThreadError (osStatus status) {
|
|
// To Do
|
|
}
|
|
|
|
__NO_RETURN void osThreadExit (void);
|
|
|
|
// Thread Service Calls declarations
|
|
SVC_2_1(svcThreadCreate, osThreadId, const osThreadDef_t *, void *, RET_pointer)
|
|
SVC_0_1(svcThreadGetId, osThreadId, RET_pointer)
|
|
SVC_1_1(svcThreadTerminate, osStatus, osThreadId, RET_osStatus)
|
|
SVC_0_1(svcThreadYield, osStatus, RET_osStatus)
|
|
SVC_2_1(svcThreadSetPriority, osStatus, osThreadId, osPriority, RET_osStatus)
|
|
SVC_1_1(svcThreadGetPriority, osPriority, osThreadId, RET_osPriority)
|
|
|
|
// Thread Service Calls
|
|
|
|
/// Create a thread and add it to Active Threads and set it to state READY
|
|
osThreadId svcThreadCreate (const osThreadDef_t *thread_def, void *argument) {
|
|
P_TCB ptcb;
|
|
OS_TID tsk;
|
|
void *stk;
|
|
|
|
if ((thread_def == NULL) ||
|
|
(thread_def->pthread == NULL) ||
|
|
(thread_def->tpriority < osPriorityIdle) ||
|
|
(thread_def->tpriority > osPriorityRealtime)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if (thread_def->stacksize != 0) { // Custom stack size
|
|
stk = rt_alloc_mem( // Allocate stack
|
|
os_stack_mem,
|
|
thread_def->stacksize
|
|
);
|
|
if (stk == NULL) {
|
|
sysThreadError(osErrorNoMemory); // Out of memory
|
|
return NULL;
|
|
}
|
|
} else { // Default stack size
|
|
stk = NULL;
|
|
}
|
|
|
|
tsk = rt_tsk_create( // Create task
|
|
(FUNCP)thread_def->pthread, // Task function pointer
|
|
(thread_def->tpriority-osPriorityIdle+1) | // Task priority
|
|
(thread_def->stacksize << 8), // Task stack size in bytes
|
|
stk, // Pointer to task's stack
|
|
argument // Argument to the task
|
|
);
|
|
|
|
if (tsk == 0) { // Invalid task ID
|
|
if (stk != NULL) {
|
|
rt_free_mem(os_stack_mem, stk); // Free allocated stack
|
|
}
|
|
sysThreadError(osErrorNoMemory); // Create task failed (Out of memory)
|
|
return NULL;
|
|
}
|
|
|
|
ptcb = (P_TCB)os_active_TCB[tsk - 1]; // TCB pointer
|
|
|
|
*((uint32_t *)ptcb->tsk_stack + 13) = (uint32_t)osThreadExit;
|
|
|
|
return ptcb;
|
|
}
|
|
|
|
/// Return the thread ID of the current running thread
|
|
osThreadId svcThreadGetId (void) {
|
|
OS_TID tsk;
|
|
|
|
tsk = rt_tsk_self();
|
|
if (tsk == 0) return NULL;
|
|
return (P_TCB)os_active_TCB[tsk - 1];
|
|
}
|
|
|
|
/// Terminate execution of a thread and remove it from ActiveThreads
|
|
osStatus svcThreadTerminate (osThreadId thread_id) {
|
|
OS_RESULT res;
|
|
P_TCB ptcb;
|
|
void *stk;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return osErrorParameter;
|
|
|
|
stk = ptcb->priv_stack ? ptcb->stack : NULL; // Private stack
|
|
|
|
res = rt_tsk_delete(ptcb->task_id); // Delete task
|
|
|
|
if (res == OS_R_NOK) return osErrorResource; // Delete task failed
|
|
|
|
if (stk != NULL) {
|
|
rt_free_mem(os_stack_mem, stk); // Free private stack
|
|
}
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Pass control to next thread that is in state READY
|
|
osStatus svcThreadYield (void) {
|
|
rt_tsk_pass(); // Pass control to next task
|
|
return osOK;
|
|
}
|
|
|
|
/// Change priority of an active thread
|
|
osStatus svcThreadSetPriority (osThreadId thread_id, osPriority priority) {
|
|
OS_RESULT res;
|
|
P_TCB ptcb;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return osErrorParameter;
|
|
|
|
if ((priority < osPriorityIdle) || (priority > osPriorityRealtime)) {
|
|
return osErrorValue;
|
|
}
|
|
|
|
res = rt_tsk_prio( // Change task priority
|
|
ptcb->task_id, // Task ID
|
|
priority - osPriorityIdle + 1 // New task priority
|
|
);
|
|
|
|
if (res == OS_R_NOK) return osErrorResource; // Change task priority failed
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Get current priority of an active thread
|
|
osPriority svcThreadGetPriority (osThreadId thread_id) {
|
|
P_TCB ptcb;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return osPriorityError;
|
|
|
|
return (osPriority)(ptcb->prio - 1 + osPriorityIdle);
|
|
}
|
|
|
|
|
|
// Thread Public API
|
|
|
|
/// Create a thread and add it to Active Threads and set it to state READY
|
|
osThreadId osThreadCreate (const osThreadDef_t *thread_def, void *argument) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcThreadCreate(thread_def, argument);
|
|
} else {
|
|
return __svcThreadCreate(thread_def, argument);
|
|
}
|
|
}
|
|
|
|
/// Return the thread ID of the current running thread
|
|
osThreadId osThreadGetId (void) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
return __svcThreadGetId();
|
|
}
|
|
|
|
/// Terminate execution of a thread and remove it from ActiveThreads
|
|
osStatus osThreadTerminate (osThreadId thread_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcThreadTerminate(thread_id);
|
|
}
|
|
|
|
/// Pass control to next thread that is in state READY
|
|
osStatus osThreadYield (void) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcThreadYield();
|
|
}
|
|
|
|
/// Change priority of an active thread
|
|
osStatus osThreadSetPriority (osThreadId thread_id, osPriority priority) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcThreadSetPriority(thread_id, priority);
|
|
}
|
|
|
|
/// Get current priority of an active thread
|
|
osPriority osThreadGetPriority (osThreadId thread_id) {
|
|
if (__get_IPSR() != 0) return osPriorityError;// Not allowed in ISR
|
|
return __svcThreadGetPriority(thread_id);
|
|
}
|
|
|
|
/// INTERNAL - Not Public
|
|
/// Auto Terminate Thread on exit (used implicitly when thread exists)
|
|
__NO_RETURN void osThreadExit (void) {
|
|
__svcThreadTerminate(__svcThreadGetId());
|
|
for (;;); // Should never come here
|
|
}
|
|
|
|
|
|
// ==== Generic Wait Functions ====
|
|
|
|
// Generic Wait Service Calls declarations
|
|
SVC_1_1(svcDelay, osStatus, uint32_t, RET_osStatus)
|
|
#if osFeature_Wait != 0
|
|
SVC_1_3(svcWait, os_InRegs osEvent, uint32_t, RET_osEvent)
|
|
#endif
|
|
|
|
// Generic Wait Service Calls
|
|
|
|
/// Wait for Timeout (Time Delay)
|
|
osStatus svcDelay (uint32_t millisec) {
|
|
if (millisec == 0) return osOK;
|
|
rt_dly_wait(rt_ms2tick(millisec));
|
|
return osEventTimeout;
|
|
}
|
|
|
|
/// Wait for Signal, Message, Mail, or Timeout
|
|
#if osFeature_Wait != 0
|
|
os_InRegs osEvent_type svcWait (uint32_t millisec) {
|
|
osEvent ret;
|
|
|
|
if (millisec == 0) {
|
|
ret.status = osOK;
|
|
return osEvent_ret_status;
|
|
}
|
|
|
|
/* To Do: osEventSignal, osEventMessage, osEventMail */
|
|
rt_dly_wait(rt_ms2tick(millisec));
|
|
ret.status = osEventTimeout;
|
|
|
|
return osEvent_ret_status;
|
|
}
|
|
#endif
|
|
|
|
|
|
// Generic Wait API
|
|
|
|
/// Wait for Timeout (Time Delay)
|
|
osStatus osDelay (uint32_t millisec) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcDelay(millisec);
|
|
}
|
|
|
|
/// Wait for Signal, Message, Mail, or Timeout
|
|
os_InRegs osEvent osWait (uint32_t millisec) {
|
|
osEvent ret;
|
|
|
|
#if osFeature_Wait == 0
|
|
ret.status = osErrorOS;
|
|
return ret;
|
|
#else
|
|
if (__get_IPSR() != 0) { // Not allowed in ISR
|
|
ret.status = osErrorISR;
|
|
return ret;
|
|
}
|
|
return __svcWait(millisec);
|
|
#endif
|
|
}
|
|
|
|
|
|
// ==== Timer Management ====
|
|
|
|
// Timer definitions
|
|
#define osTimerInvalid 0
|
|
#define osTimerStopped 1
|
|
#define osTimerRunning 2
|
|
|
|
// Timer structures
|
|
|
|
typedef struct os_timer_cb_ { // Timer Control Block
|
|
struct os_timer_cb_ *next; // Pointer to next active Timer
|
|
uint8_t state; // Timer State
|
|
uint8_t type; // Timer Type (Periodic/One-shot)
|
|
uint16_t reserved; // Reserved
|
|
uint16_t tcnt; // Timer Delay Count
|
|
uint16_t icnt; // Timer Initial Count
|
|
void *arg; // Timer Function Argument
|
|
const osTimerDef_t *timer; // Pointer to Timer definition
|
|
} os_timer_cb;
|
|
|
|
// Timer variables
|
|
os_timer_cb *os_timer_head; // Pointer to first active Timer
|
|
|
|
|
|
// Timer Helper Functions
|
|
|
|
// Insert Timer into the list sorted by time
|
|
static void rt_timer_insert (os_timer_cb *pt, uint32_t tcnt) {
|
|
os_timer_cb *p, *prev;
|
|
|
|
prev = NULL;
|
|
p = os_timer_head;
|
|
while (p != NULL) {
|
|
if (tcnt < p->tcnt) break;
|
|
tcnt -= p->tcnt;
|
|
prev = p;
|
|
p = p->next;
|
|
}
|
|
pt->next = p;
|
|
pt->tcnt = (uint16_t)tcnt;
|
|
if (p != NULL) {
|
|
p->tcnt -= pt->tcnt;
|
|
}
|
|
if (prev != NULL) {
|
|
prev->next = pt;
|
|
} else {
|
|
os_timer_head = pt;
|
|
}
|
|
}
|
|
|
|
// Remove Timer from the list
|
|
static int rt_timer_remove (os_timer_cb *pt) {
|
|
os_timer_cb *p, *prev;
|
|
|
|
prev = NULL;
|
|
p = os_timer_head;
|
|
while (p != NULL) {
|
|
if (p == pt) break;
|
|
prev = p;
|
|
p = p->next;
|
|
}
|
|
if (p == NULL) return -1;
|
|
if (prev != NULL) {
|
|
prev->next = pt->next;
|
|
} else {
|
|
os_timer_head = pt->next;
|
|
}
|
|
if (pt->next != NULL) {
|
|
pt->next->tcnt += pt->tcnt;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
// Timer Service Calls declarations
|
|
SVC_3_1(svcTimerCreate, osTimerId, const osTimerDef_t *, os_timer_type, void *, RET_pointer)
|
|
SVC_2_1(svcTimerStart, osStatus, osTimerId, uint32_t, RET_osStatus)
|
|
SVC_1_1(svcTimerStop, osStatus, osTimerId, RET_osStatus)
|
|
SVC_1_1(svcTimerDelete, osStatus, osTimerId, RET_osStatus)
|
|
SVC_1_2(svcTimerCall, os_InRegs osCallback, osTimerId, RET_osCallback)
|
|
|
|
// Timer Management Service Calls
|
|
|
|
/// Create timer
|
|
osTimerId svcTimerCreate (const osTimerDef_t *timer_def, os_timer_type type, void *argument) {
|
|
os_timer_cb *pt;
|
|
|
|
if ((timer_def == NULL) || (timer_def->ptimer == NULL)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
pt = timer_def->timer;
|
|
if (pt == NULL) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if ((type != osTimerOnce) && (type != osTimerPeriodic)) {
|
|
sysThreadError(osErrorValue);
|
|
return NULL;
|
|
}
|
|
|
|
if (osThreadId_osTimerThread == NULL) {
|
|
sysThreadError(osErrorResource);
|
|
return NULL;
|
|
}
|
|
|
|
if (pt->state != osTimerInvalid){
|
|
sysThreadError(osErrorResource);
|
|
return NULL;
|
|
}
|
|
|
|
pt->state = osTimerStopped;
|
|
pt->type = (uint8_t)type;
|
|
pt->arg = argument;
|
|
pt->timer = timer_def;
|
|
|
|
return (osTimerId)pt;
|
|
}
|
|
|
|
/// Start or restart timer
|
|
osStatus svcTimerStart (osTimerId timer_id, uint32_t millisec) {
|
|
os_timer_cb *pt;
|
|
uint32_t tcnt;
|
|
|
|
pt = rt_id2obj(timer_id);
|
|
if (pt == NULL) return osErrorParameter;
|
|
|
|
tcnt = rt_ms2tick(millisec);
|
|
if (tcnt == 0) return osErrorValue;
|
|
|
|
switch (pt->state) {
|
|
case osTimerRunning:
|
|
if (rt_timer_remove(pt) != 0) {
|
|
return osErrorResource;
|
|
}
|
|
break;
|
|
case osTimerStopped:
|
|
pt->state = osTimerRunning;
|
|
pt->icnt = (uint16_t)tcnt;
|
|
break;
|
|
default:
|
|
return osErrorResource;
|
|
}
|
|
|
|
rt_timer_insert(pt, tcnt);
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Stop timer
|
|
osStatus svcTimerStop (osTimerId timer_id) {
|
|
os_timer_cb *pt;
|
|
|
|
pt = rt_id2obj(timer_id);
|
|
if (pt == NULL) return osErrorParameter;
|
|
|
|
if (pt->state != osTimerRunning) return osErrorResource;
|
|
|
|
pt->state = osTimerStopped;
|
|
|
|
if (rt_timer_remove(pt) != 0) {
|
|
return osErrorResource;
|
|
}
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Delete timer
|
|
osStatus svcTimerDelete (osTimerId timer_id) {
|
|
os_timer_cb *pt;
|
|
|
|
pt = rt_id2obj(timer_id);
|
|
if (pt == NULL) return osErrorParameter;
|
|
|
|
switch (pt->state) {
|
|
case osTimerRunning:
|
|
rt_timer_remove(pt);
|
|
break;
|
|
case osTimerStopped:
|
|
break;
|
|
default:
|
|
return osErrorResource;
|
|
}
|
|
|
|
pt->state = osTimerInvalid;
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Get timer callback parameters
|
|
os_InRegs osCallback_type svcTimerCall (osTimerId timer_id) {
|
|
os_timer_cb *pt;
|
|
osCallback ret;
|
|
|
|
pt = rt_id2obj(timer_id);
|
|
if (pt == NULL) {
|
|
ret.fp = NULL;
|
|
ret.arg = NULL;
|
|
return osCallback_ret;
|
|
}
|
|
|
|
ret.fp = (void *)pt->timer->ptimer;
|
|
ret.arg = pt->arg;
|
|
|
|
return osCallback_ret;
|
|
}
|
|
|
|
static __INLINE osStatus isrMessagePut (osMessageQId queue_id, uint32_t info, uint32_t millisec);
|
|
|
|
/// Timer Tick (called each SysTick)
|
|
void sysTimerTick (void) {
|
|
os_timer_cb *pt, *p;
|
|
|
|
p = os_timer_head;
|
|
if (p == NULL) return;
|
|
|
|
p->tcnt--;
|
|
while ((p != NULL) && (p->tcnt == 0)) {
|
|
pt = p;
|
|
p = p->next;
|
|
os_timer_head = p;
|
|
isrMessagePut(osMessageQId_osTimerMessageQ, (uint32_t)pt, 0);
|
|
if (pt->type == osTimerPeriodic) {
|
|
rt_timer_insert(pt, pt->icnt);
|
|
} else {
|
|
pt->state = osTimerStopped;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Timer Management Public API
|
|
|
|
/// Create timer
|
|
osTimerId osTimerCreate (const osTimerDef_t *timer_def, os_timer_type type, void *argument) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcTimerCreate(timer_def, type, argument);
|
|
} else {
|
|
return __svcTimerCreate(timer_def, type, argument);
|
|
}
|
|
}
|
|
|
|
/// Start or restart timer
|
|
osStatus osTimerStart (osTimerId timer_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcTimerStart(timer_id, millisec);
|
|
}
|
|
|
|
/// Stop timer
|
|
osStatus osTimerStop (osTimerId timer_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcTimerStop(timer_id);
|
|
}
|
|
|
|
/// Delete timer
|
|
osStatus osTimerDelete (osTimerId timer_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcTimerDelete(timer_id);
|
|
}
|
|
|
|
/// INTERNAL - Not Public
|
|
/// Get timer callback parameters (used by OS Timer Thread)
|
|
os_InRegs osCallback osTimerCall (osTimerId timer_id) {
|
|
return __svcTimerCall(timer_id);
|
|
}
|
|
|
|
|
|
// Timer Thread
|
|
__NO_RETURN void osTimerThread (void const *argument) {
|
|
osCallback cb;
|
|
osEvent evt;
|
|
|
|
for (;;) {
|
|
evt = osMessageGet(osMessageQId_osTimerMessageQ, osWaitForever);
|
|
if (evt.status == osEventMessage) {
|
|
cb = osTimerCall(evt.value.p);
|
|
if (cb.fp != NULL) {
|
|
(*(os_ptimer)cb.fp)(cb.arg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// ==== Signal Management ====
|
|
|
|
// Signal Service Calls declarations
|
|
SVC_2_1(svcSignalSet, int32_t, osThreadId, int32_t, RET_int32_t)
|
|
SVC_2_1(svcSignalClear, int32_t, osThreadId, int32_t, RET_int32_t)
|
|
SVC_2_3(svcSignalWait, os_InRegs osEvent, int32_t, uint32_t, RET_osEvent)
|
|
|
|
// Signal Service Calls
|
|
|
|
/// Set the specified Signal Flags of an active thread
|
|
int32_t svcSignalSet (osThreadId thread_id, int32_t signals) {
|
|
P_TCB ptcb;
|
|
int32_t sig;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return 0x80000000;
|
|
|
|
if (signals & (0xFFFFFFFF << osFeature_Signals)) return 0x80000000;
|
|
|
|
sig = ptcb->events; // Previous signal flags
|
|
|
|
rt_evt_set(signals, ptcb->task_id); // Set event flags
|
|
|
|
return sig;
|
|
}
|
|
|
|
/// Clear the specified Signal Flags of an active thread
|
|
int32_t svcSignalClear (osThreadId thread_id, int32_t signals) {
|
|
P_TCB ptcb;
|
|
int32_t sig;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return 0x80000000;
|
|
|
|
if (signals & (0xFFFFFFFF << osFeature_Signals)) return 0x80000000;
|
|
|
|
sig = ptcb->events; // Previous signal flags
|
|
|
|
rt_evt_clr(signals, ptcb->task_id); // Clear event flags
|
|
|
|
return sig;
|
|
}
|
|
|
|
/// Wait for one or more Signal Flags to become signaled for the current RUNNING thread
|
|
os_InRegs osEvent_type svcSignalWait (int32_t signals, uint32_t millisec) {
|
|
OS_RESULT res;
|
|
osEvent ret;
|
|
|
|
if (signals & (0xFFFFFFFF << osFeature_Signals)) {
|
|
ret.status = osErrorValue;
|
|
return osEvent_ret_status;
|
|
}
|
|
|
|
if (signals != 0) { // Wait for all specified signals
|
|
res = rt_evt_wait(signals, rt_ms2tick(millisec), __TRUE);
|
|
} else { // Wait for any signal
|
|
res = rt_evt_wait(0xFFFF, rt_ms2tick(millisec), __FALSE);
|
|
}
|
|
|
|
if (res == OS_R_EVT) {
|
|
ret.status = osEventSignal;
|
|
ret.value.signals = signals ? signals : os_tsk.run->waits;
|
|
} else {
|
|
ret.status = millisec ? osEventTimeout : osOK;
|
|
ret.value.signals = 0;
|
|
}
|
|
|
|
return osEvent_ret_value;
|
|
}
|
|
|
|
|
|
// Signal ISR Calls
|
|
|
|
/// Set the specified Signal Flags of an active thread
|
|
static __INLINE int32_t isrSignalSet (osThreadId thread_id, int32_t signals) {
|
|
P_TCB ptcb;
|
|
int32_t sig;
|
|
|
|
ptcb = rt_tid2ptcb(thread_id); // Get TCB pointer
|
|
if (ptcb == NULL) return 0x80000000;
|
|
|
|
if (signals & (0xFFFFFFFF << osFeature_Signals)) return 0x80000000;
|
|
|
|
sig = ptcb->events; // Previous signal flags
|
|
|
|
isr_evt_set(signals, ptcb->task_id); // Set event flags
|
|
|
|
return sig;
|
|
}
|
|
|
|
|
|
// Signal Public API
|
|
|
|
/// Set the specified Signal Flags of an active thread
|
|
int32_t osSignalSet (osThreadId thread_id, int32_t signals) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return isrSignalSet(thread_id, signals);
|
|
} else { // in Thread
|
|
return __svcSignalSet(thread_id, signals);
|
|
}
|
|
}
|
|
|
|
/// Clear the specified Signal Flags of an active thread
|
|
int32_t osSignalClear (osThreadId thread_id, int32_t signals) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcSignalClear(thread_id, signals);
|
|
}
|
|
|
|
/// Wait for one or more Signal Flags to become signaled for the current RUNNING thread
|
|
os_InRegs osEvent osSignalWait (int32_t signals, uint32_t millisec) {
|
|
osEvent ret;
|
|
|
|
if (__get_IPSR() != 0) { // Not allowed in ISR
|
|
ret.status = osErrorISR;
|
|
return ret;
|
|
}
|
|
return __svcSignalWait(signals, millisec);
|
|
}
|
|
|
|
|
|
// ==== Mutex Management ====
|
|
|
|
// Mutex Service Calls declarations
|
|
SVC_1_1(svcMutexCreate, osMutexId, const osMutexDef_t *, RET_pointer)
|
|
SVC_2_1(svcMutexWait, osStatus, osMutexId, uint32_t, RET_osStatus)
|
|
SVC_1_1(svcMutexRelease, osStatus, osMutexId, RET_osStatus)
|
|
SVC_1_1(svcMutexDelete, osStatus, osMutexId, RET_osStatus)
|
|
|
|
// Mutex Service Calls
|
|
|
|
/// Create and Initialize a Mutex object
|
|
osMutexId svcMutexCreate (const osMutexDef_t *mutex_def) {
|
|
OS_ID mut;
|
|
|
|
if (mutex_def == NULL) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
mut = mutex_def->mutex;
|
|
if (mut == NULL) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if (((P_MUCB)mut)->cb_type != 0) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
rt_mut_init(mut); // Initialize Mutex
|
|
|
|
return mut;
|
|
}
|
|
|
|
/// Wait until a Mutex becomes available
|
|
osStatus svcMutexWait (osMutexId mutex_id, uint32_t millisec) {
|
|
OS_ID mut;
|
|
OS_RESULT res;
|
|
|
|
mut = rt_id2obj(mutex_id);
|
|
if (mut == NULL) return osErrorParameter;
|
|
|
|
if (((P_MUCB)mut)->cb_type != MUCB) return osErrorParameter;
|
|
|
|
res = rt_mut_wait(mut, rt_ms2tick(millisec)); // Wait for Mutex
|
|
|
|
if (res == OS_R_TMO) {
|
|
return (millisec ? osErrorTimeoutResource : osErrorResource);
|
|
}
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Release a Mutex that was obtained with osMutexWait
|
|
osStatus svcMutexRelease (osMutexId mutex_id) {
|
|
OS_ID mut;
|
|
OS_RESULT res;
|
|
|
|
mut = rt_id2obj(mutex_id);
|
|
if (mut == NULL) return osErrorParameter;
|
|
|
|
if (((P_MUCB)mut)->cb_type != MUCB) return osErrorParameter;
|
|
|
|
res = rt_mut_release(mut); // Release Mutex
|
|
|
|
if (res == OS_R_NOK) return osErrorResource; // Thread not owner or Zero Counter
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Delete a Mutex that was created by osMutexCreate
|
|
osStatus svcMutexDelete (osMutexId mutex_id) {
|
|
OS_ID mut;
|
|
|
|
mut = rt_id2obj(mutex_id);
|
|
if (mut == NULL) return osErrorParameter;
|
|
|
|
if (((P_MUCB)mut)->cb_type != MUCB) return osErrorParameter;
|
|
|
|
rt_mut_delete(mut); // Release Mutex
|
|
|
|
return osOK;
|
|
}
|
|
|
|
|
|
// Mutex Public API
|
|
|
|
/// Create and Initialize a Mutex object
|
|
osMutexId osMutexCreate (const osMutexDef_t *mutex_def) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcMutexCreate(mutex_def);
|
|
} else {
|
|
return __svcMutexCreate(mutex_def);
|
|
}
|
|
}
|
|
|
|
/// Wait until a Mutex becomes available
|
|
osStatus osMutexWait (osMutexId mutex_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcMutexWait(mutex_id, millisec);
|
|
}
|
|
|
|
/// Release a Mutex that was obtained with osMutexWait
|
|
osStatus osMutexRelease (osMutexId mutex_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcMutexRelease(mutex_id);
|
|
}
|
|
|
|
/// Delete a Mutex that was created by osMutexCreate
|
|
osStatus osMutexDelete (osMutexId mutex_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcMutexDelete(mutex_id);
|
|
}
|
|
|
|
|
|
// ==== Semaphore Management ====
|
|
|
|
// Semaphore Service Calls declarations
|
|
SVC_2_1(svcSemaphoreCreate, osSemaphoreId, const osSemaphoreDef_t *, int32_t, RET_pointer)
|
|
SVC_2_1(svcSemaphoreWait, int32_t, osSemaphoreId, uint32_t, RET_int32_t)
|
|
SVC_1_1(svcSemaphoreRelease, osStatus, osSemaphoreId, RET_osStatus)
|
|
SVC_1_1(svcSemaphoreDelete, osStatus, osSemaphoreId, RET_osStatus)
|
|
|
|
// Semaphore Service Calls
|
|
|
|
/// Create and Initialize a Semaphore object
|
|
osSemaphoreId svcSemaphoreCreate (const osSemaphoreDef_t *semaphore_def, int32_t count) {
|
|
OS_ID sem;
|
|
|
|
if (semaphore_def == NULL) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
sem = semaphore_def->semaphore;
|
|
if (sem == NULL) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if (((P_SCB)sem)->cb_type != 0) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if (count > osFeature_Semaphore) {
|
|
sysThreadError(osErrorValue);
|
|
return NULL;
|
|
}
|
|
|
|
rt_sem_init(sem, count); // Initialize Semaphore
|
|
|
|
return sem;
|
|
}
|
|
|
|
/// Wait until a Semaphore becomes available
|
|
int32_t svcSemaphoreWait (osSemaphoreId semaphore_id, uint32_t millisec) {
|
|
OS_ID sem;
|
|
OS_RESULT res;
|
|
|
|
sem = rt_id2obj(semaphore_id);
|
|
if (sem == NULL) return -1;
|
|
|
|
if (((P_SCB)sem)->cb_type != SCB) return -1;
|
|
|
|
res = rt_sem_wait(sem, rt_ms2tick(millisec)); // Wait for Semaphore
|
|
|
|
if (res == OS_R_TMO) return 0; // Timeout
|
|
|
|
return (((P_SCB)sem)->tokens + 1);
|
|
}
|
|
|
|
/// Release a Semaphore
|
|
osStatus svcSemaphoreRelease (osSemaphoreId semaphore_id) {
|
|
OS_ID sem;
|
|
|
|
sem = rt_id2obj(semaphore_id);
|
|
if (sem == NULL) return osErrorParameter;
|
|
|
|
if (((P_SCB)sem)->cb_type != SCB) return osErrorParameter;
|
|
|
|
if (((P_SCB)sem)->tokens == osFeature_Semaphore) return osErrorResource;
|
|
|
|
rt_sem_send(sem); // Release Semaphore
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Delete a Semaphore that was created by osSemaphoreCreate
|
|
osStatus svcSemaphoreDelete (osSemaphoreId semaphore_id) {
|
|
OS_ID sem;
|
|
|
|
sem = rt_id2obj(semaphore_id);
|
|
if (sem == NULL) return osErrorParameter;
|
|
|
|
if (((P_SCB)sem)->cb_type != SCB) return osErrorParameter;
|
|
|
|
rt_sem_delete(sem); // Delete Semaphore
|
|
|
|
return osOK;
|
|
}
|
|
|
|
|
|
// Semaphore ISR Calls
|
|
|
|
/// Release a Semaphore
|
|
static __INLINE osStatus isrSemaphoreRelease (osSemaphoreId semaphore_id) {
|
|
OS_ID sem;
|
|
|
|
sem = rt_id2obj(semaphore_id);
|
|
if (sem == NULL) return osErrorParameter;
|
|
|
|
if (((P_SCB)sem)->cb_type != SCB) return osErrorParameter;
|
|
|
|
if (((P_SCB)sem)->tokens == osFeature_Semaphore) return osErrorResource;
|
|
|
|
isr_sem_send(sem); // Release Semaphore
|
|
|
|
return osOK;
|
|
}
|
|
|
|
|
|
// Semaphore Public API
|
|
|
|
/// Create and Initialize a Semaphore object
|
|
osSemaphoreId osSemaphoreCreate (const osSemaphoreDef_t *semaphore_def, int32_t count) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcSemaphoreCreate(semaphore_def, count);
|
|
} else {
|
|
return __svcSemaphoreCreate(semaphore_def, count);
|
|
}
|
|
}
|
|
|
|
/// Wait until a Semaphore becomes available
|
|
int32_t osSemaphoreWait (osSemaphoreId semaphore_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) return -1; // Not allowed in ISR
|
|
return __svcSemaphoreWait(semaphore_id, millisec);
|
|
}
|
|
|
|
/// Release a Semaphore
|
|
osStatus osSemaphoreRelease (osSemaphoreId semaphore_id) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return isrSemaphoreRelease(semaphore_id);
|
|
} else { // in Thread
|
|
return __svcSemaphoreRelease(semaphore_id);
|
|
}
|
|
}
|
|
|
|
/// Delete a Semaphore that was created by osSemaphoreCreate
|
|
osStatus osSemaphoreDelete (osSemaphoreId semaphore_id) {
|
|
if (__get_IPSR() != 0) return osErrorISR; // Not allowed in ISR
|
|
return __svcSemaphoreDelete(semaphore_id);
|
|
}
|
|
|
|
|
|
// ==== Memory Management Functions ====
|
|
|
|
// Memory Management Helper Functions
|
|
|
|
// Clear Memory Box (Zero init)
|
|
static void rt_clr_box (void *box_mem, void *box) {
|
|
uint32_t *p, n;
|
|
|
|
if (box) {
|
|
p = box;
|
|
for (n = ((P_BM)box_mem)->blk_size; n; n -= 4) {
|
|
*p++ = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Memory Management Service Calls declarations
|
|
SVC_1_1(svcPoolCreate, osPoolId, const osPoolDef_t *, RET_pointer)
|
|
SVC_2_1(sysPoolAlloc, void *, osPoolId, uint32_t, RET_pointer)
|
|
SVC_2_1(sysPoolFree, osStatus, osPoolId, void *, RET_osStatus)
|
|
|
|
// Memory Management Service & ISR Calls
|
|
|
|
/// Create and Initialize memory pool
|
|
osPoolId svcPoolCreate (const osPoolDef_t *pool_def) {
|
|
uint32_t blk_sz;
|
|
|
|
if ((pool_def == NULL) ||
|
|
(pool_def->pool_sz == 0) ||
|
|
(pool_def->item_sz == 0) ||
|
|
(pool_def->pool == NULL)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
blk_sz = (pool_def->item_sz + 3) & ~3;
|
|
|
|
_init_box(pool_def->pool, sizeof(struct OS_BM) + pool_def->pool_sz * blk_sz, blk_sz);
|
|
|
|
return pool_def->pool;
|
|
}
|
|
|
|
/// Allocate a memory block from a memory pool
|
|
void *sysPoolAlloc (osPoolId pool_id, uint32_t clr) {
|
|
void *ptr;
|
|
|
|
if (pool_id == NULL) return NULL;
|
|
|
|
ptr = rt_alloc_box(pool_id);
|
|
if (clr) {
|
|
rt_clr_box(pool_id, ptr);
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
/// Return an allocated memory block back to a specific memory pool
|
|
osStatus sysPoolFree (osPoolId pool_id, void *block) {
|
|
int32_t res;
|
|
|
|
if (pool_id == NULL) return osErrorParameter;
|
|
|
|
res = rt_free_box(pool_id, block);
|
|
if (res != 0) return osErrorValue;
|
|
|
|
return osOK;
|
|
}
|
|
|
|
|
|
// Memory Management Public API
|
|
|
|
/// Create and Initialize memory pool
|
|
osPoolId osPoolCreate (const osPoolDef_t *pool_def) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcPoolCreate(pool_def);
|
|
} else {
|
|
return __svcPoolCreate(pool_def);
|
|
}
|
|
}
|
|
|
|
/// Allocate a memory block from a memory pool
|
|
void *osPoolAlloc (osPoolId pool_id) {
|
|
if ((__get_IPSR() != 0) || ((__get_CONTROL() & 1) == 0)) { // in ISR or Privileged
|
|
return sysPoolAlloc(pool_id, 0);
|
|
} else { // in Thread
|
|
return __sysPoolAlloc(pool_id, 0);
|
|
}
|
|
}
|
|
|
|
/// Allocate a memory block from a memory pool and set memory block to zero
|
|
void *osPoolCAlloc (osPoolId pool_id) {
|
|
if ((__get_IPSR() != 0) || ((__get_CONTROL() & 1) == 0)) { // in ISR or Privileged
|
|
return sysPoolAlloc(pool_id, 1);
|
|
} else { // in Thread
|
|
return __sysPoolAlloc(pool_id, 1);
|
|
}
|
|
}
|
|
|
|
/// Return an allocated memory block back to a specific memory pool
|
|
osStatus osPoolFree (osPoolId pool_id, void *block) {
|
|
if ((__get_IPSR() != 0) || ((__get_CONTROL() & 1) == 0)) { // in ISR or Privileged
|
|
return sysPoolFree(pool_id, block);
|
|
} else { // in Thread
|
|
return __sysPoolFree(pool_id, block);
|
|
}
|
|
}
|
|
|
|
|
|
// ==== Message Queue Management Functions ====
|
|
|
|
// Message Queue Management Service Calls declarations
|
|
SVC_2_1(svcMessageCreate, osMessageQId, const osMessageQDef_t *, osThreadId, RET_pointer)
|
|
SVC_3_1(svcMessagePut, osStatus, osMessageQId, uint32_t, uint32_t, RET_osStatus)
|
|
SVC_2_3(svcMessageGet, os_InRegs osEvent, osMessageQId, uint32_t, RET_osEvent)
|
|
|
|
// Message Queue Service Calls
|
|
|
|
/// Create and Initialize Message Queue
|
|
osMessageQId svcMessageCreate (const osMessageQDef_t *queue_def, osThreadId thread_id) {
|
|
|
|
if ((queue_def == NULL) ||
|
|
(queue_def->queue_sz == 0) ||
|
|
(queue_def->pool == NULL)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
if (((P_MCB)queue_def->pool)->cb_type != 0) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
rt_mbx_init(queue_def->pool, 4*(queue_def->queue_sz + 4));
|
|
|
|
return queue_def->pool;
|
|
}
|
|
|
|
/// Put a Message to a Queue
|
|
osStatus svcMessagePut (osMessageQId queue_id, uint32_t info, uint32_t millisec) {
|
|
OS_RESULT res;
|
|
|
|
if (queue_id == NULL) return osErrorParameter;
|
|
|
|
if (((P_MCB)queue_id)->cb_type != MCB) return osErrorParameter;
|
|
|
|
res = rt_mbx_send(queue_id, (void *)info, rt_ms2tick(millisec));
|
|
|
|
if (res == OS_R_TMO) {
|
|
return (millisec ? osErrorTimeoutResource : osErrorResource);
|
|
}
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Get a Message or Wait for a Message from a Queue
|
|
os_InRegs osEvent_type svcMessageGet (osMessageQId queue_id, uint32_t millisec) {
|
|
OS_RESULT res;
|
|
osEvent ret;
|
|
|
|
if (queue_id == NULL) {
|
|
ret.status = osErrorParameter;
|
|
return osEvent_ret_status;
|
|
}
|
|
|
|
if (((P_MCB)queue_id)->cb_type != MCB) {
|
|
ret.status = osErrorParameter;
|
|
return osEvent_ret_status;
|
|
}
|
|
|
|
res = rt_mbx_wait(queue_id, &ret.value.p, rt_ms2tick(millisec));
|
|
|
|
if (res == OS_R_TMO) {
|
|
ret.status = millisec ? osEventTimeout : osOK;
|
|
return osEvent_ret_value;
|
|
}
|
|
|
|
ret.status = osEventMessage;
|
|
|
|
return osEvent_ret_value;
|
|
}
|
|
|
|
|
|
// Message Queue ISR Calls
|
|
|
|
/// Put a Message to a Queue
|
|
static __INLINE osStatus isrMessagePut (osMessageQId queue_id, uint32_t info, uint32_t millisec) {
|
|
|
|
if ((queue_id == NULL) || (millisec != 0)) {
|
|
return osErrorParameter;
|
|
}
|
|
|
|
if (((P_MCB)queue_id)->cb_type != MCB) return osErrorParameter;
|
|
|
|
if (rt_mbx_check(queue_id) == 0) { // Check if Queue is full
|
|
return osErrorResource;
|
|
}
|
|
|
|
isr_mbx_send(queue_id, (void *)info);
|
|
|
|
return osOK;
|
|
}
|
|
|
|
/// Get a Message or Wait for a Message from a Queue
|
|
static __INLINE os_InRegs osEvent isrMessageGet (osMessageQId queue_id, uint32_t millisec) {
|
|
OS_RESULT res;
|
|
osEvent ret;
|
|
|
|
if ((queue_id == NULL) || (millisec != 0)) {
|
|
ret.status = osErrorParameter;
|
|
return ret;
|
|
}
|
|
|
|
if (((P_MCB)queue_id)->cb_type != MCB) {
|
|
ret.status = osErrorParameter;
|
|
return ret;
|
|
}
|
|
|
|
res = isr_mbx_receive(queue_id, &ret.value.p);
|
|
|
|
if (res != OS_R_MBX) {
|
|
ret.status = osOK;
|
|
return ret;
|
|
}
|
|
|
|
ret.status = osEventMessage;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
// Message Queue Management Public API
|
|
|
|
/// Create and Initialize Message Queue
|
|
osMessageQId osMessageCreate (const osMessageQDef_t *queue_def, osThreadId thread_id) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcMessageCreate(queue_def, thread_id);
|
|
} else {
|
|
return __svcMessageCreate(queue_def, thread_id);
|
|
}
|
|
}
|
|
|
|
/// Put a Message to a Queue
|
|
osStatus osMessagePut (osMessageQId queue_id, uint32_t info, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return isrMessagePut(queue_id, info, millisec);
|
|
} else { // in Thread
|
|
return __svcMessagePut(queue_id, info, millisec);
|
|
}
|
|
}
|
|
|
|
/// Get a Message or Wait for a Message from a Queue
|
|
os_InRegs osEvent osMessageGet (osMessageQId queue_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return isrMessageGet(queue_id, millisec);
|
|
} else { // in Thread
|
|
return __svcMessageGet(queue_id, millisec);
|
|
}
|
|
}
|
|
|
|
|
|
// ==== Mail Queue Management Functions ====
|
|
|
|
// Mail Queue Management Service Calls declarations
|
|
SVC_2_1(svcMailCreate, osMailQId, const osMailQDef_t *, osThreadId, RET_pointer)
|
|
SVC_4_1(sysMailAlloc, void *, osMailQId, uint32_t, uint32_t, uint32_t, RET_pointer)
|
|
SVC_3_1(sysMailFree, osStatus, osMailQId, void *, uint32_t, RET_osStatus)
|
|
|
|
// Mail Queue Management Service & ISR Calls
|
|
|
|
/// Create and Initialize mail queue
|
|
osMailQId svcMailCreate (const osMailQDef_t *queue_def, osThreadId thread_id) {
|
|
uint32_t blk_sz;
|
|
P_MCB pmcb;
|
|
void *pool;
|
|
|
|
if ((queue_def == NULL) ||
|
|
(queue_def->queue_sz == 0) ||
|
|
(queue_def->item_sz == 0) ||
|
|
(queue_def->pool == NULL)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
pmcb = *(((void **)queue_def->pool) + 0);
|
|
pool = *(((void **)queue_def->pool) + 1);
|
|
|
|
if ((pool == NULL) || (pmcb == NULL) || (pmcb->cb_type != 0)) {
|
|
sysThreadError(osErrorParameter);
|
|
return NULL;
|
|
}
|
|
|
|
blk_sz = (queue_def->item_sz + 3) & ~3;
|
|
|
|
_init_box(pool, sizeof(struct OS_BM) + queue_def->queue_sz * blk_sz, blk_sz);
|
|
|
|
rt_mbx_init(pmcb, 4*(queue_def->queue_sz + 4));
|
|
|
|
|
|
return queue_def->pool;
|
|
}
|
|
|
|
/// Allocate a memory block from a mail
|
|
void *sysMailAlloc (osMailQId queue_id, uint32_t millisec, uint32_t isr, uint32_t clr) {
|
|
P_MCB pmcb;
|
|
void *pool;
|
|
void *mem;
|
|
|
|
if (queue_id == NULL) return NULL;
|
|
|
|
pmcb = *(((void **)queue_id) + 0);
|
|
pool = *(((void **)queue_id) + 1);
|
|
|
|
if ((pool == NULL) || (pmcb == NULL)) return NULL;
|
|
|
|
if (isr && (millisec != 0)) return NULL;
|
|
|
|
mem = rt_alloc_box(pool);
|
|
if (clr) {
|
|
rt_clr_box(pool, mem);
|
|
}
|
|
|
|
if ((mem == NULL) && (millisec != 0)) {
|
|
// Put Task to sleep when Memory not available
|
|
if (pmcb->p_lnk != NULL) {
|
|
rt_put_prio((P_XCB)pmcb, os_tsk.run);
|
|
} else {
|
|
pmcb->p_lnk = os_tsk.run;
|
|
os_tsk.run->p_lnk = NULL;
|
|
os_tsk.run->p_rlnk = (P_TCB)pmcb;
|
|
// Task is waiting to allocate a message
|
|
pmcb->state = 3;
|
|
}
|
|
rt_block(rt_ms2tick(millisec), WAIT_MBX);
|
|
}
|
|
|
|
return mem;
|
|
}
|
|
|
|
/// Free a memory block from a mail
|
|
osStatus sysMailFree (osMailQId queue_id, void *mail, uint32_t isr) {
|
|
P_MCB pmcb;
|
|
P_TCB ptcb;
|
|
void *pool;
|
|
void *mem;
|
|
int32_t res;
|
|
|
|
if (queue_id == NULL) return osErrorParameter;
|
|
|
|
pmcb = *(((void **)queue_id) + 0);
|
|
pool = *(((void **)queue_id) + 1);
|
|
|
|
if ((pmcb == NULL) || (pool == NULL)) return osErrorParameter;
|
|
|
|
res = rt_free_box(pool, mail);
|
|
|
|
if (res != 0) return osErrorValue;
|
|
|
|
if (pmcb->state == 3) {
|
|
// Task is waiting to allocate a message
|
|
if (isr) {
|
|
rt_psq_enq (pmcb, (U32)pool);
|
|
rt_psh_req ();
|
|
} else {
|
|
mem = rt_alloc_box(pool);
|
|
if (mem != NULL) {
|
|
ptcb = rt_get_first((P_XCB)pmcb);
|
|
if (pmcb->p_lnk == NULL) {
|
|
pmcb->state = 0;
|
|
}
|
|
rt_ret_val(ptcb, (U32)mem);
|
|
rt_rmv_dly(ptcb);
|
|
rt_dispatch(ptcb);
|
|
}
|
|
}
|
|
}
|
|
|
|
return osOK;
|
|
}
|
|
|
|
|
|
// Mail Queue Management Public API
|
|
|
|
/// Create and Initialize mail queue
|
|
osMailQId osMailCreate (const osMailQDef_t *queue_def, osThreadId thread_id) {
|
|
if (__get_IPSR() != 0) return NULL; // Not allowed in ISR
|
|
if (((__get_CONTROL() & 1) == 0) && (os_running == 0)) {
|
|
// Privileged and not running
|
|
return svcMailCreate(queue_def, thread_id);
|
|
} else {
|
|
return __svcMailCreate(queue_def, thread_id);
|
|
}
|
|
}
|
|
|
|
/// Allocate a memory block from a mail
|
|
void *osMailAlloc (osMailQId queue_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return sysMailAlloc(queue_id, millisec, 1, 0);
|
|
} else { // in Thread
|
|
return __sysMailAlloc(queue_id, millisec, 0, 0);
|
|
}
|
|
}
|
|
|
|
/// Allocate a memory block from a mail and set memory block to zero
|
|
void *osMailCAlloc (osMailQId queue_id, uint32_t millisec) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return sysMailAlloc(queue_id, millisec, 1, 1);
|
|
} else { // in Thread
|
|
return __sysMailAlloc(queue_id, millisec, 0, 1);
|
|
}
|
|
}
|
|
|
|
/// Free a memory block from a mail
|
|
osStatus osMailFree (osMailQId queue_id, void *mail) {
|
|
if (__get_IPSR() != 0) { // in ISR
|
|
return sysMailFree(queue_id, mail, 1);
|
|
} else { // in Thread
|
|
return __sysMailFree(queue_id, mail, 0);
|
|
}
|
|
}
|
|
|
|
/// Put a mail to a queue
|
|
osStatus osMailPut (osMailQId queue_id, void *mail) {
|
|
if (queue_id == NULL) return osErrorParameter;
|
|
if (mail == NULL) return osErrorValue;
|
|
return osMessagePut(*((void **)queue_id), (uint32_t)mail, 0);
|
|
}
|
|
|
|
/// Get a mail from a queue
|
|
os_InRegs osEvent osMailGet (osMailQId queue_id, uint32_t millisec) {
|
|
osEvent ret;
|
|
|
|
if (queue_id == NULL) {
|
|
ret.status = osErrorParameter;
|
|
return ret;
|
|
}
|
|
|
|
ret = osMessageGet(*((void **)queue_id), millisec);
|
|
if (ret.status == osEventMessage) ret.status = osEventMail;
|
|
|
|
return ret;
|
|
}
|