ADC+DMA采集并传输完成后的中断问题

2019-07-14 15:55发布

现在需要单路AD连续采集2048个数据存入数组并通过DMA传输至存储器并进行FFT变换,AD采集和FFT变换两个分开的部分已经可以了,但是加在一起的话就不知道数据采集和传输是怎样的一个顺序了,AD采集到的电压值是正确的,但是传递到fft的输入就完全变了,这是什么原因呢,下面附下我的配置函数,希望知道的朋友帮帮忙看下,谢谢!(文字复制过来就乱码了,不过应该不影响看懂程序吧)#include "bsp_adc.h"
#include "ARM_math.h"

__IO uint16_t ADC_ConvertedValue[2048];

static void ADC_GPIO_Config(void)
{
                GPIO_InitTypeDef GPIO_InitStructure;
        
        // ʹÄÜ GPIO ʱÖÓ
        RCC_AHB1PeriphClockCmd(RHEOSTAT_ADC_GPIO_CLK, ENABLE);
               
        // ÅäÖÃ IO
        GPIO_InitStructure.GPIO_Pin = RHEOSTAT_ADC_GPIO_PIN;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;            
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ; //²»ÉÏÀ­²»ÏÂÀ­
        GPIO_Init(RHEOSTAT_ADC_GPIO_PORT, &GPIO_InitStructure);               
}

void NVIC_Config(void)
{
  NVIC_InitTypeDef   NVIC_InitStructure;

  /* Enable the DMA2_Stream0 gloabal Interrupt */
  NVIC_InitStructure.NVIC_IRQChannel = DMA_IRQ;
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);
}

static void ADC_Mode_Config(void)
{
        DMA_InitTypeDef DMA_InitStructure;
        ADC_InitTypeDef ADC_InitStructure;
  ADC_CommonInitTypeDef ADC_CommonInitStructure;

        
  // ------------------DMA Init ½á¹¹Ìå²ÎÊý ³õʼ»¯--------------------------

  RCC_AHB1PeriphClockCmd(RHEOSTAT_ADC_DMA_CLK, ENABLE);

  DMA_InitStructure.DMA_Channel = RHEOSTAT_ADC_DMA_CHANNEL;
        // ÍâÉè»ùַΪ£ºADC Êý¾Ý¼Ä´æÆ÷µØÖ·
        DMA_InitStructure.DMA_PeripheralBaseAddr = RHEOSTAT_ADC_DR_ADDR;        
  // ´æ´¢Æ÷µØÖ·£¬Êµ¼ÊÉϾÍÊÇÒ»¸öÄÚ²¿SRAMµÄ±äÁ¿        
        DMA_InitStructure.DMA_Memory0BaseAddr = (u32)&ADC_ConvertedValue;
  // Êý¾Ý´«Êä·½ÏòΪÍâÉèµ½´æ´¢Æ÷        
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;        
        // »º³åÇø´óСΪ£¬Ö¸Ò»´Î´«ÊäµÄÊý¾ÝÁ¿£¨Ò»´Î´«ÊäÍê³É¼´½øÈëÖжϣ©
        DMA_InitStructure.DMA_BufferSize = 2048;        
        // ÍâÉè¼Ä´æÆ÷Ö»ÓÐÒ»¸ö£¬µØÖ·²»ÓõÝÔö
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  // ´æ´¢Æ÷µØÖ·¹Ì¶¨
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  // // ÍâÉèÊý¾Ý´óСΪ°ë×Ö£¬¼´Á½¸ö×Ö½Ú
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
  //        ´æ´¢Æ÷Êý¾Ý´óСҲΪ°ë×Ö£¬¸úÍâÉèÊý¾Ý´óСÏàͬ
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;        
        // Ñ­»·´«Êäģʽ
        DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  // DMA ´«ÊäͨµÀÓÅÏȼ¶Îª¸ß£¬µ±Ê¹ÓÃÒ»¸öDMAͨµÀʱ£¬ÓÅÏȼ¶ÉèÖò»Ó°Ïì
        DMA_InitStructure.DMA_Priority = DMA_Priority_High;
  // ½ûÖ¹DMA FIFO        £¬Ê¹ÓÃÖ±Á¬Ä£Ê½
  DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;  
  // FIFO ´óС£¬FIFOģʽ½ûֹʱ£¬Õâ¸ö²»ÓÃÅäÖà       
  DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
  DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
  DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;  
  //³õʼ»¯DMAÁ÷£¬Á÷Ï൱ÓÚÒ»¸ö´óµÄ¹ÜµÀ£¬¹ÜµÀÀïÃæÓкܶàͨµÀ
        DMA_Init(RHEOSTAT_ADC_DMA_STREAM, &DMA_InitStructure);
        // ʹÄÜDMAÁ÷
  DMA_Cmd(RHEOSTAT_ADC_DMA_STREAM, ENABLE);
        
        // ¿ªÆôADCʱÖÓ
        RCC_APB2PeriphClockCmd(RHEOSTAT_ADC_CLK , ENABLE);
  // -------------------ADC Common ½á¹¹Ìå ²ÎÊý ³õʼ»¯------------------------
        // ¶ÀÁ¢ADCģʽ
  ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
  // ʱÖÓΪfpclk=90MHZ 4·ÖƵ        
  ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;
  // ½ûÖ¹DMAÖ±½Ó·ÃÎÊģʽ        
  ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;
  // ²ÉÑùʱ¼ä¼ä¸ô        
  ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles;  
  ADC_CommonInit(&ADC_CommonInitStructure);
        
  // -------------------ADC Init ½á¹¹Ìå ²ÎÊý ³õʼ»¯--------------------------

  ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;                                 // ADC ·Ö±æÂÊ
  ADC_InitStructure.ADC_ScanConvMode = DISABLE;                                                                  // ½ûֹɨÃèģʽ£¬¶àͨµÀ²É¼¯²ÅÐèÒª        
  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;                                                  // Á¬Ðøת»»        
  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;                     //ת»»Í¨µÀ 1¸ö                     
  ADC_Init(RHEOSTAT_ADC, &ADC_InitStructure);
  //---------------------------------------------------------------------------

  ADC_RegularChannelConfig(RHEOSTAT_ADC, RHEOSTAT_ADC_CHANNEL, 1, ADC_SampleTime_56Cycles);


  ADC_DMARequestAfterLastTransferCmd(RHEOSTAT_ADC, ENABLE);

  ADC_DMACmd(RHEOSTAT_ADC, ENABLE);
        DMA_ITConfig(DMA2_Stream0,DMA_IT_TC , ENABLE);
  /* Enable ADC3 */
  ADC_Cmd(RHEOSTAT_ADC, ENABLE);

  ADC_Cmd(RHEOSTAT_ADC, ENABLE);

  ADC_SoftwareStartConv(RHEOSTAT_ADC);
}



