本人在MINISTM32 V2.0基础上实现了带DMA的双通道同步规则通道ADC,使用的是Keil3.80编译并在MINISTM32 V2.0成功实现,而根据论坛上的说明在Keil5中将一些启动文件更换,编译通过,但是在板子上跑其他正常,只有上述功能无法实现,一直没有查出问题,今天特此请教!
说明:1、采用定时器3的TRGO事件触发同步规则通道ADC(在STM32中文手册第162页表65);
2、ADC采集完成通过DMA自动存储到变量以便随时使用,但这个变量中一直是0。
代码如下:
main()中初始化:
MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)&ADC1_Value,1); //ADC1_Value是32位变量
Adc_Init();
Timer3_Init(10,7199);
定时器3初始化:
void Timer3_Init(u16 arr,u16 psc) //DMA ADC 1ms
{
RCC->APB1ENR|=1<<1;
TIM3->ARR=arr;
TIM3-> SC=psc;
TIM3->CR2|=2<<4; //更新触发TRGO使能
TIM3->CR1|=0x01; //
}
DMA初始化:
void MYDMA_Config(DMA_Channel_TypeDef*DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
u32 DR_Base; //做缓冲用,不知道为什么.非要不可
RCC->AHBENR|=1<<0;//开启DMA1时钟
DR_Base=cpar;
DMA_CHx->CPAR=DR_Base; //DMA1 外设地址
DMA_CHx->CMAR=(u32)cmar; //DMA1,存储器地址
DMA1_MEM_LEN=cndtr; //保存DMA传输数据量
DMA_CHx->CNDTR=cndtr; //DMA1,传输数据量
DMA_CHx->CCR=0X00000000;//复位
DMA_CHx->CCR|=0<<4; //从外设读
DMA_CHx->CCR|=1<<5; //循环模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=0<<7; //存储器非增量模式
DMA_CHx->CCR|=2<<8; //外设数据宽度为32位
DMA_CHx->CCR|=2<<10; //存储器数据宽度32位
DMA_CHx->CCR|=3<<12; //最高优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
ADC初始化:
void Adc_Init(void)
{
//先初始化IO口
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
RCC->APB2ENR|=1<<4; //使能PORTC口时钟
GPIOA->CRL&=0XFFFFFFF0;//PA0 anolog输入 电压
GPIOC->CRL&=0XFFFFFFF0;//PC0 anolog输入 电流
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
RCC->APB2ENR|=1<<10; //ADC2时钟使能
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->APB2RSTR|=1<<10; //ADC2复位
RCC->APB2RSTR&=~(1<<10);//复位结束
RCC->CFGR&=~(3<<14); //分频因子清零
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=2<<14;
ADC1->CR1&=0XF0FFFF; //工作模式清零
ADC1->CR1|=6<<16; //规则同步工作模式
ADC2->CR1&=0XF0FFFF; //工作模式清零
ADC2->CR1|=6<<16; //规则同步工作模式
// ADC1->CR1&=~(1<<8); //非扫描模式
ADC1->CR2&=~(1<<1); //单次转换模式
ADC2->CR2&=~(1<<1); //单次转换模式
ADC1->CR2&=~(7<<17);
ADC1->CR2&=~(7<<17); //清外部事件位
ADC2->CR2&=~(7<<17);
ADC2->CR2&=~(7<<17); //清外部事件位
ADC1->CR2|=4<<17; //定时器3的CC4控制转换
ADC2->CR2|=4<<17; //定时器3的CC4控制转换
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC2->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐
ADC2->CR2&=~(1<<11); //右对齐
ADC1->SQR1&=~(0XF<<20);
ADC2->SQR1&=~(0XF<<20);
ADC1->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1
ADC2->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1
ADC1->SQR3&=~(0XF<<0);
ADC1->SQR3|=0<<0; //ADC1第一个转换的是1通道
ADC2->SQR3&=~(0XF<<0);
ADC2->SQR3|=13<<0; //ADC2第一个转换的是11通道
//设置通道0~3的采样时间
// ADC1->SMPR2&=0XFFFFF000;//通道0,1,2,3采样时间清空
ADC1->SMPR2|=7<<3; //通道1 239.5周期,提高采样时间可以提高精确度
ADC2->SMPR1|=7<<3; //通道11 239.5周期,提高采样时间可以提高精确度
// ADC1->SMPR2|=7<<3; //通道1 239.5周期,提高采样时间可以提高精确度
// ADC1->SMPR2|=7<<0; //通道0 239.5周期,提高采样时间可以提高精确度
ADC1->CR2|=1<<8; //DMA传输使能
ADC1->CR2|=1<<0; //开启AD转换器
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待校准结束
ADC2->CR2|=1<<0; //开启AD转换器
ADC2->CR2|=1<<3; //使能复位校准
while(ADC2->CR2&1<<3); //等待校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
ADC2->CR2|=1<<2; //开启AD校准
while(ADC2->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
}
|