经典STM32 ADC多通道转换

发布者:qiuxubiao最新更新时间:2018-05-13 来源: eefocus关键字:STM32  ADC  多通道转换 手机看文章 扫描二维码
随时随地手机看文章
  1. STM32 ADC多通道转换  

  2. 描述:用ADC连续采集11路模拟信号,并由DMA传输到内存。ADC配置为扫描并且连续转换模式,ADC的时钟配置为12MHZ。在每次转换结束后,由DMA循环将转换的数据传输到内存中。ADC可以连续采集N次求平均值。最后通过串口传输出最后转换的结果。  

  3. 程序如下:  

  4. #i nclude "stm32f10x.h" //这个头文件包括STM32F10x所有外围寄存器、位、内存映射的定义  

  5. #i nclude "eval.h" //头文件(包括串口、按键、LED的函数声明)  

  6. #i nclude "SysTickDelay.h"  

  7. #i nclude "UART_INTERFACE.h"  

  8. #i nclude   

  9.   

  10. #define N 50 //每通道采50次  

  11. #define M 12 //为12个通道  

  12.   

  13. vu16 AD_Value[N][M]; //用来存放ADC转换结果,也是DMA的目标地址  

  14. vu16 After_filter[M]; //用来存放求平均值之后的结果  

  15. int i;  

  16.   

  17.   

  18.   

  19. void GPIO_Configuration(void)  

  20. {  

  21. GPIO_InitTypeDef GPIO_InitStructure;  

  22.   

  23. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;  

  24. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //因为USART1管脚是以复用的形式接到GPIO口上的,所以使用复用推挽式输出  

  25. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  

  26. GPIO_Init(GPIOA, &GPIO_InitStructure);  

  27.   

  28.   

  29. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;  

  30. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;  

  31. GPIO_Init(GPIOA, &GPIO_InitStructure);  

  32.   

  33.   

  34.   

  35. //PA0/1/2 作为模拟通道输入引脚  

  36. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0| GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;  

  37. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚  

  38. GPIO_Init(GPIOA, &GPIO_InitStructure);  

  39.   

  40. //PB0/1 作为模拟通道输入引脚  

  41. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;  

  42. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚  

  43. GPIO_Init(GPIOB, &GPIO_InitStructure);  

  44.   

  45. //PC0/1/2/3/4/5 作为模拟通道输入引脚  

  46. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;  

  47. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚  

  48. GPIO_Init(GPIOC, &GPIO_InitStructure);  

  49. }  

  50.   

  51. }  

  52.   

  53.   

  54. void RCC_Configuration(void)  

  55. {  

  56. ErrorStatus HSEStartUpStatus;  

  57.   

  58. RCC_DeInit(); //RCC 系统复位  

  59. RCC_HSEConfig(RCC_HSE_ON); //开启HSE  

  60. HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等待HSE准备好  

  61. if(HSEStartUpStatus == SUCCESS)  

  62. {  

  63. FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //Enable Prefetch Buffer  

  64. FLASH_SetLatency(FLASH_Latency_2); //Set 2 Latency cycles  

  65. RCC_HCLKConfig(RCC_SYSCLK_Div1); //AHB clock = SYSCLK  

  66. RCC_PCLK2Config(RCC_HCLK_Div1); //APB2 clock = HCLK  

  67. RCC_PCLK1Config(RCC_HCLK_Div2); //APB1 clock = HCLK/2  

  68. RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_6); //PLLCLK = 12MHz * 6 = 72 MHz  

  69. RCC_PLLCmd(ENABLE); //Enable PLL  

  70. while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //Wait till PLL is ready  

  71. RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //Select PLL as system clock source  

  72. while(RCC_GetSYSCLKSource() != 0x08); //Wait till PLL is used as system clock source  

  73.   

  74. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB  

  75. | RCC_APB2Periph_GPIOC |RCC_APB2Periph_ADC1 | RCC_APB2Periph_AFIO |RCC_APB2Periph_USART1, ENABLE ); //使能ADC1通道时钟,各个管脚时钟  

  76.   

  77. RCC_ADCCLKConfig(RCC_PCLK2_Div6); //72M/6=12,ADC最大时间不能超过14M  

  78. RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输  

  79.   

  80. }  

  81. }  

  82.   

  83.   

  84. void ADC1_Configuration(void)  

  85. {  

  86. ADC_InitTypeDef ADC_InitStructure;  

  87.   

  88. ADC_DeInit(ADC1); //将外设 ADC1 的全部寄存器重设为缺省值  

  89.   

  90.   

  91. ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式  

  92. ADC_InitStructure.ADC_ScanConvMode =ENABLE; //模数转换工作在扫描模式  

  93. ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //模数转换工作在连续转换模式  

  94. ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //外部触发转换关闭  

  95. ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐  

  96. ADC_InitStructure.ADC_NbrOfChannel = M; //顺序进行规则转换的ADC通道的数目  

  97. ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器  

  98.   

  99.   

  100. //设置指定ADC的规则组通道,设置它们的转化顺序和采样时间  

  101. //ADC1,ADC通道x,规则采样顺序值为y,采样时间为239.5周期  

  102. ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 );  

  103. ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_239Cycles5 );  

  104. ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 );  

  105. ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_239Cycles5 );  

  106. ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 5, ADC_SampleTime_239Cycles5 );  

  107. ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 6, ADC_SampleTime_239Cycles5 );  

  108. ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 7, ADC_SampleTime_239Cycles5 );  

  109. ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 8, ADC_SampleTime_239Cycles5 );  

  110. ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 9, ADC_SampleTime_239Cycles5 );  

  111. ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 10, ADC_SampleTime_239Cycles5 );  

  112. ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 11, ADC_SampleTime_239Cycles5 );  

  113. ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 12, ADC_SampleTime_239Cycles5 );  

  114.   

  115. // 开启ADC的DMA支持(要实现DMA功能,还需独立配置DMA通道等参数)  

  116. ADC_DMACmd(ADC1, ENABLE);  

  117.   

  118.   

  119. ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1  

  120.   

  121. ADC_ResetCalibration(ADC1); //复位指定的ADC1的校准寄存器  

  122.   

  123. while(ADC_GetResetCalibrationStatus(ADC1)); //获取ADC1复位校准寄存器的状态,设置状态则等待  

  124.   

  125.   

  126. ADC_StartCalibration(ADC1); //开始指定ADC1的校准状态  

  127.   

  128. while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状态则等待  

  129.   

  130.   

  131. }  

  132.   

  133.   

  134. void DMA_Configuration(void)  

  135. {  

  136.   

  137. DMA_InitTypeDef DMA_InitStructure;  

  138. DMA_DeInit(DMA1_Channel1); //将DMA的通道1寄存器重设为缺省值  

  139. DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1->DR; //DMA外设ADC基地址  

  140. DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&AD_Value; //DMA内存基地址  

  141. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //内存作为数据传输的目的地  

  142. DMA_InitStructure.DMA_BufferSize = N*M; //DMA通道的DMA缓存的大小  

  143. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址寄存器不变  

  144. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址寄存器递增  

  145. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为16位  

  146. DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位  

  147. DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //工作在循环缓存模式  

  148. DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA通道 x拥有高优先级  

  149. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA通道x没有设置为内存到内存传输  

  150. DMA_Init(DMA1_Channel1, &DMA_InitStructure); //根据DMA_InitStruct中指定的参数初始化DMA的通道  

  151.   

  152. }  

  153.   

  154.   

  155. //配置所有外设  

  156. void Init_All_Periph(void)  

  157. {  

  158.   

  159. RCC_Configuration();  

  160.   

  161. GPIO_Configuration();  

  162.   

  163. ADC1_Configuration();  

  164.   

  165. DMA_Configuration();  

  166.   

  167. //USART1_Configuration();  

  168. USART_Configuration(9600);  

  169.   

  170.   

  171. }  

  172.   

  173.   

  174.   

  175. u16 GetVolt(u16 advalue)  

  176.   

  177. {  

  178.   

  179. return (u16)(advalue * 330 / 4096); //求的结果扩大了100倍,方便下面求出小数  

  180.   

  181. }  

  182.   

  183.   

  184.   

  185.   

  186. void filter(void)  

  187. {  

  188. int sum = 0;  

  189. u8 count;  

  190. for(i=0;i<12;i++)  

  191.   

  192. {  

  193.   

  194. for ( count=0;count

  195.   

  196. {  

  197.   

  198. sum += AD_Value[count][i];  

  199.   

  200. }  

  201.   

  202. After_filter[i]=sum/N;  

  203.   

  204. sum=0;  

  205. }  

  206.   

  207. }  

  208.   

  209.   

  210.   

  211.   

  212. int main(void)  

  213. {  

  214.   

  215. u16 value[M];  

  216.   

  217. init_All_Periph();  

  218. SysTick_Initaize();  

  219.   

  220.   

  221. ADC_SoftwareStartConvCmd(ADC1, ENABLE);  

  222. DMA_Cmd(DMA1_Channel1, ENABLE); //启动DMA通道  

  223. while(1)  

  224. {  

  225. while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//等待传输完成否则第一位数据容易丢失  

  226.   

  227. filter();  

  228. for(i=0;i<12;i++)  

  229. {  

  230. value[i]= GetVolt(After_filter[i]);  

  231.   

  232. printf("value[%d]:\t%d.%dv\n",i,value[i]/100,value[i]0) ;  

  233. delay_ms(100);  

  234. }  

  235. }  

  236.   

  237. }  

  238. 总结  

  239. 该程序中的两个宏定义,M和N,分别代表有多少个通道,每个通道转换多少次,可以修改其值。  

  240. 曾出现的问题:配置时钟时要知道外部晶振是多少,以便准确配置时钟。将转换值由二进制转换为十进制时,要先扩大100倍,方便显示小数。最后串口输出时在 printf语句之前加这句代码,防止输出的第一位数据丢失:while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);  


