原创 [原创]STM32学习笔记之ADC

2008-3-28 10:49 6575 8 19 分类: MCU/ 嵌入式

程序功能是把ADC1模块里通道14的输入电压转换后通过USART2发送到PC,在PC机上用串口调试助手观察接收数据:


9aa2875f-e14a-406c-a8d5-453e99a52d81.jpg


STM32是12位ADC,测量结果基本还可以!程序用了DMA来传输ADC转换值,调高了读取速度。串口部分用是上一篇串口调试笔记里的代码。


/************************************************************************
     Copyright (c) 2008 wormchen            
     All rights reserved            
文 件 名: main.c             
说    明: 将ADC转换值通过串口发送到PC端         
主要硬件: EMSTM32V1+miniSTMV100(外部8MRC)        
编译环境: MDK3.10             
当前版本: 1.0              
作    者: 陈崇              
完成日期: 2008年3月24日9:08:41          
取代版本: 1.0              
原作  者: 陈崇             
完成日期: 2008年3月24日9:08:46         
************************************************************************/
#include <stm32f10x_lib.h>
#include <stdio.h>


#define ADC1_DR_Address    ((u32)0x4001244C)


#ifdef __GNUC__
  /* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
     set to 'Yes') calls __io_putchar() */
  #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
  #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif /* __GNUC__ */


vu16 ADC_ConvertedValue;


