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

第七十六章:CH32V103应用教程——USART-数据收发(带缓冲区)

[复制链接]

  离线 

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

    [LV.4]

    发表于 2021-5-1 15:31:52 | 显示全部楼层 |阅读模式

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

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

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

    本章教程主要在第三章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、下载验证

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

    CH32V CH573单片机芯片-第七十六章:CH32V103应用教程——USART-数据收发(带缓冲区)risc-v单片机中文社区(1)

    75、USART-数据收发(带缓冲区)1和2.rar
    CH32V CH573单片机芯片-第七十六章:CH32V103应用教程——USART-数据收发(带缓冲区)risc-v单片机中文社区(2) 75、USART-数据收发(带缓冲区)1.rar (447.85 KB, 下载次数: 21)
    CH32V CH573单片机芯片-第七十六章:CH32V103应用教程——USART-数据收发(带缓冲区)risc-v单片机中文社区(3) 75、USART-数据收发(带缓冲区)2.rar (439.11 KB, 下载次数: 22)
    链接:https://pan.baidu.com/s/1Wq0thtE2zm52V9K_A_gtVw
    提取码:ga0g
    复制这段内容后打开百度网盘手机App,操作更方便哦





    上一篇:第七十五章:CH32V103应用教程——舵机控制
    下一篇:第七十七章:CH32V103应用教程——USART-指令控制LED灯
    RISCV作者优文
    全球首家只专注于RISC-V单片机行业应用的中文网站
    回复

    使用道具 举报

    高级模式
    B Color Image Link Quote Code Smilies

    本版积分规则

    关闭

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



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

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

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