STM32第八节:位带操作——GPIO输出和输入

前言

        我们讲了GPIO的输出,虽然我们使用的是固件库编程,但是最底层的操作是什么呢?对,我们学习过51单片机的同学肯定学习过 sbit 修改某一位的高低电平,从而实现对于硬件的控制。那么我们现在在STM32中有没有相似的操作呢?答案肯定是有的。那么我们今天就来讲讲位带操作。


创作不易,点个三连不迷路!!!

STM32第八节:位带操作——GPIO输出和输入

位带

位带简介

P0=0xFE;    //总线操作

sbit LED1 = P0^0;   //位操作
LED1 = 0;

        如此这般,就是总线操作与位操作的区别(在51单片机中)。那么我们在32中该如何操作呢          位操作就是可以单独的对一个bit位进行读和写的过程。51 单片机中通过关键字 sbit 来实现位定义,而 STM32 是通过访问位带别名区来实现。
        在 STM32 中,有两个地方实现了位带,一个是 SRAM 区的最低 1MB 空间,另一个是外设区最低 1MB 空间。这两个 1MB 的空间除了可以像正常的 RAM 一样操作外,他们还有自己的位带别
名区,位带别名区把这 1MB 的空间的每一个位膨胀成一个 32 位的字,当访问位带别名区的这些
字时,就可以达到访问位带区某bit位的目的。

        如图所示, 我们以ODR寄存器为例,而位带操作就是把寄存器中的每一个位都重新找了个地址。这个地址在位带别名区内,而且在位带别名区里会膨胀成4个字节,但是操作的时候只有最低位有效(ODR0)。

位带区分布 

         在位带操作中,不止止是片上外设会有位带操作,而且SRAM也会有1MB的位带区,位带区里面的每一个位都可以通过位带别名区的地址来访问(一位为四个字节)。

         结合上述例子,我们知道GPIO_ODR的基地址,那么我们怎么知道每一位所对应的地址呢?那么我们就有了位带区与位带别名区地址转换。

位带区与位带别名区地址转换

位带区地址与位带别名区的地址之间的转换        

         接着我就来给大家讲解一下转换的公式的具体含义及代码展示。

        前一个呢是位带别名区地址,0x 4200 0000 和 0x 2200 0000.接着呢就是(A-0x 2000 0000),这个算出来的是偏移地址(字节)。一个字节有八个位,而一个位是四个字节。n*4为字节的偏移数,n为位号。从公式来看,我们只需要知道A以及n就可以算出位带地址。再编程上来说,可以统一用一个公式表示:

// AliasAddr= =0x42000000+ (A-0x40000000)*8*4 +n*4
// AliasAddr= =0x22000000+ (A-0x20000000)*8*4 +n*4

// 把“位带地址 + 位序号”转换成别名地址的宏
#define BITBAND(addr, bitnum) 
    ((addr & 0xF0000000)+0x02000000+((addr & 0x00FFFFFF)<<5)+(bitnum<<2))

         为了方便操作,我们可以把这两个公式合并成一个公式,把“位带地址 + 位序号”转换成别名区地址统一成一个宏。

        addr & 0xF0000000 是为了区别 SRAM 还是外设,实际效果就是取出 4 或者 2,如果是外设,则取出的是 4,+0X02000000 之后就等于 0X42000000,0X42000000 是外设别名区的起始地址。如果是 SRAM,则取出的是 2,+0X02000000 之后就等于 0X22000000,0X22000000 是 SRAM 别名区的起始地址。addr & 0x00FFFFFF 屏蔽了高三位,相当于减去 0X20000000 或者 0X40000000。

        外设的最高地址是:0X20100000,跟起始地址 0X20000000 相减的时候,总是低5位才有效,所以就把高三位屏蔽掉来达到减去起始地址的效果,具体屏蔽掉多少位跟最高地址有关。SRAM 同理分析。«5 相当于 *8*4,«2 相当于 *4。

        最后我们就可以通过指针的形式操作这些位带别名区地址,最终实现位带区的bit位操作。

// 把一个地址转换成一个指针
#define MEM_ADDR(addr) *((volatile unsigned long *)(addr))

// 把位带别名区地址转换成指针
#define BIT_ADDR(addr, bitnum) MEM_ADDR(BITBAND(addr, bitnum))

编写程序——实现GPIO上的位带输出操作

