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

嵌入式软件--stm32 DAY 6 USART串口通讯(下)

1.寄存器轮询_收发字符串

通过寄存器轮询方式实现了收发单个字节之后,我们趁热打铁,争上游,进阶到字符串。字符串就是多个字符。很明显可以循环收发单个字节实现。

然后就是接收字符串。如果接受单个字符的函数放在while里,它也可以实现一个一个的接收字符串,在一定时间戳下,效果等同。

然后我们要想想如何整合,写成一个函数。

2.问题解答,状态位清零

我们的编程方式,一个是寄存器写法,一个是hal库写法。还有的就是实现思路。

//发送一个字符
void USART_SendChar(uint8_t ch)
{//判断TDR是否为空,必须等待TDR为空才能写入数据,也就是继续发送while((USART1->SR & USART_SR_TXE)==0){}//将要发送的数据写入TDRUSART1->DR = ch;
}

初始的时候,TDR的存储寄存器肯定为空

关于状态位清零,TXE和RXNE都可以自动清零。

只要发数据,就自动清零。

RXNE也是一样,只要读RDR,就会清零。

3.串口通讯 寄存器中断方式

复制上一个工程文件,把改删除的删掉。

主要配置中断管理NVIC

中断服务函数

main.c

#include "usart.h"
#include "delay.h"
#include <string.h>
//定义接收缓冲区和接收数据长度
uint8_t buff[100];
uint8_t size;
int main(void)
{//1.初始化USART_Init();//发送单个字符USART_SendChar('A');USART_SendChar('\n');//发送字符串uint8_t str[]="Hello World!\n";USART_SendString(str,strlen((char *)str));while(1){}}

usart.c

#include "usart.h"//初始化
void USART_Init(void)
{//开启时钟RCC->APB2ENR |= RCC_APB2ENR_USART1EN;RCC->APB2ENR |= RCC_APB2ENR_IOPAEN;//GPIO工作模式// PA9:TX 复用推挽输出,CNF=10,MODE=11GPIOA->CRH |= GPIO_CRH_MODE9;GPIOA->CRH |= GPIO_CRH_CNF9_1;GPIOA->CRH &= ~GPIO_CRH_CNF9_0;// PA10:RX 浮空输入,CNF=01,MODE=00GPIOA->CRH &= ~GPIO_CRH_MODE10;GPIOA->CRH &= ~GPIO_CRH_CNF10_1;GPIOA->CRH |= GPIO_CRH_CNF10_0;//串口模块设置//设置波特率 115200USART1->BRR = 0x271;//使能串口和收发模块USART1->CR1 |= USART_CR1_UE;USART1->CR1 |= (USART_CR1_TE|USART_CR1_RE);//配置数据帧格式USART1->CR1 &=~ USART_CR1_M;   //长度为8位的数据,没有校验位USART1->CR1 &=~ USART_CR1_PCE;  //无校验,不使用校验位USART1->CR2 &=~ USART_CR2_STOP;  //1位停止位//使能串口接收中断USART1->CR1 |= USART_CR1_RXNEIE;USART1->CR1 |= USART_CR1_IDLEIE;//配置NVICNVIC_SetPriorityGrouping(3);NVIC_SetPriority (USART1_IRQn,2);NVIC_EnableIRQ(USART1_IRQn);}//发送一个字符
void USART_SendChar(uint8_t ch)
{//判断TDR是否为空,必须等待TDR为空才能写入数据,也就是继续发送while((USART1->SR & USART_SR_TXE)==0){}//将要发送的数据写入TDRUSART1->DR = ch;
}//发送字符串,数组可以用指针表示,形参就是数组和字长
void USART_SendString(char *str,uint8_t size)
{uint8_t i = 0;for ( i = 0; i < size; i++){USART_SendChar(str[i]);}}//引入外部变量
extern uint8_t buff[100];
extern uint8_t size;//中断服务程序
void USART1_IRQHandler(void)
{//判断是RXNE=1还是IDLE=1if (USART1->SR & USART_SR_RXNE){//如果RXNE=1,表示接收到一个字符
buff[size]=USART1->DR;
size++;}else if (USART1->SR & USART_SR_IDLE){USART1->DR;//如果IDLE=1,表示检测到空闲帧,字符串接收完毕USART_SendString(buff,size);size=0;}}

usart.h

#ifndef __USART_H
#define __USART_H
#include "string.h"
#include "stm32f10x.h"
//初始化
void USART_Init(void);//发送一个字符
void USART_SendChar(uint8_t ch);//发送一个字符串,数组可以用指针表示,形参就是数组和字长
void USART_SendString(char *str,uint8_t size);#endif

 

我们可以将接收字符串的标志位在外面定义出来。

 

4.串口案例一串口通讯_hal库轮询方式

我们有usart1配置。

我们反正还得需要配,这个时候我们可以不管他,直接去左边框栏中点点点。图形化配置的精髓就是方便。

还是调用hal库简单。

轮询方式使用效率不是很高,中断方式才高。

5.HAL库中断方式—定长数据接收

记住USART1用异步模式,要打开中断实现

中断方式要和回调函数结合。

接收定长数据,会和之前接收的数据结合分析

6.HAL库中断方式--变长数据接收

接收变长数据,size是可接收的字长上限。

她的回调不一样,是下面这个。在函数面前加void,我忘了加,编译出错。

先检查是什么模块

7.重定向printf—寄存器方式

用printf输出调试信息,打印到屏幕,没有屏幕也可重定向printf,把数据打印到串口,从而在电脑端接收调试信息,这是一种有效调试手段。

printf大法对任何语言都是有用的。

printf的底层函数:fputc

把字符一个个发到控制台文件里。 

重写也在usart.c里,记得引入<stdio.h>

8.重定向——hal

usart.c里

在main函数来随便打印。

http://www.lryc.cn/news/2379717.html

相关文章:

  • 问题处理——在ROS2(humble)+Gazebo+rqt下,无法显示仿真无人机的相机图像
  • 69、微服务保姆教程(十二)容器化与云原生
  • 朱老师,3518e系列,第六季
  • ElasticSearch-集群
  • 一文掌握工业相机选型计算
  • 记录心态和工作变化
  • 深入理解 TypeScript 中的 unknown 类型:安全处理未知数据的最佳实践
  • LabVIEW机械振动信号分析与故障诊断
  • Helm配置之为特定Deployment配置特定Docker仓库(覆盖全局配置)
  • 【Spring】Spring中的适配器模式
  • GO学习指南
  • 2、ubuntu系统配置OpenSSH | 使用vscode或pycharm远程连接
  • MySQL面试知识点详解
  • 小白入门:GitHub 远程仓库使用全攻略
  • RPC与SOAP的区别
  • Day11-苍穹外卖(数据统计篇)
  • Tomcat简述介绍
  • 《从零开始:Spring Cloud Eureka 配置与服务注册全流程》​
  • 如何保证RabbitMQ消息的顺序性?
  • FPGA学习知识(汇总)
  • c语言 写一个五子棋
  • Redisson分布式锁-锁的可重入、可重试、WatchDog超时续约、multLock联锁(一文全讲透,超详细!!!)
  • Python爬虫实战:研究源码还原技术,实现逆向解密
  • WordPress Relevanssi插件时间型SQL注入漏洞(CVE-2025-4396)
  • Adobe Illustrator学习备忘
  • C#中的dynamic与var:看似相似却迥然不同
  • 求职困境:开发、AI、运维、自动化
  • 语言模型:AM-Thinking-v1 能和大参数语言模型媲美的 32B 单卡推理模型
  • ChatGPT:OpenAI Codex—一款基于云的软件工程 AI 代理,赋能 ChatGPT,革新软件开发模式
  • docker compose up -d 是一个用于 通过 Docker Compose 在后台启动多容器应用 的命令