当前位置: 首页 > news >正文

【BUG】记STM32F030多通道ADC DMA读取乱序问题

STM32F0多通道ADC的校准顺序与DMA乱序问题的本质

声明:本段转载:https://www.cnblogs.com/chihirosan/p/5458673.html

  • 问题描述
    通过 uint16_t ConvData[8]保存DMA搬运的ADC转换数值,但是这个数组数值的顺序总是和ADC不是顺序对应的。比如用7个通道的ADC,

    • 当设置ADC_InitStructure.ADC_ScanDirection = ADC_ScanDirection_Backward,是对应顺序是:0->0,1->7,2->6…7->1 ;
    • 当设置ADC_InitStructure.ADC_ScanDirection = ADC_ScanDirection_Upward,是对应顺序是:0->7,1->0,2->1…7->6 。
  • 问题原因
    F0的ADC在使用之前需要校准。这个7位的校准值也是放在ADC_DR中的,它也会触发DMA请求。
    可以参照F0的ADC-DMA例程,先做ADC校准、然后再设置DMA,再使能ADC的DMA。

  • 实例代码

void ADC1_DMA_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;DMA_InitTypeDef DMA_InitStructure;ADC_InitTypeDef ADC_InitStructure;RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);GPIO_InitStructure.GPIO_Pin = 0x00ff;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;GPIO_Init(GPIOA, &GPIO_InitStructure);RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);ADC_DeInit(ADC1); //ADC恢复默认设置ADC_StructInit(&ADC_InitStructure); //初始化ADC结构ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b; //12位精度ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //规定模式装换工作在连续模式ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //数据对其为右对齐ADC_InitStructure.ADC_ScanDirection = ADC_ScanDirection_Upward; // ADC_ScanDirection_Backward; //ADC的扫描方向ADC_Init(ADC1, &ADC_InitStructure);ADC_ChannelConfig(ADC1, ADC_Channel_0, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_1, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_2, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_3, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_4, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_5, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_6, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_ChannelConfig(ADC1, ADC_Channel_7, ADC_SampleTime_239_5Cycles); /* Convert the ADC1 Channel 11 with 239.5 Cycles as sampling time */ADC_GetCalibrationFactor(ADC1); /* ADC Calibration */ADC_Cmd(ADC1, ENABLE); /* Enable ADCperipheral[PerIdx] */while (!ADC_GetFlagStatus(ADC1, ADC_FLAG_ADRDY)); /* Wait the ADCEN falg *///设置DMA要在校准ADC之后DMA_DeInit(DMA1_Channel1); /* DMA1 Channel1 Config */DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t) 0x40012440; //ADC1->DR; //外设地址DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t) RegularConvData_Tab; //内存地址DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外设作为数据传输的来源DMA_InitStructure.DMA_BufferSize = 8; //DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址寄存器不变DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址寄存器不变DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为16位DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //DMA_Mode_Circular;DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA_Priority设定DMA通道x的软件优先级DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA通道x没有设置为内存到内存传输DMA_Init(DMA1_Channel1, &DMA_InitStructure);DMA_Cmd(DMA1_Channel1, ENABLE);/* DMA1 Channel1 enable */DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, ENABLE);ADC_DMARequestModeConfig(ADC1, ADC_DMAMode_Circular); /* Enable ADC_DMA */ADC_DMACmd(ADC1, ENABLE);ADC_StartOfConversion(ADC1); /* ADC1 regular Software Start Conv */}
原问题现象与深层原因
  1. DMA在ADC中的应用

    • ADC多通道扫描时,DMA用于自动搬运转换结果。但STM32F0系列的ADC包含校准值存储机制
    • 校准阶段:调用ADC_GetCalibrationFactor()时,会将一个7位的校准值写入ADC_DR寄存器(与转换结果共用同一地址),并触发DMA请求。
  2. DMA乱序的成因

    • 过早启用DMA(如在校准前)会导致DMA将校准值当做首个有效数据搬运,后续通道数据依次错位。
    • 数值错位实例
      • 若ADC通道0的值在ADC_DR中的实际存储顺序应为第1个,但因校准值的介入,DMA搬运时该值会出现在数组的第二个位置。
  3. 解决方案的工程意义

    • 校准后配置DMA:确保DMA仅搬运有效转换结果。关键代码如下:
      ADC_GetCalibrationFactor(ADC1);   // 先校准
      DMA_Init(DMA1_Channel1, ...);     // 后配置DMA
      ADC_DMACmd(ADC1, ENABLE);         // 最后使能ADC的DMA请求
      
  4. ADC扫描方向的补充解释
    ADC_ScanDirection参数定义通道扫描的物理顺序:

    • Upward模式:从通道编号低到高扫描(如0→1→2)。
    • Backward模式:从编号高到低扫描(如2→1→0)。
      此参数需要与DMA数组的预期存储顺序一致,否则需软件层调整数组索引。
扩展思考:其他引发DMA错位的可能
  1. 未清除DMA缓存:DMA传送前后未重置缓存区,残留数据可能导致混淆。
  2. 中断抢占冲突:若ADC中断优先级低于其他中断,可能因响应延迟导致数据覆盖。

总结

  1. 编码规范的重要性:结构体声明的位置不仅是语法问题,更影响代码可移植性。
  2. 硬件机制的深度理解:结合芯片手册分析异常(如STM32F0校准值特性),能快速定位隐蔽问题。
http://www.xdnf.cn/news/943921.html

相关文章:

  • 2025年能源电力系统与流体力学国际会议 (EPSFD 2025)
  • 曲面的存在性定理
  • 【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
  • 【医疗电子技术】新型医疗电子和医学人工智能发展现状和趋势
  • 【异常】极端事件的概率衰减方式(指数幂律衰减)
  • 漏洞检测方案如何选工具?开源与商业工具适用环境大不同
  • 【时序预测】-Transformer系列
  • Hibernate Validator 数据验证
  • pymongo配置事务环境并封装事务功能
  • JDBC基础关键_001_认识
  • Spring类型转换器相关接口和实现原理
  • 【JavaScript】利用`localStorage`实现多窗口数据交互同步【附完整源码】
  • OD 算法题 B卷【删除字符串中出现次数最少的字符】
  • 如何禁用windows server系统自动更新并防止自动重启
  • 推理式奖励模型:使用自然语言反馈改进强化学习效果
  • 卫星接收天线G/T值怎么计算?附G/T计算excel表格链接
  • 打卡day48
  • 12.7Swing控件5 JProgressBar
  • Spring AI中使用ChatMemory实现会话记忆功能
  • 算法打卡第18天
  • 【CUDA 】第5章 共享内存和常量内存——5.3减少全局内存访问(2)
  • Linux 环境配置
  • 【立体匹配】:双目立体匹配SGBM:(1)运行
  • 深入解析JavaScript构造函数与原型链
  • JavaScript 自定义对象详解
  • AI医生时代来临!o1模型在医疗诊断中超越人类医生
  • 查看进程线程的方法
  • 进制符号表示
  • 【阿里巴巴 x 浙江大学】信息与交互设计 - 信息设计漫谈
  • AIGC 基础篇 Python基础 02