【STM32H7教程】第12章 STM32H7的HAL库框架设计学习

发布者:dswecd最新更新时间:2019-08-27 来源: eefocus关键字:STM32H7  HAL库  框架设计 手机看文章 扫描二维码
随时随地手机看文章

12.1 初学者重要提示

学习使用HAL库前,有必要对他们的基本设计框架有所了解,然后深入学习,效果更好。


为了方便调用,HAL库为各种外设基本都配了三套API,查询,中断和DMA。


12.2 HAL库的配置文件

HAL库有一个专门的配置文件叫stm32h7xx_hal_conf.h,这个文件里面有一个很重要的参数,就是HSE_VALUE,大家所设计板子使用的实际晶振大小一定要与这个数值一致。比如V7的外置晶振是25MHz,那么这里就务必配置宏定义为:


#define HSE_VALUE    ((uint32_t)25000000)

完整的代码如下:


/* ########################## Module Selection ############################## */

/**

  * @brief This is the list of modules to be used in the HAL driver 

  */

#define HAL_MODULE_ENABLED

#define HAL_ADC_MODULE_ENABLED

#define HAL_CEC_MODULE_ENABLED

 

 /* 省略未写 */

 

/* ########################## Oscillator Values adaptation ####################*/

/**

  * @brief Adjust the value of External High Speed oscillator (HSE) used in your application.

  *        This value is used by the RCC HAL module to compute the system frequency

  *        (when HSE is used as system clock source, directly or through the PLL).  

  */

#if !defined  (HSE_VALUE) 

#define HSE_VALUE    ((uint32_t)25000000) /*!< Value of the External oscillator in Hz */

#endif /* HSE_VALUE */

 

#if !defined  (HSE_STARTUP_TIMEOUT)

  #define HSE_STARTUP_TIMEOUT    ((uint32_t)5000)   /*!< Time out for HSE start up, in ms */

#endif /* HSE_STARTUP_TIMEOUT */

 

/**

  * @brief Internal  oscillator (CSI) default value.

  *        This value is the default CSI value after Reset.

  */

#if !defined  (CSI_VALUE)

  #define CSI_VALUE    ((uint32_t)4000000) /*!< Value of the Internal oscillator in Hz*/

#endif /* CSI_VALUE */

   

/**

  * @brief Internal High Speed oscillator (HSI) value.

  *        This value is used by the RCC HAL module to compute the system frequency

  *        (when HSI is used as system clock source, directly or through the PLL). 

  */

#if !defined  (HSI_VALUE)

  #define HSI_VALUE    ((uint32_t)64000000) /*!< Value of the Internal oscillator in Hz*/

#endif /* HSI_VALUE */

 

/**

  * @brief External Low Speed oscillator (LSE) value.

  *        This value is used by the UART, RTC HAL module to compute the system frequency

  */

#if !defined  (LSE_VALUE)

  #define LSE_VALUE    ((uint32_t)32768) /*!< Value of the External oscillator in Hz*/

#endif /* LSE_VALUE */

 

   

#if !defined  (LSE_STARTUP_TIMEOUT)

  #define LSE_STARTUP_TIMEOUT    ((uint32_t)5000)   /*!< Time out for LSE start up, in ms */

#endif /* LSE_STARTUP_TIMEOUT */

 

/**

  * @brief External clock source for I2S peripheral

  *        This value is used by the I2S HAL module to compute the I2S clock source 

  *        frequency, this source is inserted directly through I2S_CKIN pad. 

  */

#if !defined  (EXTERNAL_CLOCK_VALUE)

  #define EXTERNAL_CLOCK_VALUE    12288000U /*!< Value of the External clock in Hz*/

#endif /* EXTERNAL_CLOCK_VALUE */

 

/* Tip: To avoid modifying this file each time you need to use different HSE,

   ===  you can define the HSE value in your toolchain compiler preprocessor. */

 

/* ########################### System Configuration ######################### */

/**

  * @brief This is the HAL system configuration section

  */     

#define  VDD_VALUE                    ((uint32_t)3300) /*!< Value of VDD in mv */

#define  TICK_INT_PRIORITY            ((uint32_t)0x0F) /*!< tick interrupt priority */

#define  USE_RTOS                     0

/* #define  USE_SD_TRANSCEIVER           1U   */            /*!< use uSD Transceiver */

 

/* ########################### Ethernet Configuration ######################### */

#define ETH_TX_DESC_CNT         4  /* number of Ethernet Tx DMA descriptors */

