STM32单片机芯片与内部33 ADC 单通道连续DMA

embedded/2024/12/21 13:40:12/

目录

一、ADC DMA配置——标准库

1、ADC配置

2、DMA配置

二、ADC DMA配置——HAL库

1、ADC配置

2、DMA配置

三、用户侧

1、DMA开关

(1)、标准库

(2)、HAL库

2、DMA乒乓

(1)、标准库

(2)、HAL库


上文提到了当转换速度较高的时候需要由DMA进行搬运。

一、ADC DMA配置——标准库

1、ADC配置

        可以看到ADC配置几乎不用变。

	// 使能ADC DMA 请求ADC_DMACmd(ADCx, ENABLE);

2、DMA配置

        最重要的是源地址、目的地址、传输大小。如下配置为将每次ADC的数据从DR源地址搬运到ADC_ConvertedValue变量,因为只有一个大小长度,因此设定为1,大小设置为两个字节。

__IO uint16_t ADC_ConvertedValue;DMA_InitTypeDef DMA_InitStructure;// 打开DMA时钟RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);// 打开ADC时钟ADC_APBxClock_FUN ( ADC_CLK, ENABLE );// 复位DMA控制器DMA_DeInit(ADC_DMA_CHANNEL);// 配置 DMA 初始化结构体// 外设基址为:ADC 数据寄存器地址DMA_InitStructure.DMA_PeripheralBaseAddr = ( uint32_t ) ( & ( ADCx->DR ) );// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADC_ConvertedValue;// 数据源来自外设DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;// 缓冲区大小为1,缓冲区的大小应该等于存储器的大小DMA_InitStructure.DMA_BufferSize = 1;// 外设寄存器只有一个,地址不用递增DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;// 存储器地址固定DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable; // 外设数据大小为半字,即两个字节DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;// 存储器数据大小也为半字,跟外设数据大小相同DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;// 循环传输模式DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;	// DMA 传输通道优先级为高,当使用一个DMA通道时,优先级设置不影响DMA_InitStructure.DMA_Priority = DMA_Priority_High;// 禁止存储器到存储器模式,因为是从外设到存储器DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;// 初始化DMADMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);// 使能 DMA 通道DMA_Cmd(ADC_DMA_CHANNEL , ENABLE);

        有人会好奇,DR寄存器32位,为什么定义的16位,不会丢失什么嘛,前面数据手册介绍过。高16位用于双ADC,单ADC仅用到低16位,且一般右端对齐,则为低12位有效。

二、ADC DMA配置——HAL库

1、ADC配置

        同样不修改配置。

    HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t*)&ADC_ConvertedValue, 1);

2、DMA配置

        可以看到源地址和目的地址和长度最重要的被放在了HAL_ADC_Start_DMA中,这样可以实现不修改初始化的情况下,进行地址、长度的修改。

DMA_HandleTypeDef hdma_adcx;
__IO uint16_t ADC_ConvertedValue;// 开启DMA时钟RHEOSTAT_ADC_DMA_CLK_ENABLE();// 数据传输通道hdma_adcx.Instance = RHEOSTAT_ADC_DMA_STREAM;hdma_adcx.Init.Direction=DMA_PERIPH_TO_MEMORY;;            //存储器到外设hdma_adcx.Init.PeriphInc=DMA_PINC_DISABLE;                 //外设非增量模式hdma_adcx.Init.MemInc=DMA_MINC_ENABLE;                     //存储器增量模式 hdma_adcx.Init.PeriphDataAlignment=DMA_PDATAALIGN_HALFWORD;//外设数据长度:16位hdma_adcx.Init.MemDataAlignment=DMA_MDATAALIGN_HALFWORD;   //存储器数据长度:16位hdma_adcx.Init.Mode= DMA_CIRCULAR;                         //外设普通模式hdma_adcx.Init.Priority=DMA_PDATAALIGN_HALFWORD;               //中等优先级//初始化DMA流,流相当于一个大的管道,管道里面有很多通道HAL_DMA_Init(&hdma_adcx); __HAL_LINKDMA( &ADC_Handle,DMA_Handle,hdma_adcx);

三、用户侧

还是前面的问题,如果需要处理1000个点,该怎么办?

        如果不需要前1000和后1000连续,则可以进行DMA的开关或ADC的开关,如果要求连续则开启DMA乒乓切换。

1、DMA开关

        说明不需要DMA的连续转换,而是传输1000个点则停止,处理后再进行一次DMA传输。

(1)、标准库

        需要修改为单次的缓冲区大小、单次传输模式。

define max_size 1000
__IO uint16_t ADC_ConvertedValue[max_size ];// 存储器地址,实际上就是一个内部SRAM的数组DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADC_ConvertedValue;// 缓冲区大小为1,缓冲区的大小应该等于存储器的大小DMA_InitStructure.DMA_BufferSize = max_size ;// 单次传输模式DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;	// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue;// 存储器地址递增DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; 

        DMA中断赋值flag,主函数用户检测到flag=1说明单次1000个点的数据好了,就可以进行处理,然后再次开启DMA和ADC。

        DMA进入中断,关闭ADC转换并将数据转换结果置1。当然也可以直接在主函数检测DMA的传输完成TCIF。

// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除 DMA 中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 关闭 ADC 转换ADC_SoftwareStartConvCmd(ADCx, DISABLE);flag=1;}
}

        主函数检测到flag置位1后,说明可以进行数据处理,处理完成后,就可以重启DMA和ADC转换了。

