基于uFUN开发板的心率计(一)DMA方式获取传感器数据

简介: 基于uFUN开发板的心率计(一)DMA方式获取传感器数据

640.jpg


前言

从3月8号收到板子,到今天算起来,uFUN到手也有两周的时间了,最近利用下班后的时间,做了个心率计,从单片机程序到上位机开发,到现在为止完成的差不多了,实现很简单,uFUN开发板外加一个PulseSensor传感器就行,又开发了配套的串口上位机,实现数据的解析和显示,运行界面如下:

640.gif


其实PulseSensor官方已经配备的了Processing语言编写的上位机软件,串口协议的,界面还蛮好看,只要按照它的通信协议,就可以实现心跳波形和心率的显示。刚好最近学习了Qt,所以就用这个小软件来练手了。本篇文章是这个小项目的第一篇,介绍一下如何使用DMA方式获取传感器的数据,至于后面几篇文章会写什么,欢迎大家保持关注哈!


传感器介绍

640.jpg


PulseSensor 是一款用于脉搏心率测量的光电反射式模拟传感器。将其佩戴于手指、耳垂等处,利用人体组织在血管搏动时造成透光率不同来进行脉搏测量。传感器对光电信号进行滤波、放大,最终输出模拟电压值。单片机通过将采集到的模拟信号值转换为数字信号,再通过简单计算就可以得到心率数值。


信号输出引脚连接到示波器,看一下是什么样的信号:

640.jpg


可以看出信号随着心跳起伏变化,周期大概为:1.37/2 = 0.685s。计算出心率值为:600 / 0.685 = 87,我的心率在正常范围内(废话!),这个传感器测心率还是可以的。手头上没有传感器的朋友,可以看一下这篇自制心率传感器的教程:手指检测心跳设计——传感器制作篇,这篇文章介绍的使用一个红外发射管和一个红外接收管,外加放大滤波电路,效果还是挺不错的。


AD采集电路的分析

大家在使用ADC接口的时候要注意了,线别插错了。我第一次使用就是测不到电压值,后来用万用表量了一下,才发现是入门指南中引脚功能标示错了,要采集AD电压,输入脚应该接DCIN这个,对应的是PC3-ADC_IN13。如下图。可能是由于原理图版本的迭代,入门指南没有来得及更新吧!手动@管理员 更改一下。

640.jpg


从原理图中可以看出,直流电压采集电路前级采用双T陷波滤波器滤除50Hz工频干扰,后级为运放电路:

640.jpg


关于前级的双T陷波滤波器S域分析,可以参考这篇文章:双T陷波器s域计算分析(纯手算,工程版!)


大学期间学得信号与系统都忘了,所以这部分计算我没有看懂。其实了解电路的S域分析,更有利于理解电路的特性,大家还是要掌握好理论基础。


后面的运放电路,还是大概能看懂的,下面来分析一下直流通路,把电容看作断路:

640.jpg


所有的运放电路分析,就记住两个要点就行了:虚短和虚断。(感觉又回到了大学。。。。)


虚短:理解成短路,运放处于线性状态时,把两输入端视为等电位,即运放正输入端和负输入端的电压相等,即U+ = U-。

虚断:理解成断路,运放处于线性状态时,把两输入端视为开路,即流入正负输入端的电流为零。


总结一句话:虚短即U+=U-;虚断即净输入电流为0。


好了,有了这两把利器,我们来看一下这部分电路的分析,直流通路可进一步简化为:

640.jpg


很明显,可计算出

U+ = 0.5 * VCC = 1.65v


应用虚短:

U- = U+ = 1.65v


应用虚断,即没有电流流入运放,根据串联电流相等:

640.jpg


以上三式联立,可得:

Uo = 3.368 - 1.205*Ui


即:

Ui  = 3 - 0.83 * Uo


只要得到单片机采集到的电压值Uo,就可以反推出实际的传感器电压值Ui。


通过使用示波器测量Ui和Uo的波形,近似可以认为是反向的,但是明显可以看出,Uo的峰值比Ui的峰值小一点。

640.jpg


而且通过绘制 Ui=3-0.83*UoUi=3.3-Uo的曲线,也可以看出,两条直线几乎重合,即输入和输出近似为反向。

640.jpg


DMA简介