void RCC_Config(void);
void GPIO_Config(void);
void USART_Config(void);
void DMA_Config(void);
void ADC_Config(void);
void Put_String(u8 *p);
void Delay(vu32 nCount);
int main(void)
{
 RCC_Config();
 GPIO_Config();
 USART_Config();
 DMA_Config();
 ADC_Config();


 
 while(1)
 { 
 
  Delay(0x8FFFF);  
   printf("ADC = %X Volt = %d mv\r\n", ADC_ConvertedValue, ADC_ConvertedValue*3300/4096);
  
 }
}
/*************************************************
函数: void RCC_Config(void)
功能: 配置系统时钟
参数: 无
返回: 无
**************************************************/
void RCC_Config(void)
{
 ErrorStatus HSEStartUpStatus;//定义外部高速晶体启动状态枚举变量
 RCC_DeInit();//复位RCC外部设备寄存器到默认值
 RCC_HSEConfig(RCC_HSE_ON); //打开外部高速晶振
 HSEStartUpStatus = RCC_WaitForHSEStartUp();//等待外部高速时钟准备好
 if(HSEStartUpStatus == SUCCESS)//外部高速时钟已经准别好
    {


   RCC_HCLKConfig(RCC_SYSCLK_Div1);//配置AHB(HCLK)时钟=SYSCLK
   RCC_PCLK2Config(RCC_HCLK_Div1); //配置APB2(PCLK2)钟=AHB时钟
   RCC_PCLK1Config(RCC_HCLK_Div2);//配置APB1(PCLK1)钟=AHB 1/2时钟
      RCC_ADCCLKConfig(RCC_PCLK2_Div4);//配置ADC时钟=PCLK2 1/4
     
   RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
   //配置PLL时钟 == 外部高速晶体时钟*9
   RCC_ADCCLKConfig(RCC_PCLK2_Div4);//配置ADC时钟= PCLK2/4


      RCC_PLLCmd(ENABLE);//使能PLL时钟
   while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET) //等待PLL时钟就绪
       {
       }
      RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);//配置系统时钟 = PLL时钟
  
      while(RCC_GetSYSCLKSource() != 0x08) //检查PLL时钟是否作为系统时钟
       {
       }
  }
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA, ENABLE);//使能DMA时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);
  //使能ADC1,GPIOC时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD | RCC_APB2Periph_AFIO, ENABLE);
  //打开GPIOD和AFIO时钟


  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);//使能串口2时钟
}
/*************************************************
函数: void GPIO_Config(void)
功能: GPIO配置
参数: 无
返回: 无
**************************************************/
void GPIO_Config(void)
{
 //设置RTS(PD.04),Tx(PD.05)为推拉输出模式
 GPIO_InitTypeDef GPIO_InitStructure; //定义GPIO初始化结构体
 GPIO_PinRemapConfig(GPIO_Remap_USART2, ENABLE);//使能GPIO端口映射USART2
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;//选择PIN4 PIN5
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //引脚频率50M
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//引脚设置推拉输出
 GPIO_Init(GPIOD, &GPIO_InitStructure);//初始化GPIOD
 //配置CTS (PD.03),USART2 Rx (PD.06)为浮点输入模式
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_6;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
 GPIO_Init(GPIOD, &GPIO_InitStructure);
 //配置PC4为模拟输入
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
 GPIO_Init(GPIOC, &GPIO_InitStructure);


}
/*************************************************
函数: void DMA_Config(void)
功能: DMA配置
参数: 无
返回: 无
**************************************************/
void DMA_Config(void)
{
 DMA_InitTypeDef DMA_InitStructure;//定义DMA初始化结构体
 DMA_DeInit(DMA_Channel1);//复位DMA通道1
 DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; //定义 DMA通道外设基地址=ADC1_DR_Address
 DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; //定义DMA通道存储器地址
 DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;//指定外设为源地址
 DMA_InitStructure.DMA_BufferSize = 1;//定义DMA缓冲区大小1
 DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//当前外设寄存器地址不变
 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;//当前存储器地址不变
 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//定义外设数据宽度16位
 DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //定义存储器数据宽度16位
 DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//DMA通道操作模式位环形缓冲模式
 DMA_InitStructure.DMA_Priority = DMA_Priority_High;//DMA通道优先级高
 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//禁止DMA通道存储器到存储器传输
 DMA_Init(DMA_Channel1, &DMA_InitStructure);//初始化DMA通道1
 DMA_Cmd(DMA_Channel1, ENABLE); //使能DMA通道1
}
/*************************************************
函数: void ADC_Config(void)
功能: ADC配置
参数: 无
返回: 无
**************************************************/
void ADC_Config(void)
{
  ADC_InitTypeDef ADC_InitStructure;//定义ADC初始化结构体变量
  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC1和ADC2工作在独立模式
  ADC_InitStructure.ADC_ScanConvMode = ENABLE; //使能扫描
  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//ADC转换工作在连续模式
  ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//有软件控制转换
  ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//转换数据右对齐
  ADC_InitStructure.ADC_NbrOfChannel = 1;//转换通道为通道1
  ADC_Init(ADC1, &ADC_InitStructure); //初始化ADC
  ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 1, ADC_SampleTime_28Cycles5);
  //ADC1选择信道14,音序器等级1,采样时间239.5个周期
  ADC_DMACmd(ADC1, ENABLE);//使能ADC1模块DMA
  ADC_Cmd(ADC1, ENABLE);//使能ADC1
  ADC_ResetCalibration(ADC1); //重置ADC1校准寄存器
  while(ADC_GetResetCalibrationStatus(ADC1));//等待ADC1校准重置完成
  ADC_StartCalibration(ADC1);//开始ADC1校准
  while(ADC_GetCalibrationStatus(ADC1));//等待ADC1校准完成
  ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能ADC1软件开始转换
}
/*************************************************
函数: void USART_Config(void)
功能: USART配置
参数: 无
返回: 无
**************************************************/
void USART_Config(void)
{
 USART_InitTypeDef USART_InitStructure; //定义串口初始化结构体
 USART_InitStructure.USART_BaudRate = 115200;//波特率9600
 USART_InitStructure.USART_WordLength = USART_WordLength_8b;//8位数据
 USART_InitStructure.USART_StopBits = USART_StopBits_1;//1个停止位
 USART_InitStructure.USART_Parity = USART_Parity_No ; //无校验位
 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
 //禁用RTSCTS硬件流控制
 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//使能发送接收
 USART_InitStructure.USART_Clock = USART_Clock_Disable; //串口时钟禁止
 USART_InitStructure.USART_CPOL = USART_CPOL_Low; //时钟下降沿有效
 USART_InitStructure.USART_CPHA = USART_CPHA_2Edge;//数据在第二个时钟沿捕捉
 USART_InitStructure.USART_LastBit = USART_LastBit_Disable;
 //最后数据位的时钟脉冲不输出到SCLK引脚
 USART_Init(USART2, &USART_InitStructure);//初始化串口2
 USART_Cmd(USART2, ENABLE);//串口2使能
}
/*************************************************
函数: void Put_String(void)
功能: 向串口输出字符串
参数: 无
返回: 无
**************************************************/
void Put_String(u8 *p)
{
 while(*p)
 {
  USART_SendData(USART2, *p++);
  while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET)
  {
   
  }
 }
}
/*****************************************************
函数: void Delay(vu32 nCount)
功能: 延时指定时间
参数: vu32 nCount 延时指定时间
返回: 无
******************************************************/
void Delay(vu32 nCount)
{
  for(; nCount != 0; nCount--);
}
/*****************************************************
函数:PUTCHAR_PROTOTYPE
功能: 重定向C库printf函数
参数: 无
返回: 无
*****************************************************/
PUTCHAR_PROTOTYPE
{
 USART_SendData(USART2, (u8) ch);//发送一字节数据
 while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET)
 {
 }//等待发送完成
 return ch;
}

