[RISC-V MCU 应用开发] 第六十七章、CH32V103应用教程——USART-DMA

[复制链接]
 楼主| RISCVLAR 发表于 2021-1-27 10:43 | 显示全部楼层 |阅读模式
本帖最后由 RISCVLAR 于 2021-1-27 10:43 编辑

CH32V103应用教程——USART-DMA

本章教程主要使用USART2和USART3通过DMA进行数据收发。

1、USART简介及相关函数介绍
USART模块支持DMA功能,可以利用DMA实现快速连续收发。当启用DMA时,USART状态寄存器(R32_USARTx_STATR)TXE被置1时,DMA就会从设定的内存空间向发送缓冲区写数据。当使用DMA接收时,每次RXNE置1后,DMA就会将接收缓冲区里的数据转移到特定的内存空间。

使用DMA进行发送
使用DMA进行发送,可以通过设置USART控制寄存器3(USARTx_CTLR3)上的DMAT位激活。当TXE位被置1时,DMA就从指定的SRAM区传送数据到USART数据寄存器(USARTx_DATAR)。为USART的发送分配一个DMA通道的步骤如下:
1. 在DMA控制寄存器上将USARTx_DATAR寄存器地址配置成DMA传输的目的地址。在每个TXE事件后,数据将被传送到这个地址。
2. 在DMA控制寄存器上将存储器地址配置成DMA传输的源地址。在每个TXE事件后,将从此存储器区读出数据并传送到USARTx_DATAR寄存器。
3. 在DMA控制寄存器中配置要传输的总的字节数。
4. 在DMA寄存器上配置通道优先级。
5. 根据应用程序的要求,配置在传输完成一半还是全部完成时产生DMA中断。
6. 在DMA寄存器上激活该通道。
当传输完成DMA控制器指定的数据量时, DMA控制器在该DMA通道的中断向量上产生一中断。
在发送模式下,当DMA传输完所有要发送的数据时,DMA控制器设置DMA 中断状态寄存器(DMA_INTFR)的TCIFx标志;监视USART状态寄存器(R32_USARTx_STATR)的TC标志可以确认USART通信是否结束,这样可以在关闭USART或进入停机模式之前避免破坏最后一次传输的数据;软件需要先等待TXE=1,再等待TC=1。

使用DMA进行接收
可以通过设置USART_CTLR3寄存器的DMAR位激活使用DMA进行接收,每次接收到一个字节,DMA控制器就就把数据从USARTx_DATAR寄存器传送到指定的SRAM区(参考DMA相关说明)。为USART的接收分配一个DMA通道的步骤如下:
1. 通过DMA控制寄存器把USARTx_DATAR寄存器地址配置成传输的源地址。在每个RXNE事件后,将从此地址读出数据并传输到存储器。
2. 通过DMA控制寄存器把存储器地址配置成传输的目的地址。在每个RXNE事件后,数据将从USARTx_DATAR传输到此存储器区。
3. 在DMA控制寄存器中配置要传输的总的字节数。
4. 在DMA寄存器上配置通道优先级。
5. 根据应用程序的要求配置在传输完成一半还是全部完成时产生DMA中断。
6. 在DMA控制寄存器上激活该通道。
当接收完成DMA控制器指定的传输量时,DMA控制器在该DMA通道的中断矢量上产生一中断。

本章教程不使用中断进行数据收发,因此不需要在程序中进行中断配置。
关于CH32V103 USART具体信息,可参考CH32V103应用手册。USART标准库函数在第三章节已介绍,在此不再赘述。

2、硬件设计
本章教程主要使用USART2和USART3通过DMA进行数据收发。将开发板USART2与USART3连接起来即可,具体连接方式如下:
硬件连线:PA2 —— PB11
      PA3 —— PB10

