完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
现在需要单路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(); } |
|
相关推荐
15个回答
|
|
|
|
|
|
程序的执行时间、时序及 DMA状态控制,这是解决的方法.
|
|
|
|
这个考虑过,但我目前测试AD采集端给的是固定电压,我采集2048个点,值都是一样的,而且我是直接用赋值符传递到了fft输入端,不可能会变得啊,有一个问题就是AD采集值得数据类型是uint_16,而fft输入端数据的类型是float_32的,但是我要是AD采集端的数据类型改为浮点型的,那采集到的结果也错了,,, |
|
|
|
没有看到你的主程序是如何安排ADC和FFT的。你应该在ADC采集2048个点数据DMA产生中断后停止ADC转换,进行FFT处理后重新启动新一轮ADC转换。否则当你在进行FFT处理时,有可能新的ADC数据已经更新缓冲区了。
|
|
|
|
ctwewer 发表于 2018-11-20 13:22 这是我的主函数和中断向量函数,按照你说的,在存满2048个数据后,进入中断,将ADC失能,清除标志位,将StartFlag标志位置1,再将存储的数据送入FFT_input数组内,然后进行fft变换,感觉没问题啊 |
|
|
|
lxz0404 发表于 2018-11-20 13:41 感觉你应该在DMA中断函数中就应该立即停止ADC了,然后在FFT完成后再启动ADC。 |
|
|
|
|
|
|
|
谢谢你的帮助,已经改好了, |
|
|
|
|
|
|
|
|
|
|
|
楼主也可以定义4096长度的数组,使用DMA半满和全满两个中断信号,利用乒乓方式,达到ADC采集,FFT变换同时进行,又不互相干扰的效果!
|
|
|
|
mark
|
|
|
|
程序的执行时间、时序及 DMA状态控制,这是解决的方法
|
|
|
|
456162dss 发表于 2018-11-20 14:53 什么意思?可以说详细点吗,在程序里面怎么实现 |
|
|
|
能有什么问题
|
|
|
|
你正在撰写答案
如果你是对答案或其他答案精选点评或询问,请使用“评论”功能。
2049 浏览 1 评论
AD7686芯片不传输数据给STM32,但是手按住就会有数据。
1893 浏览 3 评论
4485 浏览 0 评论
如何解决MPU-9250与STM32通讯时,出现HAL_ERROR = 0x01U
2040 浏览 1 评论
hal库中i2c卡死在HAL_I2C_Master_Transmit
2549 浏览 1 评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-21 22:07 , Processed in 0.884158 second(s), Total 99, Slave 83 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号