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

第六十一章:CH32V103应用教程——USART-单线半双工

[复制链接]

  离线 

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

    [LV.4]

    发表于 2021-4-30 18:05:14 | 显示全部楼层 |阅读模式

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

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

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

    本章教程主要通过UART2和USART3进行单线半双工模式数据收发。

    1、USART简介及相关函数介绍


    半双工模式支持使用单个引脚(只使用TX引脚)来接收和发送,TX引脚和 RX引脚在芯片内部连接。
    单线半双工模式通过设置USART控制寄存器3(USARTx_CTLR3)的HDSEL位选择。在这个模式里,下面的位必须保持清零状态:

    ● USARTx_CTLR2寄存器的LINEN和CLKEN位
    ● USARTx_CTLR3寄存器的SCEN和IREN位
    USART可以配置成遵循单线半双工协议。在单线半双工模式下,TX和RX引脚在芯片内部互连。通过使用控制位”HALF DUPLEX SEL”(USARTx_CTLR3寄存器中的HDSEL位)选择半双工和全双工通信。
    当HDSEL位置1时,即选择半双工通信方式,在这种方式下:
    ● RX不再被使用
    ● 当没有数据传输时,TX总是被释放。因此,它在空闲状态的或接收状态时表现为一个标准I/O口。这就意味该I/O在不被USART驱动时,必须配置成悬空输入(或开漏的输出高)。

    除此以外,通信与正常USART模式类似。由软件来管理线上的冲突(例如通过使用一个中央仲裁器)。特别的是,发送从不会被硬件所阻碍。当USART控制寄存器 1(USARTx_CTLR1)的TE位被置1时,只要数据一写到数据寄存器上,发送就继续。
    关于CH32V103 USART具体信息,可参考CH32V103应用手册。USART标准库函数在第三章节已介绍,在此不再赘述。


    2、硬件设计

    本章教程主要通过UART2和USART3进行单线半双工模式数据收发。将开发板USART2与USART3连接起来即可,具体连接方式如下:

    硬件连线:PA2 —— PB10  
       

    3、软件设计

    本章教程主要通过UART2和USART3进行单线半双工模式数据收发,具体程序如下:
    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. void USARTx_CFG(void);
    11. TestStatus Buffercmp(uint8_t* Buf1, uint8_t* Buf2, uint16_t BufLength);

    12. #endif
    复制代码
    usart.h文件主要进行相关定义和函数声明;
    usart.c文件
    1. #include "usart.h"

    2. /*******************************************************************************
    3. * Function Name  : USARTx_CFG
    4. * Description    : Initializes the USART2 & USART3 peripheral.
    5. * Input          : None
    6. * Return         : None
    7. *******************************************************************************/
    8. void USARTx_CFG(void)
    9. {
    10.   GPIO_InitTypeDef  GPIO_InitStructure;
    11.   USART_InitTypeDef USART_InitStructure;

    12.   RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2|RCC_APB1Periph_USART3, ENABLE);
    13.   RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_GPIOB , ENABLE);

    14.   //因为配置为单线半双工通信,因此此处只需对USART2和USART3的TX引脚进行配置,RX引脚不再被使用
    15.   /* USART2 TX-->A.2 */
    16.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;                /* Only Configure TX Pin */
    17.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    18.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    19.   GPIO_Init(GPIOA, &GPIO_InitStructure);

    20.   /* USART3 TX-->B.10 */
    21.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;               /* Only Configure TX Pin */
    22.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    23.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    24.   GPIO_Init(GPIOB, &GPIO_InitStructure);

    25.   USART_InitStructure.USART_BaudRate = 115200;                 //设置串口波特率为115200
    26.   USART_InitStructure.USART_WordLength = USART_WordLength_8b;  //字长为8位数据格式
    27.   USART_InitStructure.USART_StopBits = USART_StopBits_1;       //1个停止位
    28.   USART_InitStructure.USART_Parity = USART_Parity_No;          //无奇偶校验位
    29.   USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件流控制
    30.   USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; //发送和接收模式

    31.   USART_Init(USART2, &USART_InitStructure);                    //初始化串口
    32.   USART_Init(USART3, &USART_InitStructure);
    33.   USART_Cmd(USART2, ENABLE);
    34.   USART_Cmd(USART3, ENABLE);

    35.   //使能开启半双工通信,开启半双工模式的方式是对控制寄存器 3(R16_USARTx_CTLR3)的 HDSEL 位置位
    36.   USART_HalfDuplexCmd(USART2, ENABLE);
    37.   USART_HalfDuplexCmd(USART3, ENABLE);

    38. }

    39. /*******************************************************************************
    40. * Function Name  : Buffercmp
    41. * Description    : Compares two buffers
    42. * Input          : Buf1,Buf2:buffers to be compared
    43. *                  BufferLength: buffer's length
    44. * Return         : PASSED: Buf1 identical to Buf2
    45. *                  FAILED: Buf1 differs from Buf2
    46. *******************************************************************************/
    47. TestStatus Buffercmp(uint8_t* Buf1, uint8_t* Buf2, uint16_t BufLength)
    48. {
    49.   while(BufLength--)
    50.   {
    51.     if(*Buf1 != *Buf2)
    52.     {
    53.       return FAILED;
    54.     }
    55.     Buf1++;
    56.     Buf2++;
    57.   }
    58.   return PASSED;
    59. }
    复制代码
    usart.c文件主要包括2个函数:USARTx_CFG函数、Buffercmp函数。USARTx_CFG函数主要进行串口初始化配置;Buffercmp函数主要进行发送数据和接收数据的比较。

    main.c文件
    1. /********************************** (C) COPYRIGHT *******************************
    2. * File Name          : main.c
    3. * Author             : WCH
    4. * Version            : V1.0.0
    5. * Date               : 2020/04/30
    6. * Description        : Main program body.
    7. *******************************************************************************/

    8. /*
    9. *@Note
    10. 单线半双工模式,Master/Slave 模式收发例程:
    11. Master:USART2_Tx(PA2)
    12. Slave :USART3_Tx(PB10)

    13. 本例程演示 UART2 和 USART3 单线半双工模式数据收发。
    14. 注:
    15.      硬件连线:PA2 —— PB10

    16. */

    17. #include "debug.h"
    18. #include "usart.h"

    19. /* Global define */
    20. #define TxSize1   (size(TxBuffer1))
    21. #define TxSize2   (size(TxBuffer2))
    22. #define size(a)   (sizeof(a) / sizeof(*(a)))

    23. /* Global Variable */
    24. u8 TxBuffer1[] = "*Buffer1 Send from USART2 to USART3 using HalfDuplex Mode!";     /* Send by UART2 */
    25. u8 TxBuffer2[] = "#Buffer2 Send from USART3 to USART2 using HalfDuplex Mode!";     /* Send by UART3 */
    26. u8 RxBuffer1[TxSize1]={0};                                                         /* USART2 Using */
    27. u8 RxBuffer2[TxSize2]={0};                                                         /* USART3 Using  */

    28. u8 TxCnt1 = 0, RxCnt1 = 0;
    29. u8 TxCnt2 = 0, RxCnt2 = 0;

    30. TestStatus TransferStatus1 = FAILED;
    31. TestStatus TransferStatus2 = FAILED;

    32. /*******************************************************************************
    33. * Function Name  : main
    34. * Description    : Main program.
    35. * Input          : None
    36. * Return         : None
    37. *******************************************************************************/
    38. int main(void)
    39. {
    40.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    41.     Delay_Init();
    42.     USART_Printf_Init(115200);
    43.     printf("SystemClk:%d\r\n",SystemCoreClock);
    44.     printf("USART HalfDuplex TEST\r\n");

    45.     USARTx_CFG();                                                   /* USART2 & USART3 INIT */

    46.     while(TxCnt2<TxSize2)                                           /* USART3--->USART2 */
    47.     {
    48.         while(USART_GetFlagStatus(USART3, USART_FLAG_TXE) == RESET) /* waiting for sending finish */
    49.         {
    50.         }
    51.         USART_SendData(USART3, TxBuffer2[TxCnt2++]);         //串口3向串口2发送数据

    52.         while(USART_GetFlagStatus(USART2, USART_FLAG_RXNE) == RESET)
    53.         {
    54.         }
    55.         RxBuffer1[RxCnt1++] = USART_ReceiveData(USART2);     //串口3接收来自串口2的数据
    56.     }

    57.     while(TxCnt1<TxSize1)                                           /* USART2--->USART3 */
    58.     {
    59.         while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET) /* waiting for sending finish */
    60.         {
    61.         }
    62.         USART_SendData(USART2, TxBuffer1[TxCnt1++]);         //串口2向串口3发送数据

    63.         while(USART_GetFlagStatus(USART3, USART_FLAG_RXNE) == RESET)
    64.         {
    65.         }
    66.         RxBuffer2[RxCnt2++] = USART_ReceiveData(USART3);     //串口2接收来自串口3的数据
    67.     }

    68.     //将两个缓冲区数据进行比较验证单线半双工通信是否有误
    69.     TransferStatus1=Buffercmp(TxBuffer1,RxBuffer2,TxSize1);
    70.     TransferStatus2=Buffercmp(TxBuffer2,RxBuffer1,TxSize2);

    71.     //根据比较值输出相应值,并打印输出缓冲区的值
    72.     if(TransferStatus1&&TransferStatus2)
    73.     {
    74.       printf("\r\nSend Success!\r\n");
    75.     }
    76.     else
    77.     {
    78.       printf("\r\nSend Fail!\r\n");
    79.     }
    80.     printf("TxBuffer1---->RxBuffer2     TxBuffer2---->RxBuffer1\r\n");
    81.     printf("TxBuffer1:%s\r\n",TxBuffer1);
    82.     printf("RxBuffer1:%s\r\n",RxBuffer1);
    83.     printf("TxBuffer2:%s\r\n",TxBuffer2);
    84.     printf("RxBuffer2:%s\r\n",RxBuffer2);

    85.     while(1)
    86.     {
    87.     }
    88. }

    复制代码
    main.c文件主要进行数据发送和接收,同时打印输出发送数据和接收数据。


    4、下载验证

    将编译好的程序下载到开发版并复位,串口打印如下:

    CH32V CH573单片机芯片-第六十一章:CH32V103应用教程——USART-单线半双工risc-v单片机中文社区(1)

    60、USART-单线半双工通信.rar
    CH32V CH573单片机芯片-第六十一章:CH32V103应用教程——USART-单线半双工risc-v单片机中文社区(2) 60、USART-单线半双工通信.rar (475.01 KB, 下载次数: 11)
    链接:https://pan.baidu.com/s/1_ZaHVYiU66wYnQlYLkGGIw
    提取码:dct7
    复制这段内容后打开百度网盘手机App,操作更方便哦







    上一篇:第六十章:CH32V103应用教程——TIM-DMA
    下一篇:第六十二章:CH32V103应用教程——USART-硬件流控制
    RISCV作者优文
    全球首家只专注于RISC-V单片机行业应用的中文网站
    回复

    使用道具 举报

    高级模式
    B Color Image Link Quote Code Smilies

    本版积分规则

    关闭

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



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

    GMT+8, 2025-1-10 22:36 , Processed in 0.324672 second(s), 48 queries .

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