void ADC_Configuration(void)
{
        NVIC_Config();
        ADC_GPIO_Config();
        ADC_Mode_Config();
        
}
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
14条回答
peternie
1楼-- · 2019-07-14 20:03
程序的执行时间、时序及 DMA状态控制,这是解决的方法.
lxz0404
2楼-- · 2019-07-14 20:55
peternie 发表于 2018-11-20 12:47
程序的执行时间、时序及 DMA状态控制,这是解决的方法.

这个考虑过,但我目前测试AD采集端给的是固定电压,我采集2048个点,值都是一样的,而且我是直接用赋值符传递到了fft输入端,不可能会变得啊,有一个问题就是AD采集值得数据类型是uint_16,而fft输入端数据的类型是float_32的,但是我要是AD采集端的数据类型改为浮点型的,那采集到的结果也错了,,,
ctwewer
3楼-- · 2019-07-14 21:55
没有看到你的主程序是如何安排ADC和FFT的。你应该在ADC采集2048个点数据DMA产生中断后停止ADC转换,进行FFT处理后重新启动新一轮ADC转换。否则当你在进行FFT处理时,有可能新的ADC数据已经更新缓冲区了。
lxz0404
4楼-- · 2019-07-14 22:40
 精彩回答 2  元偷偷看……
ctwewer
5楼-- · 2019-07-15 04:35
lxz0404 发表于 2018-11-20 13:41
这是我的主函数和中断向量函数,按照你说的,在存满2048个数据后,进入中断,将ADC失能,清除标志位,将StartFlag标志位置1,再将存储的数据送入FFT_input数组内,然后进行fft变换,感觉没问题啊

感觉你应该在DMA中断函数中就应该立即停止ADC了,然后在FFT完成后再启动ADC。
peternie
6楼-- · 2019-07-15 09:25
ctwewer 发表于 2018-11-20 13:57
感觉你应该在DMA中断函数中就应该立即停止ADC了,然后在FFT完成后再启动ADC。

正解~~~~~~

一周热门 更多>