#define ETH_RX_DESC_CNT         4  /* number of Ethernet Rx DMA descriptors */

 

#define ETH_MAC_ADDR0    ((uint8_t)0x02)

#define ETH_MAC_ADDR1    ((uint8_t)0x00)

#define ETH_MAC_ADDR2    ((uint8_t)0x00)

#define ETH_MAC_ADDR3    ((uint8_t)0x00)

#define ETH_MAC_ADDR4    ((uint8_t)0x00)

#define ETH_MAC_ADDR5    ((uint8_t)0x00)

 

/* ########################## Assert Selection ############################## */

/**

  * @brief Uncomment the line below to expanse the "assert_param" macro in the 

  *        HAL drivers code

  */

/* #define USE_FULL_ASSERT    1 */

 

 

/* ################## SPI peripheral configuration ########################## */

/** 

  * @brief Used to activate CRC feature inside HAL SPI Driver

  *        Activated   (1U): CRC code is compiled within HAL SPI driver

  *        Deactivated (0U): CRC code excluded from HAL SPI driver

  */

 

#define USE_SPI_CRC                   1U

 

 

/* Includes ------------------------------------------------------------------*/

/**

  * @brief Include module's header file 

  */

 

#ifdef HAL_RCC_MODULE_ENABLED

  #include "stm32h7xx_hal_rcc.h"

#endif /* HAL_RCC_MODULE_ENABLED */

/* 省略未写 */

 

   

/* Exported macro ------------------------------------------------------------*/

#ifdef  USE_FULL_ASSERT

/**

  * @brief  The assert_param macro is used for function's parameters check.

  * @param  expr: If expr is false, it calls assert_failed function

  *         which reports the name of the source file and the source

  *         line number of the call that failed. 

  *         If expr is true, it returns no value.

  * @retval None

  */

  #define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t *)__FILE__, __LINE__))

/* Exported functions ------------------------------------------------------- */

  void assert_failed(uint8_t* file, uint32_t line);

#else

  #define assert_param(expr) ((void)0)

#endif /* USE_FULL_ASSERT */

除了HSE_VALUE,下面几个也要作为了解:


  第72行,滴答定时器的优先级设置。这个优先级的设置至关重要。因为HAL库中各个外设驱动里面的延迟实现是基于此文件提供的时间基准。

如果在中断服务程序里面调用基于此时间基准的延迟函数HAL_Delay要特别注意,因为这个函数的时间基准是基于滴答定时器或者其他通用定时器实现,实现方式是滴答定时器或者其他通用定时器里面对变量计数。如此以来,结果是显而易见的,如果其他中断服务程序调用了此函数,且中断优先级高于滴答定时器,会导致滴答定时器中断服务程序一直得不到执行,从而卡死在里面。所以滴答定时器的中断优先级一定要比他们高。


另外这个时间基准既可以使用滴答定时器实现也可以使用通用的定时器实现,默认情况下是用的滴答定时器。


  第73行,当前HAL库还不支持RTOS方式。

  第74行,STM32H7的SDIO外接支持UHS-I 模式 (SDR12, SDR25, SDR50, SDR104和DDR50)的SD卡,需要1.8的电平转换器,此选项就是来使能此功能用的。

  第92行,用于使能断言功能,在HAL库的API里面都有用到,用来判断函数形参是否有效。默认情况下是关闭的。

  第126行,使能断言功能后,实际对应的代码位置。这里没有对函数void assert_failed(uint8_t* file, uint32_t line)做具体实现,大家可以根据自己的需求去实现,这里提供一个参考:

/* 

ST库函数使用了C编译器的断言功能,如果定义了USE_FULL_ASSERT,那么所有的ST库函数将检查函数形参

是否正确。如果不正确将调用 assert_failed() 函数,这个函数是一个死循环,便于用户检查代码。

关键字 __LINE__ 表示源代码行号。

关键字__FILE__表示源代码文件名。

断言功能使能后将增大代码大小,推荐用户仅在调试时使能,在正式发布软件是禁止。

用户可以选择是否使能ST固件库的断言供能。使能断言的方法有两种:

(1) 在C编译器的预定义宏选项中定义USE_FULL_ASSERT。

(2) 在本文件取消"#define USE_FULL_ASSERT    1"行的注释。

*/

 

