从单片机的启动说起一个单片机到点灯发生了什么下——使用GPIO点一个灯

news/2025/2/26 5:48:47/

目录

前言

GPIO%E7%9A%84%E6%8A%BD%E8%B1%A1-toc" name="tableOfContents" style="margin-left:40px">HAL库对GPIO的抽象

GPIO_Init-toc" name="tableOfContents" style="margin-left:40px">核心分析:HAL_GPIO_Init

前言

我们终于到达了熟悉的地方,对GPIO的初始化。经过漫长的铺垫,我们终于历经千辛万苦,来到了这里。关于GPIO的八种模式等更加详细的细节,由于只是点个灯,我们不做所有的分析。

GPIO%E7%9A%84%E6%8A%BD%E8%B1%A1" name="HAL%E5%BA%93%E5%AF%B9GPIO%E7%9A%84%E6%8A%BD%E8%B1%A1">HAL库对GPIO的抽象

HAL库对GPIO的抽象可以说是到了一个巅峰。使能一个GPIO,被化简到了一个非常显然的步骤。

  • 使能对应GPIO所在的Port的时钟

  • 设置GPIO对应的模式——点灯的时候,我们是推挽强力的控制外设

  • 使用HAL_GPIO_Init函数注册到寄存器当中

  • 做一些Post Init工作。比如说,我们明确的要求拉高拉低GPIO。这个,需要根据外设电路来实现

以我板子上外接的PA9作为一个例子把!

static void __open_gpioclk(){__HAL_RCC_GPIOF_CLK_ENABLE();
}
​
CCGPIOInitTypeDef led0_init = {.type = {.Pin    = GPIO_PIN_9,.Mode   = GPIO_MODE_OUTPUT_PP,.Speed  = GPIO_SPEED_HIGH,.Pull   = GPIO_PULLUP},.open_clock = __open_gpioclk,.post_init  = __post_init,.port = GPIOF
};
​
void configure_ccgpio(CCGPIOType* type,  CCGPIOInitTypeDef* initer)
{type->port = initer->port;type->pinType = initer->type.Pin;
​// oh shit, the open clock is missing initediniter->open_clock ? initer->open_clock() : __die();
​HAL_GPIO_Init(type->port, &initer->type);
​if(initer->post_init) initer->post_init(type);
}

GPIO_Init" name="%E6%A0%B8%E5%BF%83%E5%88%86%E6%9E%90%EF%BC%9AHAL_GPIO_Init">核心分析:HAL_GPIO_Init

我们只是简单的点个灯,这个函数就可被化简为如下的逻辑

