STM32F4xx开发学习_USART串口通讯

USART串口通讯

USART简介

USART(universal synchronous asynchronous receiver transmitter),通用同步异步接收发射机,是一种全双工异步通信串行通讯方式,是STM32内部集成的硬件外设,以帧格式传输数据。搭配DMA进行多缓冲配置,可进行高速数据通信。

异步通信接线图如下
USART接线图

这里给出码元传输速率 R B R_B RB和信息传输速率 R b R_b Rb的定义

  • 码元传输速率
    亦称传码率、波特率,单位时间内传输了多少个码元
  • 信息传输速率
    亦称传信率、比特率,单位时间内传递多少比特数

在M进制中,每个码元携带 log ⁡ 2 M \log_2M log2M比特的信息量,二者满足: R b = R B log ⁡ 2 M R_b = R_B \log_2M Rb=RBlog2M,二进制中两者相等。
USART框图

USART协议

下面根据串口数据帧格式进行说明,可配置为8位或9位长,如下图
USART数据帧

参数

  • 起始位:低电平为一个数据帧的开始
  • 数据位:可配置为8位或9位,低位先行
  • 校验位:可选择进行奇偶校验,也可不进行校验,位于最后一个数据位之后,停止位之前
  • 停止位:高电平为一个数据帧的结束,长度可设置为0.5、1(默认)、1.5、2位宽

发射端

发送器根据是否设置校验位(即USART_CR1寄存器的M位)发送8位或9位的数据字,USART_CR1寄存器的发送使能位TE置1时,发送移位寄存器的数据会在TX引脚输出,相应的时钟脉冲输出到CK引脚。在数据传输过程中为保证不丢失数据TE位不应被重置

USART_发生器

实现步骤:1、通过USART_CR1寄存器中的UE位使能USART;2、通过USART_CR1寄存器中的M位设置数据位长度;3、通过USART_CR2寄存器设置停止位长度;4、通过USART_CR3寄存器中的DMAT位选择使能DMA;5、通过USART_BRR寄存器设置波特率;6、通过USART_CR1中的TE位发送空闲帧作为第一次传输;7、将要发送的数据写入USART_DR寄存器(这回自动清除TXE位);8、在写完最后一个数据到USART_DR寄存器中,等待TC=1。
简述为:TXE=1,先发送一个空闲帧,数据就送入TDR同时TXE=0;等到开始位时TDR内数据送入移位寄存器,恢复TXE=1;TXE=1,下一个数据送入TDR同时TXE=0。依次类推。

接收端

接收器根据校验位(即USART_CR1寄存器的M位)接收8位或9位的数据字,前提是保证和发送器相同的波特率,并且要求每次采样的位置正好处于每一位的正中间,同时还要对噪声有一定的判断能力。起始位侦测如下

USART起始位检测
检测到正确的起始位后,就是进行数据接收。实现步骤:1、通过USART_CR1寄存器中的UE位使能USART;2、通过USART_CR1寄存器中的M位设置数据位长度;3、通过USART_CR2寄存器设置停止位长度;4、通过USART_CR3寄存器中的DMAT位选择使能DMA;5、通过USART_BRR寄存器设置波特率;6、通过USART_CR1寄存器中的RE位使能USART接收,RX引脚检测起始位。
当完整收到一个字节数据时,RXNE位置1、如果使能了RXNEIE将会产生中断、发生错误将产生错误标志位、在配置了DMA时RXNE由DMA清除、未配置DMA时RXNE是在RDR读完后自动清除。

选择合适的过采样法

为保证接收端接收到正确的数据帧,需配置合适的过采样技术,通过USART_CR1寄存器这OVER8位进行选择。

  • 选择 8(OVER8=1)的过采样以实现更高的速度(最高 fPCLK/8),在这种情况下,接收机对时钟偏差的最大容差会降低
  • 选择过采样 16 (OVER8=0) 以增加接收器对时钟偏差的容差。在这种情况下,最大速度限制为最大 fPCLK/16

小数波特率

波特率指数据信号对载波的调制速率,计算公式如下

