USART串口协议全解析

引言

        在嵌入式系统开发中,通信是至关重要的一部分。USART(通用同步 / 异步收发传输器)作为一种常见的串行通信接口,广泛应用于各种设备之间的数据传输。本文将详细介绍 USART 的工作原理、特点,并给出基于 STM32 单片机的代码实现示例。

一、USART 原理

1. 基本概念

        USART 通信通常需要两根数据线,分别是发送线(TX)和接收线(RX)。在异步通信模式下,这两根线就足以完成数据的双向传输;而在同步通信模式下,还需要额外的时钟线(CLK)来同步发送和接收端的操作。此外,如果需要进行硬件流控制,还会用到请求发送(RTS)和清除发送(CTS)两根线。

2. 以 STM32 为例的引脚说明

        以 STM32F103 系列单片机为例,不同的 USART 通道对应着不同的 GPIO 引脚。以下是 USART1 的引脚分配情况:

  • 发送引脚(TX):通常连接到 PA9 引脚。当单片机要发送数据时,数据会通过这个引脚以串行的方式发送出去。
  • 接收引脚(RX):一般连接到 PA10 引脚。外部设备发送过来的数据会通过这个引脚被单片机接收。
  • 时钟引脚(CLK,同步模式使用):如果使用同步通信模式,USART1 的时钟信号会从 PA8 引脚输出,用于同步发送和接收端的操作。
  • 请求发送引脚(RTS,硬件流控制使用):在需要硬件流控制时,RTS 信号从 PA12 引脚输出,用于向对方设备请求发送数据。
  • 清除发送引脚(CTS,硬件流控制使用):CTS 信号从 PA11 引脚输入,用于接收对方设备的允许发送信号。

3. 电平匹配问题

        在进行 USART 通信时,要特别注意发送端和接收端的电平匹配问题。常见的电平标准有 TTL(晶体管 - 晶体管逻辑)和 RS - 232 等。

  • TTL 电平:一般单片机的 USART 接口输出的是 TTL 电平,高电平通常为 3.3V 或 5V,低电平为 0V。如果通信双方都是单片机或者使用 TTL 电平的设备,那么可以直接连接。
  • RS - 232 电平:RS - 232 是一种常用的串行通信标准,其电平范围与 TTL 不同,高电平为 - 3V 到 - 15V,低电平为 + 3V 到 + 15V。如果要实现单片机与 RS - 232 设备的通信,就需要使用电平转换芯片,如 MAX232 等,将 TTL 电平转换为 RS - 232 电平,反之亦然。

4. 数据格式

USART 传输的数据通常以帧为单位,每一帧包含起始位、数据位、校验位和停止位。

  • 起始位:用于标识一帧数据的开始,通常为一个逻辑低电平。
  • 数据位:要传输的实际数据,常见的数据位长度有 5 位、6 位、7 位和 8 位。
  • 校验位:用于检测数据传输过程中是否发生错误,常见的校验方式有奇校验、偶校验和无校验。
  • 停止位:用于标识一帧数据的结束,通常为 1 位、1.5 位或 2 位逻辑高电平。

5. 波特率

        波特率是指每秒传输的二进制位数,它决定了数据传输的速度。常见的波特率有 9600、115200 等。在进行 USART 通信时,发送端和接收端的波特率必须保持一致,否则会导致数据传输错误。

二、USART 的特点

1. 灵活性高

        USART 支持同步和异步两种通信模式,这使得它能够适应不同的应用场景。例如,在与一些简单的外部设备(如温度传感器、湿度传感器等)进行通信时,可以选择异步通信模式,以简化电路设计和降低成本;而在与高速数据采集设备或其他需要精确时序控制的设备通信时,则可以选择同步通信模式,以满足高速和高精度的要求。

2. 可靠性强

        通过校验位的设置,USART 可以在一定程度上检测数据传输过程中的错误。当校验结果不一致时,接收端可以采取相应的措施,如请求重传数据,从而提高数据传输的可靠性。此外,USART 还具备硬件流控制功能,可以在数据传输过程中进行流量控制,避免数据丢失。

3. 应用广泛

        由于其简单易用、成本低等特点,USART 被广泛应用于各种嵌入式系统中。在工业自动化领域,USART 可以用于 PLC(可编程逻辑控制器)与传感器、执行器之间的通信;在智能家居领域,它可以实现智能门锁、智能插座等设备与网关之间的数据交互;在医疗设备中,USART 也可以用于数据的采集和传输。

三、基于 STM32 的 USART 代码实现

1. 初始化 USART

//USART.c


#include "stm32f10x.h"

void USART1_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;

    // 使能 GPIOA 和 USART1 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1, ENABLE);

    // 配置 USART1 Tx (PA9) 为复用推挽输出
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 配置 USART1 Rx (PA10) 为浮空输入
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // USART1 配置
    USART_InitStructure.USART_BaudRate = 115200;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    USART_Init(USART1, &USART_InitStructure);

    // 使能 USART1 接收中断
    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

    // 使能 USART1
    USART_Cmd(USART1, ENABLE);

    // 配置 NVIC
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
}

配置原理图如下:

2. 发送数据

void USART1_SendByte(uint8_t byte)
{
    while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
    USART_SendData(USART1, byte);
}

void USART1_SendString(char* str)
{
    while (*str)
    {
        USART1_SendByte(*str++);
    }
}

3. 接收数据中断处理函数

void USART1_IRQHandler(void)
{
    if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
    {
        uint8_t data = USART_ReceiveData(USART1);
        // 处理接收到的数据
        // 这里简单地将接收到的数据原样发送回去
        USART1_SendByte(data);
        USART_ClearITPendingBit(USART1, USART_IT_RXNE);
    }
}

4. 主函数

int main(void)
{
    USART1_Init();

    while (1)
    {
        USART1_SendString("Hello, USART!\r\n");
        for (int i = 0; i < 1000000; i++); // 简单延时
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值