1.GPIO

news2025/1/14 18:02:38

理论说明

输入

上拉输入:拉高电平

下拉输入:拉低电平

浮空输入:不拉高也不拉低电平

输出

开漏输出:不能输出高电平(P-MOS不可用,则只能低电平)

推挽输出:可输出高低电平

输入输出电路图

输出速率

  • Low:2MHz
  • Medium:10MHz
  • High:50MHz

代码编写

几个功能例程的配置及代码

Cube IDE代码

LED灯闪烁

配置引脚功能:

GPIO配置
在main.c文件,大概94行开始

  /* USER CODE BEGIN WHILE */
while (1)
{
	//参数1:GPIO口名字(PB),参数2:GPIO数字(PB5中的5),参数3:高低电平(0,1)
	HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 0);
	HAL_Delay(1000);
	HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 1);
	
	//HAL_GPIO_TogglePin(LED1_GPIO_Port, LED1_Pin);	//电平反转,无需设置高低电平
	HAL_Delay(1000);
	/* USER CODE END WHILE */
	
	/* USER CODE BEGIN 3 */
}
	/* USER CODE END 3 */

实验二

按键PE3控制蜂鸣器开关、按键PE4控制 LED2(PE5) 亮灭、光敏传感器控制 LED1(PB5) 亮灭

按键:设置成上拉输入,PE4则为高电平,当按键按下时,PE4为0,则按键按下(其它按钮同理)

按键电路图

LED灯、蜂鸣器(PB8):输出高低电平,设置推挽输出

光敏传感器(PB11):高低电平由外界光照强度来决定,设置浮空输入

按键配置:

按钮配置

LED灯配置方面已经说明,直接配置

蜂鸣器配置(PB8):

蜂鸣器配置图

光敏传感器配置(PB11):

光敏传感器配置图

光敏传感器AO DO有什么区别?

AO:输出模拟信号

DO:输出数字信号(高低电平)

在main.c文件,大概94行开始

/* USER CODE BEGIN WHILE */
while (1)
{
	if(0 == HAL_GPIO_ReadPin(Key1_GPIO_Port, Key1_Pin))	//检测读取按键高低电平
	{
		HAL_Delay(120);
		if(0 == HAL_GPIO_ReadPin(Key1_GPIO_Port, Key1_Pin))	//检测读取按键高低电平
		{
			HAL_GPIO_TogglePin(BEEP_GPIO_Port, BEEP_Pin);	//电平反转,无需设置高低电平
		}
	}

	if(0 == HAL_GPIO_ReadPin(Key2_GPIO_Port, Key2_Pin))	//检测读取按键输入高低电平
	{
		HAL_Delay(120);
		if(0 == HAL_GPIO_ReadPin(Key2_GPIO_Port, Key2_Pin))	//检测读取按键高低电平
		{
			HAL_GPIO_TogglePin(LED2_GPIO_Port, LED2_Pin);	//电平反转,无需设置高低电平
		}
	}

	if(0 == HAL_GPIO_ReadPin(SUN_GPIO_Port, SUN_Pin))	//检测读取光敏传感器输入高低电平
	{
		HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 1);
	}else{
		HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 0);
	}
   /* USER CODE END WHILE */

   /* USER CODE BEGIN 3 */
}

效果展示:

代码过程

初始化时钟

  1. 初始化时钟

APB2ENR:外设时钟使能寄存器

外设时钟使能寄存器图

大概功能:将APB2ENR(外设时钟使能寄存器)初始化IO端口B时钟开启

//初始化代码,在gpio.c中48行
__HAL_RCC_GPIOE_CLK_ENABLE();
__HAL_RCC_GPIOB_CLK_ENABLE();
__HAL_RCC_GPIOA_CLK_ENABLE();

