基于STM32的心率计(一)DMA方式获取传感器数据

原创 电子电路开发学习 2021-10-01 22:19


前言

最近利用下班后的时间,使用STM32做了个心率计,从单片机程序到上位机开发,到现在为止完成的差不多了,实现很简单,STM32开发板外加一个PulseSensor传感器就行,这里我选择的是uFUN开发板,又开发了配套的串口上位机,实现数据的解析和显示,运行界面如下:

其实PulseSensor官方已经配备的了Processing语言编写的上位机软件,串口协议的,界面还蛮好看,只要按照它的通信协议,就可以实现心跳波形和心率的显示。刚好最近学习了Qt,所以就用这个小软件来练手了。

传感器介绍

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

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

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

AD采集电路的分析

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

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

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

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

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

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

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

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

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

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

很明显,可计算出

  1. U+ = 0.5 * VCC = 1.65v

应用虚短:

  1. U- = U+ = 1.65v

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

以上三式联立,可得:

  1. Uo = 3.368 - 1.205*Ui

即:

  1. Ui = 3 - 0.83 * Uo

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

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

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

DMA简介

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

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

STM32 DMA程序配置

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

0.定义一个全局变量

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

  1. uint16_t ADC_ConvertedValue;

1.配置GPIO和使能时钟

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

  1. RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

  2. RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);

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

  1. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;

  2. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //设置为模拟输入

  3. GPIO_Init(GPIOC, &GPIO_InitStructure);

2.配置DMA

配置ADC对应的DAM1通道1:

  1. DMA_DeInit(DMA1_Channel1);

  2. DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&(ADC1->DR)); //设置源地址

  3. DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; //设置内存地址

  4. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; // 设置传输方向

  5. DMA_InitStructure.DMA_BufferSize = 1;

  6. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

  7. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;

  8. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;

  9. DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;

  10. DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环模式

  11. DMA_InitStructure.DMA_Priority = DMA_Priority_High; //高优先级

  12. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;

  13. DMA_Init(DMA1_Channel1, &DMA_InitStructure);

  14. DMA_Cmd(DMA1_Channel1, ENABLE); //使能DMA1通道1

3.配置ADC

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

  1. ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;

  2. ADC_InitStructure.ADC_ScanConvMode = DISABLE ;

  3. ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;

  4. ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;

  5. ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;

  6. ADC_InitStructure.ADC_NbrOfChannel = 1;

  7. ADC_Init(ADC1, &ADC_InitStructure);

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

  1. ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 1, ADC_SampleTime_55Cycles5);

  2. ADC_DMACmd(ADC1, ENABLE);

  3. ADC_Cmd(ADC1, ENABLE);

  4. ADC_ResetCalibration(ADC1);

  5. while(ADC_GetResetCalibrationStatus(ADC1));

  6. ADC_StartCalibration(ADC1);

  7. while(ADC_GetCalibrationStatus(ADC1));

  8. ADC_SoftwareStartConvCmd(ADC1, ENABLE);

4.主程序调用

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

  1. int main(void)

  2. {

  3. float Sensor_Voltage;

  4. float Uo_Voltage;

  5. delay_init();

  6. UART1_Config(115200);

  7. ADC1_Init();

  8. while(1)

  9. {

  10. Uo_Voltage = ADC_ConvertedValue * 3.3 / 4096;

  11. Sensor_Voltage = 3.3 - Uo_Voltage; //近似值

  12. // Sensor_Voltage = 3 - 0.83 * Uo_Voltage; //实际传感器输出电压值

  13. ANO_SendFloat(0xA1, Sensor_Voltage);

  14. delay_ms(10);

  15. }

  16. }

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

函数实现

  1. //匿名上位机,波形显示一个浮点型数据ANO_SendFloat(0xA1, ad);

  2. void ANO_SendFloat(int channel, float f_dat)

  3. {

  4. u8 tbuf[8];

  5. int i;

  6. unsigned char* p;

  7. for(i = 0; i <= 7; i++)

  8. tbuf[i] = 0;

  9. p = (unsigned char*)&f_dat;

  10. tbuf[0] = 0x88;

  11. tbuf[1] = channel; //0xA1

  12. tbuf[2] = 4;

  13. tbuf[3] = (unsigned char)(*(p + 3)); //取float类型数据存储在内存中的四个字节

  14. tbuf[4] = (unsigned char)(*(p + 2));

  15. tbuf[5] = (unsigned char)(*(p + 1));

  16. tbuf[6] = (unsigned char)(*(p + 0));

  17. for(i = 0; i <= 6; i++)

  18. tbuf[7] += tbuf[i]; //校验和

  19. printf("%s", tbuf);

  20. }

