STM32F1定时器触发AD采样+DMA中断和JATG引脚PB3 PB4作为普通IO代码

本文主要是介绍STM32F1定时器触发AD采样+DMA中断和JATG引脚PB3 PB4作为普通IO代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一基于STM32定时器触发AD采样+DMA中断例程实现参考

采用ADC的定时器触发ADC转换的功能,然后使用DMA进行数据的搬运!这样只要设置好定时器的触发间隔,就能实现ADC定时采样转换的功能,使能DMA转换完成中断,这样每次转换完成就会产生中断。在DMA中断程序即可实现ADC数据的读取,不用定时器干预。

/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
#define ADC1_DR_Address    ((uint32_t)0x4001244C)/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
ADC_InitTypeDef           ADC_InitStructure;
DMA_InitTypeDef           DMA_InitStructure;
TIM_TimeBaseInitTypeDef   TIM_TimeBaseStructure;
TIM_OCInitTypeDef         TIM_OCInitStructure;
__IO uint16_t AD_ConvertedValue;

在主程序中:调用下面子程序即可;

void TIMTriggerAdc(void)
{ 
ADC_GPIO_Configuration();         //ADC IOC配置函数ADC_Config(); //ADC 配置函数ADC_DMA_Config(); //ADC DMA配置函数TIM2_Configuration(); //定时器触发采样配置函数//TIM_Cmd(TIM2, ENABLE); //最后面打开定时器使能//DMA_Cmd(DMA1_Channel1, ENABLE); //使能DMA  
while(1);
}

第一步:ADC IO配置函数

//ADC IO配置函数
void ADC_GPIO_Configuration(void) 
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1|RCC_APB2Periph_GPIOA| RCC_APB2Periph_AFIO, ENABLE);   //使能ADC和GPIOA时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;         //管脚2
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;   //模拟输入模式
GPIO_Init(GPIOA, &GPIO_InitStructure);     //GPIO组
}

第二步:ADC配置函数

//ADC配置函数
void ADC_Config(void) 
{
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //独立的转换模式 ADC_DUALMOD[3:0]=0000;
ADC_InitStructure.ADC_ScanConvMode =DISABLE; //关闭扫描模式 因为只有一个通道
ADC_InitStructure.ADC_ContinuousConvMode =DISABLE; //关闭连续转换模式 否则只要触发一次,
//后续的转换就会永不停歇(除非CONT清0),这样第一次以后的ADC,就不是由TIM2_CC2来触发了
ADC_InitStructure.ADC_ExternalTrigConv =ADC_ExternalTrigConv_T2_CC2;//定时器2触发
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//对齐方式,ADC为12位中,右对齐方式 ADC_ALIGN=0;
ADC_InitStructure.ADC_NbrOfChannel = 1; //开启通道数,1个  ADC_SQR1[23:20]=0000;
//ADC_SQR1[23:20] 设置通道数目的选择
ADC_Init(ADC1, &ADC_InitStructure);
// RCC_ADCCLKConfig(RCC_PCLK2_Div6); //配置时钟(12MHz),在RCC里面还应配置APB2=AHB时钟72MHzADC_RegularChannelConfig(ADC1, ADC_Channel_3, 1,ADC_SampleTime_1Cycles5);
//ADC_SMPR2 ADC_SMPR1 设置每个通道的采样时间 
//ADC_SQR1[19:0]DC_SQR1[29:0]DC_SQR3[29:0]  设置对应通道的转换顺序  适用于多通道采样
//ADC通道组, 第3个通道 采样顺序1,转换时间
ADC_ExternalTrigConvCmd(ADC1, ENABLE); //设置外部触发模式使能(这个“外部“其实仅仅是相//对于ADC模块的外部,ADC_DMACmd(ADC1, ENABLE); ADC_Cmd(ADC1, ENABLE);   //ADC命令,使能  ADC_ADON=1ADC_ResetCalibration(ADC1);   //重新校准while(ADC_GetResetCalibrationStatus(ADC1));   //等待重新校准完成ADC_StartCalibration(ADC1);   //开始校准  ADC_RSTCAL=1; 初始化校准寄存器while(ADC_GetCalibrationStatus(ADC1));     //等待校准完成  ADC_CAL=0; //ADC_SoftwareStartConvCmd(ADC1, ENABLE); //连续转换开始,ADC通过DMA方式不断的更新RAM区。
//ADC_SWSTART=1 开始规则转换 切记 软件触发也属于外部事件  要设置  ADC_EXTTRIG=1
  //实际上还是在STM32内部)
} 

