最近需要用到两个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. 用SQL语句将远程SQL Server数据库中表数据导入到本地数据库相应的表中

    一.方法一 访问不同电脑上的数据库(远程访问,只好联好网就一样),如果经常访问或数据量较大,建议用链接服务器方法. 1.创建链接服务器 exec sp_addlinkedserver ‘srv_lnk ...

  2. Source Insight中代码块注释

    转载:http://blog.csdn.net/cindylx422/article/details/7560786 修改工程目录下的utils.em文件,将下面代码添加到末尾,然后将此文件添加到工程 ...

  3. 一点做用户画像的人生经验:ID强打通

    1. 背景 在构建精准用户画像时,面临着这样一个问题:日志采集不能成功地收集用户的所有ID,且每条业务线有各自定义的UID用来标识用户,从而造成了用户ID的零碎化.因此,为了做用户标签的整合,用户ID ...

  4. ElementUI 按需引入坑爹的点记录

    官网说的是这样的: 但实际上,应该是这样修改: { "presets": [ ["env", { "targets": { "br ...

  5. hibernate显示完整的sql(转)

    不完整的SQL Hibernate: /* insert com.test.bean.User */ insert into user (username, password, email, sex) ...

  6. JavaScript监听手机物理返回键的两种解决方法

    JavaScript没有监听物理返回键的API,所以只能使用 popstate 事件监听. 有两个解决办法: 1.返回到指定的页面 pushHistory(); window.addEventList ...

  7. 精确光源(Punctual Light Sources)

    <Physically-Based Shading Models in Film and Game Production>(SIGGRAPH 2010 Course Notes) (地址: ...

  8. JDK1.6新特性,网络增强(Networking features and enhancements)

    参考: http://docs.oracle.com/javase/6/docs/technotes/guides/net/enhancements-6.0.html http://blog.csdn ...

  9. Android 关于Fragment一些简单注意事项

    1.Fragment的生命周期一定要了解清楚: 其实没有什么好说的. 注意关键字所执行的方法: 注意下面基本fragmentTransaction.replace();运行 当FragmentA(初始 ...

  10. eclipse下SVN subclipse插件

    本文目的 让未使用过版本控制器软件或者未使用过subversion软件的人员尽快上手. subversion的使用技巧很多,这里只总结了最小使用集,即主要的基本功能,能够用来应付日常工作. 因此不涉及 ...