使用位带的方式访问GPIO的ODR寄存器

        拷贝一份上节课的代码,并稍作修改,使用条件编译使得上一部分代码编译,下一部分代码不编译。我们看到上一部分代码是使得LED2(即绿色)自动亮灭亮灭,中间稍作迟缓:

#include "stm32f10x.h"   // 相当于51单片机中的  #include <reg51.h>
#include "bsp_led.h"
#include "bsp_key.h"

void Delay(uint32_t count)
{
	for(;count!=0;count--);
}

int main(void)
{	
	LED_GPIO_Config();
	LED_KEY_Config();
#if 1
	while(1)
	{
		//GPIO_SetBits(LED_G_GPIO_PORT,LED_G_GPIO_PIN);
		LED2(OFF);
		Delay(0xFFFFF);
		//GPIO_ResetBits(LED_G_GPIO_PORT,LED_G_GPIO_PIN);
		LED2(ON);
		Delay(0xFFFFF);
	}
#else
	while(1)                            
	{	   
		if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON  )
		{
			LED1(ON);
		}
		if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON  )
		{
			LED2_TOGGLE;
		}	
	}
#endif
}

        那么输出怎么写呢?我们现在开始操作ODR寄存器,从公式上看,我们要先写进去通用的公式,然后用n代替参量写一个带参宏出来,然后强制转换为地址,并使用指针操作加上 * 。然后再定义一个GPIOB_ODR_Addr用来表示(GPIOB_BASE+0x0c)基地址加偏移量:

#define GPIOB_ODR_Addr       (GPIOB_BASE+0x0c)
#define PBOut(n)             *(unsigned int*)((GPIOB_ODR_Addr & 0xF0000000)+0x02000000+((GPIOB_ODR_Addr & 0x00FFFFFF)<<5)+(n<<2))

        然后我们就可以在main函数中做修改,如此这般,我们就通过位带操作实现了操作寄存器从而实现LED灯的亮灭:

while(1)
	{
		PBOut(0) = 1;
		Delay(0xFFFFF);
		PBOut(0) = 0;
		Delay(0xFFFFF);
	}

完整代码展示

#include "stm32f10x.h"   // 相当于51单片机中的  #include <reg51.h>
#include "bsp_led.h"
#include "bsp_key.h"

#define GPIOB_ODR_Addr       (GPIOB_BASE+0x0c)
#define PBOut(n)             *(unsigned int*)((GPIOB_ODR_Addr & 0xF0000000)+0x02000000+((GPIOB_ODR_Addr & 0x00FFFFFF)<<5)+(n<<2))

void Delay(uint32_t count)
{
	for(;count!=0;count--);
}

int main(void)
{	
	LED_GPIO_Config();
	LED_KEY_Config();
	
#if 1
	while(1)
	{
		PBOut(0) = 1;
		Delay(0xFFFFF);
		PBOut(0) = 0;
		Delay(0xFFFFF);
	}
#else
	while(1)                            
	{	   
		if(Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN)==KEY_ON)
			LED1(ON);
		if(Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN)==KEY_ON)
			LED2_TOGGLE;
	}
#endif
}

 同样的原理打开别的颜色的灯

        从原理图可知,我们要打开BLUE灯,就是打开PB1口,即把PBOut(0)改为PBOut(1)即可,其他的以此类推。

编写程序——实现GPIO上的位带输入操作

使用位带的方式访问GPIO的IDR寄存器

        输入的话就该操作下面一部分代码了。首先我们要算出IDR的这个第0位地址,为:

#define GPIOA_IDR_Addr       (GPIOA_BASE+0x08)
#define PAin(n)              *(unsigned int*)((GPIOA_IDR_Addr & 0xF0000000)+0x02000000+((GPIOA_IDR_Addr & 0x00FFFFFF)<<5)+(n<<2))

#define GPIOC_IDR_Addr       (GPIOA_BASE+0x08)
#define PCin(n)              *(unsigned int*)((GPIOC_IDR_Addr & 0xF0000000)+0x02000000+((GPIOC_IDR_Addr & 0x00FFFFFF)<<5)+(n<<2))

         那么我们发现我们写的Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN)和Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN)函数就没有用了,我们在这里修改为:

while(1)                            
	{	   
//		if(Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN)==KEY_ON)
//			LED1(ON);
//		if(Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN)==KEY_ON)
//			LED2_TOGGLE;
		if(PAin(0)==KEY_ON)
		{
			while(PAin(0)==KEY_ON);
			LED1(ON);
		}
		if(PCin(13)==KEY_ON)
		{
			while(PCin(13)==KEY_ON);
			LED2_TOGGLE;
		}
	}

        这里我们使用了两种方式来实现按键控制LED灯的亮灭,分别控制PA0和PC13端口。 