关键字:STM32  ADC  多通道转换 引用地址: 经典STM32 ADC多通道转换

上一篇:STM32之ADC_3(内部通道)
下一篇:stm32ADC非DMA模式的多通道采集与注入通道的采集

推荐阅读最新更新时间:2024-03-16 16:02

STM32F103VET6——ADC单通道中断读取实验讲解
编程要点 Define GPIO ADC_InitTypeDef ADC时钟 通道的转换顺序和采样时间 使能ADC转换完成中断 使能ADC转换开始 校准ADC 选取ADC转化触发源 配置中断优先级 编写中断服务函数 main函数 /*-----GPIO-----*/ static void ADC_GPIO_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; ADC_GPIO_APBxClock_FUN ( ADC_GPIO_CLK, ENABLE ); GPIO_InitStructure.GPIO_Pin = ADC_PIN; //必须为模拟输入 GPIO
[单片机]
STM32之JTAG、SWD模式
说在前面的话 最近YKY项目做了新的硬件设计,其中键盘接口采用矩阵式键盘(4*4),有两个接口使用了STM32 (PB3、PB4), 调试中发现,这两个接口对应的两行均不能正常扫描到按键值,查看数据手册才知道这是系统接口,需要进行设置才可以 作为普通IO口使用。如图是数据手册对这两个接口的介绍: 解决方法 寄存器版本: 1 //JTAG模式设置,用于设置JTAG的模式 2 //mode:jtag,swd模式设置;00,全使能;01,使能SWD;10,全关闭; 3 //#define JTAG_SWD_DISABLE 0X02 4 //#define SWD_ENABLE 0X01
[单片机]
ARM开发步步深入之掌握ADC和触摸屏
实验目的:通过串口显示输入的电压值及采集按下触摸屏的(x,y)坐标值借此掌握S3C2410的ADC和触摸屏的使用。 实验环境及说明:恒颐S3C2410开发板H2410。H24X0E扩展板上AIN0~AIN1输出悬空,通过外接可变电阻电路采样电压值;外接的触摸屏接口实现扩展触摸屏完成相应操作本实验基于夏普3.5英寸LQ035Q7DB02。 实 验思路:开发板上电启动后,自动将NandFlash开始的4K数据复制到SRAM中,然后跳转到0地址开始执行。关闭看门狗、初始化SDRAM及 NandFlash控制器、设置MPLL来改变FCLK、HCLK、PCLK的值,设置堆栈,复制4KB后的16KB数据到SDRAM,之后进入main 函数中进
[单片机]
STM32中断优先级和开关总中断开关
简介:STM32(Cortex-M3)中有两个优先级的概念——抢占式优先级和响应优先级,有人把响应优先级称作'亚优先级'或'副优先级',每个中断源都需要被指定这两种优先级。 具有高抢占式优先级的中断可以在具有低抢占式优先级的中断处理过程中被响应,即中断嵌套,或者说高抢占式优先级的中断可以嵌套低抢占式优先级的中断。 一,中断优先级: 当两个中断源的抢占式优先级相同时: 1.这两个中断将没有嵌套关系,当一个中断到来后,如果正在处理另一个中断,这个后到来的中断就要等到前一个中断处理完之后才能被处理。 2.如果这两个中断同时到达,则中断控制器根据他们的响应优先级高低来决定先处理哪一个;
[单片机]
STM32 定时器输入捕获注意点
使用的是stm32f103rct6. 最近在写一个单定时器捕获4路pwm波的程序,那就必须在中断函数中改变检测边沿,通过手动记录捕获值来手动计算pwm的频率和脉宽值。 网上也有很多相关程序,但是我在写的时候却碰到了一个对于初学者来说非常不起眼的问题,导致结果怎么都不对: 如图中所示,我正在用TIM4_CH1,因为看见其他的结构体都是自己定义一边再使用的,于是我将‘TIM_TypeDef’这个结构体也自己定义了一个对象‘TIM4_Type’然后就发现结果不对。 后来仔细分析后发现这个结构体已在stm32f10x.h中有了对象实例:TIM4。并且这个TIM4所在的地方正是那些寄存器所处的地方,而我自己定义的对象‘TIM4_Type’是
[单片机]
<font color='red'>STM32</font> 定时器输入捕获注意点
STM32串口中断的一些资料
在研究STM32串口接收发送中断的时候找到不少不错的资料,现在备份在这里。以供自己查阅,以及方便其他人。 TC ====TXE 顺便预告下最近会写个有关串口处理数据的帖子,从查询和中断方面以及数据处理的方式,从队列以及FIFO方面写起。 SECTION 1 SECTION 2 先说TC。即Transmission Complete。发送一个字节后才进入中断,这里称为 发送后中断 。和原来8051的TI方式一样,都是发送后才进中断,需要在发送函数中先发送一个字节触发中断。发送函数如下 /* 功能:中断方式发送字符串.采用判断TC的方式.即 判断 发送后中断 位. 输入:字符串的首地址 输出:无 */ void U
[单片机]
STM32 ADC采样配置
简介: 本次ADC采样主要采用stm32 103XB 芯片,用于对温度和漏电的采样。此次会进行最多16路的采样。ADC采集16路模拟信号,并由DMA传输到内存,之后从内存中提取数据进行计算。此次记录主要以配置为主,不同的情况下,采用的计算方式不同没有太大必要。系统时钟是72MHz。 正文程序及解释: #include stm32f10x.h #include #include ADC.h #define ADC_COUNT 320 //每通道采 320次 次数可修改 //0-15通道单次转换(0-4095) 定义范围(-32768~+32768)有符号16位 volatile int16_t ADC_V
[单片机]
用于时间交织ADC的高精度开环跟踪保持电路设计
0  引言   随着数字通信系统的发展,高速数字处理系统对模拟信号和数字信号之间的转换要求越来越高。目前高性能模数转换器(ADC)的两大主要发展方向是高速、中低精度ADC和低速、高精度ADC。前端T&H电路通常是ADC设计的一个关键,其动态精度的好坏直接影响着ADC性能的优劣。   1 开环T&H电路   在超高速ADC的设计中,一般多采用全并行Flash结构或者是时间交织结构。   而在时间交织结构中,其前端T&H电路则可以根据设计要求分别采用开环或者闭环结构。闭环结构速度较低,精度较高,而开环结构速度高,但本身精度较低。本设计中采用后者。   T&H电路的失真主要来源于非线性MOS开关电阻、开关寄生电容和开
[电源管理]
用于时间交织<font color='red'>ADC</font>的高精度开环跟踪保持电路设计
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved