最近需要用到两个ADC对电压电流进行同步采样,看了一下STM32的ADC介绍,发现STM32最多有3个独立ADC,有在双AD模式下可以进行同步测量,正好满足我的要求。参考官方给的例子在结合自己的需要写了一下配置程序。

程序1 根据官方例子写的:

#include"adc.h"

__IO uint32_t ADC_DualConvertedValueTab[ADC_BufferLength];

void ADC1_2_Init(void)

{

ADC_InitTypeDef ADC_InitStructure;

DMA_InitTypeDef DMA_InitStructure;

GPIO_InitTypeDef GPIO_InitStructure;

RCC_ADCCLKConfig(RCC_PCLK2_Div4);

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_ADC2 | RCC_APB2Periph_GPIOA, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_4;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

GPIO_Init(GPIOA, &GPIO_InitStructure);

DMA_DeInit(DMA1_Channel1);                                                   //DMA通道1恢复到默认状态

DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)ADC1_DR_Address;        //设置数据来源地址

DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_DualConvertedValueTab;  //设置copy目标地址

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;                           //设置copy方向

DMA_InitStructure.DMA_BufferSize = 16;                                       //设置数组大小

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;             //数据来源不变

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;                      //数据保存的数组地址自增

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;      //DMA操作字长 32位

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;              //DMA操作字长 32位

DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;                                //普通的DMA操作模式:当DMA操作转移够 数组大小                                                                                 //所限定的数据总量后,则不再进行DMA操作

DMA_InitStructure.DMA_Priority = DMA_Priority_High;                          //DMA优先级

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;                                 //关闭mTM

DMA_Init(DMA1_Channel1, &DMA_InitStructure);                                 //使能dma

/* Enable DMA1 Channel1 */

DMA_Cmd(DMA1_Channel1, ENABLE);

/* ADC1 configuration ------------------------------------------------------*/

ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;                              //ADC1同步规则组模式

ADC_InitStructure.ADC_ScanConvMode = ENABLE;                                  //启动扫描

ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;                            //连续转换

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;           //无外部触发

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                        //结果右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1;                                       //转换通道数2

ADC_Init(ADC1, &ADC_InitStructure);                                           //配置ADC1

/* ADC1 regular channels configuration */

ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_239Cycles5); //通道配置

/* Enable ADC1 DMA */

ADC_DMACmd(ADC1, ENABLE);

/* ADC2 configuration ------------------------------------------------------*/

ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;                              //ADC2同步规则组模式

ADC_InitStructure.ADC_ScanConvMode = ENABLE;                                  //启动扫描

ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;                            //连续转换

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;           //无外部触发

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                        //结果右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1;                                       //转换通道数

ADC_Init(ADC2, &ADC_InitStructure);                                           //配置ADC2

/* ADC2 regular channels configuration */

ADC_RegularChannelConfig(ADC2, ADC_Channel_4, 1, ADC_SampleTime_239Cycles5); //通道配置

/* Enable ADC2 external trigger conversion */

ADC_ExternalTrigConvCmd(ADC2, ENABLE);                                        //使能ADC2的外部触发模式

/* Enable ADC1 */

ADC_Cmd(ADC1, ENABLE);                                                        //使能ADC1

/* Enable Vrefint channel17 */

// ADC_TempSensorVrefintCmd(ENABLE);                                             // 使能温度传感器内部参考电压通道

/* Enable ADC1 reset calibration register */

ADC_ResetCalibration(ADC1);                                                   //ADC1的校准

/* Check the end of ADC1 reset calibration register */

while(ADC_GetResetCalibrationStatus(ADC1));

/* Start ADC1 calibration */

ADC_StartCalibration(ADC1);

/* Check the end of ADC1 calibration */

while(ADC_GetCalibrationStatus(ADC1));

/* Enable ADC2 */

ADC_Cmd(ADC2, ENABLE);                                                      //使能ADC2

/* Enable ADC2 reset calibration register */

ADC_ResetCalibration(ADC2);                                                 //ADC2的校准

/* Check the end of ADC2 reset calibration register */

