码迷,mamicode.com
首页 > 其他好文 > 详细

1.7 ADC模数转换测电平(普通和DMA模式)

时间:2015-08-18 13:35:07      阅读:350      评论:0      收藏:0      [点我收藏+]

标签:

  嵌入式系统在微控制领域(温度,湿度,压力检测,四轴飞行器)中占据着重要地位,这些功能的实现是由微处理器cpu(如stm32)和传感器以及控制器共同完成的,而连接他们,使它们能够互相正常交流的正是本小节要讲诉的模块,ADC模数转换外设。下面从最简单的实验说起,逐渐深入了解这个神奇的外设。

    本次ADC模数转换设计实现并不复杂,步骤可简化为以下三步:

  1. 接收板上电位器的输入电压

    2. 经过A/D转换获得数字量,并传送给cpu

    3. 通过串口在PC机上输出。

    解析上面三个步骤,分析要求,就会发现ADC、GPIO、USART以及RCC模块就是本次实验所需要的用到的外设,因为除ADC模块,其它外设前面已经学习和实践了,那么理解和学习ADC模块,就可开始程序的设计实现了。

  根据stm32f系列微控制器手册ADC章节

   技术分享

 

   技术分享

 

    ADC转换的后数字量为12位(分辨率),在参考开发板用户手册和原理图,可知电位器的端口为PC0,输入电压范围0~3.3V,可知精度为3.3/(2^12)V.

    技术分享

    查询stm32f107的引脚定义分配,可知PC0对应ADC12_IN10,也就是说采集电位器电压用ADC1和ADC2都可以,但必须采用通道10。

目前来说,用库函数操作可以避免出现漏错,因此我还是推荐使用库函数配置寄存器,但是了解库函数的含义还是十分有必要的:

typedef struct
{
  u32  ADC_Mode;                            //明确ADC1和ADC2的工作方式,独立或其它组合

  FunctionalState  ADC_ScanConvMode;      //通道工作方式,单通道还是多通道(扫描)

  FunctionalState  ADC_ContinuousConvMode;  //工作在连续还是单次模式(ADC转换工作在连续模式

  u32 ADC_ExternalTrigConv;              //A/D转换启动规则

  u32  ADC_DataAlign;                      //判断转换数据的对齐方式

  u8  ADC_NbrOfChannel;                    //明确规则转换通道的具体数目1~16

}ADC_InitTypeDef

了解上述结构体代表含义,下面就可以初始化相关寄存器实现ADC外设的配置:

 

GPIO_InitTypeDef GPIO_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
ADC_DeInit(ADC1); 
    
//ADC模块外设时钟需在APB2时钟基础上设置,决定单个周期的时钟长度(因为ADC时钟不能大于14MHZ,注意)
RCC_ADCCLKConfig(RCC_PCLK2_Div4);
//使能ADC对应GPIO口,外设区域及复用功能时钟 RCC_APB2PeriphClockCmd(RCC_ADC1, ENABLE); //初始化ADC模块对应GPIO GPIO_InitStructure.GPIO_Pin = GPIO_ADC1_Pin; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIO_ADC1, &GPIO_InitStructure); ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC1和ADC2工作在独立模式 ADC_InitStructure.ADC_ScanConvMode = ENABLE; //工作在扫描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //转换工作在连续模式

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件触发,启动需调用ADC_Cmd程序 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐 ADC_InitStructure.ADC_NbrOfChannel = 1; //ADC通道数目为1 ADC_Init(ADC1, &ADC_InitStructure); //指定ADC转换的通道和转换周期 ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_55Cycles5);
#ifdef USE_ADC1_DMA_PER ADC_DMACmd(ADC1, ENABLE);
//ADC1 DMA请求使能 #endif ADC_Cmd(ADC1, ENABLE); //ADC1使能

ADC_ResetCalibration(ADC1); //重置ADC1校准寄存器 while (ADC_GetCalibrationStatus(ADC1)) //等待ADC1校准寄存器重置完成 { } ADC_StartCalibration(ADC1); //ADC1进入校准状态 while(ADC_GetCalibrationStatus(ADC1)) //等待ADC1校准完成 { }

ADC_SoftwareStartConvCmd(ADC1, ENABLE);
//ADC1软件触发方式启动

这里有一个重要知识点:ADC通道的规则组和注入组

技术分享

      在AD转换中,规则组定义的是ADC扫描通道的顺序,按照规则组配置时的采样顺序从小到大依次扫描ADC通道,而注入组的优先级高于规则组,当注入组转换触发时就打断规则组的扫描而执行注入组的通道扫描,具体流程类似于中断中的抢占。本次ADC的转换仅仅使用到一个端口,这些不用考虑,但是在多通道AD/DA采集时,规则组和注入组要根据实际情况进行配置。

注意:配置通道的规则组和注入组是一定要在使能ADC转换之前的。

完成了初始化后,剩下的就简单了,只要获得ADC处理后的数字量,在转换成整形变量,就可以通过串口发送接收了,如下:

//直接获得当前ADC转换后的值,转换并输出,CPU参与传送
ADValue = ADC_GetConversionValue(ADC1);
Precent = (ADValue*100/0x1000);
Voltage = Precent*33;
printf("\r\n\n ADCConvertedValue is 0x%x, Percent is %d%%, voltage is %d.%d%dV",
      ADValue,Precent,Voltage/1000,(Voltage%1000)/100,(Voltage%100)/10);
