STM32--基于STM32F103的MAX30102心率血氧测量

本文介绍基于STM32F103ZET6+MAX30102心率血氧测量+0.96寸OLED(7针)显示(完整程序代码见文末链接)

一、简介

MAX30102是一个集成的脉搏血氧仪和心率监测仪生物传感器的模块。它集成了一个红光LED和一个红外光LED、光电检测器、光器件,以及带环境光抑制的低噪声电子电路。

MAX30102采用一个1.8V电源和一个独立的5.0V用于内部LED的电源,应用于可穿戴设备进行心率和血氧采集检测,佩戴于手指点耳垂和手腕处。

标准的I2C兼容的通信接口可以将采集到的数值传输给Arduino、STM32、STC51等单片机进行心率和血氧计算。此外,该芯片还可以通过软件关断模块,待机电流接近为零,实现电源始终维持供电状态。
在这里插入图片描述
模块主要参数:

LED峰值波长器 :660nm/880nm
LED供电电压 : 3.3 ~ 5V
检测信号类型 : 光反射信号(PPG)
输出信号接口 : I2C接口
通信接口电压 : 1.8 ~ 3.3V ~ 5V(可选)

工作原理

光溶积法:利用人体组织在血管搏动时造成透光率不同来进行脉搏和血氧饱和度测量

光源:采用对动脉血中痒合血红蛋白(HbO2)和血红蛋白(Hb)有选择性的特定波长的发光二极管

透光率转化为电信号:动脉搏动充血容积变化导致这束光的透光率发送改变,此时由光电变换接收经人体组织反射光线,转变为电信号并将其放大输出。

在这里插入图片描述
模块原理图

在这里插入图片描述

引脚说明

管脚定义说明
VIN电源输入 1.6V-5.5V,一般3.3V
SDAIIC-SDA
SCLIIC-SCL
GND
INTINT 低电平有效中断(漏极开路)MAX30102 的中断引脚
IRDIR_DRV IR LED阴极和LED驱动器连接点,一般NC
RDR_DRV 红色LED阴极和LED驱动器连接点,一般NC

通讯协议

本模块采用I2C通讯方式,具有I2C/SMBus兼容的2线制由串行数据线(SDA)和串行时钟线(SCL),时钟速率高达400kHz。

设备地址:
在这里插入图片描述

本文主要介绍基于STM32F103ZET6和实现的MAX30102的心率和血氧测量,并显示在0.96寸的OLED(7针)屏幕上

二、接线图

设计中涉及STM32和MAX30102链接以及和OLED的管脚连接,具体连接脚说明如下:

STM32F103ZET6+MAX30102接线

STM32F103ZET6MAX30102
3.3VVin
GNDGND
PC7SCL
PC8SDA
PC9INT

STM32F103ZET6+0.96寸OLED接线,其中OLED选用的是7针OLED-4SPI通信

STM32F103ZET60.96寸OLED(7针)
3.3VVCC
GNDGND
PA5SCL/D0
PA6SDA/D1
PA3RST
PA4DC
PA2CS

在这里插入图片描述

三、代码函数(完整工程文件见文末连接)

主函数,主要实现心率和血氧测量并显示在OLED屏幕,且当超过阈值后会蜂鸣器报警

int main(void)
{ //variables to calculate the on-board LED brightness that reflects the heartbeatsuint32_t un_min, un_max, un_prev_data;  int i;int32_t n_brightness;float f_temp;
//	u8 temp_num=0;u8 temp[6];u8 str[100];u8 dis_hr=0,dis_spo2=0;NVIC_Configuration();delay_init();	    	 //延时函数初始化	  uart_init(115200);	 	//串口初始化为115200LED_Init();BEEP_Init();//OLEDOLED_Init();OLED_ShowString(0,0,"  initializing  ",16);OLED_Refresh_Gram();//更新显示到OLED	 max30102_init();printf("\r\n MAX30102  init  \r\n");un_min=0x3FFFF;un_max=0;n_ir_buffer_length=500; //buffer length of 100 stores 5 seconds of samples running at 100sps//read the first 500 samples, and determine the signal rangefor(i=0;i<n_ir_buffer_length;i++){while(MAX30102_INT==1);   //wait until the interrupt pin assertsmax30102_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 numberaun_ir_buffer[i] = (long)((long)((long)temp[3] & 0x03)<<16) |(long)temp[4]<<8 | (long)temp[5];   // Combine values to get the actual numberif(un_min>aun_red_buffer[i])un_min=aun_red_buffer[i];    //update signal minif(un_max<aun_red_buffer[i])un_max=aun_red_buffer[i];    //update signal max}un_prev_data=aun_red_buffer[i];//calculate heart rate and SpO2 after first 500 samples (first 5 seconds of samples)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); while(1){i=0;un_min=0x3FFFF;un_max=0;//dumping the first 100 sets of samples in the memory and shift the last 400 sets of samples to the topfor(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 maxif(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];}//take 100 sets of samples before calculating the heart rate.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 numberaun_ir_buffer[i] = (long)((long)((long)temp[3] & 0x03)<<16) |(long)temp[4]<<8 | (long)temp[5];   // Combine values to get the actual numberif(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;}//send samples and calculation result to terminal program through UARTif(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;}printf("HR=%i, ", n_heart_rate); printf("HRvalid=%i, ", ch_hr_valid);printf("SpO2=%i, ", n_sp02);printf("SPO2Valid=%i\r\n", ch_spo2_valid);}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);//显示刷新LED0=0;if(dis_hr == 0 && dis_spo2 == 0)  //**dis_hr == 0 && dis_spo2 == 0{sprintf((char *)str,"HR:--- SpO2:--- ");//**HR:--- SpO2:--- }else{sprintf((char *)str,"HR:%3d SpO2:%3d ",dis_hr,dis_spo2);//**HR:%3d SpO2:%3d }OLED_ShowString(0,0,str,16);OLED_Fill(0,23,127,63,0);//红光在上,红外在下dis_DrawCurve(aun_red_buffer,20);dis_DrawCurve(aun_ir_buffer,0);OLED_Refresh_Gram();//更新显示到OLED
}
}