void HAL_GPIO_Init(GPIO_TypeDef  *GPIOx, GPIO_InitTypeDef *GPIO_Init)
{uint32_t position;uint32_t ioposition = 0x00U;uint32_t iocurrent = 0x00U;uint32_t temp = 0x00U;
​/* 检查参数 */assert_param(IS_GPIO_ALL_INSTANCE(GPIOx));assert_param(IS_GPIO_PIN(GPIO_Init->Pin));assert_param(IS_GPIO_MODE(GPIO_Init->Mode));
​/* 配置GPIO引脚 */for(position = 0U; position < GPIO_NUMBER; position++){/* 获取IO引脚位置 */ioposition = 0x01U << position;/* 获取当前IO引脚的状态 */iocurrent = (uint32_t)(GPIO_Init->Pin) & ioposition;
​...
​/* 配置引脚为高电平来点亮LED */GPIOx->ODR |= iocurrent; // 设置为高电平}}
}

这就是为什么我们可以使用Pin9 | Pin10完成我们的组操作,因为内部,我们是逐个比特的完成我们对GPIO的设置,对于每一个满足——的确是我们要设置的GPIO

  
  if(iocurrent == ioposition){/* --------------------- GPIO模式配置 ------------------------*//* 仅当输出模式时才需要配置 */if(((GPIO_Init->Mode & GPIO_MODE) == MODE_OUTPUT)){/* 配置IO速度 */assert_param(IS_GPIO_SPEED(GPIO_Init->Speed));temp = GPIOx->OSPEEDR;temp &= ~(GPIO_OSPEEDER_OSPEEDR0 << (position * 2U));temp |= (GPIO_Init->Speed << (position * 2U));GPIOx->OSPEEDR = temp;
​/* 配置IO输出类型 */temp = GPIOx->OTYPER;temp &= ~(GPIO_OTYPER_OT_0 << position);temp |= (((GPIO_Init->Mode & OUTPUT_TYPE) >> OUTPUT_TYPE_Pos) << position);GPIOx->OTYPER = temp;
​/* 配置IO输出模式 */temp = GPIOx->MODER;temp &= ~(GPIO_MODER_MODER0 << (position * 2U));temp |= ((GPIO_Init->Mode & GPIO_MODE) << (position * 2U));GPIOx->MODER = temp;}

也就是说,设置我们的输出类型是下面的:

  • 上拉是指在GPIO引脚与电源(通常是3.3V或5V)之间连接一个电阻,这样当引脚处于输入状态时,如果没有外部信号驱动该引脚,它会自动被拉到高电平。也就是被高电平拽上去了!

  • 下拉是指在GPIO引脚与地(0V)之间连接一个电阻,这样当引脚处于输入状态时,如果没有外部信号驱动该引脚,它会自动被拉到低电平。也就是被低电平拽下去了!

  • 无上下拉配置意味着不连接任何上拉或下拉电阻。当GPIO引脚处于输入模式时,它的电平状态将取决于外部电路。如果没有外部驱动信号,这个引脚将处于浮空状态(Hi-Z),可能会导致电平不稳定,容易受到噪声干扰。

中的一种。至于GPIO的速度,则是分为低速中速高速。我们的输出模式大致分两种:

  • 推挽输出(Push-pull):这种输出类型意味着GPIO引脚可以驱动电流流向负载,并且在输出高电平和低电平时都会主动提供电流。即引脚会主动拉高电平和拉低电平。它是最常用的输出类型。

  • 开漏输出(Open-drain):这种输出类型意味着GPIO引脚只有在输出低电平时才会提供电流(拉低电平),而在输出高电平时,它不会输出电流,而是处于高阻态(Hi-Z)。通常需要外部上拉电阻来将引脚拉到高电平。你会在使用软件IIC的时候,再看到它。

所以,笔者按照给出的这个原理图:配置为上拉的情况,确保初始化后不会立马被点亮。

很好,现在,我们终于来到了点灯!

set_ccgpio_state(&led0, CCGPIO_LOW);
​
void set_ccgpio_state(CCGPIOType* type, CCGPIOState state)
{HAL_GPIO_WritePin(type->port, type->pinType, (state ? (GPIO_PIN_SET) : (GPIO_PIN_RESET)));
}

此时此刻,我们的GPIO就会被拉到低,形成一个高低的电压差,存在的电流就把我们的LED导通了!


http://www.ppmy.cn/news/1574957.html

相关文章

谈谈 ES 6.8 到 7.10 的功能变迁(2)- 字段类型篇

我们继续来了解一下从 ES 6.8 到 ES 7.10 新增的功能。本篇主要介绍新增的字段类型&#xff0c;会简要概述一下新增字段类型的使用场景和限制&#xff0c;提供简单的测试代码。 Flattened 扁平化对象字段 功能说明 解决场景 该功能主要用于处理具有大量不确定键的 JSON 对象…

常见排序算法以及实现

在本文中&#xff0c;所有排序算法考虑的都是升序情况。只要我们能搞懂算法原理&#xff0c;逆序也是很容易就能实现的。所有的排序算法的代码&#xff0c;都可以在下面这道题中测试。&#xff08;当然有些排序实现的结果会导致不能AC&#xff0c;但并不能说明是错的&#xff0…

linux 里vi编辑器的使用

Vi 编辑器的三种模式及关系 Vim 是 Linux 系统中常用的文本编辑器&#xff0c;它有三种主要模式&#xff1a;命令模式、插入模式和底线模式。这三种模式之间相互切换&#xff0c;用于不同的编辑操作。 1. 命令模式&#xff08;Command Mode&#xff09; 特点&#xff1a; 默认…

【Python爬虫(44)】分布式爬虫:筑牢安全防线,守护数据之旅

【Python爬虫】专栏简介&#xff1a;本专栏是 Python 爬虫领域的集大成之作&#xff0c;共 100 章节。从 Python 基础语法、爬虫入门知识讲起&#xff0c;深入探讨反爬虫、多线程、分布式等进阶技术。以大量实例为支撑&#xff0c;覆盖网页、图片、音频等各类数据爬取&#xff…

洛谷每日1题-------Day1__超级玛丽游戏

# P1000 超级玛丽游戏 ## 题目背景 本题是洛谷的试机题目&#xff0c;可以帮助了解洛谷的使用。 建议完成本题目后继续尝试 [P1001](/problem/P1001)、[P1008](/problem/P1008)。 另外强烈推荐[新用户必读贴](/discuss/show/241461) ## 题目描述 超级玛丽是一个非常经典…

lua垃圾回收机制

文章目录 前言一、垃圾回收机制概述二、底层原理三、GC 控制与调优四、GC 的局限性总结 前言 Lua 的垃圾回收&#xff08;Garbage Collection, GC&#xff09;机制是一种自动内存管理技术&#xff0c;主要基于标记-清除&#xff08;Mark-and-Sweep&#xff09;算法&#xff0c…

【Gin-Web】Bluebell社区项目梳理6:限流策略-漏桶与令牌桶

本文目录 一、限流二、漏桶三、令牌桶算法四、Gin框架中实现令牌桶限流 一、限流 限流又称为流量控制&#xff0c;也就是流控&#xff0c;通常是指限制到达系统的并发请求数。 限流虽然会影响部分用户的使用体验&#xff0c;但是能一定程度上保证系统的稳定性&#xff0c;不至…

Websock Demo(二) Java后端代码

1.WebSocket配置类。开启WebSocket的支持 Configuration public class WebSocketConfig {/*** bean注册&#xff1a;会自动扫描带有ServerEndpoint注解声明的Websocket Endpoint(端点)&#xff0c;注册成为Websocket bean。* 要注意&#xff0c;如果项目使用外置的servlet容器&…