//__HAL_RCC_GPIOB_CLK_ENABLE(); 如下过程
#define __HAL_RCC_GPIOB_CLK_ENABLE()   do { \
			 __IO uint32_t tmpreg;  /*定义一个变量*/\
			SET_BIT(RCC->APB2ENR, RCC_APB2ENR_IOPBEN);\
			/* Delay after an RCC peripheral clock enabling(下方两个延时作用) */\
			tmpreg = READ_BIT(RCC->APB2ENR, RCC_APB2ENR_IOPBEN);\
			UNUSED(tmpreg); \
	} while(0U)
	
//SET_BIT(RCC->APB2ENR, RCC_APB2ENR_IOPBEN); 的解释

//1.RCC_APB2ENR_IOPBEN
#define RCC_APB2ENR_IOPBEN_Pos		(3U)                              
#define RCC_APB2ENR_IOPBEN_Msk		(0x1UL << RCC_APB2ENR_IOPBEN_Pos)	/*RCC_APB2ENR_IOPBEN_Msk 等于0x1<< 3 结果:0x00000008 */
//0x1 = 0000 0001 << 3 = 0000 1000(对应上图,位3为1) = 0x00000008 
#define RCC_APB2ENR_IOPBEN			RCC_APB2ENR_IOPBEN_Msk		/* RCC_APB2ENR_IOPBEN 等于0x00000008 */

//2.RCC寄存器,RCC_BASE强转成这个RCC_TypeDef类型的结构体
#define RCC                 ((RCC_TypeDef *)RCC_BASE)
//RCC_BASE = 由总线矩阵偏移到AHB系统总线再偏移到复位和时钟控制(RCC)(可看参考手册的系统结构)(下面三行从下往上看)
#define RCC_BASE              (AHBPERIPH_BASE + 0x00001000UL)
#define AHBPERIPH_BASE        (PERIPH_BASE + 0x00020000UL)
#define PERIPH_BASE           0x40000000UL
//RCC_TypeDef(结构体)
typedef struct
{
  __IO uint32_t CR;
  __IO uint32_t CFGR;
  __IO uint32_t CIR;
  __IO uint32_t APB2RSTR;
  __IO uint32_t APB1RSTR;
  __IO uint32_t AHBENR;
  __IO uint32_t APB2ENR;
  __IO uint32_t APB1ENR;
  __IO uint32_t BDCR;
  __IO uint32_t CSR;
} RCC_TypeDef;

//3.RCC->APB2ENR
RCC指向APB2ENR,则就是结构体内的__IO uint32_t APB2ENR;

//将APB2ENR的位3置1
  1. 初始化引脚

GPIOx_BSRR:端口位设置/清除寄存器

低16位中若有某一位 置1,则这个引脚为1(高电平),若都为0,则不影响

高16位中若有某一位 置1,则这个引脚位0(低电平),若都为0,则不影响

端口位设置/清除寄存器图

//gpio.c中第53行
/*Configure GPIO pin Output Level */
HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_SET);

void HAL_GPIO_WritePin(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin, GPIO_PinState PinState)
{
  /* Check the parameters */
  assert_param(IS_GPIO_PIN(GPIO_Pin));
  assert_param(IS_GPIO_PIN_ACTION(PinState));

  if (PinState != GPIO_PIN_RESET)	//GPIO_PIN_RESET为0
  {
    GPIOx->BSRR = GPIO_Pin;	//给BSRR寄存器 低16位赋值 置1
  }
  else
  {
    GPIOx->BSRR = (uint32_t)GPIO_Pin << 16u;	//给BSRR寄存器 高16位赋值 置0
  }
}

//GPIO_TypeDef *GPIOx 等于 LED2_GPIO_Port
#define LED2_GPIO_Port GPIOE
#define GPIOE		((GPIO_TypeDef *)GPIOE_BASE)
//GPIO_TypeDef 强转
typedef struct
{
  __IO uint32_t CRL;
  __IO uint32_t CRH;
  __IO uint32_t IDR;
  __IO uint32_t ODR;
  __IO uint32_t BSRR;
  __IO uint32_t BRR;
  __IO uint32_t LCKR;
} GPIO_TypeDef;
#define GPIOE_BASE            (APB2PERIPH_BASE + 0x00001800UL)
#define APB2PERIPH_BASE       (PERIPH_BASE + 0x00010000UL)
#define PERIPH_BASE           0x40000000UL

