#include <stm32f10x_lib.h>
#include "adc.h"
#include "delay.h"
//////////////////////////////////////////////////////////////////////////////////
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//Mini STM32开发板
//ADC 驱动代码
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//修改日期:2010/6/7
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) 正点原子 2009-2019
//All rights reserved
//////////////////////////////////////////////////////////////////////////////////
//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道0~3
void Adc_Init(void)
{
//先初始化IO口
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
GPIOA->CRL&=0XFFFF000F;//PA1 2 3 anolog输入
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
__nop();
__nop();
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=3<<14;
ADC1->CR1&=0XF0FFFF; //独立工作模式
ADC1->CR1|=1<<8; //扫描模式
ADC1->CR2|=1<<1; //连续转换模式
ADC1->CR2|=0x000E0000; //软件控制转换 由bit21控制
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐
ADC1->SQR1&=~(0XF<<20);
ADC1->SQR1&=2<<20; //3个转换在规则序列中
//设置通道1~3的采样时间
ADC1->SMPR2&=~(7<<3); //清除通道1的采样时间
ADC1->SMPR2|=7<<3; //设置通道1采样时间为293.5个周期
ADC1->SMPR2&=~(7<<6); //清除通道2的采样时间
ADC1->SMPR2|=7<<6; //设置通道2采样时间为293.5个周期
ADC1->SMPR2&=~(7<<9); //清除通道3的采样时间
ADC1->SMPR2|=7<<9; //设置通道3采样时间为293.5个周期
ADC1->SQR3&=~(0x1f<<0); //清除规则序列1通道
ADC1->SQR3|=1<<0; //设置规则序列1通道
ADC1->SQR3&=~(0x1f<<5); //清除规则序列2通道
ADC1->SQR3|=2<<5; //设置规则序列2通道
ADC1->SQR3&=~(0x1f<<10);//清除规则序列3通道
ADC1->SQR3|=3<<10; //设置规则序列3通道
ADC1->CR2|=1<<0; //开启AD转换器
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
ADC1->CR2|=1<<8; //开启ADC DMA转换
delay_ms(1);
}
//获得ADC值
//ch:通道值 1~3
u16 Get_Adc(u8 ch)
{
ADC1->CR2|=1<<22; //启动规则转换通道
while(!(ADC1->SR&1<<1));//等待转换结束
delay_ms(100);
return ADC1->DR; //返回adc值
}
#include "dma.h"
//////////////////////////////////////////////////////////////////////////////////
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//Mini STM32开发板
//DMA 驱动代码
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//修改日期:2010/6/7
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) 正点原子 2009-2019
//All rights reserved
//////////////////////////////////////////////////////////////////////////////////
//#define DMA1_CPAR_ADD &ADC1->DR
//u16 ADC_Buf[300];
//#define DMA1_CMAR_ADD &ADC_Buf
u16 DMA1_MEM_LEN;
//u16 ADC_Buf[300]; //保存DMA每次数据传送的长度
//DMA1的各通道配置
//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHx  MA通道CHx
//cpar:外设地址
//cmar:存储器地址
//cndtr:数据传输量
void MYDMA_Config(DMA_Channel_TypeDef*DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
// u32 DR_Base; //做缓冲用,不知道为什么.非要不可
RCC->AHBENR|=1<<0;//开启DMA1时钟
__nop(); //等待 DMA1 时钟稳定
__nop(); //经测试最少 2 个 nop
__nop();
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|=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<<13; //高等优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
}
//开启 DMA传输
void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
//开启一次DMA传输
//单次DMA转换之后,必须把DMA关闭,再次启动,才能实现第二次DMA传输!!!
void MYDMA_GOnes(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_CHx->CCR&=~(1<<0); //关闭DMA传输
DMA_CHx->CNDTR=DMA1_MEM_LEN ; //DMA1,传输数据量
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
//#define USART1_DR_Base 0x40013804
//#define ADC1_DR_Address 0x4001244C //0x40012400+0x4C
u16 ADC_Buf[300];
void Uart1_PutChar(u8 ch);
void Uart1_PutString(u8 *Buf, u8 Len);
int main(void)
{
u16 i;
u16 adc1,adc2,adc3;
u32 Sum1=0,Sum2=0,Sum3=0;
float temp1,temp2,temp3;
float temperature,temperature1,temperature2,pressure,pressure1,pressure2,humidity,humidity1,humidity2;
u8 table[8];
u8 *p;
p= table;
Stm32_Clock_Init(9);//88M
delay_init(72); //延时初始化
uart_init(72,9600);
LCD_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,"2013/10/24");
//显示提示信息
POINT_COLOR=BLUE;//设置字体为蓝色
LCD_ShowString(60,130,"ADC_CH1_VAL:");
LCD_ShowString(60,150,"ADC_CH1_VOL:0.000V");
LCD_ShowString(60,170,"ADC_CH1_TEM: 00.00C");
LCD_ShowString(60,190,"ADC_CH2_VAL:");
LCD_ShowString(60,210,"ADC_CH2_VOL:0.000V");
LCD_ShowString(60,230,"ADC_CH2_PRE: 0.000MPa");
LCD_ShowString(60,250,"ADC_CH3_VAL:");
LCD_ShowString(60,270,"ADC_CH3_VOL:0.000V");
LCD_ShowString(60,290,"ADC_CH3_HUM:0.00%RH");
Adc_Init();
MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR,(u32)&ADC_Buf,300);//配置DMA通道1
ADC1->CR2|=1<<8; //开启ADC DMA转换
MYDMA_Enable(DMA1_Channel1);//开启DMA通道1
ADC1->CR2|=1<<0; //开启AD转换
while (1)
{
if(DMA1->ISR&(1<<1)) //传输完成了
{
for(i=0;i<300;i+=3)
{
Sum1+=ADC_Buf[i+0];
Sum2+=ADC_Buf[i+1];
Sum3+=ADC_Buf[i+2];
}
adc1=Sum1/100;
adc2=Sum2/100;
adc3=Sum3/100; //取100次均值
LCD_ShowNum(156,130,adc1,4,16);//显示ADC的值
Sum1=0;
LCD_ShowNum(156,190,adc2,4,16);//显示ADC的值
Sum2=0;
LCD_ShowNum(156,250,adc3,4,16);//显示ADC的值
Sum3=0;
DMA1->IFCR|=1<<1;
LED0=!LED0;
delay_ms(1000);
MYDMA_Enable(DMA1_Channel1);
}
}
还请大家帮忙看看,指出错误之处,感激不尽~
|