基于STM32的睡眠心率血氧监测应用

心率血氧计算程序代码(带串口调试):

#include "blood.h"
#include "usart.h"
#include "esp8266.h"
#include "onenet.h"
uint16_t g_fft_index = 0;         	 	//fft输入输出下标

struct compx s1[FFT_N+16];           	//FFT输入和输出:从S[1]开始存放,根据大小自己定义
struct compx s2[FFT_N+16];           	//FFT输入和输出:从S[1]开始存放,根据大小自己定义

struct
{
	float 	Hp	;			//血红蛋白	
	float 	HpO2;			//氧合血红蛋白
	
}g_BloodWave;//血液波形数据

BloodData g_blooddata = {0};					//血液数据存储

#define CORRECTED_VALUE			47   			//标定血液氧气含量

/*funcation start ------------------------------------------------------------*/
//血液检测信息更新
void blood_data_update(void)
{
	//标志位被使能时 读取FIFO
	g_fft_index=0;
	while(g_fft_index < FFT_N)
	{
		while(MAX30102_INTPin_Read()==0)
		{
			//读取FIFO
			max30102_read_fifo();  //read from MAX30102 FIFO2
			//将数据写入fft输入并清除输出
			if(g_fft_index < FFT_N)
			{
				//将数据写入fft输入并清除输出
				s1[g_fft_index].real = fifo_red;
				s1[g_fft_index].imag= 0;
				s2[g_fft_index].real = fifo_ir;
				s2[g_fft_index].imag= 0;
				g_fft_index++;
			}
		}
	}
}

//血液信息转换
void blood_data_translate(void)
{	
	float n_denom;
	uint16_t i;
	//直流滤波
	float dc_red =0; 
	float dc_ir =0;
	float ac_red =0; 
	float ac_ir =0;
	
	for (i=0 ; i<FFT_N ; i++ ) 
	{
		dc_red += s1[i].real ;
		dc_ir +=  s2[i].real ;
	}
		dc_red =dc_red/FFT_N ;
		dc_ir =dc_ir/FFT_N ;
	for (i=0 ; i<FFT_N ; i++ )  
	{
		s1[i].real =  s1[i].real - dc_red ; 
		s2[i].real =  s2[i].real - dc_ir ; 
	}
	
	//移动平均滤波
	UsartPrintf(USART_DEBUG,"***********8 pt Moving Average red*******************\r\n");
//	
	for(i = 1;i < FFT_N-1;i++) 
	{
			n_denom= ( s1[i-1].real + 2*s1[i].real + s1[i+1].real);
			s1[i].real=  n_denom/4.00; 
			
			n_denom= ( s2[i-1].real + 2*s2[i].real + s2[i+1].real);
			s2[i].real=  n_denom/4.00; 			
	}
	//八点平均滤波
	for(i = 0;i < FFT_N-8;i++) 
	{
			n_denom= ( s1[i].real+s1[i+1].real+ s1[i+2].real+ s1[i+3].real+ s1[i+4].real+ s1[i+5].real+ s1[i+6].real+ s1[i+7].real);
			s1[i].real=  n_denom/8.00; 
			
			n_denom= ( s2[i].real+s2[i+1].real+ s2[i+2].real+ s2[i+3].real+ s2[i+4].real+ s2[i+5].real+ s2[i+6].real+ s2[i+7].real);
			s2[i].real=  n_denom/8.00; 
		
			UsartPrintf(USART_DEBUG,"%f\r\n",s1[i].real);		
	}
	UsartPrintf(USART_DEBUG,"************8 pt Moving Average ir*****************\r\n");
	for(i = 0;i < FFT_N;i++) 
	{
		UsartPrintf(USART_DEBUG,"%f\r\n",s2[i].real);	
	}
	UsartPrintf(USART_DEBUG,"***************************************************\r\n");
	//开始变换显示	
	g_fft_index = 0;	
	//快速傅里叶变换
	FFT(s1);
	FFT(s2);
	//解平方
	UsartPrintf(USART_DEBUG,"开始FFT算法****************************************\r\n");
	for(i = 0;i < FFT_N;i++) 
	{
		s1[i].real=sqrtf(s1[i].real*s1[i].real+s1[i].imag*s1[i].imag);
		s1[i].real=sqrtf(s2[i].real*s2[i].real+s2[i].imag*s2[i].imag);
	}
	//计算交流分量
	for (i=1 ; i<FFT_N ; i++ ) 
	{
		ac_red += s1[i].real ;
		ac_ir +=  s2[i].real ;
	}
	
	for(i = 0;i < FFT_N/2;i++) 
	{
		UsartPrintf(USART_DEBUG,"%f\r\n",s1[i].real);
	}
	UsartPrintf(USART_DEBUG,"**************************************************\r\n");
	for(i = 0;i < FFT_N/2;i++) 
	{
		UsartPrintf(USART_DEBUG,"%f\r\n",s2[i].real);
	}
	
	UsartPrintf(USART_DEBUG,"结束FFT算法***************************************\r\n");
	
	UsartPrintf(USART_DEBUG,"%d\r\n",(int)i);
	//读取峰值点的横坐标  结果的物理意义为 
	int s1_max_index = find_max_num_index(s1, 30);
	int s2_max_index = find_max_num_index(s2, 30);
	UsartPrintf(USART_DEBUG,"%d\r\n",s1_max_index);
	UsartPrintf(USART_DEBUG,"%d\r\n",s2_max_index);
    
    
	float Heart_Rate = 60.00 * ((100.0 * s1_max_index )/ 512.00)+20;
	g_blooddata.heart = Heart_Rate;
	
	float R = (ac_ir*dc_red)/(ac_red*dc_ir);
	float sp02_num =-45.060*R*R+ 30.354 *R + 94.845;
	g_blooddata.SpO2 = sp02_num;
			
}