DMA,即直接存储器,用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须 CPU任何干预,通过DMA数据可以快速地移动。这就节省了CPU的资源来做其他操作。STM32共有两个DMA控制器有12个通道(DMA1有7个通道,DMA2有5个通道),每个通道专门用来管理来自于一个或多个外设对存储器访问的请求。还有一个仲裁器来协调各个DMA请求的优先权。


关于DMA通道和外设的对应,可以查看STM32参考手册,心率传感器使用的PC3-ADC_IN13,对应的是DMA1的通道1

640.jpg


STM32 DMA程序配置

获取ADC通道的电压值主要有两种方式,一种是直接使用ADC,然后在需要使用的地方,先启动AD转换,然后读取AD值。另一种更好的方式是使用DMA方式,就是先定义一个保存AD值的全局变量,而全局变量是对应内存中的一个地址的。只要初始时,把DMA和ADC配置好了,DMA会自动把获取到的AD值,存入这个地址中,我们在需要的时候,直接读取这个值就可以了。


0.定义一个全局变量

必须是全局变量,用于存放AD值。

uint16_t ADC_ConvertedValue;


1.配置GPIO和使能时钟

使能外设对应的时钟,注意时钟总线的不同:

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);


引脚配置成模拟输入模式:

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;  
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;  //设置为模拟输入
GPIO_Init(GPIOC, &GPIO_InitStructure);


2.配置DMA

配置ADC对应的DAM1通道1:

DMA_DeInit(DMA1_Channel1); 
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&(ADC1->DR));    //设置源地址
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; //设置内存地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  // 设置传输方向
DMA_InitStructure.DMA_BufferSize = 1;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; 
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;  
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; 
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; 
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;          //循环模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High;    //高优先级
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;     
DMA_Init(DMA1_Channel1, &DMA_InitStructure);  
DMA_Cmd(DMA1_Channel1, ENABLE);    //使能DMA1通道1


3.配置ADC

由于只有1个通道,不需要配置成扫描模式:

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;   
ADC_InitStructure.ADC_ScanConvMode = DISABLE ;    
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;  
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; 
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;  
ADC_InitStructure.ADC_NbrOfChannel = 1;      
ADC_Init(ADC1, &ADC_InitStructure);


PC3对应ADC输入通道13,注意采样周期不能太短:

ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 1, ADC_SampleTime_55Cycles5); 
ADC_DMACmd(ADC1, ENABLE);  
ADC_Cmd(ADC1, ENABLE);  
ADC_ResetCalibration(ADC1);   
while(ADC_GetResetCalibrationStatus(ADC1)); 
ADC_StartCalibration(ADC1);  
while(ADC_GetCalibrationStatus(ADC1)); 
ADC_SoftwareStartConvCmd(ADC1, ENABLE);


4.主程序调用

DMA和ADC配置好之后,只需要初始化一次。然后就可以随时获取电压值了。

int main(void)
{
    float Sensor_Voltage;
    float Uo_Voltage;
    delay_init();           
    UART1_Config(115200);       
    ADC1_Init();
    while(1)
    {
        Uo_Voltage = ADC_ConvertedValue * 3.3 / 4096;       
        Sensor_Voltage = 3.3 - Uo_Voltage;      //近似值
    //      Sensor_Voltage = 3 - 0.83 * Uo_Voltage; //实际传感器输出电压值
        ANO_SendFloat(0xA1, Sensor_Voltage);
        delay_ms(10);
    }
}


为了方便查看数据的波形,这里直接使用了匿名上位机来显示电压值的波形。

函数实现

//匿名上位机,波形显示一个浮点型数据ANO_SendFloat(0xA1, ad);
void ANO_SendFloat(int channel, float f_dat)
{
    u8 tbuf[8];
    int i;
    unsigned char* p;
    for(i = 0; i <= 7; i++)
        tbuf[i] = 0;
    p = (unsigned char*)&f_dat;
    tbuf[0] = 0x88;
    tbuf[1] = channel;  //0xA1
    tbuf[2] = 4;
    tbuf[3] = (unsigned char)(*(p + 3));    //取float类型数据存储在内存中的四个字节
    tbuf[4] = (unsigned char)(*(p + 2));
    tbuf[5] = (unsigned char)(*(p + 1));
    tbuf[6] = (unsigned char)(*(p + 0));
    for(i = 0; i <= 6; i++)
        tbuf[7] += tbuf[i];     //校验和
    printf("%s", tbuf);
}


实际的显示

没有调试器,如何下载程序呢?可以参考我之前发的一篇帖子:如何使用串口来给STM32下载程序,详细介绍了如何通过串口来给uFUN开发板下载程序。