波特率 = f C K 8 × ( 2 − O V E R 8 ) × U S A R T D I V 波特率 = \frac{f_{CK}}{8\times (2 - OVER8)\times USARTDIV} 波特率=8×(2OVER8)×USARTDIVfCK

其中 f C K f_{CK} fCK是USART时钟即所在总线时钟频率,USARTDIV是一个存放在USART_BRR寄存器中的无符号定点数。其中DIV_Mantissa[11:0]位定义USARTDIV的整数部分, DIV_Fraction[3:0]位定义USARTDIV的小数部分,DIV_Fraction[3]位只有在OVER8位为0时有效,否则必须清零。

校验位控制

通过USART_CR1寄存器中的PCE位启用校验。分为奇校验和偶校验,如果使用了奇校验,则9位数据中会出现奇数个1;如果使用了偶校验,则9位数据中会出现偶数个1。

配置DMA

USART能够使用DMA进行连续通信。Rx 缓冲区和 Tx 缓冲区的 DMA 请求是独立生成的。

  • 发送端配置DMA
    通过USART_CR3寄存器中的DMAT位配置DMA进行发送。TXE位置位时,SRAM内数据都会通过DMA外设加载到TDR中。通过以下流程配置
    • 在DMA控制寄存器中写入TDR地址,将其作为传输目的地
    • 在DMA控制寄存器中写入SRAM地址,将其作为传输起点
    • 配置DMA其他配置

USART_DMA发送

  • 接收端配置DMA
    通过USART_CR3寄存器中的DMAR位配置DMA进行接收。RXNE位置位时,RDR内数据都会通过DMA外设加载到SRAM中。通过以下流程配置
    • 在DMA控制寄存器中写入RDR地址,将其作为传输起始地
    • 在DMA控制寄存器中写入SRAM地址,将其作为传输目的地
    • 配置DMA其他配置

USART_DMA接收

代码实现

  1. 单字节串口发送
  • 开启USART时钟和GPIO引脚时钟
  • 配置GPIO引脚
  • 配置USART
  • 使能USART
  • 功能函数编写

代码如下

/*
**********************************************************************************
*   @brief  USART1串口初始化
*			PA9是USART1_TX,PA10是USART1_RX
*			GPIO是AHB1总线,USART1是APB2总线
*   @param  none
*   @return none
*   @use	Serial_Init()
**********************************************************************************
*/
void Serial_Init()
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
	//USART1对应引脚复用映射
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1);
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1);
	//配置PA9、PA10引脚
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	//配置USART1
	USART_InitTypeDef USART_InitStructure;
	USART_InitStructure.USART_BaudRate = 9600;											//波特率
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;		//硬件流控制,这里不选用
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;						//双工
	USART_InitStructure.USART_Parity = USART_Parity_No;									//校验位,这里不选用
	USART_InitStructure.USART_StopBits = USART_StopBits_1;								//1位长的停止位
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;							//字长8位
	USART_Init(USART1, &USART_InitStructure);
	USART_Cmd(USART1, ENABLE);
}

/*
**********************************************************************************
*   @brief  发送单个字节数据
*   @param  一个十六进制数据
*   @return none
*   @use	Serial_SendOneByte(uint8_t Byte)
**********************************************************************************
*/
void Serial_SendOneByte(uint8_t Byte)
{
	//数据先被写入TDR,然后发到发送移位寄存器才能通过GPIO口输出
	USART_SendData(USART1, Byte);
	//此处作等待标志位之用,即发送数据寄存器空标志位,标志位会自动置0
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) != SET);
}

/*
**********************************************************************************
*   @brief  发送字符串
*   @param  一个十六进制数组
*   @return none
*   @use	Serial_SendArray(uint8_t *String)
**********************************************************************************
*/
void Serial_SendString(uint8_t *String)
{
	for(uint8_t i = 0; String[i] != '\0'; i++)
	{
		Serial_SendOneByte(String[i]);
	}
}
  1. 单字节串口接收
    有两种接收方法,查询法和中断法
  • 使用查询
    不需要配置中断,需要在主函数中不断判断RXNE标志位。
