热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

STM32从固件库到HAL库

让坚持成为一种热爱,极致成为一种精神。历时10个月,目前我又重新回到了程序员的身份,2023想玩不一样的嵌入式。🚀目录前言



让坚持成为一种热爱,极致成为一种精神。历时10个月,目前我又重新回到了程序员的身份,2023想玩不一样的嵌入式。🚀




目录


  • 前言
  • 一、安装STM32CubeMX
  • 二、配置自己的HAL库MDK工程
  • 三、HAL库与固件库的区别
    • 1.句柄
    • 2.回调函数
    • 3.时钟配置
    • 4.HAL_Init()
    • 5.其他

  • 结语






前言

我学STM32是基于固件库的,之后的一年都是用固件库开发STM32。其实固件库还是挺好用的,很稳定,即使ST已经放弃很多年了。
为什么改用HAL库?
1)CubeMX+HAL库是大势所趋。
2)CubeMX可以直接生成驱动代码,而且CubeMX是很好的工具,即使不用HAL库,里面的一些工具对开发也很有帮助。
3)HAL库效率不如固件库,一些关键代码可以改为寄存器操作,比如串口、DMA、ADC。
4)现在很多优秀的开源项目都是基于HAL库开发,看不懂代码怎么CV[doge]
一些个人做法
直接用CubeMX生成的MDK工程开发我认为是不可取的。开发过程中随时会修改驱动参数,而CubeMX每次修改都会覆盖原工程,只能在固定区域编程,极其不优雅。因此我的做法是用自己配置的工程开发,然后把CubeMX生成的驱动代码复制到自己工程里。本文基于这种做法展开。

开发环境
操作系统:Windows 11
JAVA 1.8.0_351
STM32CubeMX 6.5
STM32Cube_FW_F1_V 1.8.4
STM32Cube_FW_F4_V 1.26.1
keil MDK 5.36


一、安装STM32CubeMX

CubeMX需要JAVA环境,首先需要安装JAVA。适用于 Windows 的 64位 Java
然后安装CubeMX,直接在ST官网搜即可。ST官网
安装完配置仓库地址,然后下载对应的芯片包即可。
第一次打开设置时会报错,等几分钟再打开
在这里插入图片描述
在这里插入图片描述
下载芯片包
在这里插入图片描述
在这里插入图片描述
开启工程后第一步应该先设置RCC时钟源和SYS调试接口,根据我的做法这部分可以直接跳过,因为只需要驱动代码,所以直接配置IO口生成即可。



二、配置自己的HAL库MDK工程

这部分跟配置固件库工程步骤相同,只是把对应文件改为HAL库,这里我直接在正点原子的例程代码基础上进行修改,HAL库版本的原子例程依旧带有system文件夹。然后建立BSP文件,把CubeMX生成的驱动代码复制过来即可。下面讲下HAL库跟固件库的一些区别。



三、HAL库与固件库的区别

这里简单说下除命名外的一些区别。


1.句柄

所谓句柄,在HAL库里就是一个外设的结构体。
在固件库中我们是这样初始化外设的:先定义一个函数,在函数里定义一个结构体,开启时钟,然后操作这个结构体成员,最后利用这个结构体初始化外设。
在HAL库中,需要定义一个全局变量结构体,也就是句柄,为什么是全局变量?因为不只是初始化外设要用到,后面对外设的一系列操作都需要用到,比如说ADC的校正、读取,串口的接发。


2.回调函数

在固件库初始化外设中,一般都是先打开时钟,配置GPIO,配置复用功能。而在HAL库中时钟、GPIO和中断的配置都放到了回调函数中(带有MspInit结尾),因为HAL库在Init外设后会自动调用回调函数,回调函数是weak(弱定义)函数,默认为空,也就是ST让用户重新定义的函数。既然回调函数中放入了时钟和GPIO配置,那么初始化函数中就只剩下外设的复用配置。
还有一个问题,例如串口只有一个回调函数HAL_UART_MspInit,在配置USART1时会调用函数HAL_UART_MspInit,配置USART2时也会调用函数HAL_UART_MspInit,所以HAL_UART_MspInit中必须判断目前在配置哪个串口,这样才不会出错。
另外,回调函数默认是一个空函数,不一定要按照ST的思路来编程,我们也可以把回调函数中的代码放回到初始化函数中,也就是像用固件库编程那样。下面是串口的回调函数代码。