完整代码展示

#include "stm32f10x.h"   // 相当于51单片机中的  #include <reg51.h>
#include "bsp_led.h"
#include "bsp_key.h"

#define GPIOA_IDR_Addr       (GPIOA_BASE+0x08)
#define PAin(n)              *(unsigned int*)((GPIOA_IDR_Addr & 0xF0000000)+0x02000000+((GPIOA_IDR_Addr & 0x00FFFFFF)<<5)+(n<<2))

#define GPIOC_IDR_Addr       (GPIOA_BASE+0x08)
#define PCin(n)              *(unsigned int*)((GPIOC_IDR_Addr & 0xF0000000)+0x02000000+((GPIOC_IDR_Addr & 0x00FFFFFF)<<5)+(n<<2))

int main(void)
{	
	LED_GPIO_Config();
	LED_KEY_Config();
	while(1)                            
	{	   
		if(PAin(0)==KEY_ON)
		{
			while(PAin(0)==KEY_ON);
			LED1(ON);
		}
		if(PCin(13)==KEY_ON)
		{
			while(PCin(13)==KEY_ON);
			LED2_TOGGLE;
		}
	}
}

小结

        到这里我们的课程就结束啦,从下一节开始就是中级篇的讲解了,我们下次见咯!


创作不易,点个三连不迷路!!!

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

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

相关文章

【数据结构】Set的使用

文章目录 一、Set的使用1.Set的常用方法&#xff1a;1.boolean add(E e)2.void clear()3.boolean contains(Object o)4.boolean remove(Object o)5.int size()6.boolean isEmpty()7.Object[] toArray()8.boolean containsAll(Collection<?> c)9.boolean addAll(Collecti…

学习Vue(1)环境搭建与运行一个vue项目

下载node.js 下载地址&#xff1a;下载 | Node.js 中文网 安装 双击下载好的安装文件&#xff0c;选择安装路径即可。 安装完成&#xff0c;输入命令&#xff1a;nodel -v&#xff0c;查看版本&#xff0c;正常显示版本即安装成功。 自定义全局安装路径和缓存路径&#xff0…

SpringCloud-深度理解ElasticSearch

一、Elasticsearch概述 1、Elasticsearch介绍 Elasticsearch&#xff08;简称ES&#xff09;是一个开源的分布式搜索和分析引擎&#xff0c;构建在Apache Lucene基础上。它提供了一个强大而灵活的工具&#xff0c;用于全文搜索、结构化搜索、分析以及数据可视化。ES最初设计用…

用Origin快速拟合荧光寿命、PL Decay (TRPL)数据分析处理

需要准备材料&#xff1a;Origin、PL Decay数据txt文件 首先打开Origin画图软件 导入数据&#xff0c;按照下图箭头操作直接导入 双击你要导入的PL Decay的txt数据文件&#xff0c;然后点OK 继续点OK 数据导入后首先删除最大光子数之前的无效数据&#xff0c;分析的时候用…

每天五分钟计算机视觉:图像数据不足带来的问题和解决办法

本文重点 在当今的数字时代,图像数据的应用已经渗透到各个领域,包括但不限于计算机视觉、机器学习、自动驾驶、医疗诊断等。然而,当图像数据不足时,会引发一系列问题,对相关应用产生负面影响。 尤其是计算机视觉领域,图像数据尤为珍贵和稀缺,如果计算机视觉的任务中,如…

政务云安全风险分析与解决思路探讨

1.1概述 为了掌握某市政务网站的网络安全整体情况&#xff0c;在相关监管机构授权后&#xff0c;我们组织人员抽取了某市78个政务网站进行安全扫描&#xff0c;通过安全扫描&#xff0c;对该市政务网站的整体安全情况进行预估。 1.2工具扫描结果 本次利用漏洞扫描服务VSS共扫…

基于Spring Boot的疗养院管理系统的设计与实现

传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff0c;在计算机上安装疗养院管理系统软件来发挥其高效地信息处理的作用&#xff0c;可以…

使用html+css制作一个发光立方体特效

使用htmlcss制作一个发光立方体特效 <!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>Documen…

【机器学习300问】36、什么是集成学习?

