STM32中串口DMA实验里,为啥选择通道4(选择串口的发送端,而不是接收端)?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了STM32中串口DMA实验里,为啥选择通道4(选择串口的发送端,而不是接收端)?相关的知识,希望对你有一定的参考价值。

STM32中串口DMA实验里,为什么选择通道4(选择串口的发送端,而不是接收端)?另外,DMA请求只能是外设发起吗?如果是存储器到存储器模式的话,该选通道几呢?

你说的应该是串口1的吧。看下面这张图就明白了,TX和RX对应不同的通道。(STM32F103中文教程及参考手册.pdf,第99页)

选什么通道就看用的是什么外设;存储器到存储器没用过不懂(如果有答案求私信)。

追问

那为什么不选USART1_RX,即通道5呢?我想问的就是 要实现串口1DMA,为什么选的是USART1_TX而不是USART1_RX?

追答

那是它只让在发送时使用DMA,没有使用DMA接收。

参考技术A DMA 每个都有映射,比如stm32f103 usart1 tx在DMA1 channel 4,RX在DMA1 channel 5,具体可以参照下http://blog.csdn.net/xiaoxiaopengbo/article/details/77472093这篇文章

CH559L单片机DMA方式在手工选择通道模式下AD采样数据串口输出

【CH559L单片机】DMA方式在手工选择通道模式下AD采样数据串口输出


✨本篇属于上一篇背景下的细分篇。

  • ADC数据串口打印:

👉通过一个10K可调定位器,连接到P11端口,一端接3.3V一端接GND.

🧾有人会有疑惑,前面都实现了USB CDC虚拟串口信息输出,为什么还要占用一个串口来作为调试数据信息的输出,这不仅浪费了硬件资源,也将调试配置增添了成本。其实尝试了将其整合到一块,调试时,发现在通过DMA方式ADC采用时打印不出数据。时有时无,整得令人无语,后面看了一下USB CDC相关的驱动代码好像也用到了DMA资源,可能是这个方面的原因导致在使用USB CDC虚拟串口输出 ADC数据时有时无的情况。

🌿DMA ADC采样方式:手工选择通道模式自动交替通道模式

在官方给出的参考案例当中,已经给出了 自动交替通道模式下的使用案例,这里就不在表述了,注意在使用官方案例的时,在串口打印的数据问题,这里也不在表述了,具体细节问题请看前面那篇文章。

🍃手工选择通道模式代码

  • 📌案例来源WCH官网地址:https://www.wch.cn/products/CH559.html
  • ADCAutoDMA.C修改而来。
  • 手工选择通道,配置方法如下:

ADC_CTRL &= ~bADC_CHANN_MOD0; //	手工选择通道
//ADC采样有8个通道,在P1口,通道n 分别ADC_CHANN对应 (0x01 << n) ,如通道0是0x01,通道1是0x02依次类推	
ADC_CHANN =0x01 << 1;//手工指定通道1
//ADC_CHANN = 0x02;	//同上
  • 📢采用数据说明:

📗每次采样6次,抛弃第一次和最后一次采样的数据,将第2次到第5次的数据求和,再取平均值。数据没有像官方案例那样存进Xdata区。

  • 🎈ADC采样引脚P11

/********************************** (C) COPYRIGHT *******************************
* File Name          : ADCAutoDMA.C
* Author             : WCH
* Version            : V1.3
* Date               : 2016/6/24
* Description        : CH559的ADC自动DMA方式采样操作,手工选择通道模式,采样位数设置为11位 ,采样引脚P11             				   
*******************************************************************************/
#include "..\\DEBUG.C"                                                          //调试信息打印
#include "..\\DEBUG.H"
#include <string.h>
#define ADCCount 6
UINT16X	ADCbuf[ ADCCount ];// _at_ 0x0040 ; //存储ADC采样数据
UINT8 Flag;

#pragma  NOAREGS