3软件设计
本章教程主要进行串口轮询收发模式演示,具体程序如下:
usart.h文件
  1. #ifndef __USART_H
  2. #define __USART_H

  3. #include "ch32v10x_conf.h"

  4. /* Global typedef */
  5. typedef enum
  6. {
  7.     FAILED = 0,
  8.     PASSED = !FAILED
  9. } TestStatus;

  10. /* Global define */
  11. #define TxSize1   100
  12. #define TxSize2   100

  13. extern u8 TxBuffer1[];
  14. extern u8 TxBuffer2[];
  15. extern u8 RxBuffer1[];
  16. extern u8 RxBuffer2[];

  17. void USARTx_CFG(void);
  18. void DMA_INIT(void);
  19. TestStatus Buffercmp(uint8_t* Buf1, uint8_t* Buf2, uint16_t BufLength);

  20. #endif
usart.h文件主要进行相关定义和函数声明;
usart.c文件

  1. #include "usart.h"

  2. /* Global Variable */
  3. u8 TxBuffer1[] = "*Buffer1 Send from USART2 to USART3 using DMA!";     /* Send by UART2 */
  4. u8 TxBuffer2[] = "#Buffer2 Send from USART3 to USART2 using DMA!";     /* Send by UART3 */
  5. u8 RxBuffer1[TxSize1]={0};                                             /* USART2 Using  */
  6. u8 RxBuffer2[TxSize2]={0};                                             /* USART3 Using  */

  7. /*******************************************************************************
  8. * Function Name  : USARTx_CFG
  9. * Description    : Initializes the USART2 & USART3 peripheral.
  10. * Input          : None
  11. * Return         : None
  12. *******************************************************************************/
  13. void USARTx_CFG(void)
  14. {

  15.     GPIO_InitTypeDef  GPIO_InitStructure;
  16.     USART_InitTypeDef USART_InitStructure;

  17.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2|RCC_APB1Periph_USART3, ENABLE);
  18.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_GPIOB , ENABLE);

  19.     /* USART2 TX-->A.2   RX-->A.3 */
  20.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  21.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  22.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  23.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  24.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  25.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  26.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  27.     /* USART3 TX-->B.10  RX-->B.11 */
  28.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  29.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  30.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  31.     GPIO_Init(GPIOB, &GPIO_InitStructure);
  32.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
  33.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  34.     GPIO_Init(GPIOB, &GPIO_InitStructure);

  35.     USART_InitStructure.USART_BaudRate = 115200;
  36.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  37.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  38.     USART_InitStructure.USART_Parity = USART_Parity_No;
  39.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  40.     USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;

  41.     USART_Init(USART2, &USART_InitStructure);
  42.     USART_Init(USART3, &USART_InitStructure);

  43.     DMA_Cmd(DMA1_Channel7, ENABLE);                                    /* USART2 Tx */
  44.     DMA_Cmd(DMA1_Channel6, ENABLE);                                    /* USART2 Rx */
  45.     DMA_Cmd(DMA1_Channel2, ENABLE);                                    /* USART3 Tx */
  46.     DMA_Cmd(DMA1_Channel3, ENABLE);                                    /* USART3 Rx */

  47.     USART_Cmd(USART2, ENABLE);
  48.     USART_Cmd(USART3, ENABLE);
  49. }

  50. /*******************************************************************************
  51. * Function Name  : DMA_INIT
  52. * Description    : Configures the DMA for USART2 & USART3.
  53. * Input          : None
  54. * Return         : None
  55. *******************************************************************************/
  56. void DMA_INIT(void)
  57. {
  58.     DMA_InitTypeDef DMA_InitStructure;
  59.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

  60.     DMA_DeInit(DMA1_Channel7);
  61.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART2->DATAR);  /* USART2->DATAR:0x40004404 */
  62.     DMA_InitStructure.DMA_MemoryBaseAddr = (u32)TxBuffer1;
  63.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
  64.     DMA_InitStructure.DMA_BufferSize = TxSize1;
  65.     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  66.     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  67.     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  68.     DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
  69.     DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
  70.     DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;
  71.     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
  72.     DMA_Init(DMA1_Channel7, &DMA_InitStructure);

  73.     DMA_DeInit(DMA1_Channel6);
  74.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART2->DATAR);
  75.     DMA_InitStructure.DMA_MemoryBaseAddr = (u32)RxBuffer1;
  76.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
  77.     DMA_InitStructure.DMA_BufferSize = TxSize2;
  78.     DMA_Init(DMA1_Channel6, &DMA_InitStructure);

  79.     DMA_DeInit(DMA1_Channel2);
  80.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART3->DATAR);  /* USART2->DATAR:0x40004804 */
  81.     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)TxBuffer2;
  82.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
  83.     DMA_InitStructure.DMA_BufferSize = TxSize2;
  84.     DMA_Init(DMA1_Channel2, &DMA_InitStructure);

  85.     DMA_DeInit(DMA1_Channel3);
  86.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART3->DATAR);
  87.     DMA_InitStructure.DMA_MemoryBaseAddr = (u32)RxBuffer2;
  88.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
  89.     DMA_InitStructure.DMA_BufferSize = TxSize1;
  90.     DMA_Init(DMA1_Channel3, &DMA_InitStructure);
  91. }

  92. /*******************************************************************************
  93. * Function Name  : Buffercmp
  94. * Description    : Compares two buffers
  95. * Input          : Buf1,Buf2:buffers to be compared
  96. *                  BufferLength: buffer's length
  97. * Return         : PASSED: Buf1 identical to Buf2
  98. *                  FAILED: Buf1 differs from Buf2
  99. *******************************************************************************/
  100. TestStatus Buffercmp(uint8_t* Buf1, uint8_t* Buf2, uint16_t BufLength)
  101. {
  102.   while(BufLength--)
  103.   {
  104.     if(*Buf1 != *Buf2)
  105.     {
  106.       return FAILED;
  107.     }
  108.     Buf1++;
  109.     Buf2++;
  110.   }
  111.   return PASSED;
  112. }