while(ADC_GetResetCalibrationStatus(ADC2));

/* Start ADC2 calibration */

ADC_StartCalibration(ADC2);

/* Check the end of ADC2 calibration */

while(ADC_GetCalibrationStatus(ADC2));

/* Start ADC1 Software Conversion */                                        //软件启动ADC1,开始转换

ADC_SoftwareStartConvCmd(ADC1, ENABLE);

/* Test on DMA1 channel1 transfer complete flag */                          //等待一组DMA存储转换完成

while(!DMA_GetFlagStatus(DMA1_FLAG_TC1));

/* Clear DMA1 channel1 transfer complete flag */

DMA_ClearFlag(DMA1_FLAG_TC1);

DMA_Done = 1;

}

两个ADC采出来的数据存在32位数组ADC_DualConvertedValueTab[ADC_BufferLength]中,其中ADC1对应低16位,ADC2对应高16位。可通过以下程序读出:

uint16_t adc1_val[1000];

uint16_t adc2_val[1000];

for(i=0;i<1000;i++)

{

adc1_val[i] = ADC_DualConvertedValueTab[i];

adc2_val[i] = ADC_DualConvertedValueTab[i]>>16;

}

最后再附上自己写的用TIM3触发采样的程序,可以自己调用来设置采样率。

程序2:

void ADC1_2_Config(u16 SampleFreq,u16 DMA_BufferSizer)

{

ADC_InitTypeDef ADC_InitStructure;

DMA_InitTypeDef DMA_InitStructure;

GPIO_InitTypeDef GPIO_InitStructure;

TIM_TimeBaseInitTypeDef   TIM_TimeBaseStructure;

NVIC_InitTypeDef NVIC_InitStructure;

RCC_ADCCLKConfig(RCC_PCLK2_Div2);

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_ADC2, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_4;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

GPIO_Init(GPIOA, &GPIO_InitStructure);

//定时器3控制采样率

TIM_TimeBaseStructure.TIM_Prescaler= 0;

TIM_TimeBaseStructure.TIM_CounterMode= TIM_CounterMode_Up;

TIM_TimeBaseStructure.TIM_Period= 72000000 / SampleFreq - 1;

TIM_TimeBaseStructure.TIM_ClockDivision= 0;

TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;

TIM_TimeBaseInit( TIM3, &TIM_TimeBaseStructure);

TIM_SelectOutputTrigger( TIM3, TIM_TRGOSource_Update );

TIM_ARRPreloadConfig(TIM3,ENABLE);

TIM_Cmd( TIM3, ENABLE );

DMA_DeInit(DMA1_Channel1);                                                   //DMA通道1恢复到默认状态

DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)ADC1_DR_Address;        //设置数据来源地址

DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_DualConvertedValueTab;  //设置copy目标地址

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;                           //设置copy方向

DMA_InitStructure.DMA_BufferSize = ADC_BufferLength;                                       //设置数组大小

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;             //数据来源不变

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;                      //数据保存的数组地址自增

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;      //DMA操作字长 32位

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;              //DMA操作字长 32位

DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;                                //普通的DMA操作模式:当DMA操作转移够 数组大小                                                                                 //所限定的数据总量后,则不再进行DMA操作

DMA_InitStructure.DMA_Priority = DMA_Priority_High;                          //DMA优先级

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;                                 //关闭mTM

DMA_Init(DMA1_Channel1, &DMA_InitStructure);                                 //使能dma

/* Enable DMA1 Channel1 */

DMA_Cmd(DMA1_Channel1, ENABLE);

/* DMA Transfer Complete Interrupt enable */

NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;   //通道

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //占先级

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;        //响应级

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;           //启动

NVIC_Init(&NVIC_InitStructure);                           //初始化

DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE);

/* ADC1 configuration ------------------------------------------------------*/

ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;                              //ADC1同步规则组模式

ADC_InitStructure.ADC_ScanConvMode = ENABLE;                                  //启动扫描

ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;                            //连续转换

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO;           //无外部触发

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                        //结果右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1;                                       //转换通道数2

ADC_Init(ADC1, &ADC_InitStructure);                                           //配置ADC1

