STM32F0 定时器触发ADC,多通道采样、DMA传输数据的配置

发布者:TranquilDreamer最新更新时间:2018-09-11 来源: eefocus关键字:STM32F0  定时器触发  ADC  多通道采样 手机看文章 扫描二维码
随时随地手机看文章

本块代码实现了定时器定时触发ADC,多通道采样,并通过DMA进行数据传输到内存的操作。

#include    "adc.h"  

[cpp] view plain copy

#define DMA_BUFFER_SIZE     6  

uint8 sample_finish = 0;  

int16 adc_dma_tab[6] = { 0 };  

uint8 sample_index = 0;  

  

//采样点数据  

int16 sample_1[128] = { 0 };  

int16 sample_2[128] = { 0 };  

int16 sample_3[128] = { 0 };  

int16 sample_4[128] = { 0 };  

int16 sample_5[128] = { 0 };  

int16 sample_6[128] = { 0 };  

  

void user_adc_init()  

{  

    adc_gpio_init();  

    adc_config();               // 注意此处的初始化顺序,否则采样传输的数据容易出现数据错位的结果  

    adc_dma_init();             //  

    adc_timer_init();           //  

  

}  

  

void adc_config()  

{  

    ADC_InitTypeDef adc_init_structure;  

  

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);            //使能ADC时钟  

  

    ADC_DeInit(ADC1);                                               //复位ADC  

    ADC_StructInit(&adc_init_structure);                            //初始化ADC结构体  

  

    adc_init_structure.ADC_ContinuousConvMode = DISABLE;            //禁用连续转换模式  

    adc_init_structure.ADC_DataAlign = ADC_DataAlign_Right;         //采样数据右对齐  

    adc_init_structure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T2_TRGO; //外部触发设置为TIM2  

    adc_init_structure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_Rising;//上升沿触发  

    adc_init_structure.ADC_Resolution = ADC_Resolution_12b;         //12位分辨率  

    adc_init_structure.ADC_ScanDirection = ADC_ScanDirection_Upward;//向上扫描0-18通道  

    ADC_Init(ADC1, &adc_init_structure);  

  

    ADC_OverrunModeCmd(ADC1, ENABLE);                               //使能数据覆盖模式  

    ADC_ChannelConfig(ADC1, ADC_Channel_0 | ADC_Channel_1 | ADC_Channel_2  

                          | ADC_Channel_8 | ADC_Channel_14 | ADC_Channel_15,  

                          ADC_SampleTime_13_5Cycles);               //配置采样通道,采样时间125nS  

    ADC_GetCalibrationFactor(ADC1);                                 //使能前校准ADC  

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

    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_ADEN) == RESET);         //等待ADC1使能完成  

  

    ADC_DMACmd(ADC1, ENABLE);                                       //使能ADC_DMA  

    ADC_DMARequestModeConfig(ADC1, ADC_DMAMode_Circular);           //配置DMA请求模式为循环模式  

    ADC_StartOfConversion(ADC1);                                    //开启一次转换(必须)  

}  

  

void adc_gpio_init()  

{  

    GPIO_InitTypeDef gpio_init_structure;  

    //使能GPIO时钟  

    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA | RCC_AHBPeriph_GPIOB | RCC_AHBPeriph_GPIOC, ENABLE);  

  

    GPIO_StructInit(&gpio_init_structure);  

    //GPIOA                                                         //PA-0~3用作ADC  

    gpio_init_structure.GPIO_Pin = (GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3);  

    gpio_init_structure.GPIO_Mode = GPIO_Mode_AN;                   //使用附加(模拟)功能  

    gpio_init_structure.GPIO_OType = GPIO_OType_PP;                 //推挽输出  

    gpio_init_structure.GPIO_Speed = GPIO_Speed_50MHz;              //Fast speed  

    gpio_init_structure.GPIO_PuPd= GPIO_PuPd_UP;                    //上拉  

    GPIO_Init(GPIOA, &gpio_init_structure);  

    //GPIOB  

    gpio_init_structure.GPIO_Pin = GPIO_Pin_0;                      //PB-0 用作ADC  

    GPIO_Init(GPIOB, &gpio_init_structure);  

    //GPIOC  

    gpio_init_structure.GPIO_Pin = (GPIO_Pin_4 | GPIO_Pin_5);       //PC-4~5用作ADC  

    GPIO_Init(GPIOC, &gpio_init_structure);  

}  

  

