//主函数
int main(void)
{
// u32 ADC1_Buf[32];
u32 ADC1_Buf[1];
u16 adcx1;
u16 adcx2;
u8 arry1[6];
u8 arry2[6];
float temp;
SystemInit();
// Stm32_Clock_Init(9); //
delay_init(72); //
KEY_Init();
LED_Init();
ADC1_Init();
//ADC2_Init();
//显示初始化
LCD12864_InitPort(); //
LCD12864_Init(); //
LCD12864_Pos(3,5);
LCD12684_Wdat(0x35);
// adcx1=0;
// adcx2=0;
//调用DMA
// MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC1_Buf,32);
MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC1_Buf,1);
MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
while(1)
{
if(DMA1->ISR&(1<<1)); //DMA1传输完
adcx1=ADC1_Buf&0xffff; // 将底16位分出来
adcx2=ADC1_Buf>>16; //分出高16位
//DMA1->IFCR|=1<<1;
MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
//显示
temp = (float)adcx2*(3.3/4096);
adcx2 = temp;
arry2[0]=adcx2+0X30;
arry2[1] = '.';
adcx2 = (temp - adcx2)*1000;
arry2[2]=adcx2/100+0X30;
arry2[3]=adcx2%100/10+0X30;
arry2[4]=adcx2%10+0X30;
arry2[5]=' '; */
//adcx1=Get_ADC1(ADC_CH11);
//显示
temp = (float)adcx1*(3.3/4096);
adcx1 = temp;
arry1[0]=adcx1+0X30;
arry1[1] = '.';
adcx1 = (temp - adcx1)*1000;
arry1[2]=adcx1/100+0X30;
arry1[3]=adcx1%100/10+0X30;
arry1[4]=adcx1%10+0X30;
arry1[5]=' ';
LCD_ShowString(1,0,arry1);
LCD_ShowString(4,4,arry2);
LCD_ShowString(3,0,"电子信息");
//LCD_ShowString(4,0,arry3);
//LCD_ShowString(4,4,arry4);
delay_ms(10);
}
}
//双ADC初始化
#include "stm32f10x.h"
#include "adc.h"
#include "dma.h"
//u32 ADC1_Buf;
//初始化ADC1
//这里我们仅以规则通道为例
//我们默认将开启通道10 11
void ADC1_Init(void)
{
//先初始化IO口
RCC->APB2ENR|=1<<4; //使能PORTA口时钟
GPIOC->CRL&=0XFFFF0000;//PA0 1 2 3 anolog输入//PC0 1 2 3 anolog输入
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->CFGR&=~(3<<14); //分频因子清零
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=2<<14;
ADC1->CR1&=0XF0FFFF; //工作模式清零
ADC1->CR1|=0<<16; //独立工作模式
//ADC1->CR1|=6<<16;//规则同步模式
ADC1->CR1&=~(1<<8); //非扫描模式
//ADC1->CR1|=(1<<8); //扫描模式
ADC1->CR2&=~(1<<1); //单次转换模式
//ADC1->CR2|=(1<<1); //连续转换模式
ADC1->CR2&=~(7<<17);
ADC1->CR2|=7<<17; //软件控制转换
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐
ADC1->SQR1&=~(0XF<<20);
ADC1->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1
//设置通道10 11的采样时间
// ADC1->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
// ADC1->SQR3|=11; //通道ch=11
ADC1->SMPR1&=0XFFFFFFC7;//通道11采样时间清空
ADC1->SMPR1|=7<<3; //通道11 239.5周期,提高采样时间可以提高精确度
ADC1->CR2|=1<<8;//开启ADC DMA转换
ADC1->CR2|=1<<0; //开启AD转换器
// ADC1->CR2|=1<<22; //启动规则转换通道
// while(!(ADC1->SR&1<<1));//等待转换结束
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待复位校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
// MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)&ADC1_Buf,1);//配置DMA1_CHx
MYDMA_Enable(DMA1_Channel1);//使能DMA1_CHx
// MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
// ADC1->CR2|=1<<0;//开启ADC1转换
}
//获得ADC值
//ch:通道值 10 11
/*u16 Get_ADC1(u8 ch)
{
//设置转换序列
ADC1->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
ADC1->SQR3|=ch;
ADC1->CR2|=1<<22; //启动规则转换通道
while(!(ADC1->SR&1<<1));//等待转换结束
return ADC1->DR; //返回adc值
}*/
//初始化ADC2
void ADC2_Init(void)
{
RCC->APB2ENR|=1<<10; //ADC2时钟使能
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;
//ADC2->CR1|=1<<11; //ADC2->CR1 11不能开,不然就不能转换
ADC2->CR1&=0XF0FFFF; //工作模式清零
ADC2->CR1|=0X0<<16; //独立工作模式
//ADC1->CR1|=6<<16; // 规则同步模式
//ADC2->CR1&=~(1<<8); //非扫描模式
ADC2->CR1|=(1<<8); //扫描模式
//ADC2->CR2&=~(1<<1); //单次转换模式
ADC2->CR2|=1<<1; //连续转换模式
// ADC2->CR2&=~(7<<17);
ADC2->CR2|=7<<17; //软件控制转换
ADC2->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC2->CR2&=~(1<<11); //右对齐
ADC2->SQR1&=~(0XF<<20);
ADC2->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1 V
ADC2->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
ADC2->SQR3|=10; // 通道 ch=10
//设置通道10的采样时间
ADC2->SMPR1&=0XFFFFFFF8;//通道10采样时间清空
ADC2->SMPR1|=7<<0; //通道10 239.5周期,提高采样时间可以提高精确度
ADC2->CR2|=1<<0; //开启AD转换器
// ADC2->CR2|=1<<22; //启动规则转换通道
// while(!(ADC2->SR&1<<1));//等待转换结束
ADC2->CR2|=1<<3; //使能复位校准
while(ADC2->CR2&1<<3); //等待复位校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC2->CR2|=1<<2; //开启AD校准
while(ADC2->CR2&1<<2); //等待校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
}
//获得ADC2值
//ch:通道值 10 11
/*u16 Get_ADC2(u8 ch)
{
//设置转换序列
ADC2->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
ADC2->SQR3|=ch;
ADC2->CR2|=1<<22; //启动规则转换通道
while(!(ADC2->SR&1<<1));//等待转换结束
return ADC2->DR; //返回adc值
} */
//DMA初始化
#include "dma.h"
#include "delay.h"
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)
{
RCC->AHBENR|=1<<0; //开启DMA1时钟
delay_ms(1); //等待DMA时钟稳定
DMA_CHx->CPAR=cpar; //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<<4; //从存储器读
DMA_CHx->CCR|=1<<5; //连续模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=1<<7; //存储器增量模式
DMA_CHx->CCR|=2<<8; //外设数据宽度为32位
DMA_CHx->CCR|=2<<10; //存储器数据宽度32位
DMA_CHx->CCR|=1<<12; //中等优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
DMA_CHx->CCR|=1<<0;
}
void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
//开启一次DMA传输
void MYDMA_dmago(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR&=~(1<<0); //关闭DMA传输
DMA_CHx->CNDTR=DMA1_MEM_LEN; //DMA1,传输数据量
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
//ADC和DMA的初始化参考了原子哥的程序,主函数中if(DMA1->ISR&(1<<1)); //DMA1传输完 可以进入,能分出高16位和低16位的值,但是分出的值从来都没变过,ADC1和ADC2我都接的3V的电压,采集回来的去是ADC1=0.556 ADC2=1.654,且一直都没变过。我头大了,搞了很久,麻烦各位前辈指点,感激不尽。
---------------------------------
连续转换模式我屏蔽掉了,使用单词转换模式,规则序列?我屏蔽了,还是不行。原子哥说没用到DMA,我不清楚,能帮我看一下吗,谢谢了。
---------------------------------回复【8楼】冯丑丑爱睡觉:
这是我前几天写的,可能不完善,希望对你有帮助
---------------------------------
谢谢,我研究一下。
u32 ADC_Buf[10];
int main(void)
{
u16 t;
u32 adcx;
u16 adcx1;
// u16 adcx2;
u8 arry1[6];
// u8 arry2[6];
float temp;
SystemInit();
delay_init(72); //延时初始化
ADC1_Init();
//ADC2_Init();
LCD12864_InitPort(); //端口初始化
LCD12864_Init(); //液晶初始化
LCD12864_Pos(3,5);
LCD12684_Wdat(0x35);
while(1)
{
/*****利用DMA由ADC外设向储存器传送数据********/
MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)ADC_Buf,10);
ADC1->CR2|=1<<8; //开启ADC的DMA转换
// ADC1->CR2|=1<<0; //开启AD转换
MYDMA_dmago(DMA1_Channel1); //开启一次DMA传输
/*****等待DMA传送完成*******/
while(!DMA1_Channel1->CNDTR)
{
if(DMA1->ISR&(1<<1)); //DMA1传输完
{
DMA1->IFCR|=1<<1;
break;
}
}
/***数据处理****/
for(t=0;t<10;t++)
{
adcx = adcx + ADC_Buf[t];
}
adcx = adcx /10; // 取10次平均值
adcx1=adcx&0xffff; //取低16位
//delay_ms(1);
temp = (float)adcx1*(3.3/4096);
adcx1 = temp;
arry1[0]=adcx1+0X30;
arry1[1] = '.';
adcx1 = (temp - adcx1)*1000;
arry1[2]=adcx1/100+0X30;
arry1[3]=adcx1%100/10+0X30;
arry1[4]=adcx1%10+0X30;
arry1[5]=' ';
LCD_ShowString(1,0,arry1);
LCD_ShowString(3,0,"电子信息工程学院");
delay_ms(100);
}
}
void ADC1_Init(void)
{
//先初始化IO口
RCC->APB2ENR|=1<<4; //使能PORTA口时钟
GPIOC->CRL&=0XFFFF0000;//PC0 1 2 3 anolog输入
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->CFGR&=~(3<<14); //分频因子清零
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=2<<14;
ADC1->CR1&=0XF0FFFF; //工作模式清零
ADC1->CR1|=0<<16; //独立工作模式
ADC1->CR1|=(1<<8); //扫描模式
ADC1->CR2|=(1<<1); //连续转换模式
ADC1->CR2&=~(7<<17);
ADC1->CR2|=7<<17; //软件控制转换
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐
ADC1->SQR1&=~(0XF<<20);
//设置通道10 11的采样时间
ADC1->SQR3&=0XFFFFFFE0; //规则序列1=通道0
ADC1->SQR3|=11;
ADC1->SMPR1&=0XFFFFFFC7;//通道11采样时间清空
ADC1->SMPR1|=7<<3; //通道11 239.5周期,提高采样时间可以提高精确度
ADC1->CR2|=1<<8;//开启ADC DMA转换
ADC1->CR2|=1<<0; //开启AD转换器
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待复位校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
}
//获得ADC值
u16 Get_ADC1(u8 ch)
{
//设置转换序列
ADC1->SQR3&=0XFFFFFFE0;//
ADC1->SQR3|=ch; //
ADC1->CR2|=1<<22; //启动规则转换通道
while(!(ADC1->SR&1<<1));//等待转换结束
return ADC1->DR; //返回adc值
}
回复【8楼】冯丑丑爱睡觉:
---------------------------------
我参考了你的程序,也是只用一个ADC,即ADC1,DMA有数据了,但是跳动变化直到0停止,不是想要的。还是没有调出来。急啊!!!
一周热门 更多>