四、视频演示

MAX30102心率血氧测量

附完整代码程序资料

链接:https://pan.baidu.com/s/1TaxlDzRbkNnz4guaJPrGnQ?pwd=6arw
提取码:6arw

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/342095.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何将远程桌面全屏显示在3台显示器中的2台

cmd运行&#xff1a; mstsc /l编辑*.rdp文件 screen mode id:i:2 use multimon:i:1 selectedmonitors:s:1,4 desktopwidth:i:1920 desktopheight:i:1080 session bpp:i:32 winposstr:s:0,0,1920,0,3840,1080

Matlab 使用 DH table 建立的 robot 和实际不符

机器人仿真 想借助 matlab robotics toolbox 来仿真机器人&#xff0c;但是直接输入自己的 DH table 显示出来的 robot 和实际不情况不符。 DH table 建立 robot Build Manipulator Robot Using Kinematic DH Parameters 主要使用 setFixedTransform&#xff0c;DH table 中…

谷歌开发者账号关联的主要原因有哪些?应如何解决或避免?

众所周知&#xff0c;谷歌对账号的风控非常严格&#xff0c;不允许一个开发者拥有多个开发者账号&#xff0c;且相信大多数开发者都收到过谷歌发来的账号终止的邮件&#xff0c;甚至同时连续被封多个账号。 大多数开发者账号被封的的主要原因是谷歌认为或检测出账号存在高风险行…

Inis博客系统本地部署结合内网穿透实现远程访问本地站点

文章目录 前言1. Inis博客网站搭建1.1. Inis博客网站下载和安装1.2 Inis博客网站测试1.3 cpolar的安装和注册 2. 本地网页发布2.1 Cpolar临时数据隧道2.2 Cpolar稳定隧道&#xff08;云端设置&#xff09;2.3.Cpolar稳定隧道&#xff08;本地设置&#xff09; 3. 公网访问测试总…

龙芯3A5000上使用腾讯会议

原文链接&#xff1a;龙芯3A5000上使用腾讯会议 hello&#xff0c;大家好啊&#xff01;今天我要给大家介绍的是在龙芯3A5000处理器上安装使用腾讯会议的经验分享。随着远程工作和在线会议的普及&#xff0c;腾讯会议成为了许多人日常工作不可或缺的工具。而对于使用龙芯3A5000…

JetCache源码解析——缓存处理

在Java技术体系中&#xff0c;如果想要在不改变已有代码逻辑的情况下&#xff0c;对已有的函数进行功能增强&#xff0c;一般可以使用两种方式&#xff0c;如AOP&#xff08;Aspect Oriented Programming&#xff09;&#xff0c;即面向切面编程&#xff0c;以及代理模式&#…

MacOS访问某局域网域名存在问题,但是ip可以正常访问的问题解决方案

背景&#xff1a; 公司服务器部署wiki&#xff0c;之前macos访问wiki服务都是通过ip:8090形式访问&#xff0c;后来给wiki服务配置了域名wiki:8090&#xff0c;后macos浏览器连接同样的网络却一直没办法正常域名访问wiki&#xff0c;但可以ip访问wiki&#xff0c;但是其他同事…

用通俗易懂的方式讲解:对 embedding 模型进行微调,我的大模型召回效果提升了太多了

QA对话目前是大语言模型的一大应用场景&#xff0c;在QA对话中&#xff0c;由于大语言模型信息的滞后性以及不包含业务知识的特点&#xff0c;我们经常需要外挂知识库来协助大模型解决一些问题。 在外挂知识库的过程中&#xff0c;embedding模型的召回效果直接影响到大模型的回…

四、C++运算符(5)逻辑运算符

作用&#xff1a;用于根据表达式的值返回真值或假值 #define _CRT_SECURE_NO_WARNINGS #include<iostream> #include<string> using namespace std; int main() {//逻辑运算符 非&#xff01;int a 10;int b 20;//在c中除了0都是真cout << !a << end…

小红书种草类型有哪些,小红书营销攻略

我们都知道小红书是个内容平台。用户来这可以看到各种类型的笔记&#xff0c;从笔记中获取自己想要了解的内容。这也就意味着平台上有着许多种不同的笔记类型。今天我们和大家分享下小红书种草类型有哪些&#xff0c;小红书营销攻略&#xff01; 1. 明星带货类 顾名思义&#x…

职称为什么要提前报名?⬇️ ⬇️

评职称需要一堆的材料&#xff0c;比如&#xff1a; 论文发表&#xff0c;从写作到选期刊到发表&#xff0c;需要3-12个月的时间继续教育&#xff0c;职称评审对于继续教育是有学时要求的&#xff0c;这一点申报职称的人想必都清楚&#xff0c;但具体学时要求,就要根据当时的要…

5. 属性自动填充

《阿里巴巴Java开发手册》&#xff0c;在第 5 章 MySQL 数据库可以看到这样一条规范&#xff1a; 对于一张数据表&#xff0c;它必须具备三个字段&#xff1a; id : 唯一IDgmt_create : 保存的是当前数据创建的时间gmt_modified : 保存的是更新时间 改造一下数据表&#xf…