基于STM32的血氧仪控制系统
- 前言
- 一、设计任务
- 二、系统硬件设计
- 1.元器件选用
- 2.系统模型设计
- 3.硬件连接
- 三、系统程序设计
- 1.程序流程
- 2.主程序
- 总结
- 下载
前言
本文使用搭载STM32F103VET6主控芯片的野火指南者、野火3.2寸LCD屏幕和MAX30102心率血氧传感器实现心率血氧采集并显示的效果。
(文末附完整程序下载链接)
一、设计任务
1.可以通过MAX30102心率血氧传感器对用户的心率血氧进行采集,并用实时显示在LCD液晶屏上;
2.可以设定一个固定的报警温度值显示在液晶屏上;
3.用户心率低于60或者高于阈值报警、用户血氧低于阈值报警,报警形式包括:蜂鸣器发出滴滴声,LCD中心率血氧字符根据不同情况显示相应的报警颜色,报警颜色有红、蓝、品红;
4.能够通过按键动态修改报警温度阈值;
5.LCD显示屏下方绘制心率波形图;
6.ESP8266将数据发送给TCP助手;
二、系统硬件设计
1.元器件选用
1、STM32f103VET6单片机一块
2、ESP8266一块
(此项目使用野火指南者,指南者自带ESP8266,其他型号单片机杜邦线自连即可)
3、MAX30102心率血氧传感器
4、3.2寸LCD显示器
5、杜邦线5根
2.系统模型设计
按键1设置心率阈值,按键2设置血氧阈值,MAX30102检测用户心率和血氧值。
当检测的用户心率值大于心率阈值时LCD当前心率一栏变为红色、蜂鸣器报警、LED红灯闪烁、串口输出警告。
当检测的用户心率值底于60时LCD当前心率一栏变为品红色、蜂鸣器报警、LED红灯闪烁、串口输出警告。
当检测的用户血氧浓度小于血氧阈值时LCD显示血氧过低警告、蜂鸣器报警、蓝灯闪烁、串口输出心率过低警告。
血氧波形图将检测到的血氧数据绘制出来,在未检测时y轴绘制为0。相应的颜色警告也会在波形图中显示。
ESP8266将单片机采集的数据通过联网的方式发送到TCP助手中并显示。
演示视频:
波形显示
3.硬件连接
1、本系统使用野火指南者,由于该单片机自带ESP8266,因此无需连接。如果使用其他型号单片机,根据ESP8266用户手册连接即可。
2、STM32F103与MAX30102的连接,VCC<->3.3V,GND<->GND,SCL<->PC7,SDA<->PC8,IM<->PC9。
总电路连接图:
三、系统程序设计
1.程序流程
基于STM32的血氧仪控制系统总体软件的设计,可以分为系统初始化、数据采集、数据处理、数据显示和系统控制。系统初始化包括各模块的初始化,包括MAX30102模块、LCD显示模块、按键模块、蜂鸣器模块等。同时还需要初始化串口通信模块。
采集数据主要是获取的血氧数据,进行数字转换和滤波处理,得到血氧饱和度和脉搏率数据。数据处理是对采集到的血氧饱和度和脉搏率数据进行处理。同时,还需要对数据进行校验和处理,确保数据的准确性和可靠性。将处理后的数据通过LCD显示模块显示出来。同时,还需要通过蜂鸣器模块进行声音提示,例如当血氧饱和度低于一定阈值时,发出警报声。系统控制即通过按键控制模块控制系统的工作模式,如调节警报阈值等。
程序设计流程图:
WiFi运行逻辑图:
2.主程序
项目部分程序示例。
main.c
int main()
{
/* 初始化 */
USART_Config (); //初始化串口1
NVIC_Configuration();
CPU_TS_TmrInit(); //初始化DWT计数器,用于延时函数
LED_Init(); //初始化RGB彩灯
EXTI_Key_Config(); //KEY中断初始化
BEEP_GPIO_Config(); //初始化蜂鸣器引脚
ESP8266_Init(); //初始化WiFi模块使用的接口和外设
ILI9341_Init(); //LCD 初始化
max30102_init(); //max30102初始化
ILI9341_GramScan(6); //LCD显示模式
ESP8266_StaTcpClient_Unvarnish_ConfigTest(); //对ESP8266进行配置
max30102_read(); //数据预处理
LCD_SetFont(&Font8x16); //LCD显示字体
LCD_SetColors(WHITE,BLACK);//LCD黑底白字,这样显示它别致
ILI9341_Clear(0,0,LCD_X_LENGTH,LCD_Y_LENGTH); /* 清屏,显示全黑 */
while (1)
{
max30102_control(); //max30102控制函数
ESP8266_Sendmax30102DataTest();//LCD显示及串口\网络数据发送
drawCurve(280,dis_hr);//LCD画波形,第一个参数是LCD显示起始位置,第二个是波形的数值
}
}
max30102_control(); max30102控制函数
void max30102_control()
{
i=0;
un_min=0x3FFFF;
un_max=0;
//将前100组样本转储到存储器中,并将最后400组样本移到顶部
for(i=100;i<500;i++){
aun_red_buffer[i-100]=aun_red_buffer[i];
aun_ir_buffer[i-100]=aun_ir_buffer[i];
//update the signal min and max
if(un_min>aun_red_buffer[i])
un_min=aun_red_buffer[i];
if(un_max<aun_red_buffer[i])
un_max=aun_red_buffer[i];
}
//在计算心率之前采集100组样本。
for(i=400;i<500;i++){
un_prev_data=aun_red_buffer[i-1];
while(MAX30102_INT==1);
max30102_FIFO_ReadBytes(REG_FIFO_DATA,temp);
aun_red_buffer[i] = (long)((long)((long)temp[0]&0x03)<<16) | (long)temp[1]<<8 | (long)temp[2]; // Combine values to get the actual number
aun_ir_buffer[i] = (long)((long)((long)temp[3] & 0x03)<<16) |(long)temp[4]<<8 | (long)temp[5]; // Combine values to get the actual number
if(aun_red_buffer[i]>un_prev_data){
f_temp=aun_red_buffer[i]-un_prev_data;
f_temp/=(un_max-un_min);
f_temp*=MAX_BRIGHTNESS;
n_brightness-=(int)f_temp;
if(n_brightness<0)
n_brightness=0;
}else{
f_temp=un_prev_data-aun_red_buffer[i];
f_temp/=(un_max-un_min);
f_temp*=MAX_BRIGHTNESS;
n_brightness+=(int)f_temp;
if(n_brightness>MAX_BRIGHTNESS)
n_brightness=MAX_BRIGHTNESS;
}
//通过USART将样本和计算结果发送到终端程序
if(ch_hr_valid == 1 && n_heart_rate<120){//**/ ch_hr_valid == 1 && ch_spo2_valid ==1 && n_heart_rate<120 && n_sp02<101
dis_hr = n_heart_rate;
dis_spo2 = n_sp02;
}else{
dis_hr = 0;
dis_spo2 = 0;
}
}
maxim_heart_rate_and_oxygen_saturation(aun_ir_buffer, n_ir_buffer_length, aun_red_buffer, &n_sp02, &ch_spo2_valid, &n_heart_rate, &ch_hr_valid);
if(dis_hr == 0 && dis_spo2 == 0) //**dis_hr == 0 && dis_spo2 == 0
{
sprintf((char *)str,"HR:--- SpO2:--- ");//**HR:--- SpO2:---
ILI9341_DispStringLine_EN_CH(LINE(4)," 当前心率:---");
ILI9341_DispStringLine_EN_CH(LINE(6)," 当前血氧:---");
}else{
sprintf((char *)str,"HR:%3d SpO2:%3d ",dis_hr,dis_spo2);//**HR:%3d SpO2:%3d
LCD_SetTextColor(GREEN);
sprintf(get_hr_char," 当前心率:%d ",dis_hr);
LCD_ClearLine(LINE(4));
ILI9341_DispStringLine_EN_CH(LINE(4),get_hr_char);
LCD_SetTextColor(GREEN);
sprintf(get_spo2_char," 当前血氧:%d%%",dis_spo2);
LCD_ClearLine(LINE(6));
ILI9341_DispStringLine_EN_CH(LINE(6),get_spo2_char);
}
}
ESP8266_Sendmax30102DataTest();LCD显示以及数据发送
void ESP8266_Sendmax30102DataTest(void)
{
char cStr[170]={0};
uint8_t ucStatus;
LCD_SetTextColor(WHITE);
ILI9341_DispStringLine_EN_CH(LINE(1)," 血氧仪控制系统 ");
LCD_SetTextColor(GREEN);
sprintf(set_hr_char," 心率阈值:%d ",set_hr_max);
LCD_ClearLine(LINE(3));
ILI9341_DispStringLine_EN_CH(LINE(3),set_hr_char);
LCD_SetTextColor(GREEN);
sprintf(set_spo2_char," 血氧阈值:%d%%",set_spo2);
LCD_ClearLine(LINE(5));
ILI9341_DispStringLine_EN_CH(LINE(5),set_spo2_char);
if(dis_hr == 0 && dis_spo2 == 0) //max30102未检测
{
printf("Max30102 not detected!/r/n");
LCD_SetTextColor(GREEN);
LCD_ClearLine(LINE(18));
LCD_ClearLine(LINE(19));
PBout(5)=1;//灯灭
PBout(1)=1;//灯灭
BEEP( OFF );//蜂鸣器关
}else{ //max30102开始检测
/* 显示血氧 */
if(dis_spo2 >= set_spo2){
LCD_ClearLine(LINE(19));
LCD_SetTextColor(GREEN);//当前血氧一栏显示绿色
sprintf(get_spo2_char," 当前血氧:%d%% ",dis_spo2);
LCD_ClearLine(LINE(6)); /* 清除单行文字 */
ILI9341_DispStringLine_EN_CH(LINE(6),get_spo2_char);
PBout(1)=1;//灯灭
BEEP( OFF );//蜂鸣器关
} else{ //当前血氧低于阈值时颜色变蓝
PBout(1)=0;//灯亮
BEEP( ON );//蜂鸣器开
LCD_SetTextColor(BLUE);
LCD_ClearLine(LINE(6)); /* 清除单行文字 */
sprintf(get_spo2_char," 当前血氧:%d%% ",dis_spo2);
ILI9341_DispStringLine_EN_CH(LINE(6),get_spo2_char);
ILI9341_DispStringLine_EN_CH(LINE(19)," 血氧过低 ");
printf("!!!血氧低于阈值!!!"); //串口输出
}
/* 显示心率 */
if((dis_hr <= set_hr_max) && (dis_hr >= set_hr_min)){ //当前心率一栏显示绿色
LCD_ClearLine(LINE(18));
LCD_SetTextColor(GREEN);
LCD_ClearLine(LINE(4)); /* 清除单行文字 */
sprintf(get_hr_char," 当前心率:%d ",dis_hr);
ILI9341_DispStringLine_EN_CH(LINE(4),get_hr_char);
PBout(5)=1;//灯灭
BEEP( OFF );//蜂鸣器关
}else if( ( dis_hr <= set_hr_min ) && ( dis_hr != 0) ){
LCD_SetTextColor(MAGENTA);
LCD_ClearLine(LINE(4)); /* 清除单行文字 */
sprintf(get_hr_char," 当前心率:%d ",dis_hr);
ILI9341_DispStringLine_EN_CH(LINE(4),get_hr_char);
ILI9341_DispStringLine_EN_CH(LINE(18)," 心率低于60 ");
printf("!!!心率低于60!!!"); //串口输出
BEEP( ON );//蜂鸣器开
PBout(5)=0;//灯亮
}else{ //当前心率超过阈值时颜色变红
BEEP( ON );//蜂鸣器开
PBout(5)=0;//灯亮
LCD_SetTextColor(RED);
LCD_ClearLine(LINE(4)); /* 清除单行文字 */
sprintf(get_hr_char," 当前心率:%d ",dis_hr);
ILI9341_DispStringLine_EN_CH(LINE(4),get_hr_char);
ILI9341_DispStringLine_EN_CH(LINE(18)," 心率过高 ");
printf("!!!心率超出阈值!!!"); //串口输出
}
}
if(dis_hr !=0 || dis_spo2 != 0 ){
sprintf ( cStr, "POST /devices/92648495/datapoints?type=5 HTTP/1.1\napi-key:Yn2cDHrWmsTL62QUjHYu4RNtgPw=\nHost:api.zj.cmcconenet.com\nContent-Length:23\n\n,;xinlv,%3d;xueyang,%3d",dis_hr,dis_spo2);
printf ( "%s", cStr ); //打印读取 DHT11 温湿度信息
ESP8266_SendString ( ENABLE, cStr, 0, Single_ID_0 ); //发送 DHT11 温湿度信息到网络调试助手
}
if ( ucTcpClosedFlag ){ //检测是否失去连接
ESP8266_ExitUnvarnishSend (); //退出透传模式
do ucStatus = ESP8266_Get_LinkStatus (); //获取连接状态
while ( ! ucStatus );
if ( ucStatus == 4 ){ //确认失去连接后重连
printf ( "\r\n正在重连热点和服务器 ......\r\n" );
while ( ! ESP8266_JoinAP ( macUser_ESP8266_ApSsid, macUser_ESP8266_ApPwd ) );
while ( ! ESP8266_Link_Server ( enumTCP, macUser_ESP8266_TcpServer_IP, macUser_ESP8266_TcpServer_Port, Single_ID_0 ) );
printf ( "\r\n重连热点和服务器成功\r\n" );
}
while ( ! ESP8266_UnvarnishSend () );
}
}
drawCurve(280,dis_hr);LCD画波形,第一个参数是LCD显示起始位置,第二个是波形的数值
void drawCurve(int coord_x,short int rawValue)
{
//coord_xLCD显示起始位置坐标
u16 x,y;
int rawValue_value;
rawValue_value = rawValue*280;//rawValue_value波形高度,280可修改
y = coord_x - rawValue_value/280; //数据处理代码
//这里之所以是120-rawValue/280,与屏幕的扫描方向有关,如果出现上下颠倒的情况,可以改成120 +
if(firstPoint){//如果是第一次画点,则无需连线,直接描点即可
ILI9341_SetPointPixel(10,y);
lastX=0;
lastY=y;
firstPoint=0;
}
else{
x=lastX+1;
if(x<240){ //不超过屏幕宽度
ILI9341_DrawLine(lastX,lastY,x,y);
lastX=x;
lastY=y;
}
else{ //超出屏幕宽度,清屏,从第一个点开始绘制,实现动态更新效果
ILI9341_Clear(0,0,LCD_X_LENGTH,LCD_Y_LENGTH); /* 清屏,显示全黑 */
ILI9341_SetPointPixel(10,y);
lastX=0;
lastY=y;
}
}
}