匿名上位机的帧格式配置

640.jpg


实际的显示效果:

640.jpg


总结

传感器数据的获取,只是心率计实现的第一步,传感器放置位置的不同,波形的振幅也会不同,所以,对获得数据的处理、分析,才是最关键的部分。


目录
相关文章
|
数据采集 C语言
单片机开发之ADC0808/9信号采集
本文主要介绍了单片机开发之ADC0808/9信号采集
501 0
单片机开发之ADC0808/9信号采集
|
9月前
【单片机期中测试】13.串口通信的应用(2)—— 超声波通过串口返回数据
【单片机期中测试】13.串口通信的应用(2)—— 超声波通过串口返回数据
83 0
|
传感器 芯片
可编程 USB 转串口适配器开发板与振弦传感器测量模块
当通过 IIC 接口修改 VM5xx 单个寄存器后,被修改的寄存器立即保存(断电不丢失),但连续寄存器的写入仅当时修改生效,模块重启后会自动恢复。为了能够使寄存器永久保存,可以单独向功能寄存器 03 写入指令码 0x000C 来强制保存所有寄存器。
可编程 USB 转串口适配器开发板与振弦传感器测量模块
|
12月前
|
传感器 芯片
基于51单片机的智能热水器STC89C52水位检测传感器DS18B20温度探头传感器
基于51单片机的智能热水器 由STC89C52单片机+水位检测传感器+DS18B20温度探头传感器+按键模块+继电器模块+报警及指示模块+LCD1602显示模块+电源构成。 具体功能: 1、LCD1602显示屏第一行显示温度及定时时间,第二行显示温度上限和下限值; 2、按键可以设置温度的上下限值及定时时间;定时时间到开始加热 3、用两个水位检测传感器检测水位:当水位下限传感器检测到有水时,继电器开始控制加水;当水位上限传感器检测到有水时,停止加水; 4、当温度低于设置的下限温度值时,继电器开始控制加热;当温度高于设置的上限温度值时,停止加热。
106 0
|
传感器 编解码 监控
单片机温湿度测量程序编程
单片机温湿度测量程序编程
64 0
|
传感器
工程监测多通道振弦模拟信号采集仪VTN MODBUS指令驱动测量模式
当工作模式拨码开关的第 2 和第 4 为 ON 时,设备工作于 MODBUS 指令驱动测量模式。在此模式下,设备始终处于开机状态,振弦测量停止。当收到 MODBUS 协议的传感器通道数据读取指令时自动开始测量并在测量结束后响应通道值读取指令的数据包。
工程监测多通道振弦模拟信号采集仪VTN MODBUS指令驱动测量模式
|
传感器
手持VH501TC混合传感器信号采集读数仪怎么使用
1.开机和关机 开机 在关机状态,长按【电源】 按键,屏幕显示开机画面, 当听到蜂鸣器提示音后即可松开按键,设备自动完成参数加载和系统自检进入工作首页。
手持VH501TC混合传感器信号采集读数仪怎么使用
|
传感器 存储
多功能手持VH501TC采集仪连接传感器与读数
振弦与温度传感器 振弦传感器和温度传感器(NTC)均为无源传感,不需要连接电源线。 根据前述“设备组成和接口定义” 用对应颜色的鳄鱼夹分别连接振弦传感器线圈和温度传感器两端即可。传感器连接后,屏幕自动显示实时的测量结果。一般情况下,设备配套传感测线为一根 4 芯线,红黑线连接振弦线圈,另外两根连接温度传感器。
多功能手持VH501TC采集仪连接传感器与读数
|
传感器 存储
手持VH501TC多功能混合传感器信号采集读数仪使用方法
电池仓位于设备背面下半部分, 仅当使用 5 号电池供电时需要操作电池仓,锂电池供电的设备无需操作电池仓。默认情况下,电池仓盖处于锁定状态无法直接打开,在需要安装或者更换电池时,应将水平拨动开关推至解锁侧,在电池安装完成后必须将开关推至锁定侧。注意:在安装电池时必须按照仓内+/-符号对应电池的正/负极,错误的安装极性会永久性损坏设备。
手持VH501TC多功能混合传感器信号采集读数仪使用方法
基于单片机的电压检测系统(含上位机)设计
基于单片机的电压检测系统(含上位机)设计
基于单片机的电压检测系统(含上位机)设计