原子大哥的寄存器版本的例程中,操作ADC的时候,没有使用DMA例程,所以,今天我自己将DMA和ADC两个寄存器版本的程序融合在一起,主要还是为了加深自己对这个的理解,但是出现了问题,请大家看一下,能帮忙找一下不?红色是我添加和修改的部分。
主要修改 程序如下:
1,dma.c中修改的地方:
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|=1<<1; //允许传输完中断
DMA_CHx->CCR|=0<<4; //从外设读
DMA_CHx->CCR|=1<<5; //循环模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=1<<7; //存储器增量模式
DMA_CHx->CCR|=1<<8; //外设数据宽度为16位
DMA_CHx->CCR|=1<<10; //存储器数据宽度16位
DMA_CHx->CCR|=1<<12; //中等优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
///设定中断
MY_NVIC_Init(1,3,DMA1_Channel1_IRQChannel ,2);
}
2,adc.c中修改的地方:
void Adc_Init(void)
{
//1,先初始化IO口
///ADC通道0 在PA0上,所以,我们先要使能 PORTA的时钟,然后设置PA0为模拟输入。
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
GPIOA->CRL&=0XFFFF0000;//PA0 1 2 3 anolog输入
//通道10/11设置
///使能ADC1 时钟,并设置分频因子。
RCC->APB2ENR|=1<<9; //ADC1时钟使能 ///1:ADC1接口时钟开启
RCC->APB2RSTR|=1<<9; //ADC1复位 ///1:复位ADC1 接口
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->CFGR&=~(3<<14); //分频因子清零 ///00:PCLK2 2分频后作为ADC时钟
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=2<<14; ///10:PCLK2 6分频后作为ADC时钟,这里应该是12mhz
///3,设置ADC1 的工作模式
ADC1->CR1&=0XF0FFFF; //工作模式清零
ADC1->CR1|=0<<16; //独立工作模式
ADC1->CR1&=(1<<8); //扫描模式
ADC1->CR2&=(1<<1); //连续转换模式
ADC1->CR2|=1<<8; //启用DMA
ADC1->CR2&=~(7<<17);
ADC1->CR2|=7<<17; //软件控制转换 ///111 :SWSTART
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发 ///1:使用外部事件启动转换
ADC1->CR2&=~(1<<11); //右对齐 ///0:右对齐
ADC1->SQR1&=~(0XF<<20); ///0000:1个转换
ADC1->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1 ///0000:1个转换
//4,设置通道0~3的采样时间
ADC1->SMPR2&=0XFFFFF000;//通道0,1,2,3采样时间清空
ADC1->SMPR2|=7<<9; //通道3 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<6; //通道2 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<3; //通道1 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<0; //通道0 239.5周期,提高采样时间可以提高精确度
///5,开启AD转换器,并校准
ADC1->CR2|=1<<0; //开启AD转换器
ADC1->CR2|=1<<3; //使能复位校准 ///1:初始化校准寄存器。
while(ADC1->CR2&1<<3); //等待校准结束 ///0:校准寄存器已初始化
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
}
3,中断函数:(主要还是显示)
u16 ADC1_DR, adcx1;
void DMAChannel1_IRQHandler(void)
{
u16 i;
u32 sum[6]={0};
float TEMP;
LED0 =!LED0;
ADC1->CR2&=~(1<<1); //关闭连续转换
for( i = 0; i<768 ;i+= 6)
{
sum[0] += SendBuff;
sum[1] += SendBuff[i+1];
sum[2] += SendBuff[i+2];
sum[3] += SendBuff[i+3];
sum[4] += SendBuff[i+4];
sum[5] += SendBuff[i+5];
}
for(i = 0;i <6 ;i++)
{
ADC1_DR = sum/6;
TEMP=(float)ADC1_DR*(3.3/4096);
adcx1=TEMP;
LCD_ShowNum(149,70+i*20,adcx1,1,16);//显示电压值
TEMP-=adcx1;
TEMP*=1000;
LCD_ShowNum(165,70+i*20,TEMP,3,16);
}
delay_ms(200);
DMA1->IFCR |= 1<<1; //清零通道完成中断标志位
ADC1->CR2|=1<<1; //启用连续转换
ADC1->CR2|=1<<22; //启动规则转换通道
}
4,main函数的内容:
u8 SendBuff[768];
int main(void)
{
u16 adcx;
float temp;
Stm32_Clock_Init(9);//系统时钟设置
delay_init(72); //延时初始化
uart_init(72,9600); //串口1初始化
LED_Init();
LCD_Init();
Adc_Init();
POINT_COLOR=RED;//设置字体为红色
LCD_ShowString(60,50,"Mini STM32");
LCD_ShowString(60,70,"ADC TEST");
LCD_ShowString(60,90,"ATOM@ALIENTEK");
LCD_ShowString(60,110,"2010/12/30");
//显示提示信息
POINT_COLOR=BLUE;//设置字体为蓝色
LCD_ShowString(60,130,"ADC_CH0_VAL:");
LCD_ShowString(60,150,"ADC_CH0_VOL:0.000V");
MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)SendBuff,768);
MYDMA_Enable(DMA1_Channel1);
while(1)
{
LED0=!LED0;
delay_ms(250);
}
}
上面就是小弟自己修改的地方,但是在tft屏幕上面,将PA0和3.3v的电压点接触后,数据没有一点反应,上面的中断函数的内容,也是参考了网上的写法,但是主要目的依然是在屏幕上面显示数据的。
希望各位看官和原子大哥能帮忙分析一下看是什么原因。
|