当前位置: 首页 > news >正文

STM32-USART

串口USART:同步异步串行全双公

通常为96-N-8-1(9600波特率,无校验位,8个有效数据位,1位停止位)

串行通信:一位一位进行传输(传输线少,成本低)

并行通信:多条数据线同时传递数据,一次性传递过去数据 (抗干扰能力弱)

异步通信:以字符为单位(间隔任意)字符里的数据间隔一定;发送接受时的时钟可以不一致(要相似);抗干扰能力强

同步通信:由一方控制另一方的时钟

半双工:RX-TX进行时,另一方的RX-TX不能进行,只能进行一个

全双工:RX-TX进行时,另一方的RX-TX也可以进行

速率:比特率:每秒钟传输二进制代码的位数

接收发送时双方要共地

UART:异步收发器;去掉了同步通信功能

看下图:

nRTS、nCTS硬件流控制(一般不用)(1)

数据发送:数据寄存器DR-发送寄存器TDR-发送移位寄存器(2)

数据接收:移位寄存器-接收数据寄存器RDR-数据寄存器DR(2)

发送器控制,接收器控制(3)

4为波特率

 

 

软件设计流程:

  • 使能串口时钟以及GPIO端口时钟

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);

  • GPIO端口模式设置,设置串口对应的引脚为复用功能
  • 初始化串口参数(波特率;字长;停止位;校验位;USART模式;硬件流控制)
typedef struct
{uint32_t USART_BaudRate; //配置 USART 通信波特率                                         uint16_t USART_WordLength;//传输或接收的数据位数量此参数可以是 @ref USART_Word_Length 中的一个值uint16_t USART_StopBits;//指定传输的停止位数量//此参数可以是 @ref USART_Stop_Bits 中的一个值uint16_t USART_Parity; // 指定奇偶校验模式// 此参数可以是 @ref USART_Parity 中的一个值;当启用奇偶校验时,计算出的校验位会插入到传输数据的最高有效位(MSB)位置(当字长设为9位时为第9位; 当字长设为8位时为第8位) uint16_t USART_Mode; // 指定接收或发送模式或者接发同时uint16_t USART_HardwareFlowControl; //指定硬件流控制模式参数是 USART_Hardware_Flow_Control 中的一个值 
} USART_InitTypeDef;
  • 使能串口
/*** @brief  启用或禁用 USART 外设* @param  USARTx: 指向 USART_TypeDef 结构体的指针,标识目标 USART 外设* @param  NewState: 新状态,取值为 ENABLE 或 DISABLE*/
void USART_Cmd(USART_TypeDef* USARTx, FunctionalState NewState)
  • 设置串口中断类型并使能
/*** @brief  配置 USART 的中断使能状态* @param  USARTx: 指向 USART_TypeDef 结构体的指针,标识目标 USART 外设* @param  USART_IT: 指定要配置的 USART 中断源*         可取值为 @ref USART_Interrupt_Sources 中的一个或多个值的组合* @param  NewState: 新状态,取值为 ENABLE 或 DISABLE* @retval 无*/
void USART_ITConfig(USART_TypeDef* USARTx, uint16_t USART_IT, FunctionalState NewState)
  • 设置串口中断优先级使能串口通信(NVIC模块)
  • 编写串口中断服务函数(判断用了那种串口中断类型标志位的状态)
/*** @brief  检查指定的 USART 中断状态标志* @param  USARTx: 指向 USART_TypeDef 结构体的指针,标识目标 USART 外设* @param  USART_IT: 指定要检查的 USART 中断源*         可取值为 @ref USART_Interrupt_Sources 中的一个值* @retval ITStatus: 中断状态*         - SET: 中断标志被设置*         - RESET: 中断标志未被设置*/
ITStatus USART_GetITStatus(USART_TypeDef* USARTx, uint16_t USART_IT)
  • 接收函数:uint16_t USART_ReceiveData(USART_TypeDef* USARTx)
  • 发送函数: void USART_SendData(USART_TypeDef* USARTx, uint16_t Data)
  • 串口的标志位:
  • 发送标志位:USART_IT_TXE:发送寄存器-1为空;可进行数据发送

                              USART_IT_TC:发送完成标志位-1为发送完成

  • 接收标志位:USART_IT_RXNE:接受寄存器-1为空;可进行数据接收

代码实现:USART1与PC机对话

USART1_RX_STA 就像一个 “光标”,指示下一个数据的存储位置

