OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2583|回复: 2

有关stm32F1系列双ADC模式与FSMC的问题

[复制链接]

1

主题

3

帖子

0

精华

初级会员

Rank: 2

积分
51
金钱
51
注册时间
2017-11-6
在线时间
6 小时
发表于 2018-4-28 17:00:30 | 显示全部楼层 |阅读模式
10金钱
本帖最后由 wzq11021 于 2018-4-28 17:02 编辑

想做一个简易示波器,利用双ADC模式采集两路波形数据,通过DMA传送到内部sram,然后通过FSMC驱动的TFTLCD屏显示波形。现在的问题是,每次ADC采集完数据后LCD就不能正常用了,具体表现为执行一些显示指令时LCD屏幕没有反应。
目前确定是adc转换以及dma的问题,更细节的问题根源还没有查。猜测可能是FSMC地址空间多主访问的bug,造成总线失效,但是限于知识水平,不能确定是这个原因。
这个问题困扰我好几天了,希望能有大佬帮助解决一下,指出问题的源头,谢谢啦!

ADC初始化代码
[mw_shl_code=c,true]void  Adc_Init(void)
{         
        ADC_InitTypeDef ADC_InitStructure;
        GPIO_InitTypeDef GPIO_InitStructure;
        
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1|RCC_APB2Periph_GPIOC|RCC_APB2Periph_ADC2        , ENABLE );          //使能ADC1、ADC2、GPIOC通道时钟

        RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

        //PC1作为ADC1模拟通道输入引脚,PC2作为ADC2模拟通道输入引脚           
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_2;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
        GPIO_Init(GPIOC, &GPIO_InitStructure);        

        //ADC1配置
        ADC_DeInit(ADC1);  //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值

        ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;        //ADC工作模式:ADC1和ADC2工作在双ADC模式
        ADC_InitStructure.ADC_ScanConvMode = DISABLE;        //模数转换工作在单通道模式
        ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;        //模数转换工作在连续转换模式
        ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;        //转换由软件而不是外部触发启动
        ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;        //ADC数据右对齐
        ADC_InitStructure.ADC_NbrOfChannel = 1;        //顺序进行规则转换的ADC通道的数目
        ADC_Init(ADC1, &ADC_InitStructure);        //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器

  ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 1, ADC_SampleTime_41Cycles5 );//配置ADC通道参数
        
        //ADC_TempSensorVrefintCmd(ENABLE); //开启内部温度传感器
        
        ADC_Cmd(ADC1, ENABLE);        //使能指定的ADC1
        ADC_ResetCalibration(ADC1);        //使能复位校准  
        while(ADC_GetResetCalibrationStatus(ADC1));        //等待复位校准结束
        ADC_StartCalibration(ADC1);         //开启AD校准
        while(ADC_GetCalibrationStatus(ADC1));         //等待校准结束
        delay_us(500);
        
        //ADC2配置
        ADC_DeInit(ADC2);  //复位ADC2,将外设 ADC2 的全部寄存器重设为缺省值

        ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;        //ADC工作模式:ADC1和ADC2工作在双ADC模式
        ADC_InitStructure.ADC_ScanConvMode = DISABLE;        //模数转换工作在单通道模式
        ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;        //模数转换工作在连续转换模式
        ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;        //转换由软件而不是外部触发启动
        ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;        //ADC数据右对齐
        ADC_InitStructure.ADC_NbrOfChannel = 1;        //顺序进行规则转换的ADC通道的数目
        ADC_Init(ADC2, &ADC_InitStructure);        //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器

  ADC_RegularChannelConfig(ADC2, ADC_Channel_12, 1, ADC_SampleTime_41Cycles5 );//配置ADC通道参数
        
        ADC_Cmd(ADC2, ENABLE);        //使能指定的ADC1
        ADC_ResetCalibration(ADC2);        //使能复位校准  
        while(ADC_GetResetCalibrationStatus(ADC2));        //等待复位校准结束
        ADC_StartCalibration(ADC2);         //开启AD校准
        while(ADC_GetCalibrationStatus(ADC2));         //等待校准结束
        delay_us(500);
        ADC_ExternalTrigConvCmd(ADC2, ENABLE);
        
        //DMA配置
        MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)&ADC_SampleDepth,10000);
        
        ADC_DMACmd(ADC1,ENABLE);
        ADC_DMACmd(ADC2,ENABLE);
        //        ADC_SoftwareStartConvCmd(ADC1, ENABLE);                //使能指定的ADC1的软件转换启动功能
}                                 [/mw_shl_code]
DMA配置代码
[mw_shl_code=c,true]
u16 DMA1_MEM_LEN;//保存DMA每次数据传送的长度            
//DMA1的各通道配置
//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHxMA通道CHx
//cpar:外设地址
//cmar:存储器地址
//cndtr:数据传输量

