[RISC-V MCU 应用开发] 第七十六章、CH32V103应用教程——USART-数据收发(带缓冲区)

[复制链接]
 楼主| RISCVLAR 发表于 2021-3-3 09:59 | 显示全部楼层 |阅读模式
CH32V103应用教程——USART-数据收发(带缓冲区)

本章教程主要在第三章USART原有数据收发的基础上增加了缓冲区,提高了数据传输效率,同时有效提高数据传输的准确性。

1、USART简介及相关函数介绍
关于USART相关介绍,在前面第三章以及第六十一章到第六十七章已经进行过相关介绍,在此不再赘述。

2、硬件设计
由于CH32V103系列MCU的串口1在debug文件中被用于调试打印,因此本次教程使用串口2进行收发验证。由CH32V103数据手册可知,串口2对应引脚为PA2和PA3引脚,PA2为USART2_TX,PA3为USART2_RX。使用杜邦线将WCH-Link模块与CH32V103开发板串口2连接起来,连接方式如下:
  • WCH-Link模块RX引脚与CH32V103开发板PA2引脚连接;
  • WCH-Link模块TX引脚与CH32V103开发板PA3引脚连接。

3软件设计
本章教程主要在第三章基础上进行,增加了缓冲区,提高了传输效率和准确性,具体程序如下:
usart.h文件
  1. #ifndef __USART_H
  2. #define __USART_H

  3. #include "ch32v10x_conf.h"

  4. #define buffer_len 256

  5. void USARTx_Init(void);
  6. void USARTx_SendByte(USART_TypeDef* pUSARTx, uint8_t data);
  7. void USARTx_SendStr(USART_TypeDef* pUSARTx, char *str);
  8. void USART2_IRQHandler(void);

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

  1. #include "usart.h"

  2. void USART2_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));

  3. u8 USART_Rbuffer_Num = 0;
  4. u8 USART_Tbuffer_Num = 0;
  5. u8 USART_Rbuffer[buffer_len];//接收缓冲区数组

  6. void USARTx_Init(void)
  7. {
  8.     GPIO_InitTypeDef   GPIO_InitStructure;
  9.     USART_InitTypeDef  USART_InitStructure;
  10.     NVIC_InitTypeDef   NVIC_InitStructure;

  11.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
  12.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);

  13.     /* USART2 TX-->A.2   RX-->A.3 */
  14.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  15.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  16.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;              //设置PA2为复用推挽输出
  17.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  18.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  19.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;        //设置PA3为浮空输入
  20.     GPIO_Init(GPIOA, &GPIO_InitStructure);

  21.     USART_InitStructure.USART_BaudRate = 115200;
  22.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  23.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  24.     USART_InitStructure.USART_Parity = USART_Parity_No;
  25.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  26.     USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
  27.     USART_Init(USART2, &USART_InitStructure);

  28.     NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
  29.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;      //抢占优先级为1
  30.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;           //子优先级为1
  31.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;              //IRQ通道使能
  32.     NVIC_Init(&NVIC_InitStructure);                              //中断优先级初始化

  33.     USART_ITConfig(USART2,USART_IT_RXNE,ENABLE);

  34.     USART_Cmd(USART2,ENABLE);
  35. }

  36. void USARTx_SendByte(USART_TypeDef* pUSARTx, uint8_t data)
  37. {
  38.     USART_SendData(pUSARTx, data);
  39.     while(USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
  40. }

  41. void USARTx_SendStr(USART_TypeDef* pUSARTx, char *str)
  42. {
  43.     uint8_t i = 0;
  44.     do
  45.     {
  46.        USARTx_SendByte(pUSARTx, *(str+i));
  47.        i++;
  48.     }while(*(str+i) != '\0');
  49.     while(USART_GetFlagStatus(pUSARTx, USART_FLAG_TC) == RESET);
  50. }

  51. void USART2_IRQHandler(void)
  52. {
  53.     if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) //中断产生
  54.     {
  55.         USART_ClearITPendingBit(USART2,USART_IT_RXNE);                //清除中断标志

  56.         USART_Rbuffer[USART_Rbuffer_Num] = USART_ReceiveData(USART2); //接收数据

  57.         USART_Rbuffer_Num++;
  58.     }


  59.     if(USART_GetFlagStatus(USART2,USART_FLAG_ORE) == SET) //溢出
  60.     {
  61.         USART_ClearFlag(USART2,USART_FLAG_ORE);  //清标志

  62.         USART_ReceiveData(USART2);               //读DR
  63.     }


  64.     for(USART_Tbuffer_Num=0;USART_Tbuffer_Num < USART_Rbuffer_Num;USART_Tbuffer_Num++)
  65.     {
  66.         USARTx_SendByte(USART2,USART_Rbuffer[USART_Tbuffer_Num]); //发送数据
  67.     }

  68.     USART_Rbuffer_Num = 0; //初始化
  69.     USART_Tbuffer_Num = 0;
  70. }
usart.c文件相较于第三章,主要增加了一个缓冲区数据,同时在中断服务函数中针对数据发送和接收进行了相关改动,将接收数据先放在缓冲区数组然后发送出来。
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("This is printf example\r\n");

  8.         USARTx_Init();

  9.         USARTx_SendStr(USART2, "This is a test data.\n");

  10.         while(1)
  11.         {

  12.         }
  13. }
main.c文件主要进行函数初始化。

4下载验证
将编译好的程序下载到开发板并复位,打开串口调试助手,发送数据,可以看到数据被接收并发送返回显示,如下所示:
图片1.png

75、USART-数据收发(带缓冲区)1.rar

447.85 KB, 下载次数: 255

75、USART-数据收发(带缓冲区)2.rar

439.11 KB, 下载次数: 276

jerow 发表于 2021-3-11 19:28 | 显示全部楼层
他的串口带dma缓冲区吗,如果有DMA的话就更完美了。
gejigeji521 发表于 2021-3-13 14:57 | 显示全部楼层
不是实时的?
您需要登录后才可以回帖 登录 | 注册

本版积分规则

133

主题

296

帖子

45

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