if(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == SET)
		{
			RxData = USART_ReceiveData(USART1);
			Serial_SendOneByte(RxData);
		}
  • 使用中断
    使用中断可使程序更加灵活
    代码如下
#include "Serial.h"
/*
**********************************************************************************
*   @brief  USART1串口初始化
*			PA9是USART1_TX,PA10是USART1_RX
*			GPIO是AHB1总线,USART1是APB2总线
*   @param  none
*   @return none
*   @use	Serial_Init()
**********************************************************************************
*/
void Serial_Init()
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
	//USART1对应引脚复用映射
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1);
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1);
	//配置PA9、PA10引脚
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	//配置USART1
	USART_InitTypeDef USART_InitStructure;
	USART_InitStructure.USART_BaudRate = 9600;											//波特率
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;		//硬件流控制,这里不选用
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;						//双工
	USART_InitStructure.USART_Parity = USART_Parity_No;									//校验位,这里不选用
	USART_InitStructure.USART_StopBits = USART_StopBits_1;								//1位长的停止位
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;							//字长8位
	USART_Init(USART1, &USART_InitStructure);
	//配置USART中断
	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);										//配置USART接收数据寄存器非空中断
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);										//选择第二组中断
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;									//配置USART1中断通道
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;										//使能NVIC
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;							//先占优先级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;									//子优先级
	NVIC_Init(&NVIC_InitStructure);
	
	USART_Cmd(USART1, ENABLE);
}

/*
**********************************************************************************
*   @brief  发送单个字节数据
*   @param  一个十六进制数据
*   @return none
*   @use	Serial_SendOneByte(uint8_t Byte)
**********************************************************************************
*/
void Serial_SendOneByte(uint8_t Byte)
{
	//数据先被写入TDR,然后发到发送移位寄存器才能通过GPIO口输出
	USART_SendData(USART1, Byte);
	//此处作等待标志位之用,即发送数据寄存器空标志位,标志位会自动置0
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) != SET);
}

/*
**********************************************************************************
*   @brief  发送字符串
*   @param  一个十六进制数组
*   @return none
*   @use	Serial_SendArray(uint8_t *String)
**********************************************************************************
*/
void Serial_SendString(uint8_t *String)
{
	for(uint8_t i = 0; String[i] != '\0'; i++)
	{
		Serial_SendOneByte(String[i]);
	}
}

/*
**********************************************************************************
*   @brief  发送数字
*   @param  数字,以及数字长度
*   @return none
*   @use	Serial_SendNumber(uint32_t Number, uint8_t Length)
**********************************************************************************
*/
//求x的y次方
uint32_t Serial_Pow(uint32_t X, uint32_t Y)
{
	uint32_t Result = 1;
	while (Y --)
	{
		Result *= X;
	}
	return Result;
}
void Serial_SendNumber(uint32_t Number, uint8_t Length)
{
	//需要把Number的个位、十位、百位..以十进制拆开,然后换成字符数字对应的数据依次发出去
	for(uint8_t i = 0; i < Length; i ++)
	{
		Serial_SendOneByte(Number / Serial_Pow(10, Length - i - 1) % 10 + '0');
	}
}

/*
**********************************************************************************
*   @brief  printf()函数重定向到串口,需要打开MicroLIB
*   @param  int ch, FILE *f
*   @return none
*   @use	printf("Num = %d\n", 666)
**********************************************************************************
*/
//重定向fputc函数到串口,通过串口发送。
//但此方法printf函数只能用于一个串口
//fputc是printf的底层
int fputc(int ch, FILE *f)
{
	Serial_SendOneByte(ch);
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
	return ch;
}

uint8_t Serial_RxFlag;					//读取标志位
uint8_t Serial_RxData;					//读取数据

/*
**********************************************************************************
*   @brief  是否读取数据的标志位
*   @param 	none
*   @return 1表示正常读取,0表示读取失败
*   @use	Serial_GetRxFlag()
**********************************************************************************
*/
uint8_t Serial_GetRxFlag()
{
	if(Serial_RxFlag == 1)
	{
		Serial_RxFlag = 0;
		return 1;
	}
	return 0;
}