void HAL_UART_MspInit(UART_HandleTypeDef *huart)
{
GPIO_InitTypeDef gpio_init_struct;
if(huart->Instance == USART_UX) /* 如果是串口1,进行串口1 MSP初始化 */
{
USART_UX_CLK_ENABLE(); /* USART1 时钟使能 */
USART_TX_GPIO_CLK_ENABLE(); /* 发送引脚时钟使能 */
USART_RX_GPIO_CLK_ENABLE(); /* 接收引脚时钟使能 */
gpio_init_struct.Pin = USART_TX_GPIO_PIN; /* TX引脚 */
gpio_init_struct.Mode = GPIO_MODE_AF_PP; /* 复用推挽输出 */
gpio_init_struct.Pull = GPIO_PULLUP; /* 上拉 */
gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH; /* 高速 */
gpio_init_struct.Alternate = USART_TX_GPIO_AF; /* 复用为USART1 */
HAL_GPIO_Init(USART_TX_GPIO_PORT, &gpio_init_struct); /* 初始化发送引脚 */
gpio_init_struct.Pin = USART_RX_GPIO_PIN; /* RX引脚 */
gpio_init_struct.Alternate = USART_RX_GPIO_AF; /* 复用为USART1 */
HAL_GPIO_Init(USART_RX_GPIO_PORT, &gpio_init_struct); /* 初始化接收引脚 */
#if USART_EN_RX
HAL_NVIC_EnableIRQ(USART_UX_IRQn); /* 使能USART1中断通道 */
HAL_NVIC_SetPriority(USART_UX_IRQn, 3, 3); /* 抢占优先级3,子优先级3 */
#endif
}

3.时钟配置

在固件库中,根据外设对应的AHB、APB1、APB2时钟用对应的RCC函数打开时钟。如

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );

在HAL库中把每个外设和GPIO的时钟都单独分开为一个宏。如

ADC_ADCX_CHY_CLK_ENABLE(); /* 使能ADCx时钟 */
ADC_ADCX_CHY_GPIO_CLK_ENABLE(); /* 开启GPIO时钟 */

4.HAL_Init()

main函数中需要加入HAL_Init()函数进行系统初始化。
HAL_Init()的功能是配置Flash,设置中断分组、系统时钟。默认的中断分组是4(如果后面不进行修改)

HAL_StatusTypeDef HAL_Init(void)
{
/* Configure Flash prefetch, Instruction cache, Data cache */
#if (INSTRUCTION_CACHE_ENABLE != 0U)
__HAL_FLASH_INSTRUCTION_CACHE_ENABLE();
#endif /* INSTRUCTION_CACHE_ENABLE */
#if (DATA_CACHE_ENABLE != 0U)
__HAL_FLASH_DATA_CACHE_ENABLE();
#endif /* DATA_CACHE_ENABLE */
#if (PREFETCH_ENABLE != 0U)
__HAL_FLASH_PREFETCH_BUFFER_ENABLE();
#endif /* PREFETCH_ENABLE */
/* Set Interrupt Group Priority */
HAL_NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4);
/* Use systick as time base source and configure 1ms tick (default clock after Reset is HSI) */
HAL_InitTick(TICK_INT_PRIORITY);
/* Init the low level hardware */
HAL_MspInit();
/* Return function status */
return HAL_OK;
}

不难发现,里面也有一个回调函数,这里也是一个空函数。


5.其他

以上四个是重要区别,其他基本都是命名区别,可以结合例程代码进行配置,在熟悉固件库的基础上花几个小时就可以快速入门HAL库。

