STM32学习笔记—定时器触发ADC采集+DMA转运数据(基于标准库)

2023-12-15 04:10

本文主要是介绍STM32学习笔记—定时器触发ADC采集+DMA转运数据(基于标准库),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        最近处于电赛准备期间,我们组准备做信号的题目,我作为软件打下手的,需要了解些信号处理的一些基本算法,比如说FFT。我也看了很多网上的教程,基本上也把FFT摸得大差不差,今天我们先讲一下怎么通过定时器2触发ADC1采集,然后在通过DMA1把数据转运出去。

        通过查找数据手册相关内容,我找到了定时器外部触发的条件,这样的外部触发事件一共是有8个,我打算用的是通用定时器2(TIM2)的CC2时间触发(也就是TIM的输出比较模式0),数据手册上说的是,只有它的上升沿可以启动转换,也就是说我们生成的PWM的周期就是我们ADC1的采样周期,通过设置ARR和PSC的值我们可以很方便的控制采样频率。

         现在程序的大概框架我们也就知道了,先用定时器2的输出比较模式生成一个PWM,ADC1设置成外部触发模式,然后在通过DMA1转运ADC1的采样数据。那就让我们开始代码的编写吧!

1:GPIO初始化(设置PA1为模拟输入口)

void MyGPIO_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启gpio时钟GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AIN;			//设置PA1为模拟输入GPIO_InitStruct.GPIO_Pin = GPIO_Pin_1;GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStruct);
}

2.ADC1初始化 设置为外部触发模式

void MyADC1_Init(void)
{MyGPIO_Init();										//设置PA1为ADC1模拟输入口/*==================1.开启ADC1RCC时钟===============*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);//开启ADC1时钟/*==================2.复位ADC1 设置ADC1分频因子===============*/ADC_DeInit(ADC1);								   //复位ADC1 全部寄存器设置为缺省值RCC_ADCCLKConfig(RCC_PCLK2_Div6);				   //设置ADC1分频因子 12MHz/*==================3.ADC1初始化===============*/ADC_InitTypeDef ADC_InitStruct;ADC_InitStruct.ADC_Mode = ADC_Mode_Independent;	   //设置ADC1模式:独立模式ADC_InitStruct.ADC_ScanConvMode = DISABLE;		   //不开启扫描模式ADC_InitStruct.ADC_ContinuousConvMode = DISABLE;   //不开启连续转换模式ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T2_CC2;//启动规则转换组转换时间:TIM2的通道2ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;//数据对其模式:右对齐ADC_InitStruct.ADC_NbrOfChannel = 1;			   //转换的数量:1ADC_Init(ADC1,&ADC_InitStruct);ADC_RegularChannelConfig(ADC1,ADC_Channel_1,1,ADC_SampleTime_239Cycles5);//为所选的ADC常规通道配置其在序列器中的相应等级及其采样时间。ADC_ExternalTrigConvCmd(ADC1,ENABLE);              //启用外部触发ADC1转换/*==================4.使能ADC1并校准===============*/ADC_Cmd(ADC1,ENABLE);							   //启用ADC1ADC_ResetCalibration(ADC1);						   //ADC1复位校准while(ADC_GetResetCalibrationStatus(ADC1));		   //等待复位校准完成ADC_StartCalibration(ADC1);						   //ADC1 开始校准while(ADC_GetCalibrationStatus(ADC1));             //等待ADC校准完成
}

3.定时器初始化 输出比较模式生成PWM

void MyTIM2_Init(uint16_t arr,uint16_t psc)
{RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);/*==============1.初始化TIM2=============*/TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInitStruct.TIM_Period = arr - 1;TIM_TimeBaseInitStruct.TIM_Prescaler = psc - 1;TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);/*==============2.配置TIM2PWM模式=============*/TIM_OCInitTypeDef TIM_OCInitStruct;TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;				//设置输出模式:PWM模式TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;	//设置比较输出使能:TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;		//设置ref为有效电平时 输出高电平TIM_OCInitStruct.TIM_Pulse = arr/2;							//输出占空比:50%的PWM波TIM_OC2Init(TIM2,&TIM_OCInitStruct);//TIM_CtrlPWMOutputs(TIM2,ENABLE);							//PWM输出使能TIM_Cmd(TIM2,DISABLE);										//ADC1失能	初始化全部完成在开启定时器
}

注意:在这里我看到网上有些教程说的是,这里需要加一个TIM_CtrlPWMOutputs(TIM2,ENABLE),我查了下数据手册发现这个函数是高级定时器使用的,通用定时器用了可能会让程序出现问题(这个是我自己查的 具体的我也忘记在哪了)我转到函数定义也看了下,定义上也是没有TIM2的。所以我的推荐还是不要写!!!!!

 4.DMA初始化

void MyDMA1_Init(void)
{/*===============使能DMA输出==============*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);/*===============初始化DMA通道==============*/DMA_InitTypeDef DMA_InitStruct;DMA_InitStruct.DMA_BufferSize = NPT;DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)adc_value;DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;DMA_InitStruct.DMA_Mode = DMA_Mode_Circular;DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;DMA_Init(DMA1_Channel1,&DMA_InitStruct);/*===============使能DMA中断==============*/NVIC_InitTypeDef NVIC_InitStruct;NVIC_InitStruct.NVIC_IRQChannel = DMA1_Channel1_IRQn;NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0;NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0;NVIC_Init(&NVIC_InitStruct);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);DMA_ITConfig (DMA1_Channel1,DMA_IT_TC|DMA_IT_HT,ENABLE);//使能DMA1中断/*==================使能ADC1DMA发送===================*/ADC_DMACmd(ADC1,ENABLE);/*==================使能DMA1通道1 开启传输===================*/DMA_Cmd(DMA1_Channel1,ENABLE);
}

PS:DMA这里我是给他开了个中断 是准备在回调函数里面进行数据处理

5.主函数部分

    MyTIM2_Init(10,72);//100KHzMyADC1_Init();MyDMA1_Init();TIM_Cmd(TIM2,ENABLE);//使能定时器 开始触发ADC采样

        实验现象我就不放了,因为我把核心板从面包板上拆下来了,不想在弄了,太麻蛋了,你们自己弄着测一下吧,下课!!!!! 

这篇关于STM32学习笔记—定时器触发ADC采集+DMA转运数据(基于标准库)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/495059

相关文章

C#实现将XML数据自动化地写入Excel文件

《C#实现将XML数据自动化地写入Excel文件》在现代企业级应用中,数据处理与报表生成是核心环节,本文将深入探讨如何利用C#和一款优秀的库,将XML数据自动化地写入Excel文件,有需要的小伙伴可以... 目录理解XML数据结构与Excel的对应关系引入高效工具:使用Spire.XLS for .NETC

MySQL数据目录迁移的完整过程

《MySQL数据目录迁移的完整过程》文章详细介绍了将MySQL数据目录迁移到新硬盘的整个过程,包括新硬盘挂载、创建新的数据目录、迁移数据(推荐使用两遍rsync方案)、修改MySQL配置文件和重启验证... 目录1,新硬盘挂载(如果有的话)2,创建新的 mysql 数据目录3,迁移 MySQL 数据(推荐两

Python数据验证神器Pydantic库的使用和实践中的避坑指南

《Python数据验证神器Pydantic库的使用和实践中的避坑指南》Pydantic是一个用于数据验证和设置的库,可以显著简化API接口开发,文章通过一个实际案例,展示了Pydantic如何在生产环... 目录1️⃣ 崩溃时刻:当你的API接口又双叒崩了!2️⃣ 神兵天降:3行代码解决验证难题3️⃣ 深度

Linux内核定时器使用及说明

《Linux内核定时器使用及说明》文章详细介绍了Linux内核定时器的特性、核心数据结构、时间相关转换函数以及操作API,通过示例展示了如何编写和使用定时器,包括按键消抖的应用... 目录1.linux内核定时器特征2.Linux内核定时器核心数据结构3.Linux内核时间相关转换函数4.Linux内核定时

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

MySQL中的DELETE删除数据及注意事项

《MySQL中的DELETE删除数据及注意事项》MySQL的DELETE语句是数据库操作中不可或缺的一部分,通过合理使用索引、批量删除、避免全表删除、使用TRUNCATE、使用ORDERBY和LIMI... 目录1. 基本语法单表删除2. 高级用法使用子查询删除删除多表3. 性能优化策略使用索引批量删除避免

MySQL 数据库进阶之SQL 数据操作与子查询操作大全

《MySQL数据库进阶之SQL数据操作与子查询操作大全》本文详细介绍了SQL中的子查询、数据添加(INSERT)、数据修改(UPDATE)和数据删除(DELETE、TRUNCATE、DROP)操作... 目录一、子查询:嵌套在查询中的查询1.1 子查询的基本语法1.2 子查询的实战示例二、数据添加:INSE

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p

使用MyBatis TypeHandler实现数据加密与解密的具体方案

《使用MyBatisTypeHandler实现数据加密与解密的具体方案》在我们日常的开发工作中,经常会遇到一些敏感数据需要存储,比如用户的手机号、身份证号、银行卡号等,为了保障数据安全,我们通常会对... 目录1. 核心概念:什么是 TypeHandler?2. 实战场景3. 代码实现步骤步骤 1:定义 E