/*
**********************************************************************************
*   @brief  返回所读取的数据
*   @param 	none
*   @return 所读取的一个字节数据
*   @use	Serial_GetRxData()
**********************************************************************************
*/
uint8_t Serial_GetRxData()
{
	return Serial_RxData;
}

/*
**********************************************************************************
*   @brief 	中断函数
*   @param 	none
*   @return none
*   @use	中断事件触发会自动跳转到此函数
**********************************************************************************
*/
void USART1_IRQHandler()
{
	if(USART_GetITStatus(USART1, USART_IT_RXNE) == SET)
	{
		Serial_RxData = USART_ReceiveData(USART1);
		Serial_RxFlag = 1;
		USART_ClearITPendingBit(USART1, USART_IT_RXNE);		//如果正常读取RDR数据会自动清除标志位
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/609682.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

泛微E9开发 通过点击按钮来复制选择的明细行

泛微E9开发 通过点击按钮来复制选择的明细行 复制明细行功能背景展示效果实现方法 复制明细行 功能背景 用户可以通过“复制明细”按钮来实现新增选择的明细行&#xff0c;并且新增明细行的数据跟选择的数据完全一样&#xff0c;具体操作如下图所示&#xff1a; 手动新增明细…

图像ISP——AGC参数解析

前言 AWB和AGC是两种常见的自动调整功能。AWB用于自动调整图像的白平衡&#xff0c;以确保颜色在不同光照条件下仍然看起来自然。而AGC则用于自动调整图像的增益&#xff0c;以在不同的亮度条件下保持适当的曝光。 代码例程 static AWB_AGC_TABLE_S g_stAwbAgcTable {/* bvali…

关于位操作符的实际应用<C语言>

前言 位操作符在C语言初学阶段相对其他操作符来说&#xff0c;是一种难度比较大的操作符&#xff0c;且运用较少的一类操作符&#xff0c;但是位操作符并不是“一无是处”&#xff0c;合理运用的位操作符&#xff0c;在某些场景下可以优化算法&#xff0c;提高代码的执行效率&a…

介绍适用于 Node.js 的 Elastic OpenTelemetry 发行版

作者&#xff1a;来自 Elastic Trent Mick 我们很高兴地宣布推出 Elastic OpenTelemetry Distribution for Node.js 的 alpha 版本。 该发行版是 OpenTelemetry Node.js SDK 的轻量级包装&#xff0c;可以让你更轻松地开始使用 OpenTelemetry 来观察 Node.js 应用程序。 背景 …

recycleview和banner新闻列表轮播图

说明&#xff1a;最近碰到一个需求&#xff0c;弄一个新闻列表和轮播图&#xff0c;在首页显示&#xff0c;并且需要json解析&#xff0c;图片下载&#xff0c;轮播图和新闻列表一起滑动 ui效果图&#xff1a; 文件说明&#xff1a; step1:引用依赖包 图片下载 json解析 轮播…

idea无法识别加载pom.xml文件

有时idea无法识别加载pom.xml文件&#xff0c;直接打开pom.xml文件&#xff0c;然后添加到maven就行

官方文档k8s1.30安装部署高可用集群,kubeadm安装Kubernetes1.30最新版本

文章目录 节点架构一、准备开始(每一台机器都执行)1️⃣ 检查所需端口(可以直接关闭防火墙放开所有端口)端口和协议控制面工作节点 关闭防火墙关闭 SELinux 2️⃣ 安装containerd容器containerd部署containerd切换为国内源 3️⃣ 设置/etc/hosts 二、安装 kubeadm、kubelet 和 …

C++青少年简明教程:C++程序结构

C青少年简明教程&#xff1a;C程序结构 一个简单的C程序源码如下&#xff1a; #include <iostream> using namespace std;int main() {cout << "Hello World" << endl;return 0; }下面解析一下。 1. #include <iostream> 这是一条预处理…

SSH隧道可以做什么?

SSH隧道是SSH协议服务端提供的一种扩展功能&#xff0c;一般仅在linux服务器的SSH服务端中提供&#xff0c;其它的如交换机、防火墙等网络设备中&#xff0c;虽然支持SSH协议&#xff0c;但多数并不提供SSH隧道功能。 所以&#xff0c;在通过SSH协议连接远程设备时&#xff0c…

Python远程连接Linux执行操作

一、任务要求 要使用Python编写代码来远程给Linux主机上传一个文件&#xff0c;可以使用paramiko库&#xff0c;这是一个实现了SSHv2协议的Python库&#xff0c;它支持SSH连接&#xff08;包括客户端和服务端&#xff09;&#xff0c;并且提供SFTP&#xff08;SSH File Transf…

VScode查看以十六进制查看文件的插件说明

找到插件并下载 打开指定的文件 选择打开方式即可 结果如下

STM32——GPIO输出(点亮第一个LED灯)

代码示例&#xff1a; #include "stm32f10x.h" // Device headerint main() {RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启时钟GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode GPIO_Mode_Out_PP;GPIO_InitSt…

成为计算机视觉(CV)需要掌握哪些技术知识(综述)

在CV领域&#xff0c;深度学习和机器学习技术发挥着至关重要的作用&#xff0c;它们为图像识别、目标检测、图像分割等任务提供了强大的工具和方法。本文将综述CV中需要学习的深度学习和机器学习技术。 一、深度学习技术 卷积神经网络&#xff08;Convolutional Neural Netwo…

【吊打面试官系列】Java高并发篇 - 为什么线程通信的方法 wait(), notify()和 notifyAll()被定义在 Object 类里?

大家好&#xff0c;我是锋哥。今天分享关于 【为什么线程通信的方法 wait(), notify()和 notifyAll()被定义在 Object 类里&#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; 为什么线程通信的方法 wait(), notify()和 notifyAll()被定义在 Object 类里&#xff1…

被严重低估的后端技术面经,强到离谱!

前天加完班&#xff0c;回家路上翻了下粉丝群。发现群里最近在疯传一份叫《后端 offer 收割机养成指南》的资料。 本来感觉这个 title 看起来有点离谱&#xff0c;结果没想到仔细一看&#xff0c;这份资料竟然真的有点东西。内容收纳的很全&#xff0c;而且融合了很多今年的新…

软件测试从业人员怎么看待测试工作的前途?

调查背景   工信部发布的2023年软件业经济运行情况显示&#xff0c;全年累计完成软件业务收入123258亿元&#xff0c;同比增长13.4%&#xff0c;增速较上年同期提高2.2个百分点。值得注意的是&#xff0c;亚太地区软件测试市场正在迅速崛起&#xff0c;预计到2026年复合年增长…

Django开发实战之单元测试和集成测试之编写测试用例以及代码覆盖率统计分析

这里分为了两种情况&#xff1a; E2E&#xff1a;接近真实使用情况&#xff0c;用例数量少 UT&#xff1a;执行速度快&#xff0c;用例数量多 1、整理测试点 1、注册功能 允许匿名访问URL&#xff1a;http://127.0.0.1:8000/accounts/register/get请求&#xff1a;返回htmlp…

Linux 用户进程启动与Systemd

Linux用户空间的启动 我们都知道&#xff0c;Linux在内核的初始化自身的流程结束后&#xff0c;就会将程序流的运行转到用户态&#xff0c;也就是进入init进程流之后&#xff0c;他会按照这个流程进行初始化&#xff1a; init进程初始化 基础的底层服务&#xff1a;比如说ude…

JavaScript 事件

在 Web 开发中&#xff0c;JavaScript 事件是至关重要的概念之一。通过事件&#xff0c;我们可以实现交互性和动态性&#xff0c;使用户与网页进行互动。本篇博客将介绍 JavaScript 事件的基础知识&#xff0c;并深入探讨一些高级技术。 1. 什么是事件&#xff1f; 事件是指用…
最新文章