void blood_Loop(void)
{
	blood_data_update();
	//血液信息转换
	blood_data_translate();
	//显示血液状态信息
	g_blooddata.SpO2 = (g_blooddata.SpO2 > 99.99) ? 99.99:g_blooddata.SpO2;
	UsartPrintf(USART_DEBUG,"指令心率%3d",g_blooddata.heart);
	UsartPrintf(USART_DEBUG,"指令血氧%0.2f",g_blooddata.SpO2);
}

主函数程序代码:

#include "stm32f10x.h"                  // Device header
#include <string.h>
#include "Delay.h"
#include "OLED.h"
#include "usart.h"
#include "onenet.h"
#include "esp8266.h"
#include "max30102.h"
#include "algorithm.h"
#include "xiic.h"
#include "main.h"
#include "stdio.h"
#include "blood.h"
extern uint8_t dat[5];			//温湿度存储数组
extern int32_t n_heart_rate;
extern int32_t n_sp02;
unsigned short timeCount = 0;	//发送间隔变量
unsigned char *dataPtr = NULL; 	//接受命令变量
float presshigh=130;
float Temphigh = 38;
float press=0;
float Temp = 0;
int v1,p=0;
	
uint8_t Data[]= " ";                             //创建目标数组
uint8_t Data1[]= " "; 
extern BloodData g_blooddata;

void Hardware_init(void);
void Connect(void);
void test_heard(void);    
void Refresh_Data(void);

int main(void)
{
	Hardware_init();
	test_heard();
}

void Hardware_init(void)
{
	Delay_init(72);
	OLED_Init();				//OLED初始化
	Usart1_Init(115200);		//调试串口初始化
	Usart2_Init(115200);		//esp8266串口初始化
    
	IIC_GPIO_INIT();
	MAX30102_GPIO();
	Max30102_reset();
	MAX30102_Config();
    
	Connect();
    
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级

}
void Connect(void)
{
	ESP8266_Init();				//初始化ESP8266
    
	while(OneNet_DevLink())		//接入OneNET
		Delay_ms(50);
}

void test_heard(void)
{

	OLED_Clear();

	OLED_ShowString(1, 1, "SpO2:");
    OLED_ShowString(1, 11, "%");
	OLED_ShowString(2, 1, "Heart:");

	while(1)
	{

        if(++timeCount >= 100)									//发送间隔5s
		{
			blood_Loop();				//获取血氧和心率
            
			OneNet_SendData();									//发送数据
			
			timeCount = 0;
			ESP8266_Clear();
		}
		
		
		Refresh_Data();
		
        Delay_ms(10);
		
	}
}

void Refresh_Data(void)
{
    //血氧
    sprintf((char *)Data,"%0.2f",g_blooddata.SpO2);
	OLED_ShowString(1,6,(char *)Data);
    
    //心率
	sprintf((char *)Data1,"%3d",g_blooddata.heart);
	OLED_ShowString(2,7,(char *)Data1);
	
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值