【江科大】STM32:串口HEX/文本数据接收和发送(代码部分)(下)

串口发送

在这里插入图片描述

#include "stm32f10x.h"                  // Device header#include<stdio.h>
#include<stdarg.h>
void Serial_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;  //片上外设,复用推免GPIO_InitStruct.GPIO_Pin = GPIO_Pin_9; 	GPIO_InitStruct.GPIO_Speed =  GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStruct); USART_InitTypeDef USART_InitStruct;USART_InitStruct.USART_BaudRate = 9600;  //波特率USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStruct.USART_Mode = USART_Mode_Tx;  //启用发送USART_InitStruct.USART_Parity = USART_Parity_No;USART_InitStruct.USART_StopBits = USART_StopBits_1;//指定传输的停止位数USART_InitStruct.USART_WordLength = USART_WordLength_8b;USART_Init(USART1,&USART_InitStruct);USART_Cmd(USART1,ENABLE);}void Serial_SendByte(uint8_t Byte) //发送数据函数{USART_SendData(USART1,Byte);//调用后byte就写入TDRwhile(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);	//等待标志位,且下一次标志位会清零	}void Serial_SendArray(uint8_t *Array,uint16_t  length)//传数组{for(int i= 0;i<length;++i){Serial_SendByte(Array[i]);}}void Serial_SendString(char *String)//发送字符串{uint8_t i;for(int i = 0; String[i] != '\0';++i){Serial_SendByte(String[i]);}}	
uint32_t Serial_Pow(uint32_t x,uint8_t y)
{uint32_t Result =  1;while(y--){Result *= x; //表示取x的y次方}return Result;
}
//发送数字的逻辑void Serial_SendNumber(uint32_t Num,uint8_t len)//发送一个数字以字符串返回{uint8_t i;for(i = 0;i<len;i++){Serial_SendByte(Num / Serial_Pow(10, len - i - 1) % 10 + '0');//+字符偏移}}//printf函数在打印的时候在不断调用底层fputcint fputc(int ch,FILE *f)//对printf函数重定向到串口{Serial_SendByte(ch);return ch;}//1.定义一个函数,最后一个参数为省略号,省略号前面可以设置自定义参数//2.在函数定义中创建一个 va_list 类型变量\n\n该类型是在 stdarg.h 头文件中定义的//3.使用 int 参数和 va_start 宏来初始化 va_list 变量为一个参数列表\n\n宏 va_start 是在 stdarg.h 头文件中定义的//4.使用 va_arg 宏和 va_list 变量来访问参数列表中的每个项//5.使用宏 va_end 来清理赋予 va_list 变量的内存//用来接收格式化字符串    省略号代表,传递可变数量的参数void  Serial_Printf(char *format,...)//多个串口用printf{char String[100];va_list arg;//定义参数列表变量va_start(arg,format);vsprintf(String,format,arg);va_end(arg);//释放参数表Serial_SendString(String);}

串口库函数

//配置同步时钟输出  时钟是否输出,时钟的极性相位等参数
void USART_ClockInit(USART_TypeDef* USARTx, USART_ClockInitTypeDef* USART_ClockInitStruct);
void USART_ClockStructInit(USART_ClockInitTypeDef* USART_ClockInitStruct);void USART_Cmd(USART_TypeDef* USARTx, FunctionalState NewState);
void USART_ITConfig(USART_TypeDef* USARTx, uint16_t USART_IT, FunctionalState NewState);
//开启DMA的触发通道
void USART_DMACmd(USART_TypeDef* USARTx, uint16_t USART_DMAReq, FunctionalState NewState);//设置地址
void USART_SetAddress(USART_TypeDef* USARTx, uint8_t USART_Address);
//唤醒单元
void USART_SendData(USART_TypeDef* USARTx, uint16_t Data);
uint16_t USART_ReceiveData(USART_TypeDef* USARTx);

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"
uint8_t KeyNum;
int main(void)
{//使用RCC开启GPIO的时钟//使用GPIO_Init函数初始化GPIO//使用输入输出的函数控制GPIO口OLED_Init();Serial_Init();Serial_SendByte('B');//uint8_t arr[] = {0x41,0x42,0x43};//  Serial_SendNumber(12345,5);//printf("num = %d",123);char String[100];//sprintf(String,"num = %d\n",999);//将999写入String C语言可变参数//Serial_SendString(String);//Serial_Printf("num = %d\n",999);//--no-multibyte-charsSerial_Printf("你好");while(1){}}

数据模式

在这里插入图片描述

在这里插入图片描述

#include "stm32f10x.h"                  // Device header#include<stdio.h>
#include<stdarg.h>uint8_t Serial_RxData;
uint8_t Serial_RxFlag;
void Serial_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;  //片上外设,复用推免GPIO_InitStruct.GPIO_Pin = GPIO_Pin_9; 	GPIO_InitStruct.GPIO_Speed =  GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStruct); //配置接收	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IPU;  //片上外设,上拉输入:开始为高电平GPIO_InitStruct.GPIO_Pin = GPIO_Pin_10; 	GPIO_InitStruct.GPIO_Speed =  GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStruct); USART_InitTypeDef USART_InitStruct;USART_InitStruct.USART_BaudRate = 9600;  //波特率USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStruct.USART_Mode = USART_Mode_Tx|USART_Mode_Rx;  //启用发送/和接收USART_InitStruct.USART_Parity = USART_Parity_No;USART_InitStruct.USART_StopBits = USART_StopBits_1;//指定传输的停止位数USART_InitStruct.USART_WordLength = USART_WordLength_8b;USART_Init(USART1,&USART_InitStruct);//选择中断源USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//配置优先级分组NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//使用中断NVIC_InitTypeDef NVIC_InitStruct;NVIC_InitStruct.NVIC_IRQChannel =  USART1_IRQn;NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority =  1;NVIC_InitStruct.NVIC_IRQChannelSubPriority = 1;NVIC_Init(&NVIC_InitStruct);USART_Cmd(USART1,ENABLE);}
//发送/接收和上面一样,自己添加
//.....
//.....uint8_t Serial_GetRxFlag(void){if(Serial_RxFlag  == 1){Serial_RxFlag = 0;return 1;}return 0;}uint8_t Serial_GetRxData(void){return Serial_RxData;}//配置中断函数void USART1_IRQHandler(void){if(USART_GetITStatus(USART1,USART_IT_RXNE) == SET){ //如果已经发生中断,就清空Serial_RxData = USART_ReceiveData(USART1);Serial_RxFlag = 1;USART_ClearITPendingBit(USART1,USART_IT_RXNE);}}		

HEX数据包

在这里插入图片描述

文本数据包

在这里插入图片描述

HEX数据包接收

在这里插入图片描述
在这里插入图片描述
串口收发HEX数据包
在这里插入图片描述

#include "stm32f10x.h"                  // Device header#include<stdio.h>
#include<stdarg.h>uint8_t Serial_TxPacket[4];				//FF 01 02 03 04 FE
uint8_t Serial_RxPacket[4];
uint8_t Serial_RxFlag;
void Serial_Init(void)
{		
//RCC和GPIO配置和上面一样
//串口配置和上面一样
//中断配置和上面一样USART_Cmd(USART1,ENABLE);}//数据收发部分自己写//....//....void Serial_SendPacket(void)
{Serial_SendByte(0xFF);Serial_SendArray(Serial_TxPacket, 4);Serial_SendByte(0xFE);
}uint8_t Serial_GetRxFlag(void)  //判断是否有数据报{if(Serial_RxFlag  == 1){Serial_RxFlag = 0;return 1;}return 0;}//配置中断函数void USART1_IRQHandler(void){ //利用状态姐接收数据包static uint8_t RXState = 0;static uint8_t pRxPacket = 0;  //数据包长度if(USART_GetITStatus(USART1,USART_IT_RXNE) == SET){ //如果已经发生中断,就清空uint8_t RxData = USART_ReceiveData(USART1);if(RXState == 0){if(RxData == 0xFF)  //判断包头{RXState = 1;pRxPacket = 0;}}else if(RXState ==1){Serial_RxPacket[pRxPacket] = RxData;pRxPacket++;if(pRxPacket>= 4){RXState = 2;}}else if(RXState == 2){if(RxData == 0xFE)  //判断包尾{RXState = 0;Serial_RxFlag = 1;  }}USART_ClearITPendingBit(USART1,USART_IT_RXNE);}}		
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"
#include "Key.h"
uint8_t RxDaTa;
uint8_t keynum;
int main(void)
{//使用RCC开启GPIO的时钟//使用GPIO_Init函数初始化GPIO//使用输入输出的函数控制GPIO口OLED_Init();Key_Init();Serial_Init();OLED_ShowString(1, 1, "TxPacket");OLED_ShowString(3, 1, "RxPacket");Serial_TxPacket[0] = 0x01;Serial_TxPacket[1] = 0x02;Serial_TxPacket[2] = 0x03;Serial_TxPacket[3] = 0x04;while (1){keynum = Key_GetNum();if(keynum == 1){Serial_TxPacket[0]++;Serial_TxPacket[1]++;Serial_TxPacket[2]++;Serial_TxPacket[3]++;Serial_SendPacket();OLED_ShowHexNum(2, 1, Serial_TxPacket[0], 2);OLED_ShowHexNum(2, 4, Serial_TxPacket[1], 2);OLED_ShowHexNum(2, 7, Serial_TxPacket[2], 2);OLED_ShowHexNum(2, 10, Serial_TxPacket[3], 2);}if (Serial_GetRxFlag() == 1){OLED_ShowHexNum(4, 1, Serial_RxPacket[0], 2);OLED_ShowHexNum(4, 4, Serial_RxPacket[1], 2);OLED_ShowHexNum(4, 7, Serial_RxPacket[2], 2);OLED_ShowHexNum(4, 10, Serial_RxPacket[3], 2);}}
}

串口收发文本数据包

在这里插入图片描述

void USART1_IRQHandler(void){static uint8_t RXState = 0;static uint8_t pRxPacket = 0;  //数据包长度if(USART_GetITStatus(USART1,USART_IT_RXNE) == SET){ //如果已经发生中断,就清空uint8_t RxData = USART_ReceiveData(USART1);if(RXState == 0){if(RxData == 0xFF)  //判断包头{RXState = 1;pRxPacket = 0;}}else if(RXState ==1){Serial_RxPacket[pRxPacket] = RxData;pRxPacket++;if(pRxPacket>= 4){RXState = 2;}}else if(RXState == 2){if(RxData == 0xFE)  //判断包尾{RXState = 0;Serial_RxFlag = 1;  }}USART_ClearITPendingBit(USART1,USART_IT_RXNE);}}		
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"
#include "Key.h"
uint8_t RxDaTa;
uint8_t keynum;
int main(void)
{//使用RCC开启GPIO的时钟//使用GPIO_Init函数初始化GPIO//使用输入输出的函数控制GPIO口OLED_Init();Key_Init();Serial_Init();OLED_ShowString(1, 1, "TxPacket");OLED_ShowString(3, 1, "RxPacket");Serial_TxPacket[0] = 0x01;Serial_TxPacket[1] = 0x02;Serial_TxPacket[2] = 0x03;Serial_TxPacket[3] = 0x04;while (1){keynum = Key_GetNum();if(keynum == 1){Serial_TxPacket[0]++;Serial_TxPacket[1]++;Serial_TxPacket[2]++;Serial_TxPacket[3]++;Serial_SendPacket();OLED_ShowHexNum(2, 1, Serial_TxPacket[0], 2);OLED_ShowHexNum(2, 4, Serial_TxPacket[1], 2);OLED_ShowHexNum(2, 7, Serial_TxPacket[2], 2);OLED_ShowHexNum(2, 10, Serial_TxPacket[3], 2);}if (Serial_GetRxFlag() == 1){OLED_ShowHexNum(4, 1, Serial_RxPacket[0], 2);OLED_ShowHexNum(4, 4, Serial_RxPacket[1], 2);OLED_ShowHexNum(4, 7, Serial_RxPacket[2], 2);OLED_ShowHexNum(4, 10, Serial_RxPacket[3], 2);}}
}

(FlyMcu)利用串口下载程序

在这里插入图片描述
2.
在这里插入图片描述
将跳线帽置在boot1 按复位键,点击开始编程即可。
由于该单片机只有串口1支持串口烧录。因此连接线连在串口1的所在的引脚上。
程序加载到bootloader完成后,再将跳线帽换到boot0,按复位键
为什么可以使用串口下载?
在这里插入图片描述
原理是实现程序的自我更新,即利用bootloader(程序代码),更新程序存储器。
串口下载的过程:Bootloader接收usart1的数据,刷新程序存储器。这时主程序处于瘫痪状态,主程序更新完成再启动主程序,执行新程序。
在这里插入图片描述

那为什么切换boot引脚,为什么每次要复位?

Boot0时,启动时主闪存存储器
Boot1时,启动再系统存储器
而Bootloader在更新系统存储器时需要切换到boot1,而更新完成运行更新后的主闪存存储器需要切换到Boot0。
而由于sysclk的第四个上升沿,Boot引脚会被锁定,因此每次都需要复位。配置新得启动模式。
每次下载都要切换跳线帽,太麻烦了,怎么解决?
1.设计一个外置电路。利用程序控制Boot0和Boot1 的切换

为什么要使用串口下载?

  • 比如使用烧录器烧录时,使得某个IO口失能,可以利用串口下载将它改回来。
  • 读取flash中的程序(可用于读取他人flash的程序)
    在这里插入图片描述

选项字节

存储不随程序变化而变化的参数。 就是一直爆出不变的参数
读保护,保护程序不被读出。
写保护,就无法再写入。
硬件参数,用户参数
STLK Utlity
:可以读取bin文件
:选项字节的配置

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

latex画边框框加粗的表格

示例代码 \begin{table}[H]% 这四个距离分别控制toprule和buttomrule的上、下空白间距&#xff0c;如果不是0的话以后竖线和横线连不起来\abovetopsep0pt\aboverulesep0pt\belowrulesep0pt\belowbottomsep0pt\centering% \heavyrulewidth是\toprule和\bottomrule的默认宽度&am…

如何做好培训管理?推荐使用这款培训管理系统(内附详细步骤+免费模板)

本文将为大家讲解&#xff1a;1、如何做好培训管理&#xff1f;2、如何使用零代码平台搭建培训管理系统&#xff1f; 培训管理&#xff0c;作为企业人力资源管理的核心环节&#xff0c;对于确保员工具备完成任务所需的专业知识和技能发挥着至关重要的作用。它不仅是提升员工绩…

北斗卫星为野外科考人员提供安全保障

北斗卫星为野外科考人员提供安全保障 自第二次青藏高原综合科学考察研究启动以来&#xff0c;青海不断提升科考服务保障能力&#xff0c;推动科考全程信息化&#xff0c;有效促进科考成果转化。 为保障科考人员的人身安全&#xff0c;青海省青藏科学考察服务中心开发了基于北…

第08章_面向对象编程(高级)(static,单例设计模式,理解mian方法,代码块,final,抽象类与抽象方法,接口,内部类,枚举类,注解,包装类)

文章目录 第08章_面向对象编程(高级)本章专题与脉络1. 关键字&#xff1a;static1.1 类属性、类方法的设计思想1.2 static关键字1.3 静态变量1.3.1 语法格式1.3.2 静态变量的特点1.3.3 举例1.3.4 内存解析 1.4 静态方法1.4.1 语法格式1.4.2 静态方法的特点1.4.3 举例 1.5 练习 …

UI设计中的插画运用优势(下)

6. 插画赋予设计以美学价值&#xff0c;更容易被接受 即使所有人都在分析和争论产品的可用性和易用性&#xff0c;大家在对美的追求上&#xff0c;始终保持着一致的态度。一个设计是否具备可取性&#xff0c;是否能够通过甲方、客户和实际用户&#xff0c;是每个设计人都需要面…

高频一体式读写器的应用及其原理

高频一体式读写器作为一款读写设备&#xff0c;将RFID读写模块和天线集于一体&#xff0c;通过天线与RFID标签进行无线通信&#xff0c;实现对标签的识别和内存数据的读出或写入操作。具备安全、准确、快速、扩展、兼容性强等特点&#xff0c;具备非接触识别、远距离识别、环境…

Laravel 10.x 里如何使用ffmpeg

原理上很简单&#xff0c;就是使用命令行去调用ffmpeg&#xff0c;然后分析一下输出是不是有错误。 安装 首先安装 symfony/process&#xff0c;主要用于包装一下&#xff0c;用来代替 exec, passthru, shell_exec and system 。 composer require symfony/process composer…

PowerShell install 一键部署grafana

grafana 前言 Grafana 是一款开源的数据可视化和监控仪表盘工具。它提供了丰富的数据查询、可视化和报警功能,可用于实时监控、数据分析和故障排除等领域。 通过 Grafana,您可以连接到各种不同的数据源,包括时序数据库(如 Prometheus、InfluxDB)和关系型数据库(如 MySQ…

linux性能优化-磁盘I_O优化

1.文件系统 1.1.文件系统的工作原理 文件系统是在磁盘的基础上&#xff0c;提供了一个用来管理文件的树状结构。 接下来我们就看看Linux 文件系统的工作原理。 1.1.1索引节点和目录项 在 Linux 中一切皆文件 ,文件系统,本身是对存储设备上的文件&#xff0c;进行组织管理的…

【Linux】—— 共享内存

本期我将要带大家学习的是有关进程间通信的另一种方式——共享内存。共享内存是一种用于进程间通信的高效机制&#xff0c;允许多个进程访问和操作同一块内存区域。 目录 &#xff08;一&#xff09;深刻理解共享内存 1.1 概念解释 1.2 共享内存原理 1.3 共享内存数据结构 …

基于SpringBoot的药品管理系统

文章目录 项目介绍主要功能截图&#xff1a;部分代码展示设计总结项目获取方式 &#x1f345; 作者主页&#xff1a;超级无敌暴龙战士塔塔开 &#x1f345; 简介&#xff1a;Java领域优质创作者&#x1f3c6;、 简历模板、学习资料、面试题库【关注我&#xff0c;都给你】 &…

数据治理能解决AI疲劳问题吗?

这篇文章强调了AI疲劳开始的两个阶段&#xff0c;并介绍了数据质量报告等数据治理措施如何能够推动构建值得信赖和健壮的模型。 数据治理和AI疲劳听起来像是两个不同的概念&#xff0c;但两者之间有着内在的联系。为了更好地理解它&#xff0c;让我们从它们的定义开始。 数据治…

JS 将字符串‘10.3%‘ 经过运算加2转换为 ‘12.3%‘

文章目录 需求分析 需求 已知 字符串 a ‘10.3%’&#xff0c;现需将转换为 字符串’12.3%’ 分析 去掉百分号&#xff0c;将字符串转换为数字 const aNumber parseFloat(10.3%); const resultNumber aNumber 2;将结果转换为带百分号的字符串 const resultString re…

灰度图像的自动阈值分割

第一种&#xff1a;Otsu &#xff08;大津法&#xff09; 一、基于cv2的API调用 1、代码实现 直接给出相关代码&#xff1a; import cv2 import matplotlib.pylab as pltpath r"D:\Desktop\00aa\1.png" img cv2.imread(path, 0)def main2():ret, thresh1 cv2.…

【CentOS】Linux 在线帮助文档命令:help、man 命令与文档汉化

目录 1、Linux 的命令行模式 2、help 命令 3、man 命令 4、man 命令输出文档汉化 注&#xff1a;本文档使用 Linux 版本为 CentOS 7.9 [swadianlocalhost ~]$ cat /etc/centos-release CentOS Linux release 7.9.2009 (Core) 1、Linux 的命令行模式 一般情况下&#xff0…

喜讯 | 华院计算摘得“2023大数据产业年度创新技术突破”奖

2024年1月17日&#xff0c; 由数据猿和上海大数据联盟主办&#xff0c;上海市经济和信息化委员会、上海市科学技术委员会指导的“第六届金猿季&魔方论坛——大数据产业发展论坛”在上海市四行仓库举行。论坛以“小趋势大未来”为主题&#xff0c;围绕大数据产业的各个领域展…

MySQL-SQL-DQL

DQL-介绍 DQL-语法 基本查询 1、查询多个字段 2、设置别名 3、去除重复记录 条件查询 1、语法 2、条件 聚合函数 1、介绍 2、常见的聚合函数 3、语法 分组查询 1、语法 2、where与having区别 排序查询 1、语法 2、排序方式 分页查询 1、语法 DQL-执行顺序

ubuntu 安装protobuf

apt 安装 sudo apt install protobuf-compiler 编译安装 – 方式1 资料链接&#xff1a;ubuntu环境 安装ncnn_ubuntu ncnn_jbyyy、的博客-CSDN博客 git clone https://github.com/google/protobuf.git cd protobuf git submodule update --init --recursive ./autogen.sh …

关于php8的数据类型转换

目录 1、数据类型介绍 1.1 简单数据类型&#xff1a; 1.2 复合数据类型&#xff1a; 1.3 特殊数据类型&#xff1a; 2、数据类型转换 2.1 自动转换&#xff1a; 2.2 强制&#xff08;手动&#xff09;转换&#xff1a; 3、验证数据类型途径 &#xff08;1&#xff09;var…

2024年第3周农产品价格监测报告

一、摘要 农产品价格监测主要涉及对畜禽类产品、水产品、蔬菜类产品、水果类产品的价格&#xff0c;以周为单位&#xff0c;进行变化情况的数据监测。其中&#xff0c;蔬菜类产品共8种&#xff0c;分别为菜花、韭菜、豆角、西红柿、胡萝卜、土豆、大葱、葱头。 本周重点监测的…