/*

*********************************************************************************************************

* 函 数 名: assert_failed

* 形    参:file : 源代码文件名称。关键字__FILE__表示源代码文件名。

*   line :代码行号。关键字 __LINE__ 表示源代码行号

* 返 回 值: 无

*********************************************************************************************************

*/

void assert_failed(uint8_t* file, uint32_t line)

/* 

用户可以添加自己的代码报告源代码文件名和代码行号,比如将错误文件和行号打印到串口

printf("Wrong parameters value: file %s on line %drn", file, line) */

*/

/* 这是一个死循环,断言失败时程序会在此处死机,以便于用户查错 */

while (1)

{

}

}

12.3 HAL库的时间基准

为了方便各种外设延迟的实现,HAL库专门搞了一个时间基准,默认来源是滴答定时器,也可以通过重定向使用其他定时器实现。相关函数全部集中在stm32h7xx_hal.c文件里面实现,关于这些函数在本教程的第16章有详细讲解。


12.4 HAL库的启动流程

这里通过V7板子初始化流程做个说明:


/*

******************************************************************************************************

* 函 数 名: bsp_Init

* 功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。

*              只需要调用一次

* 形    参:无

* 返 回 值: 无

******************************************************************************************************

*/

void bsp_Init(void)

{

    /* 配置MPU */

MPU_Config();

/* 使能L1 Cache */

CPU_CACHE_Enable();

 

/* 

       STM32H7xx HAL 库初始化,此时系统用的还是H7自带的64MHz,HSI时钟:

   - 调用函数HAL_InitTick,初始化滴答时钟中断1ms。

   - 设置NVIV优先级分组为4。

*/

HAL_Init();

 

/* 

       配置系统时钟到400MHz

       - 切换使用HSE。

       - 此函数会更新全局变量SystemCoreClock,并重新配置HAL_InitTick。

    */

SystemClock_Config();

 

bsp_InitKey();    /* 按键初始化,要放在滴答定时器之前,因为按钮检测是通过滴答定时器扫描 */

bsp_InitTimer();  /* 初始化滴答定时器 */

bsp_InitUart(); /* 初始化串口 */

bsp_InitExtIO(); /* 初始化FMC总线74HC574扩展IO. 必须在 bsp_InitLed()前执行 */

bsp_InitLed();    /* 初始化LED */

bsp_InitLPTIMOutPWM(); /* 低功耗定时器PWM输出 */

}

  第13 – 16行,配置MPU和使能Cache,对于H7而已,这两个函数要优先执行,因为Flash速度,SRAM速度跟CPU和TCM有些差距的,所以要使能Cache。而不同存储区的Cache策略要通过MPU分别进行配置。

  第23行,调用函数HAL_Init时,系统依然使用的64MHz HSI时钟,这点要特别注意。此函数会调用函数HAL_InitTick,初始化滴答时钟中断1ms,并设置NVIV优先级分组为4。

这里就隐含了一个知识点,就是它会开启滴答定时器中断,如果用户也要使用滴答定时器中断,此问题就要引起注意,我们的bsp_timer.C文件解决办法如下:


