电子说
在实际的开发中需要使用各式各样的传感器,这些传感器可以采集周围的环境信息,比如温度、湿度、气压、光照等,这些信息本质上是物理的模拟信号,计算机处理信息时对数字信号比较敏感,因此把模拟信号转换为数字信号是有必要的。ADC就起到了这样的作用,本节将对ADC功能原理以及ADC数据采集时涉及到的DMA技术进行详解。
量程:能测量的电压范围,一般来收量程越大越好
分辨率:ADC的分辨率通常以输出二进制数的位数表示,位数越多,分辨率越高,精度越大,外部微小的变化都可以被显示。一般来说分辨率越高,转化时间越长。
转化时间:模拟输入电压在允许的最大变化范围内,从转换开始到获得稳定的数字量输出所需要的时间称为转换时间,希望时间越短越好。
STM32F40X有3个ADC,每个可配置 12 位、10 位、8 位或 6 位分辨率
每个ADC 有16个外部通道。另外还有两个内部 ADC源(直接测量内部温度和电压信号) 和 V BAT 通道(测量电池的剩余电量)挂在 ADC1上
这些通道的 A/D 转换可以单次、连续、扫描或间断模式执行。
ADC具有独立模式(单个ADC工作)、双重模式(两个ADC配合工作完成采样)和三重模式(三个ADC配合工作完成采样),对于不同 AD转换要求几乎都有合适的模式可选
ADC 的结果可以左对齐或右对齐方式存储在 16 位数据寄存器中。采集到的数据精度最大为12位,不够16位,因此,需要选择合适的存储方式。
1、电压输入范围:ADC 输入范围为:V REF- ≤ V IN ≤ V REF+ 。由 V REF- 、V REF+ 、V DDA 、V SSA 、这四个外部引脚决定。我们在设计原理图的时候一般把 V SSA 和 V REF- 接地,把 V REF+ 和 V DDA 接 3.3V,得到ADC 的输入电压范围为:03.3V。如果我们想让输入的电压范围变宽,去到可以测试负电压或者更高的正电压,我们可以在外部加一个电压调理电路(线性的转换关系,如根据一定的转换关系将-10V10V转换为03.3V),把需要转换的电压抬升或者降压到 03.3V,这样 ADC 就可以测量了
2、输入通道:每个ADC 有16个外部通道。特别是ADC1还有两个内部 ADC 源 和 V BAT 通道挂在 ADC1上,V BAT 通道也属于内部通道,两个内部 ADC 源没有对应外部管脚,直接测量内部信息。一个连接内部温度传感器,主要测量CPU以及芯片内部的一些关键电路的温度。另一个连接VREFINT,测量电源模块的电压值,V BAT 通道用来测量备用电池的电池容量。
3、转换顺序
a.规则通道组:顾名思意,规则通道就是很规矩的意思,我们平时一般使用的就是这个通道。相当于正常运行的程序,最多16个通道,在使用过程中按顺序进行转换,按顺序采集完一个通道进行转换后再采集下一个通道进行转换。规则通道和它的转换顺序在ADC_SQRx寄存器中选择,规则组转换的总数应写入ADC_SQR1寄存器的L[3:0]中。
ADC像这样的规则序列寄存器有3个,每个规则序列寄存器都能设定第n次转换的具体通道,共有16个外部通道,最多设置16次。如上图,规则序列寄存器3(ADC_SQR3)可以设定第一次到第六次顺序转换的通道,规则序列寄存器2(ADC_SQR2)可以设定第七次到第十二次顺序转换的通道,规则序列寄存器1(ADC_SQR1)可以设定第十三次到第十六次顺序转换的通道,且ADC_SQR1寄存器的L[3:0]中(20位到23位)用来设定要顺序转换的通道个数。例:有如下转换顺序要求,有4个通道,第一次转换通道2,第二次转换通道3,第三次转换通道4,第四次转换通道1,则ADC_SQR1寄存器的L[3:0]的值位0100,ADC_SQR3中SQ1的值设为2,SQ2的值设为3,SQ3的值设为4,SQ4的值设为1。具体的操作在Cube MX中可以配置
b.注入通道组:注入,可以理解为插入,插队的意思,是一种不安分的通道。相当于中断。最多4个通道。把某一通道配置位注入通道时,它的优先级就比规则通道高,暂停规则通道组的工作,优先转换注入通道组采集到的信号。注入组和它的转换顺序在ADC_JSQR寄存器中选择。注入组里转化的总数应写入ADC_JSQR寄存器的L[1:0]中。该通道组的配置方法于规则通道组相同。
4、转换时间:ADC 输入时钟 ADC_CLK 由 PCLK2 经过分频产生,最大值是 36MHz,即最大工作时钟不能超过36MHz,典型值为30MHz。对于 STM32F407我们一般设置PCLK2=HCLK/2=84MHz,最少需要4分频。所以程序一般使用 4分频或者 6分频。
ADC 的总转换时间Tconv = 采样时间 + 12个周期(转换时间),采样时间最少不小于3个ADC_CLK,转换时间一般就是12个ADC_CLK,如果对时间没有太过苛刻的要求,应当把采样时间设置的稍长一些,这样得到的结果才会更加准确。最小采样时间: T =3 + 112= 15个周期=0.42us(ADC时钟=36MHz下得到)。
5、数据寄存器:规则数据寄存器 ADC_DR
ADC_DR只有一个,是一个32 位的寄存器,只有低 16 位有效并且只是用于独立模式存放转换完成数据。因为 ADC 的最大精度是 12 位,ADC_DR 是16 位有效,这样允许 ADC存放数据时候选择左对齐或者右对齐,具体是以哪一种方式存放,由 ADC_CR2的 11 位 ALIGN 设置。对于规则通道组来说,所有的规则通道都使用这一个数据寄存器,实际采样点时候,如果开启了多个通道,每转换完成一个通道要及时把数据从寄存器中取走。如果不及时取走,下一次转换完成的数据就会覆盖原数据。而对于注入通道组来说,每一个注入通道都对应一个数据寄存器。
使用 DMA:由于规则通道组只有一个数据寄存器,因此,对于多个规则通道的转换,使用 DMA 非常有帮助。这样可以避免丢失在下一次写入之前还未被读出的 ADC_DR 寄存器中的数据。在使能 DMA 模式的情况下(ADC_CR2 寄存器中的 DMA 位置 1),每完成规则通道组中的一个通道转换后,都会生成一个 DMA 请求。
6、中断
a.转换结束中断:规则通道和注入通道的数据转换结束后,都可以产生中断。通道转换完成后,ADC 状态寄存器 ADC_SR的EOC位会置1,可以不断轮询此位来判断是否触发中断,也可以使能中断,完成后就根据此位的状态触发中断。此中断的使用是最多的,可以在此中断处理程序中获取我们想要采集的数据。
b.模拟看门狗中断:当被 ADC 转换的模拟电压低于低阈值或者高于高阈值时,就会产生中断
c.溢出中断:如果发生 DMA传输数据丢失,会置位 ADC 状态寄存器 ADC_SR的 OVR位,如果同时使能了溢出中断,那在转换结束后会产生一个溢出中断。
d.DMA 请求:规则和注入通道转换结束后,除了产生中断外,还可以产生 DMA请求,把转换好的数据直接存储在内存里面。
7、触发源
a.软件触发:ADC转换可以由ADC 控制寄存器 2: ADC_CR2的 ADON这个位来控制,写 1的时候开始转换,写 0 的时候停止转换。此方法使用最多
b.外部事件触发:触发包括内部定时器触发和外部 IO触发。触发源有很多,具体选择哪一种触发源,由 ADC控制寄存器ADC_CR2的 EXTSEL[2:0]和 JEXTSEL[2:0]位来控制
单次转换模式:ADC每次工作只转换一次,至于这一次转换几个通道并不关心,如果想要再次转换,必须重新启动ADC。
连续转换模式:在连续转换模式下,ADC结束一个转换后立即启动一个新的转换,直到用户想要停止为止。(仅适用于规则通道)。
扫描模式:该模式主要针对多个通道进行转换。例如使用单次转换模式又开启扫描模式,从第1个通道开始扫描,扫描第n个通道结束后不再扫描;如果使用单次转换模式而不开启扫描模式,这多个通道很有可能只转换第1个通道;使用连续转换模式又开启扫描模式,从第1个通道开始扫描,扫描第n个通道结束后再从第1个通道开始扫描,循环往复。
不连续采样模式:有多个通道需要转换,此模式下一次只转换其中的几个,这个值n需要用户自己设定,假设有0、1、2、3、6、7、9、10这几个通道需要转换,n设置为3,则
实验:利用ADC采集FSM4中电位器的数据
说明:电位器本质上就是一个滑动变阻器,通过调节电阻就可以输出不同的电压值,电压值通过PC3进入ADC的输入通道。
步骤:
1.配置RCC,ADC依赖于时钟APB2
2.配置PC3为ADC1的通道13
3.配置ADC1
4.使能中断,转换完成触发中断
5.编写代码
//main.c
uint32_t adc_value = 0;//全局变量外部才可以使用
int main(){
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_ADC1_Init();//ADC1初始化配置,仍然采用结构体元素赋值的形式
MX_USART1_UART_Init();
while(){
/***方法一:轮询采集ADC的值,也就是轮询判断EOC是否置位***/
//置位说明转换完成就可以读值
//启动ADC1,软件触发方式,将对应的启动位置位
HAL_ADC_Start(&hadc1);
//HAL_ADC_PollForConversion(&hadc1, 100)该函数内部循环判断
//EOC是否置位,没有置位就一直等待,直到超时
if( HAL_ADC_PollForConversion(&hadc1, 100) == HAL_OK ){
adc_value = HAL_ADC_GetValue(&hadc1);//读取数据寄存器里的值
printf("adc_value = %dn",adc_value);
}
HAL_Delay(1000);//每过1s钟重新启动ADC
/***方法二:转换完成后触发中断,在中断中接收数值***/
HAL_ADC_Start_IT(&hadc1);//启动ADC1并使能中断
HAL_Delay(1000);//每过1s钟重新启动ADC
}
}
//adc.c
//重写转换结束中断回调函数,这里对应方法二
extern uint32_t adc_value;//全局变量
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) {
if(hadc- >Instance == ADC1) {
adc_value = HAL_ADC_GetValue(&hadc1);//读取数据寄存器里的值
printf("adc_value = %dn",adc_value);
}
}
实验要求:利用ADC采集电位器及STM32内部温度传感器的数据。
说明:采集内部温度传感器使用的是内部专用通道,选择通道16或者通道18均可采集内部温度,需要注意的是,实际中多路通道的采集一般使用DMA来进行数据的搬移,因为规则通道的数据寄存器只有一个,如果不及时取走数据,上一次采集的数据可能被覆盖,DMA的知识将在后期进行讲解,因此本节仍采用轮询的方式查询标志位,及时将数据取出。
步骤:
1.配置时钟RCC
2.配置PC3位ADC1的通道13,配置内部温度传感器通道.
3.配置ADC
4.编写代码
//mian.c
int main(){
uint32_t adc_value = 0;
uint8_t i;
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_ADC1_Init();
MX_USART1_UART_Init();
while(){
//启动ADC1,软件触发方式,将对应的启动位置位
HAL_ADC_Start(&hadc1);
for(i=0;i< 2;i++){ //使用了几个通道就应该轮询几次
//首先采集外部通道电位器的数值,第二次采集内部温度传感器的值
//HAL_ADC_PollForConversion(&hadc1, 100)该函数内部循环判断
//EOC是否置位,没有置位就一直等待,直到超时
if( HAL_ADC_PollForConversion(&hadc1, 100) == HAL_OK ){
adc_value = HAL_ADC_GetValue(&hadc1);//读取数据寄存器里的值
printf("adc_value = %dn",adc_value);
}
}
HAL_Delay(1000);//每过1s钟重新启动ADC
}
}
DMA专门用于数据搬运,在开发中经常会遇到外设与内存、内存与内存进行大量的数据交互的情况,这些情况一般都是由CPU进行控制的,如果数据交互非常频繁且数据量巨大,那么CPU就会被大量占用用于数据交互而无法完成其他工作,为了把CPU从这种单一繁琐的工作中释放出来,很多微控制器中都有专门用于数据交互的器件,就是DMA。
DMA,全称为:Direct Memory Access,即直接存储器访问。DMA 传输方式无需 CPU 直接控制传输,也没有中断处理方式那样保留现场和恢复现场的过程,通过硬件为RAM 与 I/O 设备开辟一条直接传送数据的通路,无需CPU的参与,CPU将这条通路的控制权释放,转交由DMA控制器控制,能使 CPU 的效率大为提高。DMA控制器只用于数据交互,无其他功能。
①外设通道:STM32F4xx 系列资源丰富,具有两个 DMA 控制器,同时外设繁多,为实现正常传输,DMA需要通道选择控制。每个 DMA控制器具有 8个数据流,每个数据流对应 8个外设请求。
外设通道选择要解决的主要问题是决定哪一个外设作为该数据流的源地址或者目标地址。宏观上一个数据流的所有通道可以同时工作,微观上通道异步前进进行工作,或者根据通道优先级进行工作
DMA1外设通道表如下:
DMA2外设通道表如下:
每个数据流都与一个 DMA 请求相关联,此 DMA 请求可以从 8 个可能的通道请求中选出。此选择由 DMA_SxCR 寄存器中的 CHSEL[2:0] 位控制,3位用来控制8个通道的选择。
②数据流仲裁:一个 DMA控制器对应 8个数据流,数据流包含要传输数据的源地址、目标地址、数据长度等等信息。如果我们需要同时使用同一个DMA 控制器(DMA1 或 DMA2)多个外设请求时,那必然需要同时使用多个数据流,那究竟哪一个数据流具有优先传输的权利呢?这就需要仲裁器来管理判断
每个数据流的优先级都可以通过配置 DMA_SxCR寄存器 PL[1:0]位,设置为非常高、高、中和低四个级别,由于8个数据流只有4个优先级,如果两个或以上数据流软件设置优先级一样,则他们优先级取决于数据流编号,编号越低越具有优先权。
③数据FIFO:每个数据流都有一个独立的 4 字 FIFO(先进先出存储器缓冲区), DMA传输具有 FIFO模式和直接模式。
a.直接模式:每个 DMA 请求会立即启动对存储器的传输。当在直接模式(禁止 FIFO)下将 DMA请求配置为以存储器到外设模式传输数据时,DMA 仅会将一个数据从存储器预加载到内部 FIFO,从而确保一旦外设触发 DMA 请求时则立即传输数据。
b.FIFO模式:可通过控制寄存器 DMA_SxFCR 的 FTH[1:0]位来设置FIFO 阈值级别为 FIFO 大小的 1/4、1/2 或 3/4。如果数据存储量达到阈值级别时,FIFO 内容将传输到目标中。如果对传输速度要求不是很苛刻,建议使用此模式、直接模式传输效率太低,一次传输的数据量少而工作量大。
④存储器端口:DMA 控制器提供两个 AHB 主端口,分别为AHB 存储器端口(用于连接存储器)和 AHB 外设端口(用于连接外设)。存储器端口用来连接外部存储器,实现内存与外存的数据交互。
⑤外设端口:用于连接外设,实现内存与外设的数据交互.
注意:DMA1 控制器 AHB 外设端口与 DMA2 控制器的情况不同,DMA1 控制器 AHB 外设端口没有连接到总线矩阵,因此,仅 DMA2 数据流能够执行存储器到存储器的传输,DMA1 不能实现存储器到存储器传输。
1.传输模式 :DMA1 只有外设到存储器和存储器到外设两种模式,DMA2 除前面两种传输模式外还支持存储器到存储器的传输模式。模式选择可以通过 DMA_SxCR 寄存器的 DIR[1:0]位(方向选择,规定谁是源,谁是目的)进行控制。
2.DMA传输模式的源、目的、长度:
a.DMA_SxPAR寄存器: 设置外设寄存器地址
b.DMA_SxM0AR寄存器:设置存储器地址
c.DMA_SxCR 寄存器 :DIR[1:0]位配置数据的传输方向
d.DMA_CNDTRx 寄存器:写入需要传输的数据量, (0 到 65535),注意是数据的一个量,单位并不是字节
e.DMA_SxCR 寄存器中的 PSIZE 和 MSIZE 位:设置源和目的的数据宽度,两边的位宽尽量保持一致,位宽只一次传输多大的位宽(如一次传输2字节,位宽16位),如果不能保持位宽,很有可能造成数据丢失,应当开启FIFO模式,保证数据不丢失。
3.DMA增量设置:
a.根据设置 DMA_SxCR 寄存器中 PINC 和 MINC 位的状态,外设和存储器指针在每次传输后可以自动向后递增或保持常量。
b.当设置为增量模式时,下一个要传输的地址将是前一个地址加上增量值,增量值取决与所选的数据宽度为1字节(8位) 、2 字节(16位)或 4字节(32位) 。如果不启动,每一次取很有可能只取首地址对应数据,而不会向后取值。
c.一般情况下外设并不需要递增,每一次都是同一个地址,要不要配置外设的增量模式依具体情况而定。
4.DMA循环模式:
a.循环模式用于处理循环缓冲区和连续的数据传输 ( 如 ADC 的连续循环扫描模式 ) 。可以使用 DMA_SxCR 寄存器中的 CIRC 位使能此特性。
b.当启动了循环模式,一组的数据传输完成时,计数寄存器将会自动地被恢复成配置该通道时设置的初值, DMA 操作将会继续进行。
5.DMA单次传输和突发传输:
a.DMA传输类型有单次(Single)传输和突发(Burst)传输。DMA 控制器可以产生单次传输或 4 个、8 个和 16 个节拍的增量突发传输。一个节拍对应着用户配置的一个数据宽度,一般突发传输要配合FIFO来使用,突发传输的数据量不可能超过FIFO的大小
b.突发大小通过软件针对两个 AHB 端口独立配置,配置时使用 DMA_SxCR 寄存器中的MBURST[1:0] 和 PBURST[1:0] 位。
c.为确保数据一致性,形成突发的每一组传输都不可分割:在突发传输序列期间,AHB 传输会锁定,并且 AHB 总线矩阵的仲裁器不解除对 DMA 主总线的授权。
6.DMA中断: 对于每个 DMA 数据流,可在达到半传输、传输完成、传输错误、FIFO 错误(上溢、下溢或 FIFO 级别错误)、直接模式错误时产生中断。这些标志位都在中断状态寄存器DMA_xISR中设置。
实验要求:利用ADC采集电位器以及芯片内部温度传感器的数据,在main函数中启动ADC转换以及DMA传输,最后在DMA完成中断中打印采集到的数据。
说明:之前使用ADC采集两路数据是使用轮询的方式判断采集,需要CPU一直去判断,消耗CPU资源,显然这并不是ADC多路采集最好的办法,ADC多路采集一般使用DMA来进行传输,规则通道组采集数据时只有一个数据寄存器ADC_DR,读取数据不及时很有可能发生数据覆盖。
采用DMA传输,每当一路数据采集完成后都会产生一个DMA请求,及时把数据寄存器中的值搬移到内存中相应的buffer中去
步骤:
1.配置RCC
2.配置PC3位ADC1的通道13,配置内部温度传感器通道
3.配置DMA
4.配置ADC
5.配置中断
5.编写代码
//main.c
uint16_t adc_value[2] = {0};//定义一个全局的接受区域,以便外部调用
int mian(){
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_DMA_Init();
MX_USART1_UART_Init();
MX_ADC1_Init();
printf("this is adc dma testn");
while(1){
//此函数既启动ADC,又使能DMA
HAL_ADC_Start_DMA(&hadc1, (uint32_t*) adc_value, 2);
//2表示传输了2个数据的量,一个量是我们设定的位宽
//内部有一个专门的寄存器来记录这个量,每传输完成一次,寄存器的值减1
//减到0就会触发DMA的传输完成中断
HAL_Delay(1000); //每过1s启动一次
}
}
关于DMA传输完成中断处理函数是哪一个,先将函数HAL_ADC_Start_DMA(&hadc1, (uint32_t*) adc_value, 2);追进去可以看到一段代码设置了DMA传输完成的回调函数,将这个函数赋给了一个指针,这个指针是DMA_Handle下的一个指针,DMA可以用于很多的外设,不同的外设的转换完成函数是不一样的,DMA没有一个固定的处理函数,这个函数需要在外设配置的时候自己去初始化,DMA只提供一个接口,一个传输转换完成的函数指针,不同的外设对这个结构赋不同的处理函数的指针,这里对ADC来说自己封装了一个DMA传输完成处理中断的函数,当DMA传输完成中断触发的时候自己去调用这个接口下的对应的函数。ConvCplt的意思是ConverseComplete
将void DMA2_Stream0_IRQHandler(void)这个中断处理函数入口追进去可以看到DMA有各种不同的中断处理请求,我们在传输完成中断***Transfer Complete Interrupt management***这一部分里可以看到hdma->XferCpltCallback(hdma);这样一个指针函数,但是用户无法使用这个指针函数,这个指针在HAL_ADC_Start_DMA(&hadc1, (uint32_t*) 这个函数中已经赋值,实际上这个函数最终进入了HAL_ADC_ConvCpltCallback函数,也就是ADC传输完成函数。
//adc.c
extern uint16_t adc_value[2] ;
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) {
//每一次采集完成后都会将数据传输到value中
if(hadc- >Instance == ADC1) {
printf("adc_value[0]=%d, adc_value[1]=%dn",adc_value[0],adc_value[1]);
}
}
全部0条评论
快来发表一下你的评论吧 !