查看: 1787|回复: 0
收起左侧

第六十七章:CH32V103应用教程——USART-DMA

[复制链接]

  离线 

  • TA的每日心情
    慵懒
    2021-7-23 17:16
  • 签到天数: 17 天

    [LV.4]

    发表于 2021-4-30 19:29:28 | 显示全部楼层 |阅读模式

    有人预言,RISC-V或将是继Intel和Arm之后的第三大主流处理器体系。欢迎访问全球首家只专注于RISC-V单片机行业应用的中文网站

    您需要 登录 才可以下载或查看,没有帐号?立即注册

    x
    本帖最后由 草帽王子 于 2021-9-10 17:59 编辑

    本章教程主要使用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、下载验证

    将编译好的程序下载到开发版并复位,串口打印如下:
    CH32V CH573单片机芯片-第六十七章:CH32V103应用教程——USART-DMArisc-v单片机中文社区(1)

    66、USART-DMA.rar
    CH32V CH573单片机芯片-第六十七章:CH32V103应用教程——USART-DMArisc-v单片机中文社区(2) 66、USART-DMA.rar (473.68 KB, 下载次数: 19)
    链接:https://pan.baidu.com/s/1Vesn3tgfpOBgajoCtecJmQ

    提取码:retb
    复制这段内容后打开百度网盘手机App,操作更方便哦






    上一篇:第六十六章:CH32V103应用教程——USART-轮询收发模式
    下一篇:第六十八章:CH32V103应用教程——IIC-模拟IIC驱动OLED
    RISCV作者优文
    全球首家只专注于RISC-V单片机行业应用的中文网站
    回复

    使用道具 举报

    高级模式
    B Color Image Link Quote Code Smilies

    本版积分规则

    关闭

    RISC-V单片机中文网上一条 /2 下一条



    版权及免责声明|RISC-V单片机中文网 |网站地图

    GMT+8, 2025-1-10 23:16 , Processed in 0.388268 second(s), 48 queries .

    快速回复 返回顶部 返回列表