ch340串口驱动_STM32F103 串口的使用方法

串口通讯简介:

串口通讯(Serial Communication)是一种设备间非常常用的串行通讯方式,因为它简单便捷,因此大部分电子设备都支持该通讯方式。

通讯结构

设备内部一般以TTL电平传输,设备之间是通过RS232/RS485电平标准传输。

两个设备或者器件要想实现串口通讯,要电平匹配才能够正常通讯。

电平标准

根据使用的电平标准不同,串口通讯可分为 RS232标准及TTL标准,具体标准如下:

4a1d15f732cf6faaff3d3ab09a1bcd7b.png

在电子电路中,模块之间常使用TTL的电平标准,但其抗干扰能力较弱,为了增加串口的通讯距离及抗干扰能力,使用RS-232电平标准在设备之间传输信息,经常使用MA3232芯片对TTL电平及RS-232电平进行相互转换。

协议层

1.数据包

串口通讯的数据包由发送设备通过自身的TXD接口传输到接收设备得RXD接口,在协议层中规定了数据包的内容,具体包括起始位、主体数据(8位或9位)、校验位以及停止位,通讯的双方必须将数据包的格式约定一致才能正常收发数据。

2.波特率

由于异步通信中没有时钟信号,所以接收双方要约定好波特率,即每秒传输的码元个数,以便对信号进行解码,常见的波特率有4800、9600、115200等。STM32中波特率的设置通过串口初始化结构体来实现。

3.起始和停止信号

数据包的首尾分别是起始位和停止位,数据包的起始信号由一个逻辑0的数据位表示,停止位信号可由0.5、1、1.5、2个逻辑1的数据位表示,双方需约定一致。STM32中起始和停止信号的设置也是通过串口初始化结构体来实现。

4.有效数据

有效数据规定了主题数据的长度,一般为8或9位,其在STM32中也是通过串口初始化结构体来实现的。

5.数据校验

在有效数据之后,有一个可选的数据校验位。由于数据通信相对更容易受到外部干扰导致传输数据出现偏差,可以在传输过程加上校验位来解决这个问题。校验方法有奇校验(odd)、偶校验(even)、 0 校验(space)、 1 校验(mark)以及无(noparity)。这些也都可以在串口初始化结构体中实现的。

串口是我们常用的一个数据传输接口,STM32F103系列单片机共有5个串口,

46f2ad6cf2fa1318ee23a6a487e73d21.png

其中1-3是通用同步/异步串行接口USART(Universal Synchronous/Asynchronous Receiver/Transmitter),

4、5是通用异步串行接口UART(Universal Asynchronous Receiver/Transmitter)。

STM32比51单片机好用的一个地方就是串口比较多,51单片机一般只有2个串口,有时不够用。

下面以USART1为例,说明一下STM32串口设置的一般步骤:

1) 串口时钟使能,GPIO 时钟使能

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

2) GPIO端口设置

设置发送和接收引脚的信息,将Tx(发送引脚)配置为推挽复用模式用来发送数据,Rx(接收引脚)配置为浮空输入模式用来接收数据。

GPIO_InitTypeDef GPIO_InitStructure;//USART1_TX   GPIOA.9GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;    //复用推挽输出GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9   //USART1_RX      GPIOA.10初始化GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10

3)Usart1 NVIC 配置(如果需要开启中断,才进行本步骤的设置)

NVIC_InitTypeDef NVIC_InitStructure;//Usart1 NVIC 配置NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;     //抢占优先级3NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;           //子优先级3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;              //IRQ通道使能NVIC_Init(&NVIC_InitStructure);    //根据指定的参数初始化NVIC寄存器//如果需要接收串口数据,则开启串口接收中断USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启串口接受中断

4) 串口参数初始化

USART_InitTypeDef结构体,内部包含串口通讯相关工作参数:

typedef struct {    uint32_t USART_BaudRate; // 波特率    uint16_t USART_WordLength; // 字长    uint16_t USART_StopBits; // 停止位    uint16_t USART_Parity; // 校验位    uint16_t USART_Mode; // USART 模式    uint16_t USART_HardwareFlowControl; // 硬件流控制} USART_InitTypeDef;
USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = bound;//串口波特率USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式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); //初始化串口1

5) 使能串口

USART_Cmd(USART1, ENABLE);                    //使能串口1

6) 编写串口发送函数

//发送一个字节void USART1_Send_Byte(u8 Data)                     {    USART_GetFlagStatus(USART1, USART_FLAG_TC);        USART_SendData(USART1,Data);    while( USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET );}//发送字符串,遇到字符串结尾标志'0'结束void USART1_Send_String(u8 *Data)                 {    while(*Data)        USART1_Send_Byte(*Data++);}//按长度发送字符串,这种方法可以发送含0x00的字符串void USART1_Send_String_By_Lens(u8 *Data, int Len){    int i;    for(i=0; i

7) 编写中断处理函数

//串口1中断服务程序,此接收的数据是以0x0D、0x0A结尾为标志的数据帧。void USART1_IRQHandler(void)                    {    u8 Res;    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)    {        USART_ClearFlag(USART1, USART_IT_RXNE);             //清除标志位                Res =USART_ReceiveData(USART1);    //读取接收到的数据                if((USART_RX_STA&0x8000)==0)//接收未完成        {            if(USART_RX_STA&0x4000)//接收到了0x0d            {                if(Res==0x0D)                    USART_RX_STA|=0x4000;                else if(Res!=0x0a)                    USART_RX_STA=0;//接收错误,重新开始                else                    USART_RX_STA|=0x8000;    //接收完成了            }            else //还没收到0X0D            {                    if(Res==0x0d)                    USART_RX_STA|=0x4000;                else                {                    USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;                    USART_RX_STA++;                    if(USART_RX_STA>(USART_REC_LEN-1))                        USART_RX_STA=0;//接收数据错误,重新开始接收                      }                     }        }     }}

接收完数据之后,在main函数中对接收到的数据进行处理。

if(USART_RX_STA&0x8000){    //得到此次接收到的数据长度,即USART_RX_BUF数组中的有效数据长度    uart1Len=USART_RX_STA&0x3f;                                   //对接收到的数据进行数据处理,接收的数据暂存在USART_RX_BUF数组中       //... ...                 USART_RX_STA=0;       memset(USART_RX_BUF, 0, sizeof(USART_RX_BUF));        //清空数组  }

串口应用:

  • 与TTL串口传感器或模块直接通讯;
  • 转为RS232与PC通讯;
  • 转为RS485与485部件的传感器或器件通讯;

USB转串口的原理图:

58756771b9478c2724ec7a22fbda044c.png

使用CH340C芯片的话,就可以省略外部晶振了,可以节省PCB布局空间;

win7系统一般选择CH340作为USB转串口驱动,Win10系统下选择CH341驱动作为USB转串口驱动;

TTL串口转RS232原理图:

698bed8e4449b580194dc6108b4e7afa.png

TTL串口转RS485原理图:

48fa399a40e931d5ff8a5c65afc5e22c.png

RS485总线一般使用时默认处于接收状态。

参考资料:

【正点原子】MiniSTM32开发板资料

喜欢请关注微信公众号:程序员小哈

有啥想玩的模块,留言给我,咱们一起玩

d27a644ed40e40f947e8d93a2ed4bd0d.png
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值