#include <reg52.h>
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
sbit SCL = P1^0;
sbit SDA = P1^1;
sbit LED = P2^0;
#define ADDR_PCF8591 0x90 // PCF8591的IIC地址:1001 0000
#define CMD_PCF8591_WR 0x40 // PCF8591写数据命令字:0100 CCCC,CCCC为通道选择
#define CMD_PCF8591_RD 0x41 // PCF8591读数据命令字:0100 CCCC,CCCC为通道选择
#define ADDR_OLED 0x78 // SSD1306 OLED显示屏的IIC地址:0111 1000
uchar heartRate[3]; // 存储心率值的字符串
/**
* 延时函数,控制IIC通信速度
*/
void Delay()
{
uint i, j;
for(i=0; i<50; i++)
for(j=0; j<500; j++);
}
/**
* IIC启动信号
*/
void IIC_Start()
{
SCL = 1;
SDA = 1;
Delay();
SDA = 0;
Delay();
SCL = 0;
}
/**
* IIC停止信号
*/
void IIC_Stop()
{
SCL = 0;
SDA = 0;
Delay();
SCL = 1;
SDA = 1;
Delay();
}
/**
* IIC发送一个字节的数据
* @param byte 发送的字节
* @return 接收到的应答位
*/
uchar IIC_SendByte(uchar byte)
{
uchar i, ack;
for(i=0; i<8; i++)
{
SDA = (bit)(byte & 0x80);
byte <<= 1;
Delay();
SCL = 1;
Delay();
SCL = 0;
}
SDA = 1;
Delay();
SCL = 1;
Delay();
ack = SDA;
SCL = 0;
return ack;
}
/**
* 初始化PCF8591模块
*/
void Init_PCF8591()
{
IIC_Start();
IIC_SendByte(ADDR_PCF8591);
IIC_SendByte(CMD_PCF8591_WR | 0);
IIC_Stop();
}
/**
* 读取PCF8591的AD值
* @param ch 选择的通道编号
* @return AD转换后的数值
*/
uchar Read_PCF8591(uchar ch)
{
uchar value;
IIC_Start();
IIC_SendByte(ADDR_PCF8591);
IIC_SendByte(CMD_PCF8591_WR | ch);
IIC_Stop();
IIC_Start();
IIC_SendByte(ADDR_PCF8591 | 0x01);
value = IIC_SendByte(0xFF);
IIC_Stop();
return value;
}
/**
* 初始化SSD1306 OLED显示屏
*/
void Init_OLED()
{
IIC_Start();
IIC_SendByte(ADDR_OLED);
IIC_SendByte(0xAE); // 关闭显示
IIC_SendByte(0x00); // 列地址低4位
IIC_SendByte(0x10); // 列地址高4位
IIC_SendByte(0x40); // 起始行地址
IIC_SendByte(0xB0); // 设置页地址
IIC_SendByte(0x81); // 对比度设置命令
IIC_SendByte(0xCF); // 对比度值
IIC_SendByte(0xA1); // 段复用设置
IIC_SendByte(0xA6); // 常规显示模式
IIC_SendByte(0xA8); // 多路复用设置
IIC_SendByte(0x3F); // 页面数-1
IIC_SendByte(0xC8); // 扫描方式设置
IIC_SendByte(0xD3); // 设置显示偏移
IIC_SendByte(0x00);
IIC_SendByte(0xD5); // 频率设置命令
IIC_SendByte(0x80); // 分频系数
IIC_SendByte(0xD9); // 设置预充电周期
IIC_SendByte(0xF1);
IIC_SendByte(0xDA); // 设置COM硬件连接方式
IIC_SendByte(0x12);
IIC_SendByte(0xDB); // VCOMH设置
IIC_SendByte(0x40);
IIC_SendByte(0xA4); // 全部点亮/正常显示
IIC_SendByte(0xA6); // 正常/反显示控制
IIC_SendByte(0xAF); // 开启显示
IIC_Stop();
}
/**
* 在OLED上显示字符串
* @param x 开始列地址
* @param y 开始页地址
* @param str 需要显示的字符串
*/
void ShowString_OLED(uchar x, uchar y, uchar *str)
{
uchar i = 0;
IIC_Start();
IIC_SendByte(ADDR_OLED);
IIC_SendByte(0x00); // 列地址低4位
IIC_SendByte(0x10); // 列地址高4位
IIC_SendByte(0xB0 + y);// 设置页地址
for(i=0; str[i]!='\0'; i++)
{
IIC_SendByte(0xB0 + y);
IIC_SendByte((x + 8*i) & 0x0F);
IIC_SendByte(((x + 8*i) >> 4) | 0x10);
IIC_SendByte(str[i]);
}
IIC_Stop();
}
/**
* 主函数,心率计算和显示
*/
void main()
{
Init_PCF8591(); // 初始化PCF8591模块
Init_OLED(); // 初始化OLED显示屏
while(1)
{
uchar adValue = Read_PCF8591(0); // 读取PCF8591的AD值
uint timeInterval = 100; // 设定采集心率的时间间隔,单位为毫秒
uint count = 0; // 统计脉搏跳动次数的计数器
uint heartRateValue = 0; // 计算得出的心率值
for (uint i=0; i<timeInterval; i++) // 在一定时间内采集数据
{
if (adValue > 200) // 当AD值高于阈值时,统计脉搏跳动次数
{
count++;
while(adValue > 100) // 等待一段时间,避免同一次脉搏被重复计数
{
adValue = Read_PCF8591(0);
}
}
adValue = Read_PCF8591(0); // 读取下一个AD值
}
heartRateValue = (uint)(count * 60.0 / timeInterval); // 计算心率值
sprintf(heartRate, "%d", heartRateValue); // 将心率值转换为字符串
ShowString_OLED(0, 0, "Heart Rate:"); // 在OLED上显示标题
ShowString_OLED(80, 0, heartRate); // 在OLED上显示心率值
ShowString_OLED(96, 0, "bpm"); // 在OLED上显示单位
}
}
评论