if(flag==1)
{data_process();flag=0;// 重新启动 DMADMA_Cmd(ADC_DMA_CHANNEL, ENABLE);// 重新启动 ADCADC_SoftwareStartConvCmd(ADCx, ENABLE);
}

(2)、HAL库

        需要修改为单次的缓冲区大小、单次传输模式。

define max_size 1000
__IO uint16_t ADC_ConvertedValue[max_size ];hdma_adcx.Init.MemInc=DMA_MINC_ENABLE;                     //存储器增量模式 hdma_adcx.Init.Mode= DMA_NORMAL;                         //外设普通模式HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t*)&ADC_ConvertedValue, max_size );

有StartDMA,自然也有StopDMA,在中断服务函数直接执行即可。

HAL_ADC_Stop_DMA(&ADC_Handle)

2、DMA乒乓

        每次中断后修改目的地址,并开启新的中断,只需要修改中断服务函数即可。

(1)、标准库

__IO uint16_t ADC_ConvertedValue[2][max_size];  // 定义两个缓冲区,双缓冲区// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue[currentBuffer];
// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 切换到下一个缓冲区currentBuffer = (currentBuffer + 1) % 2;  // 切换到另一个缓冲区// 重新配置DMA传输目标地址DMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);DMA_Cmd(ADC_DMA_CHANNEL, ENABLE); // 重新启动DMA}
}

        在主函数中判断,如果currentBuffer为1,说明当前在向第二部分写入此时可以处理第一部分,如果为0,则说明在向第一部分写入此时可以处理第二部分。

(2)、HAL库

        可以看到,得力于源地址和目的地址和长度最重要的被放在了HAL_ADC_Start_DMA中,可以很方便实现。

// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 切换到下一个缓冲区currentBuffer = (currentBuffer + 1) % 2;  // 切换到另一个缓冲区HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t)ADC_ConvertedValue[currentBuffer], max_size );}
}

        在主函数中判断,如果currentBuffer为1,说明当前在向第二部分写入此时可以处理第一部分,如果为0,则说明在向第一部分写入此时可以处理第二部分。


http://www.ppmy.cn/embedded/147542.html

相关文章

uniapp scroll-view 不生效排查

排查路径&#xff1a; 开启 refresher-enabled 属性使用竖向滚动时&#xff0c;需要给 <scroll-view> 一个固定高度&#xff0c;通过 css 设置 height&#xff1b;使用横向滚动时&#xff0c;需要给 <scroll-view> 添加 white-space: nowrap; 样式父元素不要设置 …

spring学习(XML中定义与配置bean(超详细)。IoC与DI入门spring案例)

目录 一、配置文件(XML)中定义与配置bean。 &#xff08;1&#xff09;bean的基础配置。&#xff08;id、class&#xff09; &#xff08;2&#xff09;bean的别名配置。 1、基本介绍。 2、demo测试。 3、NoSuchBeanDefinitionException&#xff01; &#xff08;3&#xff09;…

最大子数组和 最大子数组和(有长度限制) 最大m段子数组和

1.最大子数组和 可以用dp求解,只需要维护一个sum表示目前为止的最大子数组,遇到num[i]时若sum > 0则sum可以继续作为子数组更新sum为sum num[i]否则sum更新为num[i] 2.最大子数组和(有长度限制) 135. 最大子序和 - AcWing题库 要求长度不超过m的子数组,首先求出原数组的…

基于注意力机制的faster-rcnn算法的目标检测(源码+pytorch框架)

需要完整代码和数据集请私信或评论 网络架构设计 基于注意力机制的R-CNN网络架构在传统Faster R-CNN基础上进行了创新性改进,特别融入了卷积注意力模块(CBAM),旨在提升模型对关键特征的捕获能力和整体检测性能。这种设计巧妙地结合了注意力机制的优势,有效增强了模型对目标…

python实现word转html

目录 使用mammoth库 使用spire.doc库 使用mammoth库 mammoth库支持将word转为HTML和markdown格式的文件。 import mammothdef word_html(word_file):html_save_name fr{word_file.split(.)[0]}.htmlwith open(word_file, rb) as f:data mammoth.convert_to_html(f)with o…

数据挖掘与机器学习(part 9) 规则挖掘Rules Mining关联规则(Association Rules) Apriori算法

基于规则的分类器&#xff1a;Classification using rule based classifier 互斥规则&#xff08;Mutually exclusive rules&#xff09;&#xff1a; 分类器包含互斥规则&#xff0c;如果这些规则彼此独立。 每条记录最多被一条规则覆盖。 穷尽规则&#xff08;Exhaustive …

VarifocalLoss在Yolov8中的应用

调用VFL Loss 在ultralytics/utils/loss.py可以发现v8实现了VarifocalLoss&#xff0c;但是好像和原论文有点不一样&#xff0c;这里有待考证原文地址&#xff1a;论文在cls损失处 # Cls lossloss[1] self.varifocal_loss(pred_scores, target_scores, target_labels) / targ…

XML基础学习

参考文章链接: XML基础学习 在w3school看到了XML的教程,想到以前工作学习中也接触到了XML,但只是简单搜索了解了下,没有认真去学习XML的基础,所以现在认真看下其基础部分,并写篇博客作为笔记记录下。 XML 简介 XML 被设计用来传输和存储数据。 什么是 XML? XML 指可…