//GPIO_Pin 等于 LED2_Pin
#define LED2_Pin GPIO_PIN_5
#define GPIO_PIN_5		((uint16_t)0x0020)  /* Pin 5 selected    */

  1. 写实际需求(输入:read 判断,输出:write 0/1)

Keil代码

修改的地方如下

main.c

//main函数中
while (1)
{
	if(0 == HAL_GPIO_ReadPin(Key1_GPIO_Port, Key1_Pin))	//检测读取按键高低电平
	{
		HAL_Delay(120);
		if(0 == HAL_GPIO_ReadPin(Key1_GPIO_Port, Key1_Pin))	//检测读取按键高低电平
		{
			HAL_GPIO_TogglePin(BEEP_GPIO_Port, BEEP_Pin);	//电平反转,无需设置高低电平
		}
	}

	if(0 == HAL_GPIO_ReadPin(Key2_GPIO_Port, Key2_Pin))	//检测读取按键输入高低电平
	{
		HAL_Delay(120);
		if(0 == HAL_GPIO_ReadPin(Key2_GPIO_Port, Key2_Pin))	//检测读取按键高低电平
		{
			HAL_GPIO_TogglePin(LED2_GPIO_Port, LED2_Pin);	//电平反转,无需设置高低电平
		}
	}

	if(0 == HAL_GPIO_ReadPin(SUN_GPIO_Port, SUN_Pin))	//检测读取光敏传感器输入高低电平
	{
		HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 1);
	}
	else
	{
		HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, 0);
	}
}

gpio.c

void MDK_GPIO_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStruct = {0};

  /* GPIO Ports Clock Enable */
  __HAL_RCC_GPIOE_CLK_ENABLE();
  __HAL_RCC_GPIOB_CLK_ENABLE();
  __HAL_RCC_GPIOA_CLK_ENABLE();

  /*Configure GPIO pin Output Level */
  HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_SET);

  /*Configure GPIO pin Output Level */
  HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, GPIO_PIN_SET);

  /*Configure GPIO pin Output Level */
  HAL_GPIO_WritePin(BEEP_GPIO_Port, BEEP_Pin, GPIO_PIN_RESET);

  /*Configure GPIO pins : PEPin PEPin */
  GPIO_InitStruct.Pin = Key1_Pin|Key2_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  HAL_GPIO_Init(GPIOE, &GPIO_InitStruct);

  /*Configure GPIO pin : PtPin */
  GPIO_InitStruct.Pin = LED2_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStruct.Pull = GPIO_NOPULL;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  HAL_GPIO_Init(LED2_GPIO_Port, &GPIO_InitStruct);

  /*Configure GPIO pin : PtPin */
  GPIO_InitStruct.Pin = SUN_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
  GPIO_InitStruct.Pull = GPIO_NOPULL;
  HAL_GPIO_Init(SUN_GPIO_Port, &GPIO_InitStruct);

  /*Configure GPIO pin : PtPin */
  GPIO_InitStruct.Pin = LED1_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStruct.Pull = GPIO_NOPULL;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  HAL_GPIO_Init(LED1_GPIO_Port, &GPIO_InitStruct);

  /*Configure GPIO pin : PtPin */
  GPIO_InitStruct.Pin = BEEP_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  HAL_GPIO_Init(BEEP_GPIO_Port, &GPIO_InitStruct);
}

main.h

//60行开始
#define Key1_Pin GPIO_PIN_3
#define Key1_GPIO_Port GPIOE
#define Key2_Pin GPIO_PIN_4
#define Key2_GPIO_Port GPIOE
#define LED2_Pin GPIO_PIN_5
#define LED2_GPIO_Port GPIOE
#define SUN_Pin GPIO_PIN_11
#define SUN_GPIO_Port GPIOB
#define LED1_Pin GPIO_PIN_5
#define LED1_GPIO_Port GPIOB
#define BEEP_Pin GPIO_PIN_8
#define BEEP_GPIO_Port GPIOB