/*******************************************************************************
* Function Name  : InitADCInterrupt()
* Description    : ADC中断初始化
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
void InitADCInterrupt()

//  ADC_SETUP |= bADC_IE_FIFO_OV;                                             //使能FIFO溢出中断
//  ADC_SETUP |= bADC_IE_AIN7_LOW;                                            //使能AIN7低电平中断
//  ADC_SETUP |= bADC_IE_ACT;                                                 //ADC完成中断
    IE_ADC = 1;                                                               //使能ADC中断


/*******************************************************************************
* Function Name  : InitADCAuto()
* Description    : ADC自动采样初始化
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
void InitADCAuto()

    P1_IE = 0x00;  //关闭P1口其他数据功能,如果只用部分通道,可根据需要设定,否则影响IO正常使用
    ADC_SETUP |= bADC_POWER_EN;    //ADC电源使能
	
    ADC_CK_SE |= (MASK_ADC_CK_SE & 0x0C);  //设置12分频
    ADC_CTRL &= ~MASK_ADC_CYCLE;
    ADC_CTRL |= 0x0C;//设置ADC自动采样周期
	
    ADC_CTRL &= ~bADC_CHANN_MOD1;   //1101 1111                                    
//    ADC_CTRL |= bADC_CHANN_MOD0; //0001 0000自动通道选择通道0和通道1

	ADC_CTRL &= ~bADC_CHANN_MOD0; //	手工选择通道
	//ADC采样有8个通道,在P1口,通道n 分别ADC_CHANN对应 (0x01 << n) ,如通道0是0x01,通道1是0x02依次类推
	ADC_CHANN =0x01 << 1;//手工指定通道1
//	ADC_CHANN = 0x02;//手工指定通道1
	
	ADC_EX_SW |= bADC_RESOLUTION;//采样位数11bit:2048
	//	ADC_EX_SW &= ~bADC_RESOLUTION; //采样位数10bit:1024
    mDelayuS(100);  //确保ADC正常启动	


/*******************************************************************************
* Function Name  : InitADCDMA(UINT16 addr,UINT8 num)
* Description    : ADC的DMA初始化
* Input          : UINT16 addr,DMA起始地址
                   UINT8 num,DMA剩余计数
* Output         : None
* Return         : None
*******************************************************************************/
void InitADCDMA( )

    ADC_DMA = ADCbuf; //设置DMA起始地址
    ADC_DMA_CN = ADCCount; //设置DMA剩余计数
    ADC_SETUP |= bADC_DMA_EN;//使能DMA和中断


/*******************************************************************************
* Function Name  : ADCInterrupt(void)
* Description    : ADC 中断服务程序
*******************************************************************************/
void ADCInterrupt( void ) interrupt INT_NO_ADC using 1   //ADC中断服务程序,使用寄存器组1

  	printf("%02X  ",(UINT16)ADC_STAT);  //DC 11011100  f8 11111000
    if(ADC_STAT & bADC_IF_DMA_END)  //DMA完成中断
    
    	Flag = 1;
    	ADC_STAT |= bADC_IF_DMA_END; //清中断
			printf("DMA完成中断 \\n");  
    
    if(ADC_STAT & bADC_IF_FIFO_OV)  //ADC的FIFO溢出中断
    
    	ADC_STAT |= bADC_IF_FIFO_OV; //清中断
    
    if(ADC_STAT & bADC_IF_AIN7_LOW) //AIN7低电平中断
    
    	ADC_STAT |= bADC_IF_AIN7_LOW;//清中断
    
    if(ADC_STAT & bADC_IF_ACT) //ADC完成中断
    
    	ADC_STAT |= bADC_IF_ACT; //清中断
			printf("ADC完成中断 \\n");  
    


void main( ) 

    UINT8 i;
	UINT32X total=0;
    mDelaymS(30);//上电延时,等待内部晶振稳定,必加 
  CfgFsys( );//CH559时钟选择配置
    mInitSTDIO( );//串口0,可以用于调试
    printf("start ...\\n"); 
    Flag = 0;