第三步:ADC_DMA初始化配置

//ADC_DMA初始化配置
void ADC_DMA_Config(void)
{
DMA_InitTypeDef DMA_InitStructure; // 注:ADC为12位模数转换器,只有ADCConvertedValue的低12位有效
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA时钟//中断配置
NVIC_InitTypeDef NVIC_InitStructure; NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
NVIC_InitStructure.NVIC_IRQChannel =DMA1_Channel1_IRQn; 
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; 
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure); DMA_DeInit(DMA1_Channel1); //开启DMA1的第一通道 
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; //DMA对应的外设基地址
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_ConvertedValue; 
//内存存储基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //DMA的转换模式为SRC模式,由外设搬移到内存
DMA_InitStructure.DMA_BufferSize = 1; //DMA缓存大小,1个
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; 
//接收一次数据后,设备地址禁止后移
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable; //关闭接收一次数据后,目标内存地址后移
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
//定义外设数据宽度为16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; 
//DMA搬移数据尺寸,HalfWord就是为16位
DMA_InitStructure.DMA_Mode =DMA_Mode_Circular; //循环转换模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA优先级高
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //M2M模式禁用,MA传输类型,不是内存到内存
DMA_Init(DMA1_Channel1, &DMA_InitStructure); DMA_ITConfig(DMA1_Channel1,DMA_IT_TC, ENABLE); //使能传输完成中断 
DMA_ClearITPendingBit(DMA_IT_TC);                           //清除一次DMA中断标志 
DMA_Cmd(DMA1_Channel1, ENABLE);                             //使能DMA1 
}

第四步:定时器初始化配置

//定时器初始化配置
void TIM2_Configuration(void)
{ 
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; 
TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);        //使能TIM2时钟 TIM_TimeBaseStructure.TIM_Period = 999; //设置2ms一次TIM2比较的周期
TIM_TimeBaseStructure.TIM_Prescaler = 71; //系统主频72M,这里分频71,相当于1000K的定时器2时钟 
TIM_TimeBaseStructure.TIM_ClockDivision = 0x0; 
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM2, & TIM_TimeBaseStructure);TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //下面详细说明 
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;//TIM_OutputState_Disable; 
TIM_OCInitStructure.TIM_Pulse = 500; 
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low; //如果是PWM1要为Low,PWM2则为High 
TIM_OC2Init(TIM2, & TIM_OCInitStructure); TIM_Cmd(TIM2, ENABLE);                        //使能TIM2 
}

第五步:中断处理函数

float Ad_ConvertedData;
//中断处理函数
void  DMA1_Channel1_IRQHandler(void)
{
if(DMA_GetITStatus(DMA1_IT_TC1)!=RESET)
{
//自己的中断处理代码 但是记住程序不要太复杂  最好不要超过中断时间
//Ad_ConvertedData=AD_ConvertedValue/4095.0*3.3;
Ad_ConvertedData=ADC_GetConversionValue(ADC1)/4095.0*3.3;// TIM_Cmd(TIM2, DISABLE);                     //完成周波采样,停止定时器 
//    DMA_Cmd(DMA1_Channel1, DISABLE);            //完成周波采样,停止DMA DMA_ClearITPendingBit(DMA1_IT_TC1); 
}
else;
}

STM32F1的JATG引脚PB3 PB4作为普通IO例程参考

//AFIO时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable, ENABLE);
// 改变指定管脚的映射 GPIO_Remap_SWJ_Disable SWJ 完全禁用(JTAG+SW-DP)
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable , ENABLE);
// 改变指定管脚的映射 GPIO_Remap_SWJ_JTAGDisable ,JTAG-DP 禁用 + SW-DP 使能GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_4;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_Init(GPIOB, &GPIO_InitStructure);

这篇关于STM32F1定时器触发AD采样+DMA中断和JATG引脚PB3 PB4作为普通IO代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

jupyter代码块没有运行图标的解决方案

《jupyter代码块没有运行图标的解决方案》:本文主要介绍jupyter代码块没有运行图标的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录jupyter代码块没有运行图标的解决1.找到Jupyter notebook的系统配置文件2.这时候一般会搜索到

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.