若烧录完成后,需要按复位按键才可正常,可尝试以下设置

第一步:

设置1

第二步:

设置2

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

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

相关文章

informer中DeltaFIFO机制的实现分析与源码解读

informer中的DeltaFIFO机制的实现分析与源码解读 DeltaFIFO作为informer中重要组件&#xff0c;本文从源码层面了解是如何DelatFIFO是实现的。 DeltaFIFO的定义 找到delta_fifo.go的源码&#xff0c;位于client-go/tools/cache/delta_fifo.go 代码结构大致如下: store定义…

【priority_queue的模拟实现】

priority_queue的模拟实现 小杨 容器适配器&#xff1a;priority_queue priority_queue即优先级队列是一种容器适配器&#xff0c;根据严格的弱排序标准(即排序规则可以更改),他的第一个元素总是它所包含的元素中最大的。优先队列将特定容器类封装作为其底层容器类。标准容器类…

ElasticSearch优化实战:打造高性能搜索引擎的秘籍

在当今这个大数据时代&#xff0c;信息的海量增长对搜索技术提出了前所未有的挑战。用户不仅需要快速准确地从数以亿计的数据中找到所需信息&#xff0c;还希望搜索引擎能够提供个性化和智能化的搜索体验。ElasticSearch作为市场上领先的搜索引擎&#xff0c;因其强大的全文搜索…

Typora2024最新版破解方法(亲测可用)

此方法非常简单&#xff0c;无需安装dll补丁&#xff0c;无需修改注册表&#xff0c;无需使用老版本。仅需修改部分文件内容即可 方法步骤 步骤一 下载并安装Typora 安装Typora 打开官网 下载并安装最新版即可 点击访问Typora官网 https://typoraio.cn/ 步骤二 修改文件 …

java面向对象重点总结

文章目录 java面向对象重点总结类与实例构造方法方法重载属性与修饰符封装继承多态重构抽象类接口抽象类和接口的区别&#xff1a;集合泛型 java面向对象重点总结 对象是一个自包含的实体&#xff0c;用一组可识别的特性和行为来标识。 面向对象编程&#xff0c;英文叫Object…

mcasttest-tool组播检测工具

作者&#xff1a;广大 检测组播 mcasttest-tool是oracle组播检测工具&#xff0c;组播是oracle 11.2.0.2开始的新功能。 1、上传mcasttest工具解压并授权 [rootrac1 soft]# cd /u01/soft/ [rootrac1 soft]# tar -xvf mcasttest.tgz[rootrac1 soft]# chown -R grid:oinstall…

Animate软件基础:“分散到图层”创建的新图层

FlashASer&#xff1a;AdobeAnimate2021软件零基础入门教程https://zhuanlan.zhihu.com/p/633230084 FlashASer&#xff1a;实用的各种Adobe Animate软件教程https://zhuanlan.zhihu.com/p/675680471 FlashASer&#xff1a;Animate教程及作品源文件https://zhuanlan.zhihu.co…

Pytorch实现线性回归Linear Regression

借助 PyTorch 实现深度神经网络 - 线性回归 - 第 2 周 | Coursera 线性回归预测 用PyTorch实现线性回归模块 创建自定义模块&#xff08;内含一个线性回归&#xff09; 训练线性回归模型 对于线性回归&#xff0c;特定类型的噪声是高斯噪声 平均损失均方误差函数&#xff1a…

Mallet:一款针对任意协议的安全拦截代理工具

关于Mallet Mallet是一款功能强大的协议安全分析工具&#xff0c;该工具支持针对任意协议创建用于安全审计的拦截代理&#xff0c;该工具本质上与我们所熟悉的拦截Web代理类似&#xff0c;只是通用性更强。 工具运行机制 Mallet建立在Netty框架之上&#xff0c;并且依赖于Net…

文案人的梦工场,网易入职指南!