/*

*********************************************************************************************************

* 函 数 名: SysTick_Handler

* 功能说明: 系统嘀嗒定时器中断服务程序。启动文件中引用了该函数。

* 形    参:  无

* 返 回 值: 无

*********************************************************************************************************

[1] [2]
关键字:STM32H7  HAL库  框架设计 引用地址:【STM32H7教程】第12章 STM32H7的HAL库框架设计学习

上一篇:【STM32H7教程】第13章 STM32H7启动过程详解
下一篇:【STM32H7教程】第11章 STM32H7移植SEGGER的硬件异常分析

推荐阅读最新更新时间:2024-11-10 01:36

STM32H7教程】第15章 STM32H7的GPIO基础知识(重要)
15.1 初学者重要提示 学习本章节务必要认真学习并掌握15.3小节GPIO功能模式分析,学好它们,对后续章节中外设的学习大有裨益。 对于不使用的引脚,推荐设置为模拟模式,悬空即可。 GPIO的速度等级高的时候,最好使能IO补偿单元。 GPIO还涉及到一个注入电流的问题,此贴可以作为了解:http://forum.armfly.com/forum.php?mod=viewthread&tid=87675 。 15.2 GPIO功能简介 STM32H7的GPIO特性如下: 输出状态:开漏/推挽 + 上拉/下拉电阻。 通过输出数据寄存器(GPIOx_ODR)或者外设(GPIO设置为复用模式时)输出数据。 G
[单片机]
【<font color='red'>STM32H7</font>教程】第15章 <font color='red'>STM32H7</font>的GPIO基础知识(重要)
HAL库教程9:串口接收不定长数据
  串口收到的两组数据之间,往往会有一定的时间间隔。可以判断这个间隔,来实现无需结束符,无需指定长度,串口可接收不定长数据的功能。如果串口在一定的时间内没有收到新的数据,可以认为一组数据已经接收完毕了。思路是用定时器来设置一个“闹钟”,连续的一段时间没有收到新的数据,闹钟响起,就把已经收到的数据打包,做相应处理。 定时器溢出时间配置   首先修改定时器的溢出时间。本文规定使用5ms的间隔。在某些通信协议中,会规定间隔时间。例如Modbus规定两组数据之间要间隔3.5字符。   实际上,间隔的时间常常与通信的波特率是相关的。在9600波特率下,一个字节的数据共 起始+8数据+结束=10位,一位是104us,所以一个字节的数据是1
[单片机]
STM32HAL库ADC实验(二)——连续采样模式打印电压值
int main(void) { /* USER CODE BEGIN 1 */ uint16_t adcData; float voltage; /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BEGIN Init */ /*
[单片机]
STM32<font color='red'>HAL库</font>ADC实验(二)——连续采样模式打印电压值
【STM32CubeMX】3,STM32的HAL库运用小方法
1,当在hal头文件中看到该函数时,还不理解,直接复制百度就好 如HAL_UART_Transmit(),有些论坛讲解的非常详细 2,hal_adc.c文件一般都是详细解释, hal_adc.h文件都是关键函数的综述,这个对于快速上手非常重要
[单片机]
【STM32CubeMX】3,STM32的<font color='red'>HAL库</font>运用小方法
STM32HAL库普通模式串口收发
一.简单了解 串口模式实现有三种 1.普通模式:在主函数中接收函数 2.中断模式:产生的不影响主程序运行 3. DMA模式:与主函数互不影响,独立运行 本文记录第一种普通模式。 学习目标是:实现普通串口的数据收发,发送数据时LED灯电平翻转。 二.步骤 步骤1:在STM32CUBEMX中创建好工程,我这里用的是小熊派的STM32L431RCT6开发板,需要配置的有LED灯,PC13为output。uart1为异步收发全双工,时钟为80Mhz。 步骤2:打开工程文件,在uart.c中,包含标准头文件以及实现串口的重定向 #include stdio.h int fputc(int ch,FILE* f) { uint
[单片机]
STM32<font color='red'>HAL库</font>普通模式串口收发
STM32F1xx HAL库中文版——USART篇
38.1 UART Firmware driver registers structures //串口固件驱动寄存器结构 38.1.1 UART_InitTypeDef UART_InitTypeDef被定义在stm32f1xx_hal_uart.h头文件中 数据字段: • uint32_t BaudRate 波特率 • uint32_t WordLength 字长 • uint32_t StopBits 停止位 • uint32_t Parity 奇偶校验位 • uint32_t Mode 模式 • uint32_t HwFlowCtl 硬件流控制 • uint32_t OverSampling 过采样 字段的文档: •
[单片机]
STM32F7xx —— 启动文件和HAL库
一.库开发的优势 开发速度快,便于移植。 二.stm32F7xx几个重要文件 1. HAL库关键文件 文件 描述 stm32f7xx_hal_ppp.c/h 基本外设的操作API stm32f7xx_hal_ppp_ex.c/h 拓展外设特性的API stm32f7xx_hal.c/h HAL通用API(如HAL_Init, HAL_Delay等) stm32f7xx_hal_conf.h HAL的配置文件,选择使能某种功能 stm32f7xx_hal_def.h HAL的通用数据类型和定义 stm32f7xx_ll_ppp.c/h 复杂外设的底层实现 2.stm32f7
[单片机]
STM32标准库和HAL库的区别
大体看了下,标准库移植起来没有HAL库方便。 用HAL库移植起来不需要改动,怪不得ST推HAL库 好像标准库后面的芯片就不支持了 怪不得华清现在讲课,STM32的课还有freertos的课都是用的STM32CUBE https://blog.csdn.net/ice_masters/article/details/105644704 https://blog.csdn.net/bornpride/article/details/94601754 怪不得这里说没有可移植性,我最开始看到这句的时候很吃惊,标准库怎么没有可移植性了。
[单片机]
STM32标准库和<font color='red'>HAL库</font>的区别
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved