STM32 ADC+DMA+TIM触发采样实战:避坑指南与源码解析
知识点1【TRGO的介绍】
1、TRGO的概述
TRGO:Trigger Output(触发输出),是定时器的一种功能。
它可以作为外设的启动信号,比如ADC转换,DAC输出,DMA请求等。
对于ADC来说,可以通过TRGO信号来触发ADC开始转换
2、定时器TRGO的生成过程
以通用定时器为例,TRGO信号的来源可以是:更新事件,比较事件,其他事件。如下图
通过TIM_SelectOutputTrigger()
进行配置
3、补充
(1)ADC持续转换的两种方式
1、ContinuousConvMode = ENABLE
+ ExternalTrig = None
启动连续采集,适合实时采样
2、ContinuousConvMode = DISABLE
+ ExternalTrig = TIMx_TRGO
每次触发采样一次,适合周期性定点采样
每次定时器触发都会扫描一次所有通道
(2)DMA—TC中断的触发时机
当ADC完成一次完整的通道扫描,并且DMA将所有通道的结果都搬运到内存后,才会触发DMA传输完成中断(TC)
知识点2【代码练习】
本代码是一个将ADC(多通道),DMA(中断),TIM(TRGO)相结合的案例
实现的功能是ADC每秒转换一次,但这里我只有一个光敏元件,因此只有一个通道有数据。
main.c
#include "stm32f10x.h"
#include "stm32f10x_conf.h"
#include "delay.h"
#include "usart.h"
#include "adc.h"
#include "dma.h"
#include "tim.h"
u16 ADC3_Recv_Data;
int flag;#define ADC_NUM 3//ADC + DAM(中断) + TRGO
//多通道接收数据 用数组保存
u16 data_adc_mult[ADC_NUM] = {0};//需要配置的有
//ADC3 IN6 IN5 IN4:PF8 PF7 PF6 ok
//USART1 TX:PA9,RX:PA10 ok
//DMA2 CH5
//TIM3 TRGO okint main(void)
{//中断向量组配置Systick_Init(72000);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);Usart1_Init(9600);ADC3_Interrupt_DMA_TIM_GPIO_Init();ADC3_Interrupt_DMA_TIM_Init();DMA2_IN5_ADC3_Init();TIM3_TRGO_Init();while(1){ }
}
adc.c
#include "adc.h"
#define ADC_NUM 3void ADC3_Interrupt_DMA_TIM_Init(void)
{ADC_InitTypeDef ADC3_InitStruct;//时钟 分频RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC3,ENABLE);RCC_ADCCLKConfig(RCC_PCLK2_Div6);//ADC3初始化 配置为外部触发ADC_StructInit(&ADC3_InitStruct); ADC3_InitStruct.ADC_ContinuousConvMode = DISABLE;ADC3_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;ADC3_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO;ADC3_InitStruct.ADC_Mode = ADC_Mode_Independent;ADC3_InitStruct.ADC_NbrOfChannel = ADC_NUM;ADC3_InitStruct.ADC_ScanConvMode = ENABLE;ADC_Init(ADC3,&ADC3_InitStruct);//通道配置ADC_RegularChannelConfig(ADC3,ADC_Channel_6,1,ADC_SampleTime_55Cycles5);ADC_RegularChannelConfig(ADC3,ADC_Channel_5,2,ADC_SampleTime_55Cycles5);ADC_RegularChannelConfig(ADC3,ADC_Channel_4,3,ADC_SampleTime_55Cycles5);//使能ADC3 + DAMADC_Cmd(ADC3,ENABLE);ADC_DMACmd(ADC3,ENABLE);ADC_ExternalTrigConvCmd(ADC3, ENABLE);//外部触发//校验ADC_ResetCalibration(ADC3);while(ADC_GetResetCalibrationStatus(ADC3) == SET);ADC_StartCalibration(ADC3);while(ADC_GetCalibrationStatus(ADC3) == SET);//开始转换//ADC_SoftwareStartConvCmd(ADC3,ENABLE);//由于是外部触发 因此这句话不需要写
}void ADC3_Interrupt_DMA_TIM_GPIO_Init(void)
{GPIO_InitTypeDef GPIOF_InitStruct;//时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF,ENABLE);//配置PF8 PF7 PF6为模拟输入GPIO_StructInit(&GPIOF_InitStruct);GPIOF_InitStruct.GPIO_Mode = GPIO_Mode_AIN;GPIOF_InitStruct.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8;GPIOF_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOF,&GPIOF_InitStruct);
}
dma.c
#include "dma.h"
extern u16 data_adc_mult[ADC_NUM];void DMA2_IN5_ADC3_Init(void)
{DMA_InitTypeDef DMA2_InitStruct;NVIC_InitTypeDef NVIC_InitStruct;//时钟配置RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2,ENABLE);//初始化DMA2_InitStruct.DMA_BufferSize = ADC_NUM;DMA2_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;DMA2_InitStruct.DMA_M2M = DMA_M2M_Disable;DMA2_InitStruct.DMA_MemoryBaseAddr = (u32)data_adc_mult;DMA2_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //ADC数据12位DMA2_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;DMA2_InitStruct.DMA_Mode = DMA_Mode_Circular;//这里需要设置循环模式的原因://每次触发都需要 重新将DMA的计数器复位DMA2_InitStruct.DMA_PeripheralBaseAddr = (u32)&ADC3->DR;DMA2_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;DMA2_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;DMA2_InitStruct.DMA_Priority = DMA_Priority_High;DMA_Init(DMA2_Channel5,&DMA2_InitStruct);//中断使能DMA_ITConfig(DMA2_Channel5,DMA_IT_TC,ENABLE);NVIC_InitStruct.NVIC_IRQChannel = DMA2_Channel4_5_IRQn;NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 1;NVIC_InitStruct.NVIC_IRQChannelSubPriority = 1;NVIC_Init(&NVIC_InitStruct);//使能DMADMA_Cmd(DMA2_Channel5,ENABLE);
}//中断处理函数
void DMA2_Channel4_5_IRQHandler(void)
{if(DMA_GetITStatus(DMA2_IT_TC5) == SET){int i = 0;DMA_ClearITPendingBit(DMA2_IT_TC5);for(i = 0;i < ADC_NUM ;i++){printf("%f ",DataProcess_light(data_adc_mult[i]));}printf("\\n");}
}
tim.c
**#include "tim.h"
void TIM3_TRGO_Init(void)
{TIM_TimeBaseInitTypeDef TIM3_InitStrct;//时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);//定时器初始化TIM_TimeBaseStructInit(&TIM3_InitStrct);TIM3_InitStrct.TIM_ClockDivision = TIM_CKD_DIV1;TIM3_InitStrct.TIM_CounterMode = TIM_CounterMode_Up;TIM3_InitStrct.TIM_Period = 10000 - 1;TIM3_InitStrct.TIM_Prescaler = 7200 - 1;TIM_TimeBaseInit(TIM3,&TIM3_InitStrct);//开启外部触发TIM_SelectOutputTrigger(TIM3,TIM_TRGOSource_Update);//使能定时器TIM_Cmd(TIM3,ENABLE);
}**
usart.c
#include "usart.h"
#include "led.h"//串口2初始化
void Usart2_Init(u32 Baud)
{GPIO_InitTypeDef GPIOA_Pin2_InitStruct;GPIO_InitTypeDef GPIOA_Pin3_InitStruct;USART_InitTypeDef USART2_InitStruct;//时钟配置 USART,收(PA3)发(PA2)端口 RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_StructInit(&GPIOA_Pin3_InitStruct);GPIOA_Pin3_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;GPIOA_Pin3_InitStruct.GPIO_Pin = GPIO_Pin_3;GPIOA_Pin3_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;//收PA3端口配置GPIO_Init(GPIOA,&GPIOA_Pin3_InitStruct);GPIO_StructInit(&GPIOA_Pin2_InitStruct);GPIOA_Pin2_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;GPIOA_Pin2_InitStruct.GPIO_Pin = GPIO_Pin_2;GPIOA_Pin2_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;//发PA2端口配置GPIO_Init(GPIOA,&GPIOA_Pin2_InitStruct);//串口初始化USART2_InitStruct.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;USART2_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;USART2_InitStruct.USART_BaudRate = Baud;USART2_InitStruct.USART_Parity = USART_Parity_No;USART2_InitStruct.USART_StopBits = USART_StopBits_1;USART2_InitStruct.USART_WordLength = USART_WordLength_8b;USART_Init(USART2,&USART2_InitStruct);//使能串口USART_Cmd(USART2,ENABLE);
}//串口1初始化
void Usart1_Init(u32 Baud)
{GPIO_InitTypeDef GPIOB_InitStruct;USART_InitTypeDef USART1_InitStruct;//时钟配置 USART1,TX:PA9,RX:PA10RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//端口配置GPIOB_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;GPIOB_InitStruct.GPIO_Pin = GPIO_Pin_9;GPIOB_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIOB_InitStruct);GPIOB_InitStruct.GPIO_Pin = GPIO_Pin_10;GPIOB_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;GPIO_Init(GPIOA,&GPIOB_InitStruct);//串口初始化USART1_InitStruct.USART_BaudRate = Baud;USART1_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;USART1_InitStruct.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;USART1_InitStruct.USART_Parity = USART_Parity_No;USART1_InitStruct.USART_StopBits = USART_StopBits_1;USART1_InitStruct.USART_WordLength = USART_WordLength_8b;USART_Init(USART1,&USART1_InitStruct);//使能串口USART_Cmd(USART1,ENABLE);
}//串口1发送数据函数发送数据
void USART1_Trans(u8 c)
{while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);USART_SendData(USART1,c);while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET);
}int fputc(int c,FILE *stream)
{USART1_Trans((u8)c);return c;
}
delay.c
#include "delay.h"u32 delay_ms = 0;//系统定时器初始化函数
void Systick_Init(u32 ticks)
{SysTick_Config(ticks);
}//延时函数
void Delay_ms(u32 ms)
{u32 ticks = delay_ms + ms;while(ticks > delay_ms);
}//系统定时器中断服务函数
void SysTick_Handler(void)
{delay_ms++;
}
错误
1、DMA_BufferSize理解错误
2、DMA_Mode_Circular未配置循环模式
3、ADC_ExternalTrigConvCmd(ADC3, ENABLE);
未开启ADC3的外部触发转换
结束
代码重在练习!
代码重在练习!
代码重在练习!
今天的分享就到此结束了,希望对你有所帮助,如果你喜欢我的分享,请点赞收藏加关注,谢谢大家!!!
相关文章:

STM32 ADC+DMA+TIM触发采样实战:避坑指南与源码解析
知识点1【TRGO的介绍】 1、TRGO的概述 TRGO:Trigger Output(触发输出),是定时器的一种功能。 它可以作为外设的启动信号,比如ADC转换,DAC输出,DMA请求等。 对于ADC来说,可以通过…...

(1-4)Java Object类、Final、注解、设计模式、抽象类、接口、内部类
目录 1. Object类 1.1 equals 1.2 toString() 2.final关键字 3.注解 4. 设计模式 4.1 单例模式 4.1.1 饿汉式 4.1.3 饿汉式 VS 懒汉式 5. 抽象类&抽象方法 6. 接口 7.内部类 7.1 成员内部类 7.2 静态内部类 7.3 方法内部类 7.4 匿名内…...

在服务器上安装AlphaFold2遇到的问题(3)_cat: /usr/include/cudnn_version.h: 没有那个文件或目录
[rootlocalhost ~]# cat /usr/include/cudnn_version.h cat: /usr/include/cudnn_version.h: 没有那个文件或目录这个错误表明系统找不到 cudnn_version.h 头文件,说明 cuDNN 的开发文件(头文件)没有正确安装。以下是完整的解决方案ÿ…...

实验-实现向量点积-RISC-V(计算机组成原理)
目录 一、实验内容 二、实验步骤 三、源代码 四、实现效果 五、实验环境 六、实验小结与思考 一、实验内容 首先,我们用一个简单的“向量点积”运算作为热身。你将拿到一个不完整的汇编代码“task2-向量点积”,我们的目标是按照C语言描述的功能&a…...
5.16本日总结
一、英语 背诵list30,复习list1 二、数学 学习14讲部分内容,订正30讲13讲题目 三、408 学习计网5.3知识点,完成5.1,5.2题目并订正 四、总结 高数对于基本定义概念类题目掌握不好,做题时往往不会下手,…...

描述性统计工具 - AxureMost 落葵网
描述性统计工具是用于汇总和分析数据,以更好地了解数据特征的工具1。以下是一些常见的描述性统计工具简介: 描述性统计工具 Excel 基本统计函数:提供了丰富的函数用于计算描述性统计量。例如,AVERAGE 函数用于计算平均值…...
【AI学习】AI大模型技术发展研究月报的生成提示词
AI大模型技术发展研究月报生成提示词 请输出AI大模型技术发展研究月报,要求如下: —————————— 任务目标 在今天({{today}})往前连续 30 天内,检索已正式公开发表的、与AI大模型(参数量 ≥10B&am…...

麒麟桌面系统文件保险箱快捷访问指南:让重要文件夹一键直达桌面!
往期文章链接:统信操作系统自定义快捷键配置音量调节功能指南 Hello,大家好啊,今天给大家带来一篇麒麟桌面操作系统上配置文件保险箱内文件夹桌面快捷方式的文章,欢迎大家分享点赞,点个在看和关注吧!在日常…...
LearnOpenGL --- 你好三角形
你好,三角形的课后练习题 文章目录 你好,三角形的课后练习题一、创建相同的两个三角形,但对它们的数据使用不同的VAO和VBO 一、创建相同的两个三角形,但对它们的数据使用不同的VAO和VBO #include <glad/glad.h> #include &…...

从硬件角度理解“Linux下一切皆文件“,详解用户级缓冲区
目录 前言 一、从硬件角度理解"Linux下一切皆文件" 从理解硬件是种“文件”到其他系统资源的抽象 二、缓冲区 1.缓冲区介绍 2.缓冲区的刷新策略 3.用户级缓冲区 这个用户级缓冲区在哪呢? 解释关于fork再加重定向“>”后数据会打印两份的原因 4.内核缓冲…...
【第76例】IPD流程实战:华为业务流程架构BPA进化的4个阶段
目录 简介 第一个阶段,业务流程架构BPA1.0 第二个阶段,业务流程架构BPA2.0 BPA3.0、4.0 作者简介 简介 不管业务是复杂还是简单,企业内外的所有事情、所有业务都最终会归于流程。 甚至是大家经常说的所谓的各种方法论,具体的落脚点还是在流程上。 比如把大象放进冰…...

游戏站的几种形式
游戏站点的主要形式:单品游戏站、游戏盒子站与单类型游戏盒子站 随着互联网的普及和游戏产业的快速发展,游戏站点作为玩家获取游戏资源和信息的重要平台,呈现出多种形式。本文将分析三种常见的游戏站点形式:单品游戏站、游戏盒子站…...
OpenCV 图像透视变换详解
在计算机视觉领域,图像的视角问题常常会影响后续的分析与处理。例如,从倾斜角度拍摄的文档、带有畸变的场景图像等,都需要通过特定的方法进行矫正。OpenCV 作为计算机视觉领域的重要库,提供了强大的图像透视变换功能,能…...
AI日报 - 2024年5月16日
🌟 今日概览 (60秒速览) ▎🤖 大模型前沿 | OpenAI推出GPT-4.1及mini版,专为编码优化;Google DeepMind发布AlphaEvolve,Gemini驱动算法发现。 GPT-4.1提升编码效率与指令遵循,AlphaEvolve在矩阵乘法、数学问…...
Ubuntu 更改 Nginx 版本
将 1.25 降为 1.18 先卸载干净 # 1. 完全卸载当前Nginx sudo apt purge nginx nginx-common nginx-core# 2. 清理残留配置 sudo apt autoremove sudo rm -rf /etc/apt/sources.list.d/nginx*.list修改仓库地址 # 添加仓库(通用稳定版仓库) codename$(…...
Python requests GET 报错:ChunkedEncodingError
问题 在实际项目中遇到的这样一个问题:通过Python GET从服务器请求url列表,因为是公司内部数据,知道大概多少条数据,所以直接一次请求500条。平稳运行了一段时间之后,突然某天这个GET请求报错了… response request…...
RabbitMQ是什么?应用场景有哪些?
RabbitMQ 是一款开源的消息代理中间件,基于 AMQP(高级消息队列协议)实现,用于在分布式系统中进行异步通信和消息传递。它通过将消息的发送者和接收者解耦,提高了系统的可扩展性、可靠性和灵活性。 核心特点 多协议支持:不仅支持 AMQP,还兼容 STOMP、MQTT 等多种消息协议…...

打造智能化军工软件工厂,破解版本管理难题
在数字化浪潮席卷全球的当下,军工行业正经历着前所未有的软件工业化转型。作为这一进程的核心支撑,软件工厂模式正在重塑军工领域的研发体系。然而,传统版本管理方式已难以适应现代军工软件研发的复杂需求,成为制约行业发展的关键…...

SpringbBoot nginx代理获取用户真实IP
为了演示多级代理场景,我们分配了以下服务器资源: 10.1.9.98:充当客户端10.0.3.137:一级代理10.0.4.105:二级代理10.0.4.129:三级代理10.0.4.120:服务器端 各级代理配置 以下是各级代理的基本配…...

allure报告自定义logo和名称
根据pytest框架,做自动化测试的时候,选择的是allure测试报告,这个报告是目前所有报告中功能最强大最好用的测试报告之一 我们在使用这个测试报告的时候,怎么样去把allure的logo和名称替换成自己公司或者自己的logo呢?…...
鸿蒙OSUniApp 实现的地图定位与导航功能#三方框架 #Uniapp
UniApp 实现的地图定位与导航功能 随着移动互联网的发展,地图定位与导航功能已成为众多应用的标配。本文将详细介绍如何在 UniApp 框架下实现地图定位与导航功能,并探讨如何适配鸿蒙系统,助力开发者打造更加流畅的地图体验。 前言 最近在做一…...

【AI论文】对抗性后期训练快速文本到音频生成
摘要:文本到音频系统虽然性能不断提高,但在推理时速度很慢,因此对于许多创意应用来说,它们的延迟是不切实际的。 我们提出了对抗相对对比(ARC)后训练,这是第一个不基于蒸馏的扩散/流模型的对抗加…...
Android 中 显示 PDF 文件内容(AndroidPdfViewer 库)
PDFView 是一个用于在 Android 应用中显示 PDF 文档的库。它提供了丰富的功能和灵活的配置选项,使得开发者能够轻松地在应用中嵌入 PDF 阅读器。 一、 添加依赖 在模块的 build.gradle 文件中添加以下依赖: // pdfimplementation("com.github.bar…...
Linux 软件包|服务管理
rpm 指令备注rpm -qa查看已安装软件,可以结合grep过滤查找rpm -e firefox卸载firefoxrpm -ivh firefox-115.12.0-1.el7.centos.i686.rpm安装gcc(只能离线安装) yum 能够从指定的服务器自动下载 RPM 包并且安装 指令备注yum list列出所有可…...

测试工程师如何学会Kubernetes(k8s)容器知识
Kubernetes(K8s)作为云原生时代的关键技术之一,对于运维工程师、开发工程师以及测试工程师来说,都是一门需要掌握的重要技术。作为一名软件测试工程师,学习Kubernetes是一个有助于提升自动化测试、容器化测试以及云原生应用测试能力的重要过程…...

遥感图像露天矿区检测数据集VOC+YOLO格式1542张1类别
数据集格式:Pascal VOC格式YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):1542 标注数量(xml文件个数):1542 标注数量(txt文件个数):1542 …...