实际的显示

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

匿名上位机的帧格式配置

实际的显示效果:

总结

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

资料下载

  • STM32工程下载:

    https://wcc-blog.oss-cn-beijing.aliyuncs.com/BlogFile/STM32_DMA_ADC.rar

参考资料

  • PulseSensor官网:

    https://pulsesensor.com/

  • 手指检测心跳设计——传感器制作篇:

    https://blog.csdn.net/qq_34445388/article/details/79781181

  • 玩的就是心跳 —— 使用 PulseSensor 脉搏传感器测量心率:

    http://www.shaoguoji.cn/2017/06/29/use-pulsesensor-with-stm32/

  • 双T陷波器s域计算分析(纯手算,工程版!)

    http://bbs.eeworld.com.cn/thread-498219-1-1.html

电子电路开发学习 单片机点灯小能手,电子行业从业者。开发板评测、嵌入式开源项目分享、学习笔记记录。可能不会经常更新,但每一篇都是精心编写。
评论
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
    GIRtina 2025-01-07 11:02 124浏览
  • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
    hai.qin_651820742 2025-01-07 14:52 108浏览
  • 故障现象一辆2017款东风风神AX7车,搭载DFMA14T发动机,累计行驶里程约为13.7万km。该车冷起动后怠速运转正常,热机后怠速运转不稳,组合仪表上的发动机转速表指针上下轻微抖动。 故障诊断 用故障检测仪检测,发动机控制单元中无故障代码存储;读取发动机数据流,发现进气歧管绝对压力波动明显,有时能达到69 kPa,明显偏高,推断可能的原因有:进气系统漏气;进气歧管绝对压力传感器信号失真;发动机机械故障。首先从节气门处打烟雾,没有发现进气管周围有漏气的地方;接着拔下进气管上的两个真空
    虹科Pico汽车示波器 2025-01-08 16:51 70浏览
  • 本文介绍编译Android13 ROOT权限固件的方法,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。关闭selinux修改此文件("+"号为修改内容)device/rockchip/common/BoardConfig.mkBOARD_BOOT_HEADER_VERSION ?= 2BOARD_MKBOOTIMG_ARGS :=BOARD_PREBUILT_DTB
    Industio_触觉智能 2025-01-08 00:06 92浏览
  •  在全球能源结构加速向清洁、可再生方向转型的今天,风力发电作为一种绿色能源,已成为各国新能源发展的重要组成部分。然而,风力发电系统在复杂的环境中长时间运行,对系统的安全性、稳定性和抗干扰能力提出了极高要求。光耦(光电耦合器)作为一种电气隔离与信号传输器件,凭借其优秀的隔离保护性能和信号传输能力,已成为风力发电系统中不可或缺的关键组件。 风力发电系统对隔离与控制的需求风力发电系统中,包括发电机、变流器、变压器和控制系统等多个部分,通常工作在高压、大功率的环境中。光耦在这里扮演了
    晶台光耦 2025-01-08 16:03 61浏览
  • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
    知白 2025-01-07 15:02 141浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 204浏览
  • 「他明明跟我同梯进来,为什么就是升得比我快?」许多人都有这样的疑问:明明就战绩也不比隔壁同事差,升迁之路却比别人苦。其实,之间的差异就在于「领导力」。並非必须当管理者才需要「领导力」,而是散发领导力特质的人,才更容易被晓明。许多领导力和特质,都可以通过努力和学习获得,因此就算不是天生的领导者,也能成为一个具备领导魅力的人,进而被老板看见,向你伸出升迁的橘子枝。领导力是什么?领导力是一种能力或特质,甚至可以说是一种「影响力」。好的领导者通常具备影响和鼓励他人的能力,并导引他们朝着共同的目标和愿景前
    优思学院 2025-01-08 14:54 61浏览
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 164浏览
  • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
    丙丁先生 2025-01-07 09:25 117浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