#include "usart1.h"	//注意,读取USARTx->SR能避免莫名其妙的错误   	
u8 USART1_RX_BUF[USART1_REC_LEN];     //接收缓冲,最大USART_REC_LEN个字节.
//接收状态
//bit15,	接收完成标志
//bit14,	接收到0x0d
//bit13~0,	接收到的有效字节数目
u16 USART1_RX_STA=0;       //接收状态标记
/*******************************************************************************
* 函 数 名         : USART1_Init
* 函数功能		   : USART1初始化函数
* 输    入         : bound:波特率
* 输    出         : 无
*******************************************************************************/ 
void USART1_Init(u32 bound)
{//GPIO端口设置GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);/*  配置GPIO的模式和IO口 */GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;//TX			   //串口输出PA9GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;	    //复用推挽输出GPIO_Init(GPIOA,&GPIO_InitStructure);  /* 初始化串口输入IO */GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;//RX			 //串口输入PA10GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;		  //模拟输入GPIO_Init(GPIOA,&GPIO_InitStructure); /* 初始化GPIO *///USART1 初始化设置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); //初始化串口1USART_Cmd(USART1, ENABLE);  //使能串口1 USART_ClearFlag(USART1, USART_FLAG_TC);//清除TC位USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启相关中断//Usart1 NVIC 配置NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;//抢占优先级3NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;		//子优先级3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;			//IRQ通道使能NVIC_Init(&NVIC_InitStructure);	//根据指定的参数初始化VIC寄存器、		
}/*******************************************************************************
* 函 数 名         : USART1_IRQHandler
* 函数功能		   : USART1中断函数
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/ 
void USART1_IRQHandler(void)                	//串口1中断服务程序
{u8 r;if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断{r =USART_ReceiveData(USART1);//(USART1->DR);	//读取接收到的数据if((USART1_RX_STA&0x8000)==0)//接收未完成{if(USART1_RX_STA&0x4000)//接收到了0x0d{if(r!=0x0a)USART1_RX_STA=0;//接收错误,重新开始else USART1_RX_STA|=0x8000;	//接收完成了 }else //还没收到0X0D{	if(r==0x0d)USART1_RX_STA|=0x4000;else{USART1_RX_BUF[USART1_RX_STA&0X3FFF]=r;//前13位用来存储数据USART1_RX_STA++;//位数加1指向下一个存储位置if(USART1_RX_STA>(USART1_REC_LEN-1))USART1_RX_STA=0;//接收数据错误,重新开始接收	  }		 }}   		 } 
} 	

主函数代码

#include "system.h"
#include "SysTick.h"
#include "led.h"
#include "usart1.h"int main()
{u8 i=0; u16 t=0;u16 len=0;SysTick_Init(72);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);  //中断优先级分组 分2组USART1_Init(115200);LED_Init();while(1){if(USART1_RX_STA&0x8000)//接受完成{					   len=USART1_RX_STA&0x3fff;//得到此次接收到的数据长度for(t=0;t<len;t++){USART_SendData(USART1, USART1_RX_BUF[t]);         //向串口1发送数据while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束}USART1_RX_STA=0;//为了将15位值0方便下次接受}i++;if(i%20==0){LED1=!LED1;}delay_ms(10);}
}

printf重定义:

 要记得引用头文件

#include "stdio.h" 
在usart.c文件中添加

int fputc(int ch,FILE *p)  //函数默认的,在使用printf函数时自动调用
{USART_SendData(USART1,(u8)ch);	while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);return ch;
}

主函数

int main()
{u8 i=0; u16 data=1234;float fdata=12.34;char str[]="Hello World!";	SysTick_Init(72);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);  //中断优先级分组 分2组LED_Init();USART1_Init(115200);while(1){i++;if(i%50==0){LED1=!LED1;printf("输出整型数data=%d\r\n",data);printf("输出浮点型数fdata=%0.2f\r\n",fdata);printf("输出十六进制数data=%X\r\n",data);printf("输出八进制数data=%o\r\n",data);printf("输出字符串str=%s\r\n",str);}delay_ms(10);}
}

http://www.dtcms.com/a/270353.html

相关文章:

  • 图像噪点消除:用 OpenCV 实现多种滤波方法
  • jmeter 性能测试步骤是什么?
  • 【深度学习新浪潮】什么是上下文长度?
  • MQTT QoS 2 详细流程解析
  • 爬虫-request处理POST
  • pytorch深度学习-ResNet残差网络-CIFAR-10
  • 利用AI技术快速提升图片编辑效率的方法
  • Mapper接口是什么
  • HarmonyOS从入门到精通:自定义组件开发指南(四):组件状态管理之父子组件通信
  • 跨越十年的C++演进:C++23新特性全解析
  • VR法庭相比传统法庭有哪些优势​
  • WebClient与HTTPInterface远程调用对比
  • 第8章:应用层协议HTTP、SDN软件定义网络、组播技术、QoS
  • SPI / I2C / UART 哪个更适合初学者?
  • 通过“逆向侦测”驾驭涌现复杂性的认知架构与技术实现
  • 短视频矩阵管理平台的崛起:源头厂商的深度解析
  • C# Type.GetProperties() 获取不到值的笔记
  • SQL注入与防御-第六章-2:利用操作系统--执行操作系统命令
  • 图像梯度处理与边缘检测:OpenCV 实战指南
  • 【牛客刷题】小红的v三元组
  • FastAPI Docker环境管理脚本使用指南
  • 虚拟机忘记密码怎么办
  • nmon使用方法
  • 征程 6|工具链量化简介与代码实操
  • 云原生安全观察:零信任架构与动态防御的下一代免疫体系
  • 人物设定一秒入魂!RAIDEN-R1提出可验证奖励新范式,让CoT推理更“人格一致”
  • SpringAI学习笔记-MCP客户端简单示例
  • python采集商品详情数据接口json数据返回参考
  • 前端面试常考题目详解​
  • 解决阿里云ubuntu内存溢出导致vps死机无法访问 - 永久性增加ubuntu的swap空间 - 阿里云Linux实例内存溢出(OOM)问题修复方案