void adc_dma_init()  

{  

    DMA_InitTypeDef dma_init_structure;  

    NVIC_InitTypeDef nvic_init_structure;  

  

    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);              //使能DMA时钟  

  

    nvic_init_structure.NVIC_IRQChannel = DMA1_Channel1_IRQn;       //选择DMA1通道中断  

    nvic_init_structure.NVIC_IRQChannelCmd = ENABLE;                //中断使能  

    nvic_init_structure.NVIC_IRQChannelPriority = 0;                //优先级设为0  

    NVIC_Init(&nvic_init_structure);  

  

    DMA_DeInit(DMA1_Channel1);                                      //复位DMA1_channel1  

    DMA_StructInit(&dma_init_structure);                            //初始化DMA结构体  

  

    dma_init_structure.DMA_BufferSize = DMA_BUFFER_SIZE;            //DMA缓存数组大小设置  

    dma_init_structure.DMA_DIR = DMA_DIR_PeripheralSRC;             //DMA方向:外设作为数据源  

    dma_init_structure.DMA_M2M = DISABLE;                           //内存到内存禁用  

    dma_init_structure.DMA_MemoryBaseAddr = (uint32)&adc_dma_tab[0];//缓存数据数组起始地址  

    dma_init_structure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//数据大小设置为Halfword  

    dma_init_structure.DMA_MemoryInc = DMA_MemoryInc_Enable;        //内存地址递增  

    dma_init_structure.DMA_Mode = DMA_Mode_Circular;                //DMA循环模式,即完成后重新开始覆盖  

    dma_init_structure.DMA_PeripheralBaseAddr = (uint32) &(ADC1->DR);//取值的外设地址  

    dma_init_structure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//外设取值大小设置为Halfword  

    dma_init_structure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址递增禁用  

    dma_init_structure.DMA_Priority = DMA_Priority_High;             //DMA优先级设置为高  

    DMA_Init(DMA1_Channel1, &dma_init_structure);  

  

    DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, ENABLE);                  //使能DMA中断  

    DMA_ClearITPendingBit(DMA_IT_TC);                                //清除一次DMA中断标志  

    DMA_Cmd(DMA1_Channel1, ENABLE);                                  //使能DMA1  

}  

  

void adc_timer_init()  

{  

    TIM_TimeBaseInitTypeDef timer_init_structure;  

    NVIC_InitTypeDef nvic_init_structure;  

  

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);            //使能TIM2时钟  

  

    nvic_init_structure.NVIC_IRQChannel = TIM2_IRQn;                //选择TIM2中断通道  

    nvic_init_structure.NVIC_IRQChannelCmd = ENABLE;                //使能TIM2中断  

    nvic_init_structure.NVIC_IRQChannelPriority = 0;                //优先级为0  

    NVIC_Init(&nvic_init_structure);  

  

    TIM_DeInit(TIM2);                                               //复位TIM2  

    TIM_TimeBaseStructInit(&timer_init_structure);                  //初始化TIMBASE结构体  

  

    timer_init_structure.TIM_ClockDivision = TIM_CKD_DIV1;          //系统时钟,不分频,48M  

    timer_init_structure.TIM_CounterMode = TIM_CounterMode_Up;      //向上计数模式  

    timer_init_structure.TIM_Period = 312;                          //每312 uS触发一次中断,开启ADC  

    timer_init_structure.TIM_Prescaler = 48-1;                      //计数时钟预分频,f=1M,systick=1 uS  

    timer_init_structure.TIM_RepetitionCounter = 0x00;              //发生0+1次update事件产生中断  

    TIM_TimeBaseInit(TIM2, &timer_init_structure);  

  

    TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);                      //使能TIM2中断  

    TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_Update);           //选择TIM2的update事件更新为触发源  

  

    TIM_Cmd(TIM2, ENABLE);                                          //使能TIM2  

}  

  

  

/****************************中断服务程序****************************/  

void TIM2_IRQHandler()  

{  

    if(TIM_GetITStatus(TIM2, TIM_IT_Update))            //判断发生update事件中断  

    {  

        TIM_ClearITPendingBit(TIM2, TIM_IT_Update);     //清除update事件中断标志  

    }  

}  

  