usart.c文件主要包括两个函数:USARTx_CFG函数、DMA_INIT函数和Buffercmp函数;USARTx_CFG函数主要进行串口2和串口3的初始化配置;DMA_INIT函数主要进行DMA初始化配置;Buffercmp函数主要进行发送数据和接收数据的比较。
main.c文件
  1. int main(void)
  2. {
  3.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
  4.     Delay_Init();
  5.     USART_Printf_Init(115200);
  6.     printf("SystemClk:%d\r\n",SystemCoreClock);

  7.     printf("USART DMA TEST\r\n");
  8.     DMA_INIT();
  9.     USARTx_CFG();                                                 /* USART2 & USART3 INIT */
  10.     USART_DMACmd(USART2,USART_DMAReq_Tx|USART_DMAReq_Rx,ENABLE);
  11.     USART_DMACmd(USART3,USART_DMAReq_Rx|USART_DMAReq_Tx,ENABLE);
  12.     while (DMA_GetFlagStatus(DMA1_FLAG_TC7) == RESET)             /* Wait until USART2 TX DMA1 Transfer Complete */
  13.     {
  14.     }
  15.     while (DMA_GetFlagStatus(DMA1_FLAG_TC6) == RESET)             /* Wait until USART2 RX DMA1 Transfer Complete */
  16.     {
  17.     }
  18.     while (DMA_GetFlagStatus(DMA1_FLAG_TC2) == RESET)             /* Wait until USART3 TX DMA1 Transfer Complete */
  19.     {
  20.     }
  21.     while (DMA_GetFlagStatus(DMA1_FLAG_TC3) == RESET)             /* Wait until USART3 RX DMA1 Transfer Complete */
  22.     {
  23.     }
  24.     TransferStatus1=Buffercmp(TxBuffer1,RxBuffer2,TxSize1);
  25.     TransferStatus2=Buffercmp(TxBuffer2,RxBuffer1,TxSize2);

  26.     if(TransferStatus1 && TransferStatus2)
  27.     {
  28.       printf("Send Success!\r\n");
  29.     }
  30.     else
  31.     {
  32.       printf("Send Fail!\r\n");
  33.     }

  34.     printf("TxBuffer1:%s\r\n",TxBuffer1);
  35.     printf("RxBuffer1:%s\r\n",RxBuffer1);
  36.     printf("TxBuffer2:%s\r\n",TxBuffer2);
  37.     printf("RxBuffer2:%s\r\n",RxBuffer2);

  38.     while(1)
  39.     {
  40.     }
  41. }
