串口收发字符/字符串

news2024/12/4 0:47:40

分析过程:

        框图:

通过以上框图分析可知,需要分析芯片手册 RCC / GPIO / UART

1.RCC章节:使能对应GPIOG/GPIOB/UART4控制器

2.GPIO章节:1)设置引脚为复用功能模式 2)设置复用功能为串口功能

3.UART章节:1)串口初始化相关操作 2)数据收发

确定总线连接:

        

 

 基地址:

        

 

分析RCC章节的寄存器:

1.如何确定RCC_MP_AHB4ENSETR地址?

        RCC_MP_AHB4ENSETR地址 = 基地址 + 偏移地址 = 0x50000000 + 0xA28 = 0x50000A28

2.如何通过RCC_MP_AHB4ENSETR寄存器,设置GPIOB/GPIOG组控制器使能?

        PB2 / PG11 RCC_MP_AHB4ENSETR[1] = 1

        [0x50000A28]第1位写1 ------>GPIOB组控制器使能

        RCC_MP_AHB4ENSETR[6] = 1        

        [0x50000A28]第6位写1 ------>GPIOG组控制器使能

1.如何确定RCC_MP_APB1ENSETR地址?

        RCC_MP_APB1ENSETR地址 = 基地址 + 偏移地址 = 0x50000000 + 0xA00 = 0x50000A00

2.如何通过RCC_MP_APB1ENSETR寄存器,设置UART4组控制器使能?         RCC_MP_APB1ENSETR[16] = 1

        [0x50000A00]第16位写1 ------>UART4组控制器使能

GPIO章节的寄存器: 

GPIOx_MODER寄存器:GPIO模式寄存器  作用:设置GPIO引脚模式
1.确定GPIOB_MODER / GPIOG_MODER 地址是多少?
    GPIOB_MODER地址 = 基地址 + 偏移地址 = 0x50003000 + 0x00 = 0x50003000
    GPIOG_MODER地址 = 基地址 + 偏移地址 = 0x50008000 + 0x00 = 0x50008000
2.通过GPIOB_MODER寄存器设置PB2引脚为复用功能模式?
    GPIOB_MODER[5:4] = 10 
    0x50003000[5:4]  = 10
3.通过GPIOG_MODER寄存器设置PG11引脚为复用功能模式?
    GPIOG_MODER[23:22] = 10
    0x50008000[23:22]  = 10

GPIOx_AFRL寄存器功能:设置引脚为复用功能
1.确定GPIOB_AFRL地址是多少?
    GPIOB_AFRL地址 = 基地址 + 偏移地址 = 0x50003000 + 0x20 = 0x50003020
2.通过GPIOB_AFRL 寄存器设置PB2引脚为复用功能模式为UART4_RX?
    GPIOB_AFRL[11:8] = 1000
3.在查找对应的复用功能时,需要借助stm32mp157a.pdf中Table8和Table9
    D:\新平台\fsmp1a(学生资料)\stm32mp157-学生资料\stm32mp157-学生资料\01_参考资料\01.1_数据手册/stm32mp157a.pdf

 GPIOx_AFRH寄存器功能:设置引脚为复用功能
1.确定GPIOG_AFRH地址是多少?
    GPIOG_AFRH地址 = 基地址 + 偏移地址 = 0x50008000 + 0x24 = 0x50008024
2.通过GPIOG_AFRH寄存器设置PG11引脚为复用功能模式为UART4_TX?
    GPIOG_AFRH[15:12] = 0110
3.在查找对应的复用功能时,需要借助stm32mp157a.pdf中Table8和Table9
    D:\新平台\fsmp1a(学生资料)\stm32mp157-学生资料\stm32mp157-学生资料\01_参考资料\01.1_数据手册/stm32mp157a.pdf

USART章节:

 

通过以上章节分析可知,需要分析以下寄存器
1.USART_CR1:设置数据位宽度,以及将对应位使能
2.USART_CR2:设置停止位位数
3.USART_BRR:设置串口波特率
4.USART_TDR:发送数据寄存器
5.USART_RDR:接收数据寄存器
6.USART_PRESC:设置串口分频器
7.需要去寄存器中查找TXE和TC位标志,对应哪一个寄存器 ------>USART_ISR寄存器

USART_CR1寄存器作用:初始化串口
1.USART_CR1地址是多少?
    USART_CR1地址 = 基地址 + 偏移地址 = 0x40010000 + 0x00 = 0x40010000