网易云对于咱们一些有点文艺的文案策划来说&#xff0c;简直就是梦中情司。 在这里工作锻炼机会很多&#xff0c;也很开拓眼界&#xff0c;能获得相当于在别处3倍能力的成长速度&#xff0c;福利待遇也是很好的。 要进入网易云音乐做文案策划&#xff0c;你可以按照以下步骤进…

数据结构的基本概念与算法2

线性表 &#xff1a; 线性表 是具有相同数据类型的 n (n > 0) 个数据元素的有限序列&#xff0c;其中 n 为表长&#xff0c;当 n 0 时线性表是一个空表&#xff1b;若用 L 命名线性表&#xff0c;则其一般表示为&#xff1a;L (a1 , a2 , ... ai , ai1 , ... an) 上述中&a…

月木学途开发 2.项目架构

1.项目介绍 月木学途是一款it在线学习网站&#xff0c;项目采用前后端分离架构。前端开发主要使用vue.js&#xff0c;后端使用Spring Cloud Alibaba技术栈。项目包含学习网站的大部分功能&#xff0c;分为管理员端和用户端。管理员端有权限管理、课程管理、网站管理、求职模块管…

Shell函数和Shell 输入/输出重定向

LInux&#xff1a;Shell函数和Shell 输入/输出重定向 Shell函数 参数说明&#xff1a; 可以带function fun() 定义&#xff0c;也可以直接fun() 定义,不带任何参数。参数返回&#xff0c;可以显示加&#xff1a;return 返回&#xff0c;如果不加&#xff0c;将以最后一条命令运…

[Vue warn]: data functions should return an object:

仔细检查你的代码肯定有一个data()内忘记方return{}了

C语言程序设计23

《C程序设计教程&#xff08;第四版&#xff09;——谭浩强》 例题2.11 从键盘输入B、O、Y三个字符&#xff0c;然后把他们输出到屏幕上 代码&#xff1a; //《C程序设计教程&#xff08;第四版&#xff09;——谭浩强》 //例题2.11 从键盘输入B、O、Y三个字符&#xff0c;然…

RabbitMQ:MQ的可靠性

MQ的可靠性 在默认情况下&#xff0c;RabbitMQ会将接收到的信息保存在内存中以降低消息收发的延迟。这样会导致两个问题: 一旦MQ宕机&#xff0c;内存中的消息会丢失 内存空间有限&#xff0c;当消费者故障或处理过慢时&#xff0c;会导致消息积压&#xff0c;引发MQ阻塞。 …

高效、安全、共享|济南市升级教育城域网,重塑教育网络生态

文/济南市电化教育馆 电教教研室主任 张承强 导语: 近年来,济南市教育局以前瞻性的视野,将教育数字化转型视为推动教育高质量发展的基石,全力加速教育现代化进程。在这一蓝图下,教育城域网的升级改造项目被赋予了基础性、先导性和战略性的重要意义,成为探索教育数字化转型新路…

一键搞定PDF翻译,这四款是职场达人常备翻译工具!!!

作为外贸搬砖人的一份子&#xff0c;虽说外语功底还说地过去&#xff0c;但是每天过目大量pdf文件的翻译&#xff0c;难免还有些吃力&#xff0c;这个时候如果有可以辅助翻译的工具那就再好不过了&#xff0c;今天给大家带来四款非常适合pdf文件翻译的工具&#xff0c;总有一款…

C#中的通信

上位机应用开发-串口通信1、基于C#的串口通信对象:SerialPort 2、字段属性 PortName:获取或设置通信端口 BaudRate:获取或设置串行波特率-DataBits:获取或设置每个字节的标准数据位长度 Parity:获取或设置奇偶校验检查协仪I-StopBits;获取或设置每个字节的标准停止位数 3、…

你需要的Node版本管理神器NVM

在做项目的时候&#xff0c;很多人本地的node都是装一个固定版本&#xff0c;一旦有些项目要下的依赖需要更高版本的node支持的时候&#xff0c;此时需要升级node就得把已经安装的低版本node卸载了&#xff0c;然后再重新下载、安装高版本的node,既费时间又抓狂&#xff0c;特别…