初级会员

- 积分
- 51
- 金钱
- 51
- 注册时间
- 2017-11-6
- 在线时间
- 6 小时
|
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_CHx MA通道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指的是数据的个数。
|