每日Prompt:迷你 3D 建筑
提示词 3D Q版迷你风格,一个充满奇趣的迷你星巴克咖啡馆,外观就像一个巨大的外带咖啡杯,还有盖子和吸管。建筑共两层,大大的玻璃窗清晰地展示出内部温馨而精致的设计:木质的家具、温暖的灯光以及忙碌的咖啡师们。街道…...
解决 Sarspace 处理陆探一号辐射定标地理编码出现条纹问题:DEM 下载篇
在使用 Sarspace 处理陆探一号辐射定标地理编码时,结果出现条纹现象,经排查确定是 DEM(数字高程模型)下载存在问题。以下为利用自动下载功能解决该问题的过程。 问题排查 陆探一号数据处理中,条纹现象的出现往往与多种…...

el-breadcrumb 面包屑第一项后面怎么写没有分隔符
<el-breadcrumb separator"/"><el-breadcrumb-item>当前位置:</el-breadcrumb-item><el-breadcrumb-item :to"{ path: / }">首页</el-breadcrumb-item><el-breadcrumb-item><a href"/">活…...
Free2AI解锁教育新可能:LLM+RAG 技术驱动智能学习的关键路径
一、引言 在科技飞速发展的当下,人工智能技术正深刻地改变着各个领域,教育领域也不例外。大型语言模型(LLM)和检索增强生成(RAG)技术的兴起,为教育与智能学习带来了前所未有的机遇。LLM 凭借其强大的语言理解和生成能力,能够与用户进行自然流畅的对话,仿佛一位知识…...