近期我们的产品中用到了南京沁恒推出的低功耗蓝牙芯片CH582。这款芯片与常见的STM32的ARM Cortex-M处理器架构存在显著差异,而是采用了新兴的指令集架构----RISC-V。其中,CH582的TMOS任务管理系统给我留下了深刻的印象。尽管它与传统的操作系统有一些相似之处,但又有其独特之处。
TMOS任务管理系统是CH582的一大亮点。它类似于操作系统的概念,能够实现任务的调度、内存管理等功能。然而,与常见的操作系统相比,TMOS更加轻量级,旨在满足低功耗、实时响应和高能效的需求。这种设计使得CH582在处理任务时具有出色的性能和效率,同时保持了较低的功耗。
蓝牙要与多个设备连接并实现多功能和多任务,这就会导致了调度问题。尽管软件和协议栈可以扩展,但最底层的执行单元只有一个。为了处理多事件和多任务切换,需要将事件和任务对应起来。为此,TMOS被引入作为一个操作系统抽象层。
TMOS是调度的核心,BLE协议栈、profile定义和所有应用都围绕它实现。TMOS不同于传统的操作系统,它是一个允许软件建立和执行事件的循环。
举例来说,TMOS是通过时间片轮询的方式实现多任务调度运行,实际上每次只有一个任务运行。系统时钟来源于芯片RTC,单位为625us。用户通过注册任务(Task)将自定义的事件(Event)添加到TMOS的任务链表中,由TMOS进行调度运行。Event事件标志位,为1则运行,为0则不运行。
综上所述,TMOS通过任务注册、任务调度和时间片轮询的方式实现多任务的调度和管理。这种机制可以有效地实现多事件和多任务的切换,提高系统的效率和响应速度。
这是一款面向RISC-V,ARM等内核MCU集成开发环境,界面如下,感兴趣的可以自行下载。
ID定义是全局变量。
代码如下(示例):
uint8_t ROPE_TaskID = INVALID_TASK_ID; // Task ID for internal task/event processing
ROPE_TaskID = TMOS_ProcessEventRegister(ROPE_ProcessEvent); // 向系统注册了一个任务
/**
* @brief register process event callback function//注册处理事件回调函数
*
* @param eventCb-events callback function//事件回调函数
*
* @return 0xFF - error,others-task id//错误,其他-任务id
*/
extern tmosTaskID TMOS_ProcessEventRegister( pTaskEventHandlerFn eventCb );
编写任务初始化进程,并需要添加到TMOS初始化进程中,这就是说系统启动后不能动态添加功能(新的Task ID);
代码如下(示例):
CH58X_BLEInit();
HAL_Init();
ReadImageFlag();
rope_init();//用户定义初始化
GAPRole_PeripheralInit();
Peripheral_Init();
代码如下(示例):
/*********************************************************************
* @fn Peripheral_ProcessEvent
*
* @brief Peripheral Application Task event processor. This function
* is called to process all events for the task. Events
* include timers, messages and any other user defined events.
*
* @param task_id - The TMOS assigned task ID. TMOS分配的任务ID
* @param events - events to process. This is a bit map and can
* contain more than one event.
*
* @return events not processed
*/
uint16_t ROPE_ProcessEvent(uint8_t task_id, uint16_t events)
{
if (events & SYS_EVENT_MSG)
{
uint8_t *pMsg;
if ((pMsg = tmos_msg_receive(ROPE_TaskID)) != NULL)
{
tmos_msg_deallocate(pMsg);
}
return (events ^ SYS_EVENT_MSG);
}
........省略
}
事件名按位定义,每一层taskID最多包含1个消息事件和15个任务事件(共16位)
代码如下(示例):
#define READ_BAT_EVENT 1
#define READ_UART_EVENT 2
#define E1000MS_EVENT 4
#define E100MS_EVENT 8
#define E10MS_EVENT 16
#define SLEEP_EVENT 20
代码如下(示例):
/**
* @brief start a event immediately
*
* @param taskID - task ID of event
* @param event - event value
*
* @return 0 - SUCCESS.
*/
extern bStatus_t tmos_set_event( tmosTaskID taskID, tmosEvents event );//立即启动事件
tmos_set_event(ROPE_TaskID, READ_UART_EVENT);//立刻执行
代码如下(示例):
/**
* @brief start a event after period of time
*
* @param taskID - task ID to set event for
* @param event - event to be notified with
* @param time - timeout value
*
* @return TRUE,FALSE.
*/
extern BOOL tmos_start_task( tmosTaskID taskID, tmosEvents event, tmosTimer time );//在一段时间后开始一个事件
tmos_start_task(ROPE_TaskID, E100MS_EVENT, 80); //80 * 0.625ms后执行一次 50ms
时基函数循环
while (1)
{
if (stSysTime.flg._10ms + TEN_MILLISECOND < Time_millis()) //10ms
{
stSysTime.flg._10ms = Time_millis();
//用户代码
}
if (stSysTime.flg._50ms + FIFTY_MILLISECOND < Time_millis()) //50ms
{
stSysTime.flg._50ms = Time_millis();
//用户代码
}
if (stSysTime.flg._100ms + BEST_MILLISECOND < Time_millis()) //100ms
{
stSysTime.flg._100ms = Time_millis();
//用户代码
}
if (stSysTime.flg._1s + THOUSAND_MILLISECOND < Time_millis()) //1s
{
stSysTime.flg._1s = Time_millis();
//用户代码
}
}
其实这个任务循环写法和我们写时基函数循环执行有点相似。
uint16_t ROPE_ProcessEvent(uint8_t task_id, uint16_t events)
{
if(events & ID事件1)
{
//用户代码
tmos_start_task(ID, ID事件1, 执行时间1);
return (events ^ ID事件1);
}
if(events & ID事件2)
{
//用户代码
tmos_start_task(ID, ID事件2, 执行时间2);
return (events ^ ID事件2);
}
if(events & ID事件3)
{
//用户代码
tmos_start_task(ID, ID事件3, 执行时间3);
return (events ^ ID事件3);
}
if(events & ID事件4)
{
//用户代码
tmos_start_task(ID, ID事件4, 执行时间4);
return (events ^ ID事件4);
}
}
具体代码如下(示例):
/*********************************************************************
@fn ROPE_ProcessEvent
@brief Peripheral Application Task event processor. This function
is called to process all events for the task. Events
include timers, messages and any other user defined events.
@param task_id - The TMOS assigned task ID. TMOS分配的任务ID
@param events - events to process. This is a bit map and can
contain more than one event.
@return events not processed
*/
uint16_t ROPE_ProcessEvent(uint8_t task_id, uint16_t events)
{
if (events & SYS_EVENT_MSG)
{
uint8_t *pMsg;
if ((pMsg = tmos_msg_receive(ROPE_TaskID)) != NULL)
{
tmos_msg_deallocate(pMsg);
}
return (events ^ SYS_EVENT_MSG);
}
if (events & READ_UART_EVENT)
{
if (rope.u8UartFlg == rope.u8RxNum && rope.u8RxNum)
{
analy_recv(rope.u8RxBuf, rope.u8RxNum);
memset(rope.u8RxBuf, 0, 100);
rope.u8RxNum = 0;
}
else
{
rope.u8UartFlg = rope.u8RxNum;
}
tmos_start_task(ROPE_TaskID, READ_UART_EVENT, 8); // 8 * 0.625ms执行一次 2.5MS
return (events ^ READ_UART_EVENT);
}
if (events & E1000MS_EVENT)
{
tmos_start_task(ROPE_TaskID, E1000MS_EVENT, 1600); // 1600 * 0.625ms执行一次 1000MS
off_power_task();
return (events ^ E1000MS_EVENT);
}
if (events & E100MS_EVENT)
{
tmos_start_task(ROPE_TaskID, E100MS_EVENT, 40); // 40 * 0.625ms执行一次 25MS
if(IO_KEY_POWER.read() == 1)
{
app_bat_gather();
app_led_display();
app_adc_action();
}
return (events ^ E100MS_EVENT);
}
if (events & E10MS_EVENT)
{
tmos_start_task(ROPE_TaskID, E10MS_EVENT, 16); // 16 * 0.625ms执行一次 10MS
key_scan();
return (events ^ E10MS_EVENT);
}
}
主循环不停调用TMOS_SystemProcess,查询可执行event事件;如果开始HAL_SLEEP,芯片开启低功耗睡眠模式,Tmos会开启RTC唤醒功能,事件被执行前会自动唤醒,运行事件代码。
/*********************************************************************
@fn Main_Circulation
@brief 主循环
@return none
*/
__HIGH_CODE
__attribute__((noinline))
void Main_Circulation()
{
while(1)
{
TMOS_SystemProcess();
}
}
以上就是我对南京沁恒WCH TMOS的个人学习总结,本人能力有限,如有错误,还请见谅指出。
感谢你的观看,谢谢!