串口通信实战:从寄存器操作到数据处理的完全指南
串口通信实战:从寄存器操作到数据处理的完全指南
一、串口核心寄存器详解
1.1 状态寄存器(STA/STATUS)
状态寄存器是串口通信中最重要的寄存器之一,它实时反映串口的工作状态。以下是关键位的详细说明:
-
TXE (Transmit Data Register Empty):发送数据寄存器空标志
- 1:发送数据寄存器已空,可以写入新数据
- 0:发送数据寄存器仍有数据未发送完成
- 应用场景:发送数据前检查此位,避免数据覆盖
-
TC (Transmission Complete):发送完成标志
- 1:一帧数据(包括停止位)已完全发送
- 0:数据仍在发送中
- 应用场景:确保一帧数据完全发送后再进行后续操作
-
RXNE (Receive Data Register Not Empty):接收数据寄存器非空
- 1:接收数据寄存器中有新数据可读取
- 0:接收数据寄存器为空
- 应用场景:检查是否有新数据到达
-
ORE (Overrun Error):溢出错误
- 1:新数据到来时前一个数据尚未被读取
- 0:无溢出错误
- 应用场景:错误处理时检查此标志
1.2 数据寄存器(DA/DATA)
数据寄存器是实际存储收发数据的寄存器:
- 发送过程:向数据寄存器写入数据会启动发送流程
- 接收过程:从数据寄存器读取数据会清除RXNE标志
- 特点:同一个地址对应两个物理寄存器(发送和接收)
二、关键操作:数据掩码处理(如& 0x7FFF)
2.1 为什么需要掩码处理
在串口通信中,我们经常会看到类似这样的代码:
received_data = USART1->DR & 0x7FFF;
这是因为:
- 寄存器位宽大于实际数据位:许多MCU的数据寄存器是16位的,但实际数据可能只有8位或9位
- 状态标志共用寄存器:某些芯片将状态位和数据位放在同一个寄存器中
- 去除无效位:确保只获取有效数据部分
2.2 常见掩码值解析
-
0x7FFF:用于15位数据(保留最高位为0)
- 二进制:0111 1111 1111 1111
- 应用场景:16位寄存器中取15位有效数据
-
0xFF:用于8位数据
- 二进制:0000 0000 1111 1111
- 应用场景:标准8位数据通信
-
0x1FF:用于9位数据
- 二进制:0000 0001 1111 1111
- 应用场景:9位数据模式下的通信
2.3 实际应用示例
// 接收数据时使用掩码
uint16_t receive_data(void) {while(!(USART1->SR & 0x0020)); // 等待RXNE置位return USART1->DR & 0x07FF; // 取11位有效数据
}// 发送数据时使用掩码
void send_data(uint16_t data) {while(!(USART1->SR & 0x0080)); // 等待TXE置位USART1->DR = data & 0x07FF; // 确保只发送11位数据
}
三、串口通信完整实战流程
3.1 初始化配置
void USART1_Init(uint32_t baudrate) {// 1. 使能时钟RCC->APB2ENR |= RCC_APB2ENR_USART1EN;// 2. 配置GPIOGPIOA->MODER &= ~(GPIO_MODER_MODER9 | GPIO_MODER_MODER10);GPIOA->MODER |= GPIO_MODER_MODER9_1 | GPIO_MODER_MODER10_1; // 复用功能// 3. 设置波特率USART1->BRR = SystemCoreClock / baudrate;// 4. 配置数据格式USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; // 使能USART、发送、接收// 5. 可选:使能中断USART1->CR1 |= USART_CR1_RXNEIE;NVIC_EnableIRQ(USART1_IRQn);
}
3.2 数据收发实现
查询方式收发
// 发送一个字节
void USART1_SendByte(uint8_t data) {while(!(USART1->SR & USART_SR_TXE)); // 等待发送寄存器空USART1->DR = data & 0xFF; // 写入数据寄存器while(!(USART1->SR & USART_SR_TC)); // 等待发送完成
}// 接收一个字节
uint8_t USART1_ReceiveByte(void) {while(!(USART1->SR & USART_SR_RXNE)); // 等待接收数据return USART1->DR & 0xFF; // 读取数据并掩码
}
中断方式接收
volatile uint8_t rx_buffer[256];
volatile uint16_t rx_index = 0;void USART1_IRQHandler(void) {if(USART1->SR & USART_SR_RXNE) {// 读取数据并应用掩码uint8_t data = USART1->DR & 0xFF;rx_buffer[rx_index++] = data;// 简单的缓冲区溢出保护if(rx_index >= sizeof(rx_buffer)) {rx_index = 0;}}
}
四、常见问题与调试技巧
4.1 数据接收不完整
现象:接收到的数据总是少几位或错位
解决方案:
- 检查数据掩码是否正确
- 确认通信双方的数据位长度设置一致
- 检查波特率是否准确
4.2 发送数据丢失
现象:部分发送的数据没有出现在接收端
解决方案:
- 确保每次发送前检查TXE标志
- 重要数据发送后检查TC标志
- 检查硬件连接是否稳定
4.3 溢出错误处理
现象:频繁出现ORE错误标志
解决方案:
if(USART1->SR & USART_SR_ORE) {USART1->SR &= ~USART_SR_ORE; // 清除溢出标志uint8_t dummy = USART1->DR; // 读取数据寄存器以复位状态// 可以添加错误计数或恢复逻辑
}
五、高级应用:自定义协议实现
5.1 帧头检测
#define FRAME_HEADER 0x55AAuint8_t frame_buffer[128];
uint8_t frame_pos = 0;
bool receiving_frame = false;void process_byte(uint8_t data) {static uint16_t header_temp = 0;header_temp = (header_temp << 8) | data;if(!receiving_frame) {if(header_temp == FRAME_HEADER) {receiving_frame = true;frame_pos = 0;return;}} else {frame_buffer[frame_pos++] = data;if(frame_pos >= sizeof(frame_buffer)) {// 处理完整帧process_complete_frame(frame_buffer, frame_pos);receiving_frame = false;}}
}
5.2 校验和验证
bool verify_checksum(uint8_t *data, uint8_t length) {uint8_t sum = 0;for(uint8_t i = 0; i < length - 1; i++) {sum += data[i];}return (sum == data[length-1]);
}
六、性能优化技巧
- 使用DMA传输:对于高速或大数据量传输,配置DMA可以大幅降低CPU开销
- 双缓冲技术:实现接收缓冲区的无缝切换,避免数据丢失
- 环形缓冲区:高效管理接收数据,避免频繁内存操作
- 中断优先级优化:确保串口中断及时响应,避免数据丢失
通过深入理解串口寄存器操作和数据处理技巧,结合实际项目需求,开发者可以构建稳定高效的串口通信系统。从基础的寄存器操作到高级协议实现,串口通信技术的掌握是嵌入式开发者的必备技能。