另外我在配置时钟,发现我之前一直忽视了32的ADC时钟,以F1为例,官方给出的ADC最高采样率是1M,时钟频率最高为14MHz。
采样率 = 1 / ( 用户配置时钟周期 + 12.5(固定转换时钟周期)×(1/14MHz))
用户配置周期最小1.5,在这种情况下可以达到1M采样率。
但是当系统为72M时ADC并不能达到14MHz,如下图所示,ADC时钟最高只能达到12MHz
此时最高率为1 / ( 1.5 + 12.5(固定转换时钟周期)×(1/12MHz))≈ 857KHz
也就是说如果要ADC跑到1M(F1芯片),只能更换系统时钟频率(降频)。
在这里插入图片描述



结语

以上是本篇文章的所有内容。







推荐阅读
  • GetWindowLong函数
    今天在看一个代码里头写了GetWindowLong(hwnd,0),我当时就有点费解,靠,上网搜索函数原型说明,死活找不到第 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 拥抱Android Design Support Library新变化(导航视图、悬浮ActionBar)
    转载请注明明桑AndroidAndroid5.0Loollipop作为Android最重要的版本之一,为我们带来了全新的界面风格和设计语言。看起来很受欢迎࿰ ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 修复安装win10失败并提示“磁盘布局不受UEFI固件支持”的方法
    本文介绍了修复安装win10失败并提示“磁盘布局不受UEFI固件支持”的方法。首先解释了UEFI的概念和作用,然后提供了两种解决方法。第一种方法是在bios界面中将Boot Mode设置为Legacy Support,Boot Priority设置为Legacy First,并关闭UEFI。第二种方法是使用U盘启动盘进入PE系统,运行磁盘分区工具DiskGenius,将硬盘的分区表设置为gpt格式,并留出288MB的内存。最后,通过运行界面输入命令cmd来完成设置。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 本文介绍了Java集合库的使用方法,包括如何方便地重复使用集合以及下溯造型的应用。通过使用集合库,可以方便地取用各种集合,并将其插入到自己的程序中。为了使集合能够重复使用,Java提供了一种通用类型,即Object类型。通过添加指向集合的对象句柄,可以实现对集合的重复使用。然而,由于集合只能容纳Object类型,当向集合中添加对象句柄时,会丢失其身份或标识信息。为了恢复其本来面貌,可以使用下溯造型。本文还介绍了Java 1.2集合库的特点和优势。 ... [详细]
  • Python操作MySQL(pymysql模块)详解及示例代码
    本文介绍了使用Python操作MySQL数据库的方法,详细讲解了pymysql模块的安装和连接MySQL数据库的步骤,并提供了示例代码。内容涵盖了创建表、插入数据、查询数据等操作,帮助读者快速掌握Python操作MySQL的技巧。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 【Windows】实现微信双开或多开的方法及步骤详解
    本文介绍了在Windows系统下实现微信双开或多开的方法,通过安装微信电脑版、复制微信程序启动路径、修改文本文件为bat文件等步骤,实现同时登录两个或多个微信的效果。相比于使用虚拟机的方法,本方法更简单易行,适用于任何电脑,并且不会消耗过多系统资源。详细步骤和原理解释请参考本文内容。 ... [详细]
  • 本文主要复习了数据库的一些知识点,包括环境变量设置、表之间的引用关系等。同时介绍了一些常用的数据库命令及其使用方法,如创建数据库、查看已存在的数据库、切换数据库、创建表等操作。通过本文的学习,可以加深对数据库的理解和应用能力。 ... [详细]
  • 本文介绍了一种轻巧方便的工具——集算器,通过使用集算器可以将文本日志变成结构化数据,然后可以使用SQL式查询。集算器利用集算语言的优点,将日志内容结构化为数据表结构,SPL支持直接对结构化的文件进行SQL查询,不再需要安装配置第三方数据库软件。本文还详细介绍了具体的实施过程。 ... [详细]
author-avatar
刘小飘168585
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有