最近需要用到两个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. appium简明教程(2)——appium的基本概念

    Client/Server Architecture appium的核心其实是一个暴露了一系列REST API的server. 这个server的功能其实很简单:监听一个端口,然后接收由client发 ...

  2. VMware Workstation 11

    VMware Workstation 11开始只支持Windows7以上的64位系统,VMware Workstation 10.0.6为最后支持32位和XP的系统. 下载地址: VMware Wor ...

  3. 跟我学SharePoint 2013视频培训课程——排序、过滤在列表、库中的使用(10)

    课程简介 第10天,SharePoint 2013排序.过滤在列表.库中的使用. 视频 SharePoint 2013 交流群 41032413

  4. hive SQL 字母大小写转换

    lower(string A) lcase(string A) 将文本字符串转换成字母全部小写形式 upper(string A) ucase(string A) 将文本字符串转换成字母全部大写形式

  5. How to convert String to Date – Java

    In this tutorial, we will show you how to convert a String to java.util.Date. Many Java beginners ar ...

  6. css冻结列的效果

    <!DOCTYPE html><html lang="zh-cn"><head><meta charset="utf-8&quo ...

  7. xcode6 怎样下载ios7模拟器

    1. 怎样下载ios7模拟器 点击xcode.选择"Preferences".选择"downloads",就能够看见IOS 7.1,只是下载有点慢. 2. 怎样 ...

  8. CentOS 7 安装中文环境

    centos升级到7后,系统设置好多和6有了很大的区别,中文支持就有很大的变化. 1.安装中文语言包. yum install kde-l10n-Chinese 2.安装(已经安装的要重新安装)gli ...

  9. 深入理解Linux内核-内存管理

    内核如果给自己分配动态内存 动态内存:RAM的某些部分被永久打分配给内核,用来存放内核代码以及静态内核数据结构:剩余的部分被称为动态内存 连续物理内存区管理: 页框管理:1.页大小的选择,通常情况下主 ...

  10. C/C++中的volatile关键字

    volatile提醒编译器它后面所定义的变量随时都有可能改变,因此编译后的程序每次需要存储或读取这个变量的时候,都会直接从变量地址中读取数据. 如果没有volatile关键字,则编译器可能优化读取和存 ...