/* ADC1 regular channels configuration */

ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_28Cycles5); //通道配置

/* Enable ADC1 DMA */

ADC_DMACmd(ADC1, ENABLE);

/* ADC2 configuration ------------------------------------------------------*/

ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;                              //ADC2同步规则组模式

ADC_InitStructure.ADC_ScanConvMode = ENABLE;                                  //启动扫描

ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;                            //连续转换

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;           //无外部触发

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                        //结果右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1;                                       //转换通道数

ADC_Init(ADC2, &ADC_InitStructure);                                           //配置ADC2

/* ADC2 regular channels configuration */

ADC_RegularChannelConfig(ADC2, ADC_Channel_4, 1, ADC_SampleTime_28Cycles5); //通道配置

/* Enable ADC2 external trigger conversion */

ADC_ExternalTrigConvCmd(ADC2, ENABLE);                                        //使能ADC2的外部触发模式

/* Enable ADC1 */

ADC_Cmd(ADC1, ENABLE);                                                        //使能ADC1

/* Enable Vrefint channel17 */

// ADC_TempSensorVrefintCmd(ENABLE);                                             // 使能温度传感器内部参考电压通道

/* Enable ADC1 reset calibration register */

ADC_ResetCalibration(ADC1);                                                   //ADC1的校准

/* Check the end of ADC1 reset calibration register */

while(ADC_GetResetCalibrationStatus(ADC1));

/* Start ADC1 calibration */

ADC_StartCalibration(ADC1);

/* Check the end of ADC1 calibration */

while(ADC_GetCalibrationStatus(ADC1));

/* Enable ADC2 */

ADC_Cmd(ADC2, ENABLE);                                                      //使能ADC2

/* Enable ADC2 reset calibration register */

ADC_ResetCalibration(ADC2);                                                 //ADC2的校准

/* Check the end of ADC2 reset calibration register */

while(ADC_GetResetCalibrationStatus(ADC2));

/* Start ADC2 calibration */

ADC_StartCalibration(ADC2);

/* Check the end of ADC2 calibration */

while(ADC_GetCalibrationStatus(ADC2));

/* Start ADC1 Software Conversion */                                        //软件启动ADC1,开始转换

ADC_SoftwareStartConvCmd(ADC1, ENABLE);

// while(!DMA_GetFlagStatus(DMA1_FLAG_TC1));

// /* Clear DMA1 channel1 transfer complete flag */

// DMA_ClearFlag(DMA1_FLAG_TC1);

// DMA_Done = 1;

}

void DMA1_Channel1_IRQHandler(void)

{

if(DMA_GetITStatus(DMA1_IT_TC1) != RESET)

{

DMA_ClearITPendingBit(DMA1_IT_TC1);

DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,DISABLE);

ADC_Cmd(ADC1,DISABLE);

ADC_Cmd(ADC2,DISABLE);

DMA_Done =1;

}

}

调用方式:ADC1_2_Config(5120,512);

可以用作FFT分析前的采样。

