新手也能懂!用沁恒CH579低功耗蓝牙芯片的TMOS系统点个灯(附完整代码)

张开发
2026/4/10 19:03:50 15 分钟阅读

分享文章

新手也能懂!用沁恒CH579低功耗蓝牙芯片的TMOS系统点个灯(附完整代码)
从零玩转沁恒CH579TMOS任务管理系统点亮LED全攻略第一次拿到沁恒CH579开发板时看着官方例程里那些TMOS相关的代码我完全摸不着头脑——什么是TaskIDEventID又该怎么用直到用最基础的点灯实验跑通了整个流程才恍然大悟。这篇文章就是带你用最简单的方式理解TMOS这个任务管理系统的运作机制。1. 准备工作与环境搭建在开始之前我们需要准备好硬件和软件环境。CH579是沁恒推出的一款集成低功耗蓝牙功能的微控制器特别适合物联网设备开发。它的TMOSTask Management Operating System是一个轻量级任务管理系统能有效管理多任务调度。硬件准备清单CH579开发板建议选用官方评估板USB数据线用于供电和调试LED灯开发板通常已板载杜邦线如需外接LED软件工具链MounRiver Studio沁恒推荐的集成开发环境WCHISPTool烧录工具官方SDK包含TMOS系统库和示例代码安装完MounRiver Studio后从沁恒官网下载最新的CH579 SDK包。解压后可以在EVT/EXAM/目录下找到BLE/peripheral示例工程这是我们今天要修改的基础项目。提示初次使用CH579时建议先烧录原始peripheral例程确认开发环境和蓝牙功能正常工作。2. TMOS核心概念解析TMOS系统虽然名字里有OS但它并不是传统意义上的操作系统而是一个事件驱动的任务管理系统。理解这几个核心概念是成功使用它的关键2.1 任务标识符TaskID在TMOS中每个独立的功能模块都被视为一个任务需要一个唯一的TaskID来标识。这就像给每个人分配一个工号系统通过这个ID来管理不同的任务。#define TEST_TASK_ID 0x05 // 通常从0x05开始前几个ID被系统保留2.2 事件标识符EventID事件是触发任务执行的信号。一个任务可以响应多种事件每个事件都需要一个唯一的EventID。这相当于给每种工作指令编上号码。#define TEST_LED_TOGGLE_EVT 0x0001 // 定义LED切换事件2.3 事件处理函数这是任务响应事件时实际执行的代码。当指定的事件发生时系统会调用对应的处理函数。uint16_t Test_ProcessEvent(uint8 task_id, uint16 events) { if (events TEST_LED_TOGGLE_EVT) { // 在这里实现LED切换逻辑 return (events ^ TEST_LED_TOGGLE_EVT); // 清除已处理的事件标志 } return 0; }2.4 任务初始化在使用任务前需要向系统注册它。这包括设置任务ID和对应的事件处理函数。void Test_Init(void) { tmos_task_id taskID TMOS_ProcessEventRegister(Test_ProcessEvent); ASSERT(taskID TEST_TASK_ID); }3. 实战用TMOS实现LED闪烁现在我们把上述概念组合起来实现一个通过TMOS控制LED周期性闪烁的完整示例。3.1 修改peripheral.h文件首先在peripheral.h中添加我们的任务和事件定义// 在文件开头添加以下定义 #define TEST_TASK_ID 0x05 #define TEST_LED_TOGGLE_EVT 0x0001 // 声明任务处理函数 extern uint16_t Test_ProcessEvent(uint8 task_id, uint16 events);3.2 修改peripheral.c文件接下来在peripheral.c中实现具体功能// 添加GPIO初始化假设LED接在PB4引脚 void LED_Init(void) { GPIOB_SetBits(GPIO_Pin_4); GPIOB_ModeCfg(GPIO_Pin_4, GPIO_ModeOut_PP_5mA); } // 实现事件处理函数 uint16_t Test_ProcessEvent(uint8 task_id, uint16 events) { if (events TEST_LED_TOGGLE_EVT) { GPIOB_InverseBits(GPIO_Pin_4); // 翻转LED状态 // 设置下一次事件触发500ms后 tmos_set_event(TEST_TASK_ID, TEST_LED_TOGGLE_EVT, 500); return (events ^ TEST_LED_TOGGLE_EVT); } return 0; } // 任务初始化函数 void Test_Init(void) { tmos_task_id taskID TMOS_ProcessEventRegister(Test_ProcessEvent); ASSERT(taskID TEST_TASK_ID); // 启动第一个LED切换事件 tmos_set_event(TEST_TASK_ID, TEST_LED_TOGGLE_EVT, 500); }3.3 修改主函数最后在main()函数中调用我们的初始化函数int main(void) { // 系统初始化 SetSysClock(CLK_SOURCE_PLL_60MHz); LED_Init(); // 初始化LED GPIO Test_Init(); // 初始化我们的测试任务 while(1) { TMOS_SystemProcess(); // TMOS主循环 } }4. TMOS工作原理深度解析理解了基本用法后让我们深入看看TMOS是如何在幕后工作的。4.1 事件调度机制TMOS的核心是一个事件调度器它维护着一个事件队列。当调用tmos_set_event()时实际上是在向这个队列添加一个待处理事件。TMOS_SystemProcess()函数会不断检查并处理这些事件。事件可以设置触发时间实现定时功能。系统内部有一个硬件定时器通常是SysTick来提供时间基准。4.2 任务优先级处理虽然TMOS不是实时操作系统但它有一定的优先级处理机制定时到期的优先于未定时的先设置的事件优先处理同一任务内的事件按位顺序处理4.3 内存占用分析TMOS非常轻量主要内存消耗包括组件内存占用说明任务表每个任务约8字节存储任务ID和处理函数指针事件队列每个事件约4字节存储事件ID和定时信息系统变量约20字节各种状态标志和计数器整个TMOS内核通常占用不超过100字节RAM非常适合资源受限的嵌入式系统。5. 进阶技巧与常见问题掌握了基础用法后这里分享一些实战中总结的经验技巧。5.1 多任务协同工作当系统中有多个任务时需要注意任务间的协作// 任务A中触发任务B的事件 tmos_set_event(TASK_B_ID, SOME_EVENT, 0); // 跨任务通信可以通过全局变量但要注意数据一致性5.2 低功耗模式下的TMOSCH579的一大优势是低功耗TMOS可以很好地与低功耗模式配合在无事件处理时进入低功耗模式使用TMOS_PowerControl()函数管理功耗蓝牙事件会自动唤醒系统5.3 常见问题排查LED不闪烁检查GPIO初始化是否正确确认LED极性有些板子是低电平点亮用调试器查看事件是否被正确触发系统卡死检查是否有未处理的事件确保所有任务处理函数都正确返回避免在事件处理函数中执行耗时操作蓝牙功能异常确保没有占用蓝牙任务的时间片检查任务ID是否冲突蓝牙任务通常使用0x01-0x046. 扩展应用结合蓝牙控制LED既然CH579具有蓝牙功能我们可以轻松扩展这个例子实现通过手机APP控制LED。6.1 修改蓝牙特征值在peripheral.c中找到蓝牙特征值定义添加一个控制LED的特征// 添加特征值UUID #define LED_CONTROL_UUID {0xFF,0x10,0x00,0x01,0x12,0x23,0x34,0x45,0x56,0x67,0x78,0x89,0x9A,0xAB,0xBC,0xCD} // 在GATT服务中添加特征 uint8_t ledControlValue 0; CONST uint8_t ledControlCharProps GATT_PROP_READ | GATT_PROP_WRITE;6.2 处理蓝牙写入事件在蓝牙事件处理函数中添加对LED控制命令的响应case GATT_MSG_EVENT: if (pMsg-method ATT_READ_BY_HANDLE_RSP) { // 处理读取 } else if (pMsg-method ATT_WRITE_REQ || pMsg-method ATT_WRITE_CMD) { if (pMsg-msg.writeReq.handle ledControlHandle) { ledControlValue pMsg-msg.writeReq.pValue[0]; if (ledControlValue) { tmos_set_event(TEST_TASK_ID, TEST_LED_TOGGLE_EVT, 0); } } } break;6.3 手机APP端配置使用任何支持自定义GATT的蓝牙调试APP如nRF Connect连接设备后查找我们的LED控制服务向特征值写入0x01打开LED0x00关闭也可以读取当前LED状态注意实际产品开发中应该定义更完善的通信协议包括错误处理和安全机制。

更多文章