STM32 双ADC同步规则采样
最近需要用到两个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同步规则采样的更多相关文章
- STM32 ADC多通道规则采样和注入采样
layout: post tags: [STM32] comments: true 文章目录 layout: post tags: [STM32] comments: true 什么是ADC? STM ...
- STM32cube库配置双ADC的同步规则采样
http://www.stmcu.org/module/forum/forum.php?mod=viewthread&tid=605203&extra=page%3D&page ...
- 【stm32】ADC的规则通道和注入通道混合使用
之前完成了规则通道DMA的数据传输了,不过平时在使用ADC的时候可能就会遇到很多情况,不可能就这样简单的按规则通道来采样,DMA存储,使用数据的:可能有时候会需要立刻采样,那样我们就需要利用到注入通道 ...
- STM32f103的数电采集电路的双ADC的设计与使用
STM32F103C8T6拥有3个ADC,其独立使用已经在本文的3.1.3里面有详细的介绍,这里主要是介绍双ADC的同时使用,即STM32的同步规则模式使用.在此模式在规则通道组上执行时,外部触发来自 ...
- DSP基础学习-ADC同步采样
DSP基础学习-ADC同步采样 彭会锋 2015-04-28 20:31:06 在DSP28027 LauchPad学习过程中,关于ADC同步采样和顺序采样的区别稍加研究了一下,发现里面还真有些门道, ...
- 关于Stm32定时器+ADC+DMA进行AD采样的实现
Stm32的ADC有DMA功能这都毋庸置疑,也是我们用的最多的!然而,如果我们要对一个信号(比如脉搏信号)进行定时采样(也就是隔一段时间,比如说2ms),有三种方法: 1.使用定时器中断每隔一定时间进 ...
- STM32定时器触发ADC多通道连续采样,DMA缓存结果
STM32的ADC使用非常灵活,采样触发方面:既支持软件触发,定时器或其他硬件电路自动触发,也支持转换完成后自动触发下一通道/轮转换.转换结果存储方面:既支持软件读取和转存,也支持DMA自动存储转换结 ...
- STM32 多通道ADC采样,采用Timer1进行采样率控制,利用DMA进行传输
http://blog.csdn.net/varding/article/details/17559399 http://www.51hei.com/stm32/3842.html https://w ...
- STM32的ADC采样时间
STM32的ADC采样时间与其ADC的时钟频率密不可分. 例:STM32F103系列的ADC的时钟是在APB2(最大72MHZ)上.我们可以对其分频: RCC_PCLK2_Div2: ADC cloc ...
随机推荐
- appium简明教程(2)——appium的基本概念
Client/Server Architecture appium的核心其实是一个暴露了一系列REST API的server. 这个server的功能其实很简单:监听一个端口,然后接收由client发 ...
- VMware Workstation 11
VMware Workstation 11开始只支持Windows7以上的64位系统,VMware Workstation 10.0.6为最后支持32位和XP的系统. 下载地址: VMware Wor ...
- 跟我学SharePoint 2013视频培训课程——排序、过滤在列表、库中的使用(10)
课程简介 第10天,SharePoint 2013排序.过滤在列表.库中的使用. 视频 SharePoint 2013 交流群 41032413
- hive SQL 字母大小写转换
lower(string A) lcase(string A) 将文本字符串转换成字母全部小写形式 upper(string A) ucase(string A) 将文本字符串转换成字母全部大写形式
- 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 ...
- css冻结列的效果
<!DOCTYPE html><html lang="zh-cn"><head><meta charset="utf-8&quo ...
- xcode6 怎样下载ios7模拟器
1. 怎样下载ios7模拟器 点击xcode.选择"Preferences".选择"downloads",就能够看见IOS 7.1,只是下载有点慢. 2. 怎样 ...
- CentOS 7 安装中文环境
centos升级到7后,系统设置好多和6有了很大的区别,中文支持就有很大的变化. 1.安装中文语言包. yum install kde-l10n-Chinese 2.安装(已经安装的要重新安装)gli ...
- 深入理解Linux内核-内存管理
内核如果给自己分配动态内存 动态内存:RAM的某些部分被永久打分配给内核,用来存放内核代码以及静态内核数据结构:剩余的部分被称为动态内存 连续物理内存区管理: 页框管理:1.页大小的选择,通常情况下主 ...
- C/C++中的volatile关键字
volatile提醒编译器它后面所定义的变量随时都有可能改变,因此编译后的程序每次需要存储或读取这个变量的时候,都会直接从变量地址中读取数据. 如果没有volatile关键字,则编译器可能优化读取和存 ...