ADC-dma死循环问题

方法一

在这里插入图片描述
在这里插入图片描述
初始化dma时候,里面会先使能dma,然后打开相应的中断,然后结果会导致main函数中卡死
在这里插入图片描述
卡在箭头处,也就是打不开adc和dma

解决方案

全部注释,只留第一个配置dma的函数
在这里插入图片描述
11.16:
补充:同样的现象:程序卡死在HAL_ADC_Start_DMA这个函数里面,下面程序无法运行,但是数组里面一直会接收到数据。

出去上文提到的解决方法外,

还有解决方法2:

在这里插入图片描述
dma初始化一定要在adc初始化之前,否则死循环

解决方法3:

注意缓存的大小
在这里插入图片描述
在这里插入图片描述
内存的缓存区必须是16位的(具体原因不详,实测就是这样,如果改成32位的就是会死循环,原因不详)

cubmax配置图:

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

总结:

adc和dma一起用的时候,经常会遇到程序卡死的问题(笔者身边的人也经常遇到这类问题),以上三种解决方案一般能处理死循环的问题,如果有新的情况后续会再补充。

个人的想法:

综合上文三种解决方法可以得知,这类问题大多和dma有关,所以我猜测大概率是dma中断和adc中断冲突导致的,第一种解决方法是直接把dma中断注释了,第三种则是吧dma中断优先级调到最高。

前几天参加电赛的过程中用到adc和dma时候遇到这个问题,配置代码配置了很久,一直死循环,这几天翻博客时候发现以前遇到过这种问题,并且还整理了文章,而且我居然一点映像都没有,电赛那几天找到了后面两种解决方法,并且对这篇文章作出补充。

  • 21
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
这是一个基于STM32ADC-DMA代码示例,可以将ADC采集到的数据通过DMA传输到内存中: ```c #include "stm32f4xx.h" #include "stm32f4xx_adc.h" #include "stm32f4xx_dma.h" #include "stm32f4xx_gpio.h" #include "stm32f4xx_rcc.h" #define ADC1_DR_ADDRESS ((uint32_t)0x4001204C) __IO uint16_t ADCConvertedValue; void ADC_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; ADC_InitTypeDef ADC_InitStructure; DMA_InitTypeDef DMA_InitStructure; /* Enable ADC3, DMA2 and GPIO clocks ****************************************/ RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2 | RCC_AHB1Periph_GPIOC, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); /* DMA2 Stream0 channel0 configuration **************************************/ DMA_InitStructure.DMA_Channel = DMA_Channel_0; DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)ADC1_DR_ADDRESS; DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)&ADCConvertedValue; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory; 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_FIFOMode = DMA_FIFOMode_Disable; DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull; DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; DMA_Init(DMA2_Stream0, &DMA_InitStructure); DMA_Cmd(DMA2_Stream0, ENABLE); /* Configure ADC1 Channel11 pin as analog input ******************************/ GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ; GPIO_Init(GPIOC, &GPIO_InitStructure); /* ADC Common Init *********************************************************/ ADC_CommonInitTypeDef ADC_CommonInitStructure; ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent; ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div2; ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles; ADC_CommonInit(&ADC_CommonInitStructure); /* ADC1 Init ****************************************************************/ ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None; ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfConversion = 1; ADC_Init(ADC1, &ADC_InitStructure); /* ADC1 regular channel11 configuration *************************************/ ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 1, ADC_SampleTime_144Cycles); /* Enable DMA request after last transfer (Single-ADC mode) */ ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE); /* Enable ADC1 DMA */ ADC_DMACmd(ADC1, ENABLE); /* Enable ADC1 **************************************************************/ ADC_Cmd(ADC1, ENABLE); /* Start ADC1 Software Conversion */ ADC_SoftwareStartConv(ADC1); } int main(void) { ADC_Config(); while (1) { /* Wait DMA interrupt */ } } ``` 在这个示例中,我们使用了ADC1和DMA2,采集了PC1引脚的模拟信号。ADC1使用了单次转换模式,采样时间为144个时钟周期,并且启用了DMA传输。DMA传输是循环模式,一旦转换完成就会将结果存储到内存中。 在主循环中,我们可以插入需要执行的代码,或者直接等待DMA中断。当DMA传输完成时,可以在DMA中断处理函数中对数据进行处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值