void DMA1_Channel1_IRQHandler()  

{  

    if(DMA_GetITStatus(DMA_IT_TC))                      //判断DMA传输完成中断  

    {  

        if(sample_finish == 0)  

        {  

            sample_1[sample_index] = adc_dma_tab[0];  

            sample_2[sample_index] = adc_dma_tab[1];  

            sample_3[sample_index] = adc_dma_tab[2];  

            sample_4[sample_index] = adc_dma_tab[3];  

            sample_5[sample_index] = adc_dma_tab[5];  

            sample_6[sample_index] = adc_dma_tab[4];  

            sample_index++;  

        }  

        if(sample_index >= 128)                         //注意防止数组越界导致未知错误  

        {  

            sample_index = 0;  

            TIM_Cmd(TIM2, DISABLE);                     //完成周波采样,停止定时器  

            DMA_Cmd(DMA1_Channel1, DISABLE);            //完成周波采样,停止DMA  

            sample_finish = 1;                          //置采样完成标志位  

        }  

    }  

    DMA_ClearITPendingBit(DMA_IT_TC);                   //清除DMA中断标志位  

}


关键字:STM32F0  定时器触发  ADC  多通道采样 引用地址:STM32F0 定时器触发ADC,多通道采样、DMA传输数据的配置

上一篇:STM32 USART串口DMA 接收和发送的源码详解!
下一篇:STM32实现IAP功能之一

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

