介绍
串口(UART通用异步收发器,TTL)通讯是一种设备间的串行全双工通讯方式。由于UART是异步传输,没有传输同步时钟,为了保证数据的正确性,UART采用16倍数据波特率的时钟进行采样。因为它简便捷,因此大部分电子设备都支持该通讯方式工程师在调试设备时也经常使用该方式输出调试信息。 本文详细的介绍如何来编写一个串口收发程序,我们采用常用的收发逻辑,发送直接编写函数进行实现,而接收使用中断进行完成。接收中断使用接收到一个字节和一帧数据两种中断触发方式。
USART中断
USART 有多个中断请求事件。
之所以介绍这个USART中断请求,是因为很多人在初学阶段,对串口怎么判断串口中断的状态不太了解,所以我这里重点来介绍一下。 一般在我们开始和配置完串口中断后,进入串口中断处理程序的情况会有很多,我们也可以自己选择打开哪些串口中断情况。一般情况下,我们在接受时主要使用的中断事件标志是RXNE和IDLE。 RXNE是接收中断,每接收一个字节都会出发这个中断,也是我们用的最频繁的中断请求。 IDLE 是空闲中断,每接收完一帧数据,总线就会暂时空闲,就会触发这个中断。
串口状态
串口的状态可以通过状态寄存器 USART_SR 读取。USART_SR 的各位描述如下:
这里我们关注一下两个位,第 5、6 位 RXNE 和 TC。 RXNE(读数据寄存器非空),当该位被置 1 的时候,就是提示已经有数据被接收到了,且可以读出来了。这时候我们要做的就是尽快去读取 USART_DR,通过读 USART_DR 可以该位清零,也可以向该位写 0,直接清除。 TC(发送完成),当该位被置位的时候,表示 USART_DR 内的数据已经被发送完成了。果设置了这个位的中断,则会产生中断。该位也有两种清零方式:
- 读 USART_SR,USART_DR。
- 直接向该位写 0。
实例
需求分析
本项目主要编写一个串口收发的实例。使用STM32F103C8T6充当MCU,在PC上使用串口调试助手充当上位机。每次PC向MCU下发一帧数据, MCU每接收一个字节数据,检查一下数据中是否有指令0x23,当接收到指令0x23的时候,MCU向上位机发送“PC”。当一帧数据接收完毕后,MCU向上位机发送“Receive a frame data”.
串口初始化
串口初始化的一般步骤可以总结为如下几个步骤:
- 串口时钟使能,GPIO 时钟使能。
- 设置引脚复用器映射
- GPIO 初始化设置:要设置模式为复用功能。
- 串口参数初始化:设置波特率,字长,奇偶校验等参数。
- 开启中断并且初始化 NVIC,使能中断(如果需要开启中断才需要这个步骤)。
- 使能串口。
NVIC_InitTypeDef NVIC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输
//USART3_TX GPIOB.10
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOB, &GPIO_InitStructure)
//USART3_RX GPIOB.11初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOB, &GPIO_InitStructure)
// 配置波特率
USART_InitStructure.USART_BaudRate = bound;
// 配置 针数据字长
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(USART3, &USART_InitStructure);
// 使能串口接收中断
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);
USART_Cmd(USART3, ENABLE);
// 清除发送完成标志
NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0X06;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0X00;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
我们平时需要改的其实就是串口的一些参数配置。
- BaudRate:波特率
- WordLength;:字长
- StopBits:停止位
- Parity:奇偶校验
- Mode:收/发模式设置
- HwFlowCtl:硬件流设置
- OverSampling:过采样设置
串口发送
串口发送这里使用的非中断发送方式。
代码语言:javascript复制/*******************************************************************************
* @函数名称 USART_Send
* @函数说明 发送信息
* @输入参数 _UART:串口号
data:要发送的信息的首地址
len: 发送的长度
* @输出参数 无
* @返回参数 无
*******************************************************************************/
void USART_Send(USART_TypeDef *_UART,u8 *data, u8 len)
{
for(int i = 0; i < len; i )
{
USART_SendData(_UART, data[i]); //向串口发送数据
while(USART_GetFlagStatus(_UART, USART_FLAG_TXE) == RESET); //等待发送结束
}
}
主要使用的是USART_SendData(_UART, data)函数,USART_SendData函数是标准库中自带的函数。
- _UART:串口号
- data:发送的数据
每次发送一个字节的数据,但我们要注意,当发送多个字节的数据时,可能会造成前一个数据还没有发送完,后一个数据就已经要开始发送了,解决这个问题的方法就是使用
USART_GetFlagStatus(_UART, USART_FLAG_TXE) == RESET
判断是否发送完。
串口接收
这里串口接收使用的是中断的方式。 中断的类别在文章的最上边已经介绍过。我们在初始化时设定触发中断的类型。本文中设置的
代码语言:javascript复制 USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);
代表只有接收数据和空闲中断会触发。 在stm32f1xx_it.c中有我们的串口中断处理函数。我们将这个函数进行重构。
代码语言:javascript复制/*******************************************************************************
* @函数名称 USART3_IRQHandler
* @函数说明 串口3中断服务程序
* @输入参数 无
* @输出参数 无
* @返回参数 无
*******************************************************************************/
void USART3_IRQHandler(void)
{
u8 Res;
if(USART_GetITStatus(USART3, USART_IT_RXNE) != RESET)
{
Res = USART_ReceiveData(USART3); //读取接收到的数据
if(Res==0x23)
printf("PC");
}
else if(USART_GetITStatus(USART3, USART_IT_IDLE) != RESET) //接收中断
{
USART3->SR;//先读SR寄存器
USART3->DR; //再读DR寄存器
printf("Receive a frame data. ");
}
}
这里面的几个重点,我们来一一介绍。 首先是判断标志位,我们使用标准库中的USART_GetITStatus()函数,里面有两个参数,前者是串口号,后者是具体哪个标志位。 if(USART_GetITStatus(USART3, USART_IT_RXNE) != RESET)用来检测是否检测到有单个字节的中断。 if(USART_GetITStatus(USART3, USART_IT_IDLE) != RESET)用来检测是否有空闲中断(代表这一帧数据传输完了)。
重定向printf和scanf
还有一点需要注意的,使用 fput 和 fgetc 函数达到重定向 C 语言标准库输入输出函数必须在 MDK 的工程选项把“Use MicroLIB”勾选上, MicoroLIB 是缺省 C 库的备选库,它对标准 C 库进行了高度优化使代码更少,占用更少资源 为使用 printf、 scanf 函数需要在文件中包含 stdio.h 头文件。
代码语言:javascript复制#pragma import(__use_no_semihosting)
//标准库需要的支持函数
struct __FILE
{
int handle;
};
FILE __stdout;
//定义_sys_exit()以避免使用半主机模式
void _sys_exit(int x)
{
x = x;
}
//重定义fputc函数
int fputc(int ch, FILE *f)
{
while((USART3->SR & 0X40) == 0); //循环发送,直到发送完毕
USART3->DR = (u8) ch;
return ch;
}
效果
- PC下发:11 22 33 44
- PC下发:12 23 34 45