STM32 双ADC同步规则采样的更多相关文章

  1. STM32 ADC多通道规则采样和注入采样

    layout: post tags: [STM32] comments: true 文章目录 layout: post tags: [STM32] comments: true 什么是ADC? STM ...

  2. STM32cube库配置双ADC的同步规则采样

    http://www.stmcu.org/module/forum/forum.php?mod=viewthread&tid=605203&extra=page%3D&page ...

  3. 【stm32】ADC的规则通道和注入通道混合使用

    之前完成了规则通道DMA的数据传输了,不过平时在使用ADC的时候可能就会遇到很多情况,不可能就这样简单的按规则通道来采样,DMA存储,使用数据的:可能有时候会需要立刻采样,那样我们就需要利用到注入通道 ...

  4. STM32f103的数电采集电路的双ADC的设计与使用

    STM32F103C8T6拥有3个ADC,其独立使用已经在本文的3.1.3里面有详细的介绍,这里主要是介绍双ADC的同时使用,即STM32的同步规则模式使用.在此模式在规则通道组上执行时,外部触发来自 ...

  5. DSP基础学习-ADC同步采样

    DSP基础学习-ADC同步采样 彭会锋 2015-04-28 20:31:06 在DSP28027 LauchPad学习过程中,关于ADC同步采样和顺序采样的区别稍加研究了一下,发现里面还真有些门道, ...

  6. 关于Stm32定时器+ADC+DMA进行AD采样的实现

    Stm32的ADC有DMA功能这都毋庸置疑,也是我们用的最多的!然而,如果我们要对一个信号(比如脉搏信号)进行定时采样(也就是隔一段时间,比如说2ms),有三种方法: 1.使用定时器中断每隔一定时间进 ...

  7. STM32定时器触发ADC多通道连续采样,DMA缓存结果

    STM32的ADC使用非常灵活,采样触发方面:既支持软件触发,定时器或其他硬件电路自动触发,也支持转换完成后自动触发下一通道/轮转换.转换结果存储方面:既支持软件读取和转存,也支持DMA自动存储转换结 ...

  8. STM32 多通道ADC采样,采用Timer1进行采样率控制,利用DMA进行传输

    http://blog.csdn.net/varding/article/details/17559399 http://www.51hei.com/stm32/3842.html https://w ...

  9. STM32的ADC采样时间

    STM32的ADC采样时间与其ADC的时钟频率密不可分. 例:STM32F103系列的ADC的时钟是在APB2(最大72MHZ)上.我们可以对其分频: RCC_PCLK2_Div2: ADC cloc ...

随机推荐

  1. Python 文件 readline() 方法

    描述 Python 文件 readline() 方法用于从文件读取整行,包括 "\n" 字符.如果指定了一个非负数的参数,则返回指定大小的字符数,包括 "\n" ...

  2. 公网用户接入NAT后面的freeswitch配置

    大致网络示意和终端号码: 客户端侧: 终端号码(1019)终端IP(192.168.1.15)+ 网关(192.168.1.1) + 路由器公网IP(动态地址) 服务器侧: 防火墙(181.92.2. ...

  3. PCM、G.729等常用VoIP编码的理论带宽计算

    可能通信背景的同学,一提到PCM编码,脑海里都能跳出来一个数值64K. 一.64KB还是64Kb? 64Kb! 二.哪里来的64Kb? CCITT规定抽样率为每秒8000KHz,每抽样值编8位码,所以 ...

  4. Spring-MVC配置Gson做为Message Converter解析Json

    Spring-MVC配置Gson做为Message Converter解析Json 在学习Spring的时候看到可以使用@RequestBody 和@ResponseBody注解来是的Spring自动 ...

  5. HTTP 请求头 详解

    转载:https://kb.cnblogs.com/page/92320/ HTTP(HyperTextTransferProtocol)即超文本传输协议,目前网页传输的的通用协议.HTTP协议采用了 ...

  6. 标准差(Standard Deviation) 和 标准误差(Standard Error)

    本文摘自 Streiner DL.Maintaining standards: differences between the standard deviation and standarderror ...

  7. 怎么设置Linux swap分区?方法教程

    如何设置Linux swap分区 看到不少朋友问linux下swap分区的问题,收集到一篇 不错的文章 ,分享下. 什么是Swap?Swap,即交换区,除了安装Linux的时候,有多少人关心过它呢?其 ...

  8. Java中return返回结果的优先级

    在Java开发时,异常处理是非常普遍的.先看这样一道关于异常处理的代码 public static int getNumer() { int a = 1; try { return a; } catc ...

  9. Java 用栈实现队列

    栈是先入后出,队列是先入先出.根据这个思想,可以用一个栈作为入队,另一个栈作为出队.只要把第一个栈的栈顶的元素压入第二个栈就好了,出队的时候输出第二个栈的栈顶,如果第二个栈的空了就需要不断操作从第一个 ...

  10. python-计算数据的相关性

    先解释下什么叫数据的相关性: 计算两个数组的平均值,如果两个数组中的两个值都大于平均值或者都小于平均值,则得到true. 如果一个大于平均值一个小于平均值,则得到false.最后计算true和fals ...