何谓ADC?STM32中的ADC有什么功能?
ADC简介 STM32F103系列有3个ADC,精度为12位,每个ADC最多有16个外部通道。其中ADC1和ADC2都有16个外部通道,ADC3一般有8个外部通道,各通道的A/D转换可以单次、连续、扫描或间断执行,ADC转换的结果可以左对齐或右对齐储存在16位数据寄存器中。ADC的输入时钟不得超过14MHz,其时钟频率由PCLK2分频产生。 ADC功能框图讲解 学习STM32开发板上的外设时首先要了解其外设的功能框图,如下: 功能框图可以大体分为7部分,下面一一讲解: 电压输入范围 ADC所能测量的电压范围就是VREF- ≤ VIN ≤ VREF+,把 VSSA 和 VREF-接地,把 VREF+和 VDDA 接 3V3,
[单片机]
何谓<font color='red'>ADC</font>?STM32中的<font color='red'>ADC</font>有什么功能?
利用内置PGA的24位Σ-Δ型ADC AD7780实现电子秤设计
电路功能与优势 本电路为采用 AD7780 构建的电子秤系统。AD7780是一款引脚可编程、低功耗、低漂移24位 - 型ADC,内置PGA,采用内部时钟。该器件将大多数系统构建模块置于芯片内,因此能够简化电子秤设计。该器件的典型功耗仅为330 A,适合所有低功耗或电池供电应用。AD7780还提供省电模式,不执行转换时,用户可以切断对桥式传感器的供电,并使该器件进入省电模式,从而延长电池使用时间。 利用内置PGA的24位 - 型ADC AD7780实现电子秤设计 2011-3-20 22:04:10 上传 下载附件 (16.12 KB) 图1. 采
[测试测量]
利用内置PGA的24位Σ-Δ型<font color='red'>ADC</font> AD7780实现电子秤设计
高功耗 低噪声 8位高速模数转换器【ADI】
2011-02-17 10:00 北京2011年2月17日电 -- Analog Devices, Inc.(ADI),最近针对仪器仪表和通信应用推出两款8位、高速、低功耗、低噪声 ADC(模数转换器)AD9284和AD9286。这两款8位高速 ADC 均采用流水线架构,310 mW 的功耗创业界最低水平。 AD9284 AD9286 高速低功耗8位 ADC AD9284 是业界首款双通道、8位、250 MSPS ADC;AD9286则是业界功耗最低的8位、500 MSPS ADC,能以低于竞争产品40%的功耗提供8位分辨率。两款转换器具有 49.3 dBFS 的高信噪比 (SNR),支持宽动态信号范围,无杂
[模拟电子]
高功耗 低噪声 8位高速<font color='red'>模数转换器</font>【ADI】
STM32CubeMX系列 | ADC模数转换
ADC模数转换 1. ADC简介 ADC(analog to digital converter)即模数转换器,它可以将模拟量信号转换为数字信号,按照转换原理主要分为逐次逼近型、双积分型、电压频率转换型三种。STM32F1的ADC是12位逐次逼近型的模数转换器,它有18个通道,可测量16个外部和2个内部信号源。各通道的A/D转换可以单次、连续、扫描或间断模式执行。ADC的结果可以左对齐或右对齐方式存储在16位存储寄存器中。模拟看门狗特性允许应用程序检测输入电压是否超出用户定义的高/低阈值。ADC的时钟不要超过14M,否则将导致结果准确度下降。ADC结构框图以及ADC引脚说明如下图示: 电压输入引脚和输入通道引脚见上表;另外主
[单片机]
新型流水线实现高速低功耗ADC的原理及方法
1 MAX1200的特点功能   新型ADC正在朝着低功耗、高速、高分辨率的方向发展,新型流水线结构正是实现高速低功耗ADC的有效方法。而MAX1200则是采用这一新技术的高速、高精度、低功耗ADC的代表。   MAX1200是一种16位、采样率可达1Msps的单片集成模数转换器,其内部的CMOS积分电路采用全差分多级流水线结构,它具有快速的数字误差校正和自校准功能,能保证在全采样率时具有16位的线性度和91dB的非杂散动态范围(SFDR),以及良好的信噪比(SNR)和谐波失真(THD)特性。MAX1200主要应用于高分辨率图象系统、扫描仪、数字通讯、检测仪表和数据接收等领域;其主要技术特点如下:   ●采用单电源+5V供电
[模拟电子]
新型流水线实现高速低功耗<font color='red'>ADC</font>的原理及方法
STM32 ADC模拟看门狗应用演示过程
我们在做ADC应用时,有些场合需要精确测量出待测信号的数据,有时可能并不关心ADC结果多准确,只要满足某个范围即可,这时我们就可以考虑使用STM32ADC看门狗功能了。 STM32 ADC基本上都支持模拟看门狗功能,即ADC模块对被检测的模拟通道的转换结果基于硬件对其合法性、安全性进行监测。我们可以设置被监测通道的转换结果合法性检查的上下阈值,若结果处于阈值之外则视为异常,并可以触发中断。 尤其有些场合,我们可能使用到数个ADC通道,8个、10个甚至更多,同时程序还需要基于各通道的转换结果进行合法性或安全性监测,此时使用ADC模拟看门狗功能就很方便。我们可以先让模拟看门狗做第一步把关,只有出现异常数据时才去进一步检查确认。
[单片机]
STM32 <font color='red'>ADC</font>模拟看门狗应用演示过程
已解决STM32F0的stm32f0xx.h中没有CRH和CRL函数,该如何设置?
STM32F0的stm32f0xx.h中没有CRH和CRL函数,该如何设置 #define SDA_IN() {GPIOC- CRH&=0XFFFF0FFF;GPIOC- CRH|=8 12;} #define SDA_OUT() {GPIOC- CRH&=0XFFFF0FFF;GPIOC- CRH|=3 12;} 下面用库里面的引脚设置函数写成: void SDA_IN(void) { GPIO_InitTypeDef GPIO_InitStructure; //RCC- APB2ENR|=1 4;//先使能外设IO portc时钟 RCC_AHBPeriphClockCmd(RCC_AHBPeriph
[单片机]
STM32的ADC采样周期确定
  采样频率的确定   1.首先确定ADC 的时钟,这里需要看你的RCC的设置。在采用固件库的基础上,设定ADC的采样频率相对来说是很容易的。   (1)由时钟控制器提供的ADCCLK 时钟和PCLK2(APB2 时钟)同步。CLK 控制器为ADC 时钟提供一个专用的可编程预分频器。   (2) 一般情况下在程序 中将 PCLK2 时钟设为 与系统时钟 相同   RCC_HCLKConfig(RCC_SYSCLK_Div1);   RCC_PCLK2Config(RCC_HCLK_Div1);   RCC_PCLK1Config(RCC_HCLK_Div2);   (3)采样时间和转换时间   ADC 使用若干个ADC_CLK 周
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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