2.USART_CR1寄存器需要配置内容
    USART_CR1[28][12] = 00 ------->设置8位数据位
    USART_CR1[15] = 0 -------> 设置16倍采样率
    USART_CR1[10] = 0 -------> 设置串口无奇偶校验位
    USART_CR1[3] = 1 -------> 设置串口发送器使能
    USART_CR1[2] = 1 -------> 设置串口接收器使能
    USART_CR1[20 = 1 -------> 设置串口使能

 USART_CR2寄存器作用:初始化串口
1.USART_CR2地址是多少?
    USART_CR2地址 = 基地址 + 偏移地址 = 0x40010000 + 0x04 = 0x40010004
2.USART_CR2寄存器需要配置内容
    USART_CR2[13:12] = 00 ------->设置串口1位停止位

 USART_BRR寄存器作用:设置串口波特率为115200
1.USART_BRR地址是多少?
    USART_BRR地址 = 基地址 + 偏移地址 = 0x40010000 + 0x0C = 0x4001000C
2.如果想知道USART_BRR寄存器需要设置的值,需要参考53.5.7章节,确认串口提供的主频64MHZ,并且设置波特率为115200
    USART_BRR = 64000000 / 115200 = 0x22B

 USART_ISR[7]:判断发送数据寄存器是否为空,为空才可以发送下一个字节数据,为满需要等待发送数据寄存器为空
读0:发送数据寄存器满,需要等待发送数据寄存器为空
读1:发送数据寄存器空,可以发送下一个字节数据


USART_ISR[6]:判断发送数据是否完成,表示一帧数据是否发送完成
读0:发送数据没有完成
读1:发送数据完成

USART_ISR[5]:判断接收数据寄存器是否不为空,只有接收到数据才可以读
读0:没有接收到数据
读1:接收到数据

作业:串口实现字符串、字符的收发:

main.c:

#include "uart4.h"

extern void printf(const char *fmt, ...);

void delay_ms(int ms)

{

	int i,j;

	for(i = 0; i < ms;i++)

		for (j = 0; j < 1800; j++);

}

int main()

{

	hal_uart_init();

	while(1)

	{		

		//hal_putchar(hal_recvchar()+1);

		hal_putstring(hal_recvstring());		

	}

	return 0;

}

头文件:

#ifndef __UART4__
#define __UART4__

//初始化函数
void hal_uart_init();

//发送一个字符
void hal_putchar(const char str);

//发送一个字符串
void hal_putstring(const char *string);

//接收一个字符串
char hal_recvchar();

//接收一个字符串
char *hal_recvstring();



#endif

 函数文件:

#include "uart4.h"
#include "stm32mp1xx_gpio.h"
#include "stm32mp1xx_rcc.h"
#include "stm32mp1xx_uart.h"

//初始化函数
void hal_uart_init(){
	//RCC
	RCC->MP_AHB4ENSETR |= (1 << 1);
	RCC->MP_AHB4ENSETR |= (1 << 6);

	RCC->MP_APB1ENSETR |= (1 << 16);
	//GPIO
	GPIOB->MODER &= ~(3 << 4);
	GPIOB->MODER |= (2 << 4);

	GPIOB->AFRL &= ~(15 << 8);
	GPIOB->AFRL |= (8 << 8);

	GPIOG->MODER &= ~(3 << 22);
	GPIOG->MODER |= (2 << 22);

	GPIOG->AFRH &= ~(0 << 12);
	GPIOG->AFRH |= (6 << 12);

	//UART4
	USART4->CR1 &= ~(1 << 28);
	USART4->CR1 &= ~(1 << 12);
	
	USART4->CR1 &= ~(1 << 10);

	USART4->CR2 &= ~(3 << 12);
	
	USART4->CR1 &= ~(1 << 15);
	
	USART4->PRESC = 0x0;
	
	USART4->BRR = 0x22B;
	
	USART4->CR1 |= (1 << 3);
	
	USART4->CR1 |= (1 << 2);
	
	USART4->CR1 |= (1 << 0);
}

//发送一个字符
void hal_putchar(const char str){
	//判断发送数据寄存器是否为空
	while(!(USART4->ISR & (0x1 << 7)));
	//将要发送的数据,放入到发送数据寄存器中
	USART4->TDR = str;
	//判断发送数据寄存器是否发送完成
	while(!(USART4->ISR & (0x1 << 6)));
}

//发送一个字符串
void hal_putstring(const char *string){
	//判断是否为'\0',一个字符一个字符进行发送
	int n = 0;
	while(1){
		while(!(USART4->ISR & (0x1 << 7)));
		USART4->TDR = *(string + n);
		while(!(USART4->ISR & (0x1 << 6)));
		if(*(string+n) == '\r'){
			break;
		}
		n ++;
	}
	hal_putchar('\n');
}

//接收一个字符
char hal_recvchar(){
	//判断接收数据寄存器种是否有数据可读
	while(!(USART4->ISR &(0x1 << 5)));
	//将接收数据寄存器的内容,赋值给ch
	static char ch;
	ch = USART4->RDR;
	return ch;
}

char *hal_recvstring(){
	static char *str = "\0";
	int n = 0;
	while(1){
		while(!(USART4->ISR &(0x1 << 5)));
		*(str+n) = USART4->RDR;
		hal_putchar(*(str+n));
		if(*(str+n) == '\r'){
			hal_putchar('\n');
			break;
		}
		n ++;
	}
	return str;
}

结果如下:

字符串:

 字符:

 

 

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

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

相关文章

Docker常用操作

1、单机&#xff08;非docker&#xff09;启动java程序&#xff1a;nohup java -jar springbootstudy.jar 2、启动docker&#xff1a;service docker start &#xff08;构建好以后&#xff0c;就不需要了&#xff1a;docker build -f Dockerfile -t springboot-jar .&#x…

确定因果随机森林的树木数量 the number of trees

前言 推断因果性和分析异质性是统计学家在处理混杂任务中的圣杯。传统且主流的方法有:倾向性评分、分层分享、比例风险模型等。新的方法也有很多,代表就是:因果随机森林。这种算法,浅看难度一般,深入探索发现坑还是很多的。这篇博客不对算法做深入探讨,仅仅是我在阅读文…

Hudi数据湖技术之快速体验

目录 1 编译 Hudi1.1 第一步、Maven 安装1.2 第二步、下载源码包1.3 第三步、添加Maven镜像1.4 第四步、执行编译命令1.5 第五步、Hudi CLI测试 2 环境准备2.1 安装HDFS2.2 安装Spark 3.x 3 spark-shell 使用3.1 启动spark-shell3.2 模拟数据3.3 插入数据3.4 查询数据3.5 表数据…

TVM: An Automated End-to-End Optimizing Compiler for Deep Learning

https://www.usenix.org/conference/osdi18/presentation/chen 文章目录 TVM: An Automated End-to-End Optimizing Compiler for Deep Learning引言1. 简介2. 总览3. 优化计算图4. 生成张量运算4.1 张量表达式和调度空间4.3 嵌套并行与协作4.3 张量化4.4 显式内存延迟隐藏 5 .…

2023年-测试工程师面试题(前期面试的题目)

背景&#xff1a;小型电商公司&#xff0c;薪资&#xff1a;8-11k&#xff0c;职位&#xff1a;测试工程师&#xff0c;学历&#xff1a;本科 打开微信小程序“casa品集”&#xff0c;找出该小程序存在哪些bug&#xff1f; 并列出「商品详情页」的测试用例A,B两张表&#xff0c…

[Golang] Go语言基础一知半解??这些你容易忽视的知识点(第一期)

&#x1f61a;一个不甘平凡的普通人&#xff0c;致力于为Golang社区和算法学习做出贡献&#xff0c;期待您的关注和认可&#xff0c;陪您一起学习打卡&#xff01;&#xff01;&#xff01;&#x1f618;&#x1f618;&#x1f618; &#x1f917;专栏&#xff1a;算法学习 &am…

液压控制系列之活塞位置测量(带原点标定功能)

液压轴位置控制详细内容请参看下面博客文章: 液压轴位置闭环控制(比例伺服阀应用)_RXXW_Dor的博客-CSDN博客液压阀的基础知识请参看下面的博客文章:PLC液压控制系列之比例流量阀结构分析_RXXW_Dor的博客-CSDN博客比例流量阀液压同步控制的PID闭环调节可以参看下面这篇博文三…

生命周期引入

实例&#xff1a;mouted&#xff1a;当vue完成模板解析并把真实的DOM元素放入页面后调用mounted 生命周期&#xff1a; 又名&#xff1a;生命周期回调函数、生命周期函数、生命周期钩子是什么&#xff1a;vue在关键时刻帮我们调用的一些特殊名称的函数生命周期函数的名字不可更…

CEF与Qt 结合注意事项

默认情况下&#xff0c;CEF 是/MT&#xff0c;而Qt是/MD 那么&#xff0c;如果你没有去在意MT、MD选项&#xff0c;那么极大可能性&#xff0c;会遇到程序崩溃的问题&#xff0c;并且表象会误导你查明原因的方向。并且&#xff0c;更换多个IDE也无法查明原因。 例子 当加载网…

【Linux】Linux开发工具

Linux开发工具 前言Linux编辑器 --- vimvim长啥样vim的基本概念vim的配置 Linux编译器 --- gcc/g编译和链接预处理编译汇编链接 细&#x1f512;链接静态库和动态库 Linux调试器 --- gdbLinux项目自动化构建工具 --- make/Makefile依赖关系和依赖方法 上方工具的简单示例 前言 …

MyBatis(二)—— 进阶

一、详解配置文件 1.1 核心配置文件 官方建议命名为mybatis-config.xml&#xff0c;核心配置文件里可以进行如下的配置&#xff1a; <environments> 和 <environment> mybatis可以配置多套环境&#xff08;开发一套、测试一套、、、&#xff09;&#xff0c; 在…

c实例练习笔记(拓展)

本博文参考题目的地址看右边----》C技能树 我跟你说&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;c语言有bool类型&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;你是不是百度跟你说没有只有c才有&#xff1f; 是有的&#xff01;&#xf…

第六章 完善内核

函数调用约定简介 参数的传递方式&#xff1a;使用后栈来保存。 每个进程都有自己的栈&#xff0c;这就是每个内存自己的专用内存空间&#xff1b;保存参数的内存地址不用再花精力维护&#xff0c;已经有栈机制来维护地址变化了&#xff0c;参数在栈中的位置可以通过栈顶的偏移…

QMS-云质说质量 - 1 张小泉的质量危机

云质QMS原创 转载请注明来源 作者&#xff1a;王洪石 引言 百年老店的拍蒜质量门 最近张小泉拍蒜断刀事件&#xff0c;吸引了全民关注&#xff0c;虽然随后发布了“断刀召集令”&#xff0c;但从事件发生到后续拖沓且不专业的应对&#xff0c;张小泉肯定是“失蒜”了。 张小泉…

一篇终结synchronized

一&#xff1a;基本原理 Java对象在内存中由两部分组成 &#xff1a; 1 是成员变量 2 是对象头&#xff0c;以32位虚拟机介绍&#xff1a;此时对象头是64位&#xff0c;即8字节 其中32个字节代表 mark word 另外32个字节代表klass word分别是什么意思呢&#xff1f; 1 klass …

文心一言 VS chatgpt (12)-- 算法导论3.1 6~7题

六、证明&#xff1a;一个算法的运行时间为θ(g(n))当且仅当其最坏情况运行时间为O(g(n))&#xff0c;且其最好情况运行时间为Ω(g(n)) 。 文心一言&#xff1a; chatgpt&#xff1a; 要证明「一个算法的运行时间为θ(g(n))当且仅当其最坏情况运行时间为O(g(n))&#xff0c;且…

LeetCode213 打家劫舍 II 动态规划法

题目地址 https://leetcode.cn/problems/house-robber-ii/ 你是一个专业的小偷&#xff0c;计划偷窃沿街的房屋&#xff0c;每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 &#xff0c;这意味着第一个房屋和最后一个房屋是紧挨着的。同时&#xff0c;相邻的房屋装…

【Hive实战】Windows下 IDEA DEBUG hiveMetastore hive2.3.9

Windows下 IDEA DEBUG hiveMetastore hive2.3.9 环境准备 编译好hive2.3.9源码&#xff0c;参考文档编译hive2.3.9源码准备好Mysql库&#xff0c;并手动创建schema&#xff0c;相关sql文件&#xff1a;hive-schema-2.3.0.mysql.sql和hive-txn-schema-2.3.0.mysql.sql。 启动…

SpringCloud --- Gateway服务网关

一、简介 Spring Cloud Gateway 是 Spring Cloud 的一个全新项目&#xff0c;该项目是基于 Spring 5.0&#xff0c;Spring Boot 2.0 和 Project Reactor 等响应式编程和事件流技术开发的网关&#xff0c;它旨在为微服务架构提供一种简单有效的统一的 API 路由管理方式。 二、为…

QT实现固高运动控制卡示波器

目录 一、固高示波器 二、基于QCustomPlot实现示波器 三、完整源码 一、固高示波器 固高运动控制卡自带的软件有一个示波器功能&#xff0c;可以实时显示速度的波形&#xff0c;可辅助分析电机的运行状态。但是我们基于sdk开发了自己的软件&#xff0c;无法再使用该功能&…