一、什么是集成学习&#xff1f; &#xff08;1&#xff09;它的出现是为了解决什么问题&#xff1f; 提高准确性&#xff1a;单个模型可能对某些数据敏感或者有概念偏见&#xff0c;而集成多个模型可以提高预测的准确性。让模型变稳定&#xff1a;一些模型&#xff0c;如决策…

Css基础——溢出文字省略号表达

1. 单行文本溢出显示省略号&#xff1a; 1.1、方法&#xff1a; 1. 先强制一行内显示文本*/ white-space: nowrap; &#xff08; 默认 normal 自动换行&#xff09; 2. 超出的部分隐藏*/ overflow: hidden; 3. 文字用省略号替代超出的部分*/ text-overflow: ellipsis; 1.2、代…

【他山之石】BlueLM蓝心大模型的落地

note 预训练&#xff1a;我们采用了混合精度训练以及梯度缩放策略&#xff0c;因为这可以减少训练周期&#xff0c;训练周期较少会节省模型训练所需的时间和资源。选择回放训练能够让模型定向学会一类知识信息。Loss 预测确保在训练过程前对损失进行估计&#xff0c;以保证训练…

javaweb day16 mysql

mysql 安装&#xff1a; 企业开发使用方法 安装虚拟机代替服务器 数据模型 创建数据库 写法 sql简介

KNN算法对鸢尾花进行分类:添加网格搜索和交叉验证

优化——添加网格搜索和交叉验证 from sklearn.datasets import load_iris from sklearn.model_selection import train_test_split from sklearn.preprocessing import StandardScaler from sklearn.neighbors import KNeighborsClassifier from sklearn.model_selection imp…

GNN/GCN自己学习

一、图的基本组成 V&#xff1a;点&#xff08;特征&#xff09; E&#xff1a;边 U&#xff1a;图&#xff08;全局特征&#xff09; 二、用途 整合特征&#xff08;embedding)&#xff0c;做重构 三、邻接矩阵 以图像为例&#xff0c;每个像素点周围都有邻居&#xff0c;…

Composer创建ThinkPHP无法获取最新版本的问题

composer安装TP&#xff08;截止目前最新版本为8.0&#xff09; composer create-project topthink/think tp 一开始直接给我安装成TP6了&#xff0c;原因就是我系统的PHP版本不是8.0以上&#xff0c;所以不支持最新的TP版本&#xff0c;就会默认安装之前稳定的版本。解决这个…

说说JVM的垃圾回收机制

简介 垃圾回收机制英文为Garbage Collection, 所以我们常常称之为GC。那么为什么我们需要垃圾回收机制呢&#xff1f;如果大家有了解过Java虚拟机运行时区域的组成(JVM运行时存在&#xff0c;本地方法栈&#xff0c;虚拟机方法栈&#xff0c;程序计数器&#xff0c;堆&#xf…

linux安全--CentOS7安装Tomcat,远程管理ManagerApp

目录 1.Tomcat安装 2.Tomcat远程管理 1.Tomcat安装 下载安装包并解压 tar xf apache-tomcat-7.0.54.tar.gz -C /usr/local/apache-tomcat_7.0.54/tomcat启停 启动 ./startup.sh 停止 ./shutdown.sh 2.Tomcat远程管理 找到tomcat文件夹中webapps/manager/META-INF/contex…

(黑马出品_高级篇_04)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式

&#xff08;黑马出品_高级篇_04&#xff09;SpringCloudRabbitMQDockerRedis搜索分布式 微服务技术——可靠性消息服务 今日目标服务异步通信-高级篇1.消息可靠性1.1.生产者消息确认1.1.1.修改配置1.1.2.定义Return回调1.1.3.定义ConfirmCallbac…

temu英国电商市场洞察:2月份商品销售数据分析Python数据采集Api

文章目录 引言可视化分析数据展示商品类目占比分析销售额分析价格区间占比分析各类目新品占比分析 写在最后 引言 在美国市场稳步增长基础上&#xff0c;Temu也不断加快全球步伐&#xff0c;现已进入47个国家&#xff0c;光23年9月份就进驻了10个国家&#xff0c;创造单月扩张国…

FileZillaClient连接被拒绝,无法连接

1.ECONNREFUSED - 连接被服务器拒绝 2、无法连接FZ时&#xff0c;判断没有ssh 更新源列表&#xff1a; sudo apt-get update 安装 openssh-server &#xff1a;sudo apt-get install openssh-server 查看是否启动ssh&#xff1a;sudo ps -e | grep ssh