FreeRTOS互斥量实战:血氧监测系统设计

在FreeRTOS中,互斥量是保护共享资源、防止数据竞争的关键工具,下面结合血氧监测系统的设计来详细说明。

复制代码
/* 定义血氧数据结构体 */
typedef struct {
    uint32_t timestamp;   // 时间戳
    int16_t raw_ppg;      // 原始光电容积脉搏波
    uint8_t spo2;         // 血氧饱和度
    uint16_t heart_rate;  // 心率
} OxyData_t;

/* 共享的全局数据变量 */
OxyData_t currentOxyData;
SemaphoreHandle_t xOxyDataMutex; /* 互斥量句柄 */

/* 数据采集任务 */
void vDataAcquisitionTask(void *pvParameters) {
    OxyData_t localData; // 使用局部变量暂存数据
    for (;;) {
        /* 从传感器读取数据到局部变量... */
        read_sensors(&localData);
        
        /* 获取互斥量,保护对共享数据的写操作 */
        if (xSemaphoreTake(xOxyDataMutex, pdMS_TO_TICKS(100)) == pdTRUE) {
            /* 进入临界区,将局部数据拷贝到共享全局变量 */
            memcpy(&currentOxyData, &localData, sizeof(OxyData_t));
            /* 操作完成后立即释放互斥量 */
            xSemaphoreGive(xOxyDataMutex);
        } else {
            /* 获取互斥量超时处理,例如记录错误或重试 */
        }
        vTaskDelay(pdMS_TO_TICKS(20)); // 假设每20ms采集一次
    }
}

/* 数据处理与显示任务 */
void vDataProcessingTask(void *pvParameters) {
    OxyData_t localCopy; // 用于存放共享数据的本地副本
    for (;;) {
        /* 获取互斥量,保护对共享数据的读操作 */
        if (xSemaphoreTake(xOxyDataMutex, portMAX_DELAY) == pdTRUE) {
            /* 进入临界区,快速将共享数据拷贝到本地 */
            memcpy(&localCopy, &currentOxyData, sizeof(OxyData_t));
            /* 拷贝完成后立即释放互斥量,减少持有时间 */
            xSemaphoreGive(xOxyDataMutex);
            
            /* 在临界区外对本地副本进行计算和显示操作 */
            calculate_spo2(&localCopy);
            update_display(&localCopy);
        }
        vTaskDelay(pdMS_TO_TICKS(100)); // 假设每100ms处理一次
    }
}

/* 主函数中创建互斥量和任务 */
int main(void) {
    /* 硬件初始化... */
    
    /* 创建互斥量 */
    xOxyDataMutex = xSemaphoreCreateMutex();
    if (xOxyDataMutex == NULL) {
        /* 互斥量创建失败,错误处理 */
    }
    
    /* 创建任务 */
    xTaskCreate(vDataAcquisitionTask, "Acq", 1024, NULL, 2, NULL);
    xTaskCreate(vDataProcessingTask, "Proc", 1024, NULL, 3, NULL); /* 赋予处理任务更高优先级 */
    
    vTaskStartScheduler();
    
    for (;;);
    return 0;
}

使用互斥量的重要注意事项

在设计血氧监测这类关键系统时,以下几点尤为重要:

  • 尽快释放互斥量​:获取互斥量后,应只执行最基本的共享资源操作,然后立即释放。这能最大限度地减少对其他任务的阻塞时间。

  • 防止死锁​:如果一个任务需要获取多个互斥量,要确保所有任务都以相同的顺序获取它们。例如,总是先获取互斥量A,再获取互斥量B,避免循环等待。

  • 不得在中断服务程序中使用 ​:互斥量的获取和释放操作绝对不能在中断服务程序中调用。如果需要在中断中同步,应使用专用的二进制信号量。

  • 考虑递归互斥量 ​:如果一个任务可能会多次获取同一个互斥量(例如,在调用链中多个函数都需要访问受保护的资源),则应使用 xSemaphoreCreateRecursiveMutex()创建的递归互斥量,并配套使用 xSemaphoreTakeRecursive()xSemaphoreGiveRecursive(),以避免任务自身死锁。

相关推荐
小程同学>o<1 天前
Linux 应用层开发入门(二十五)| 网络编程
linux·网络·嵌入式软件·嵌入式应用层·应用层开发·linux应用层开发
青鱼294 天前
时间片在FreeRTOS中的含义解析
freertos·时间片轮转
济6174 天前
FreeRTOS 控制任务设计 (2)--- 运动学逆解 + PID 闭环 + PWM 驱动全流程实现
stm32·单片机·嵌入式·freertos
济6174 天前
FreeRTOS 控制任务设计 (1)--- 双模式闭环控制:IDLE/RUN 状态机与任务通知机制
stm32·单片机·嵌入式·freertos
明月清了个风6 天前
libmodbus笔记
笔记·嵌入式软件·libmodbus
W.W.H.8 天前
FreeRTOS移植(保姆级教程)
经验分享·单片机·操作系统·freertos·rtos
波特率1152008 天前
单片机启动流程以STM32举例
stm32·单片机·嵌入式硬件·嵌入式·嵌入式软件
波特率11520012 天前
FreeRTOS当中的Mail Queue使用教程(CMSIS_v1)
单片机·操作系统·freertos
济61713 天前
FreeRTOS 通信任务设计(3)---基于状态机的串口协议帧解析
stm32·嵌入式·freertos
济61713 天前
FreeRTOS 通信任务设计(4终)----从字节流到有效帧的完美闭环
stm32·嵌入式·freertos