文章评论11条评论)

登录后参与讨论

用户1079511 2009-8-6 08:14

完全可以不用固件库里的函数,自己写串口输出函数,就想用8位单片机一样!

用户213184 2009-8-5 12:59

我们应该可以在主函数里头写一个printf将串口里面的信息都打印在终端上吧?像设一个变量BufferSize,Buffer[]然后 for(i=0;i

用户1079511 2009-8-3 16:06

void Put_String(u8 *p) 是我自己写的输出字符串的函数,你可以不用,也可以用stdio.h里的printf函数来输出字符串。

用户213184 2009-8-3 11:40

你好,你文中的void Put_String(u8 *p)这个函数我不太理解,不知道这个是做什么用的。也没发现在主程序中有调用该函数。而且在void Put_String(u8 *p)下面不是已经有了一个PUTCHAR_PROTOTYPE函数了吗? 我在USART的一个例程里面,发现了它只有PUTCHAR_PROTOTYPE,它同样能在调试助手里头显示printf函数里的一些内容。所以想问一下您写这个函数有什么深意,望指教!谢谢!

用户461316 2008-8-25 08:30

路过这里,踩下留下脚印

用户1079511 2008-6-27 22:02

to:hkap 不使用DMA读采样值,就不要对DMA设置就好了,使用固件库的相应函数来读取。ADC模块可以按照事先约定好的顺序来读取相应通道的AD值,具体怎么用你可以参看硬件手册,我以前在手册上看到过相应介绍不过没有亲手试过,可以连续循环采样或只采样一次后停止。 最近应为工作的问题STM32的学修计划暂时搁置了,所以不能具体给你提供帮助抱歉!

用户1088106 2008-6-25 14:30

不小心发重了

用户1088106 2008-6-25 14:22

请问: 不通过DMA读采样值,采用扫描多通道采样法如何实现呢? 例: 要采样8个通道ADC值,做成一个函数,调用一次得到8个通道的值,平时不调用时ADC不工作,弄了半天也没成功,能否指点下呢?感激不尽!

用户1359586 2008-4-10 15:05

谢谢你的建议,最后问题找到了,是由于option的设置错误。很傻很弱智的错误。

用户1079511 2008-4-9 19:46

while(ADC_GetResetCalibrationStatus(ADC1)); 这句话的意思是等待ADC1复位完成,你的程序停在这证明一直在复位(复位没有完成)。你的ADC1时钟配置正确吗,ADC1是否已经使能,还有有GPIO配置正确吗?再仔细检查一下,看看有没有缺少步骤,祝你早日解决问题!
相关推荐阅读
用户1079511 2009-07-10 16:08
Mega16读取SD卡内的FAT16文件
最近一直在学习FAT文件系统,在网络上找了一些资料加上参考别人写的程序,现在已经可以读取SD/MMC卡指定的文件,有些东西自己也不是很懂,回头还得慢慢研究。程序比较简单,就是读取SD卡的基本信息和初始...
用户1079511 2009-04-14 08:38
[原创]AVR M16热敏电阻测温LCM12864显示
https://static.assets-stash.eet-china.com/album/old-resources/2009/4/14/d085d878-6d8f-4245-acdb-cf44...
用户1079511 2009-04-08 11:34
Mega16 Bootloader+PC端上位机
AVR的Bootloader功能很方便产品的在线升级,以前参考网上的资料写过下位机的程序,标准的XMODEM协议,用超级终端来升级程序。最近在学习C#,所以用C#写个简单的上位机软件,配合以前的boo...
用户1079511 2009-01-22 16:15
Mega16热敏电阻测温上位机曲线显示
最近在一直学习用C#做上位机,把以前的做的m16热敏电阻测温的程序翻出来,稍加修改通过串口发给PC;PC端用c#2008写程序,接收数据后通过ZedGraph绘图控件实时显示出曲线图。ZedGraph...
用户1079511 2009-01-09 09:25
[原创]c#的Hex转Bin小程序
最近在搞AVR的bootloader,程序已经写好了,参考网上的代码用的是XMODEM协议用XP自带的超级终端。问题是xmodem支持bin格式的文件,一般用编译器生成的是hex文件不用直接使用,需要...
用户1079511 2008-12-31 10:42
[学习]用595驱动LED实现灰度调节
<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />前一段时间在ouravr论坛里闲逛的...
我要评论
11
8
关闭 站长推荐上一条 /2 下一条