printf("\r\n ADC output");
            
ARM_DELAY(2000000);

注意:使用了printf函数作为输入输出时,包含头文件#include ”stdio.h” Target下要选择use MicroLib,否则是不会有输出的(串口章节已经说明,重要)

如此便实现了电位记电压的采集和输出,不过这并不是结束,因为今天我们还要学习另一个同样用途广泛的外设-DMA模块。

    首先我们要知道DMA是干什么的?DMA模块的主要作用是将内存或者外设中的数据自由移动,而不需要cpu的参与,同时通过存储指针的自偏移,实现大量数据的顺序存储(这一点在通讯领域具有重要意义)。和上面一样,学习DMA,肯定首先查询手册了:

   技术分享

   从这上面我们可以得出,ADC1对应的传输通道为通道1,因为ADC是工作模式,在了解下面的结构体后:

typedef struct
{
  u32 DMA_PeripheralBaseAddr;     //定义的外设基地址
  u32 DMA_MemoryBaseAddr;         //定义的内存基地址
  u32 DMA_DIR;                    //外设作为数据传输的来源还是目的地
  u32 DMA_BufferSize;             //DMA通道的 DMA缓存的大小,单位为数据单位
  u32 DMA_PeripheralInc;          //外设地址寄存器递增或不变
  u32 DMA_MemoryInc;              //内存地址寄存器递增或不变
  u32 DMA_PeripheralDataSize;     //外设数据宽度
  u32 DMA_MemoryDataSize;         //内存数据宽度
  u32 DMA_Mode;                   //DMA缓存工作方式
  u32 DMA_Priority;               //DMA工作优先级
  u32 DMA_M2M;                    //DMA工作是内存到内存,还是外设到内存
} DMA_InitTypeDef;

我们可以得出以下结论:

定义uint16_t  ADCConvertedValue; //接收内存地址

1.外设基地址为ADC1_DR_Address或者(uint32_t)&(ADC1->DR)

2.数据来源于ADC外设,传送地址为内存

3.工作在循环模式,且都不自增

按照上面的结构体依此配置DMA_InitStructrue的各项参数,初始化如下:

DMA_InitTypeDef DMA_InitStructure;
      
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); 
DMA_DeInit(DMA1_Channel1);                                                  //复位ADC1对应DMA通道DMA1_Channel1
       
DMA_InitStructure.DMA_PeripheralBaseAddr =(uint32_t)&(ADC1->DR);            //ADC1规则组转换值寄存器地址作为基地址
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADCConvertedValue;        //数据传输至内存的基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;                          //外设作为数据来源地
DMA_InitStructure.DMA_BufferSize = 1;                                       //可增加地址的的长度
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;            //外设地址不允许自增
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;                    //内存地址不允许自增
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //外设数据为半字16bit
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;         //内存数据为半字16bit
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(DMA1_Channel1, &DMA_InitStructure);                                  
       
DMA_Cmd(DMA1_Channel1, ENABLE);                                             /*DMA使能*/

同时在main函数中添加:

//将DMA从ADC处传送的数字量经过处理,转换并输出,DMA控制器用于传送
ADCConvertedValueLocal = ADCConvertedValue;                      
Precent = (ADCConvertedValueLocal*100/0x1000);
Voltage = Precent*33;
printf("\r\n\n ADCConvertedValue is 0x%x, Percent is %d%%, voltage is %d.%d%dV",
       ADCConvertedValueLocal,Precent,Voltage/1000,(Voltage%1000)/100,(Voltage%100)/10);
printf("\r\n ADC DMA output");

   如此便实现了通过ADC通过DMA的传输。

   代码下载:http://files.cnblogs.com/files/zc110747/7.ADC-DMA.7z

以下来自外部资料及个人总结,希望对理解DMA模块有用处:

   1.DMA传输将数据从一个地址空间复制到另外一个地址空间,这部分是由DMA控制器实现的,不需要依靠CPU的大量的数据采集传送,节省cpu资源。

   2.DMA工作包含四个过程

     DMA请求-〉DMA响应-〉DMA传输-〉DMA结束

   3.DMA传送方式有以下三种

     (1)停止CPU访内存;

     当外围器件有一批数据需要传送时,DMA给CPU发送停止信号,CPU停止访问内存,释放相关总线控制权,DMA获得总线控制权后开始传递数据,完成后将总线控制权交给CPU。一次DMA传送结束。

    优点:控制简单,用于速率很高的组传送

    缺点:内存的效能没有发挥,一部分时间内存处于空闲状态。这是因为DMA传送阶段有很多时间是在读取外设的数据,总线一段时间肯定是空闲的,而这部分时间足够CPU进行内存的访问。

    (2)周期挪用;(ADC转换采用的正是这种方式)

       当I/O设备没有DMA请求时,CPU按程序要求访问内存;一旦I/O设备有DMA请求,则由I/O设备挪用一个或几个内存周期。

    (3)DMA与CPU交替访内存.

1.7 ADC模数转换测电平(普通和DMA模式)

标签:

原文地址:http://www.cnblogs.com/zc110747/p/4738645.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!