//ADC采样有8个通道,在P1口,通道n 分别ADC_CHANN对应 (0x01 << n) ,如通道0是0x01,通道1是0x02依次类推	
    InitADCInterrupt(); //ADC自动采样初始化
    InitADCAuto();//ADC中断初始化
    InitADCDMA();

    EA = 1;//开启全局中断
    while(1)
     
      if(Flag)
	    
				ADC_SETUP &= ~bADC_DMA_EN;//关闭DMA和中断
	    	for( i=1;i<5;i++)
	    	

	    	    printf("%hu  ",ADCbuf[i]);//初次和最后一次采样值建议丢弃	
					total +=ADCbuf[i];					
	    	
				total =total/4;
				printf("ADC_value=:%lu  \\n",total);//初次和最后一次采样值建议丢弃				
//				memset(ADCbuf, 0, sizeof(ADCbuf));//清空数组
				total=0;
	    	Flag = 0;
				ADC_DMA = ADCbuf; //设置DMA起始地址
				ADC_DMA_CN=ADCCount;
				ADC_SETUP |= bADC_DMA_EN;//使能DMA和中断

//				InitADCDMA( );
	    
			 mDelaymS(1000);
			
    

时钟配置函数: CfgFsys( );//CH559时钟选择配置

我这里采用的是外部时钟源,根据自己的开发板自行选择。这里的代码是在官方的代码基础上调整过的。

/*******************************************************************************
* Function Name  : CfgFsys( )
* Description    : CH559时钟选择和配置函数,默认使用内部晶振12MHz,如果定义了FREQ_SYS可以
                   根据PLL_CFG和CLOCK_CFG配置得到,公式如下:
                   Fsys = (Fosc * ( PLL_CFG & MASK_PLL_MULT ))/(CLOCK_CFG & MASK_SYS_CK_DIV);
                   具体时钟需要自己配置
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/ 
void	CfgFsys( )  

#if OSC_EN_XT
	SAFE_MOD = 0x55;                                                           //开启安全模式
  SAFE_MOD = 0xAA;                                                 
	CLOCK_CFG |= bOSC_EN_XT;                                                   //使能外部晶振
    mDelaymS(10);
	SAFE_MOD = 0x55;                                                           //开启安全模式
  SAFE_MOD = 0xAA;   
	CLOCK_CFG &= ~bOSC_EN_INT;
	SAFE_MOD = 0x00; 

#else
	SAFE_MOD = 0x55;                                                           //开启安全模式
  SAFE_MOD = 0xAA;                                                 
 	CLOCK_CFG &= ~MASK_SYS_CK_DIV;
//  CLOCK_CFG |= 6;                                                            //配置系统时钟48MHz
//  CLOCK_CFG |= 8;                                                            //配置系统时钟36MHz
//  CLOCK_CFG |= 10;                                                           //配置系统时钟28.8MHz
  CLOCK_CFG |= 12;                                                           //配置系统时钟24MHz
//  CLOCK_CFG |= 16;                                                           //配置系统时钟18MHz  
/*56MHz
// 	CLOCK_CFG &= ~MASK_SYS_CK_DIV;
//  CLOCK_CFG |= 6;                                                            //配置系统时钟56MHz    
    PLL_CFG = 0xFC;
*/	
    SAFE_MOD = 0xFF;                                                           //关闭安全模式  
//  如果修改主频,要同时修改FREQ_SYS,否则会造成延时函数不准
#endif 


以上是关于STM32中串口DMA实验里,为啥选择通道4(选择串口的发送端,而不是接收端)?的主要内容,如果未能解决你的问题,请参考以下文章

STM32为啥不能实现485和DMA的数据收发?

stm32 嵌入式开发 中ADC独立多通道DMA传输数据实验总结

STM32的HAL库中的DMA_FLAG_TCIF3_7等几个宏定义的含义

STM32的usart的DMA方式发送 一个数 ,程序怎么写?

stm32dma串口没有收到包头

STM32 DMA使用总结2