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(),以避免任务自身死锁。

相关推荐
YONYON-R&D2 天前
vTaskDelete 的作用
freertos·vtaskdelete
冷凝雨2 天前
FreeRTOS源码学习(一)内存管理heap_1、heap_3
嵌入式·c·freertos·内存管理·源码分析
绿萝瀑布5 天前
嵌入式GPIO:迟滞与压摆控制
gpio·嵌入式软件
尘似鹤6 天前
设计一个状态机
学习·状态模式·嵌入式软件
大牛攻城狮7 天前
使用stm32cubeide stm32f103 freeRTOS 实现Modbus RTU协议寄存器读写过程详解
stm32·freertos·modbus·stm32cubeide·modbus rtu·stm32从机·工程代码
Hello_Embed11 天前
STM32 环境监测项目笔记(一):DHT11 温湿度传感器原理与驱动实现
c语言·笔记·stm32·单片机·嵌入式软件
嵌软小谭14 天前
# 超简单四步完成FreeRTOS移植到STM32
freertos
OSS_ECAL15 天前
以下將介紹TLE493D-P2B6的概要,以及針對TLE493D-P2B6提供的OSS-ECAL
oss·嵌入式软件·i2c·电子元件·3d霍爾
螺丝钉的扭矩一瞬间产生高能蛋白17 天前
STM32电机控制基础知识
stm32·单片机·嵌入式硬件·嵌入式软件·直流有刷电机控制·定时器互补输出·定时器死区控制刹车输入