main.c文件主要进行串口2、串口3的发送和接收。并将发送数据与接收数据进行比较。

4下载验证
将编译好的程序下载到开发版并复位,串口打印如下:
图片1.png

66、USART-DMA.rar

473.68 KB, 下载次数: 177

htmlme 发表于 2021-2-21 20:30 | 显示全部楼层
期待开发一个图形配置软件了。  
xinpian101 发表于 2021-2-22 10:09 | 显示全部楼层
相当于单片机的CPU请来了一个搬运工。
答案很长吧 发表于 2021-2-23 15:56 | 显示全部楼层
无论什么使用DMA都是不错的。
logan0279 发表于 2022-12-4 14:59 | 显示全部楼层
例程的传输数据长度是固定的,如果是不定长的数据呢?
如果CH32V的芯片和HK32一样有串口超时中断就好了。
kissdb 发表于 2023-3-22 22:53 | 显示全部楼层
logan0279 发表于 2022-12-4 14:59
例程的传输数据长度是固定的,如果是不定长的数据呢?
如果CH32V的芯片和HK32一样有串口超时中断就好了。 ...

有空闲IDLE中断,
timfordlare 发表于 2023-4-12 20:20 | 显示全部楼层
如何用dma控制usart发送数据
xiaoyaodz 发表于 2023-4-12 21:01 | 显示全部楼层
如何通过DMA实现不定长USART接收  
tifmill 发表于 2023-4-12 22:16 | 显示全部楼层
串口波特率最大为多少               

评论

最大可配置9Mbps  发表于 2023-4-13 10:43
kkzz 发表于 2023-6-10 12:09 | 显示全部楼层
触发DMA的方式有很多种的。比如说usart接收到数据后可以把数据通过DMA搬移出来
yorkbarney 发表于 2023-6-10 12:33 | 显示全部楼层
spi1的dma可以和usart3的dma一起用么
primojones 发表于 2023-6-10 13:52 | 显示全部楼层
配置串口和 DMA,使能相应的时钟。

配置 DMA 通道的源地址、目的地址、传输数据量、传输模式等参数。其中,源地址为 USART 的数据寄存器地址,目的地址为数据缓冲区地址,传输数据量为每次传输的字节数,传输模式可以选择单次或循环传输。

配置 DMA 通道触发源为 USART 发送完成中断,并使能相应 DMA 通道和USART 发送中断。

在串口发送时,在中断函数中开启 DMA 传输,并将要发送的数据写入到 DMA 通道的数据缓冲区中。

等待 DMA 传输完成中断,并在中断函数中关闭 DMA 传输,清除相应中断标志位。
mikewalpole 发表于 2023-6-13 20:48 | 显示全部楼层
串口波特率最大为多少              
plsbackup 发表于 2023-6-13 22:18 | 显示全部楼层
usart的DMA方式发送 一个数 ,程序怎么写?
rosemoore 发表于 2023-6-13 22:41 | 显示全部楼层
用DMA 中断方式发送,为什么发送完一帧数据后就死机
minzisc 发表于 2023-6-13 23:12 | 显示全部楼层
usart中断标志位需要手动清除吗
mollylawrence 发表于 2023-6-14 09:24 | 显示全部楼层
1、首先初始化USART,打开DMA1的外设时钟。2、其次初始化DMA数据流,使能中断其数据。3、最后即可请求dma了。
maqianqu 发表于 2023-6-14 10:17 | 显示全部楼层
在DMA传输完成中断中是否不能重新启动DMA
burgessmaggie 发表于 2023-6-14 10:24 | 显示全部楼层
DMA+串口,要先了解什么事件会触发DMA操作。
linfelix 发表于 2023-6-14 10:33 | 显示全部楼层
CH32V103有多少个DMA控制器
您需要登录后才可以回帖 登录 | 注册

本版积分规则

133

主题

296

帖子

44

粉丝
快速回复 在线客服 返回列表 返回顶部