void MYDMA_Config(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
        DMA_InitTypeDef DMA_InitStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
         RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);        //使能DMA传输
        
  DMA_DeInit(DMA_CHx);   //将DMA的通道1寄存器重设为缺省值

        DMA1_MEM_LEN=cndtr;
        DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;  //DMA外设基地址
        DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  //数据传输方向,从外设读取发送到内存
        DMA_InitStructure.DMA_BufferSize = cndtr;  //DMA通道的DMA缓存的大小
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;  //数据宽度为32位
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word; //数据宽度为16位
        DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常缓存模式
        DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh; //DMA通道 x拥有中优先级
        DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
        DMA_Init(DMA_CHx, &DMA_InitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器        
        
        NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2 ;//抢占优先级3
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;                //子优先级3
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
        NVIC_Init(&NVIC_InitStructure);        //根据指定的参数初始化VIC寄存器
        DMA_ITConfig(DMA1_Channel1,DMA1_IT_TC1,ENABLE);
        DMA_ITConfig(DMA1_Channel1,DMA_IT_TE,ENABLE);
} [/mw_shl_code]


DMA中断处理代码
[mw_shl_code=c,true]void DMA1_Channel1_IRQHandler()
{
        if(DMA_GetFlagStatus(DMA1_FLAG_TC1)==SET)
        {
                ADC1->CR2&=0xfffffffd;//清零ADC1的cont位,停止其转换
                ADC1->CR2|=0x2;//置位ADC1的cont位,保证下一次软件触发可以正常开始
                DMA_Cmd(DMA1_Channel1,DISABLE);
                DMA_SetCurrDataCounter(DMA1_Channel1,10000);//重设DMA传送的值
                OutputStatus|=0x8000;
                printf("DMA传输完成\r\n");
        }
        if(DMA_GetFlagStatus(DMA1_FLAG_TE1)==SET)
        {
                ADC1->CR2&=0xfffffffd;//清零ADC1的cont位,停止其转换
                ADC_Cmd(ADC1,DISABLE);//关闭ADC1
                ADC_Cmd(ADC2,DISABLE);//关闭ADC2
                DMA_Cmd(DMA1_Channel1,DISABLE);//终止DMA通道
                printf("DMA传输出错,剩余传输量%d\r\n",DMA_GetCurrDataCounter(DMA1_Channel1));
        }
        DMA_ClearITPendingBit(DMA1_IT_TC1|DMA1_IT_TE1);
}[/mw_shl_code]



最佳答案

查看完整内容[请看2#楼]

问题已经解决,DMA缓冲区设置为2500个32位字长的数组,但是DMA传输时设置的数据量是cndtr=10000,最后导致数组溢出。我误以为cndtr指的是数据长度,实际上cndtr指的是数据的个数。
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

1

主题

3

帖子

0

精华

初级会员

Rank: 2

积分
51
金钱
51
注册时间
2017-11-6
在线时间
6 小时
 楼主| 发表于 2018-4-28 17:00:31 | 显示全部楼层
问题已经解决,DMA缓冲区设置为2500个32位字长的数组,但是DMA传输时设置的数据量是cndtr=10000,最后导致数组溢出。我误以为cndtr指的是数据长度,实际上cndtr指的是数据的个数。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2018-4-29 01:40:08 | 显示全部楼层
帮顶
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-7-18 17:40

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表