最新教程下载:http://www.armbbs.cn/forum.php?mod=viewthread&tid=93255
本章教程为大家讲解制作一个STM32F4的例子所需的最基本API函数,对于一些常用的API函数,一定要熟练掌握这些函数都是实现了什么功能,不常用的函数有个了解即可,用到的时候再去学。
目录
第16章 STM32F407必备的HAL库API(重要)
16.1 初学者重要提示
16.2 那些是必备的API
16.3 源文件stm32f4xx_hal.c(重要)
16.3.1 函数HAL_Init
16.3.2 函数HAL_DeInit
16.3.3 函数HAL_InitTick
16.3.4 Systick的相关函数
16.3.5 低功耗状态下继续使用调试功能
16.4 源文件stm32f4xx_hal_rcc.c
16.4.1 函数HAL_RCC_DeInit
16.4.2 函数HAL_RCC_OscConfig
16.4.3 函数HAL_RCC_ClockConfig
16.4.4 函数HAL_RCC_MCOConfig
16.4.5 RCC相关函数
16.5 源文件stm3f4xx_hal_cortex.c
16.5.1 NVIC相关函数
16.5.2 SysTick相关函数
16.5.3 函数HAL_MPU_ConfigRegion
16.6 总结
1、 对于一些常用的函数,大家一定要熟练的掌握都实现了什么功能,比如HAL_Init,HAL_RCC_OscConfig,HAL_RCC_ClockConfig等。最好的办法是把这些函数的源码读一遍。
这里我们通过一个简单的初始化流程来了解STM32F4的工程模板所必备的库文件和API:
1、 第1步:系统上电复位,进入启动文件startup_stm32f407xx.s,在这个文件里面执行复位中断服务程序。
2、 第2步:进入到main函数就可以开始用户应用程序编程了。在这个函数里面要做几个重要的初始化,依次是:
前面的两步完成后,就可以开始做用户需要的按键、串口等方面的初始化和应用代码的实现了。这里把我们需要学习的几个库文件整理出来,依次有:
其中startup_stm32f429xx.s和system_stm32f4xx.c已经在第13章为大家讲解过,这里不再赘述。。本章教程重点为大家讲解文件stm32f4xx_hal.c、stm32f4xx_hal_cortex.c和sm32f4x_hal_rcc.c(core_cm4.h文件在后面章节为大家讲解)。
这个文件比较杂,像基准电压大小配置,EXTI配置,IO补偿配置等都在这个文件里面设置。学习这个文件注意事项:
函数原型:
HAL_StatusTypeDef HAL_Init(void) { /* 配置Flash的指令预取,指令Cache和数据Cache */ #if (INSTRUCTION_CACHE_ENABLE != 0U) __HAL_FLASH_INSTRUCTION_CACHE_ENABLE(); #endif #if (DATA_CACHE_ENABLE != 0U) __HAL_FLASH_DATA_CACHE_ENABLE(); #endif #if (PREFETCH_ENABLE != 0U) __HAL_FLASH_PREFETCH_BUFFER_ENABLE(); #endif /* 设置中断优先级分组 */ HAL_NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4); /* 使用滴答定时器做为默认时基,配置为1ms滴答,另外系统上电后默认使用的HSI时钟 */ HAL_InitTick(TICK_INT_PRIORITY); /* 初始化底层硬件 */ HAL_MspInit(); /* 返回HAL_OK */ return HAL_OK; }
函数描述:
此函数用于初始化HAL库,此函数主要实现如下功能:
函数参数:
注意事项:
使用举例:
此函数的使用比较简单,上电后优先调用即可。
函数原型:
HAL_StatusTypeDef HAL_DeInit(void) { /* 复位所有外设 */ __HAL_RCC_APB1_FORCE_RESET(); __HAL_RCC_APB1_RELEASE_RESET(); __HAL_RCC_APB2_FORCE_RESET(); __HAL_RCC_APB2_RELEASE_RESET(); __HAL_RCC_AHB1_FORCE_RESET(); __HAL_RCC_AHB1_RELEASE_RESET(); __HAL_RCC_AHB2_FORCE_RESET(); __HAL_RCC_AHB2_RELEASE_RESET(); __HAL_RCC_AHB3_FORCE_RESET(); __HAL_RCC_AHB3_RELEASE_RESET(); /* 复位底层硬件初始化 */ HAL_MspDeInit(); /* 返回值 */ return HAL_OK; }
函数描述:
此函数用于复位HAL库和滴答时钟。
使用举例:
此函数的使用比较简单,需要调用的时候直接调用即可。
函数原型:
__weak HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority) { /* Configure the SysTick to have interrupt in 1ms time basis*/ if (HAL_SYSTICK_Config(SystemCoreClock / (1000U / uwTickFreq)) > 0U) { return HAL_ERROR; } /* Configure the SysTick IRQ priority */ if (TickPriority < (1UL << __NVIC_PRIO_BITS)) { HAL_NVIC_SetPriority(SysTick_IRQn, TickPriority, 0U); uwTickPrio = TickPriority; } else { return HAL_ERROR; } /* Return function status */ return HAL_OK; }
函数描述:
此函数用于初始化滴答时钟,此函数相关问题如下:
函数参数:
使用举例:
此函数由HAL_Init调用,无需用户操作,除非需要重定义。
调用了函数HAL_Init后,Systick相关的函数就可以使用了。这些函数如下:
函数原型:
__weak void HAL_IncTick(void) __weak uint32_t HAL_GetTick(void) uint32_t HAL_GetTickPrio(void) HAL_StatusTypeDef HAL_SetTickFreq(HAL_TickFreqTypeDef Freq) HAL_TickFreqTypeDef HAL_GetTickFreq(void) __weak void HAL_Delay(uint32_t Delay) __weak void HAL_SuspendTick(void) __weak void HAL_ResumeTick(void)
函数描述:
这些函数就比较简单了,下面把这些函数实现的功能做个简单的说明:
注意事项:
使用举例:
这些函数都比较简单,这里就不举例了。需要的时候,直接调用即可。
如果希望在睡眠,停机和待机的低功耗模式下继续使用调试功能,调用下面的函数即可:
/* 睡眠模式下继续使用调试功能 */ void HAL_DBGMCU_EnableDBGSleepMode(void) { SET_BIT(DBGMCU->CR, DBGMCU_CR_DBG_SLEEP); } /* 停机模式下继续使用调试功能 */ void HAL_DBGMCU_EnableDBGStopMode(void) { SET_BIT(DBGMCU->CR, DBGMCU_CR_DBG_STOP); } /* 待机模式下继续使用调试功能 */ void HAL_DBGMCU_EnableDBGStandbyMode(void) { SET_BIT(DBGMCU->CR, DBGMCU_CR_DBG_STANDBY); }
这个文件主要是实现内部和外部时钟(HSE、HSI、LSE、LSI、PLL、CSS、MCO)以及总线时钟(SYSCLK、AHB1、AHB2、AHB3、APB1)的配置。
学习这个文件注意事项:
1、 系统上电复位后,通过内部高速时钟HSI运行(主频16MHz),Flash工作在0等待周期,所有外设除了SRAM、Flash、JTAG 和 PWR,时钟都是关闭的。
2、 系统上电复位后,用户需要完成以下工作:
RCC局限性:
使能了外设时钟后,不能立即操作对应的寄存器,要加延迟。不同外设延迟不同:
当前HAL库的解决方案是在使能了外设时钟后,再搞一个读操作,算是当做延迟用。
比如下面使能GPIOA的时钟:
#define __HAL_RCC_GPIOA_CLK_ENABLE() do { \ __IO uint32_t tmpreg = 0x00U; \ SET_BIT(RCC->AHB1ENR, RCC_AHB1ENR_GPIOAEN);\ /* Delay after an RCC peripheral clock enabling */ \ tmpreg = READ_BIT(RCC->AHB1ENR, RCC_AHB1ENR_GPIOAEN);\ UNUSED(tmpreg); \ } while(0U)
1、正确理解PCLK1,2和HCLK对应哪些总线的时钟,下面一张图可以说明问题:
由上图可以得出:
PCLK1、PC对应的是APB总线APB1,APB2时钟。
HCLK对应的AHB总线。
2、内部和外部时钟配置:
高速内部RC振荡器,可以直接或者通过PLL倍频后做系统时钟源。缺点是精度差些,即使经过校准。
低速内部时钟,主要用于独立看门狗和RTC的时钟源。
高速外部晶振,可接4 - 26z的晶振,可以直接或者通过PLL倍频后做系统时钟源,也可以做RTC的是时钟源。
低速外部晶振,主要用于RTC。
时钟安全系统,一旦使能后,如果HSE启动失败(不管是直接作为系统时钟源还是通过PLL输出后做系统时钟源),系统时钟将切换到HSI。如果使能了中断的话,将进入不可屏蔽中断NMI。
可以在PA8引脚输出SYSCLK、PLLI2SCLK、HSE和PLLCLK。
可以在PC9引脚输出LSE、HSE、HSI和PLLCLK。
函数原型:
__weak HAL_StatusTypeDef HAL_RCC_DeInit(void) { return HAL_OK; }
函数描述:
文件stm32f4xx_hal_rcc.c里面的此函数是空的(做了__weak弱定义,方便用户在其它文件里面重定义),ST将具体实现放在了stm32f4xx_hal_rcc_ex.c里面的此函数里面
此函数用于RCC复位函数,主要实现如下功能:
注意事项:
使用举例:
此函数的使用比较简单,需要调用的时候直接调用即可。
函数原型:
函数描述:
通过上面函数原型,我们可以一目了然的看出此函数的作用,配置了HSE、HSI、LSI、LSE和PLL。
函数参数:
/** * @brief RCC Internal/External Oscillator (HSE, HSI, LSE and LSI) configuration structure definition */ typedef struct { uint32_t OscillatorType; uint32_t HSEState; uint32_t LSEState; uint32_t HSIState; uint32_t HSICalibrationValue; uint32_t LSIState; RCC_PLLInitTypeDef PLL; }RCC_OscInitTypeDef;
注意事项:
使用举例:
RCC_OscInitTypeDef RCC_OscInitStruct; /* 使能HSE,并选择HSE作为PLL时钟源 */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; RCC_OscInitStruct.PLL.PLLM = 8; RCC_OscInitStruct.PLL.PLLN = 336; RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; RCC_OscInitStruct.PLL.PLLQ = 4; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(__FILE__, __LINE__); }
函数原型:
函数描述:
通过上面函数原型,我们可以一目了然的看出此函数的作用,配置了HCLK、SYSCLK、PLCK1和PLCK2。
函数参数:
注意事项:
使用举例:
RCC_ClkInitTypeDef RCC_ClkInitStruct; HAL_StatusTypeDef ret = HAL_OK; /* 选择PLL的输出作为系统时钟 HCLK = SYSCLK / 1 (AHB1Periph) PCLK2 = HCLK / 2 (APB2Periph) PCLK1 = HCLK / 4 (APB1Periph) */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2; /* 此函数会更新SystemCoreClock,并重新配置HAL_InitTick */ if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK) { Error_Handler(__FILE__, __LINE__); }
函数原型:
void HAL_RCC_MCOConfig(uint32_t RCC_MCOx, uint32_t RCC_MCOSource, uint32_t RCC_MCODiv) { GPIO_InitTypeDef GPIO_InitStruct; /* Check the parameters */ assert_param(IS_RCC_MCO(RCC_MCOx)); assert_param(IS_RCC_MCODIV(RCC_MCODiv)); /* RCC_MCO1 */ if(RCC_MCOx == RCC_MCO1) { /* 省略未写 */ } else { /* 省略未写 */ } }
函数描述:
此函数的作用是配置MCO1(PA8引脚)和MCO2(PC9引脚)的时钟输出以及选择的时钟源,通过下面的截图可以很好的说明此函数的作用:
函数参数:
1、 第1个形参用于选择输出的引脚,可选择RCC_MCO1(PA8引脚)或者RCC_MCO2(PC9引脚)。
2、 第2个形参用于选择输出的时钟源,MCO1可选择的时钟源如下:
MCO2可选择的时钟源如下:
3、 第3个参数用于设置输出分频,范围从RCC_MCODIV_1到RCC_MCODIV_4。
使用举例:
此函数的使用比较简单,需要在MCO引脚输出时钟的时候,直接调用即可。
函数原型:
void HAL_RCC_EnableCSS(void); void HAL_RCC_DisableCSS(void); uint32_t HAL_RCC_GetSysClockFreq(void); uint32_t HAL_RCC_GetHCLKFreq(void); uint32_t HAL_RCC_GetPCLK1Freq(void); uint32_t HAL_RCC_GetPCLK2Freq(void); void HAL_RCC_GetOscConfig(RCC_OscInitTypeDef *RCC_OscInitStruct); void HAL_RCC_GetClockConfig(RCC_ClkInitTypeDef *RCC_ClkInitStruct, uint32_t *pFLatency);
剩下的这些函数都比较简单了,我们这里就不做讲解了。
这个库文件主要功能是NVIC,MPU和Systick的配置。此文件有个臃肿的地方,里面的API其实就是将ARM的CMSIS库各种API重新封装了一遍。这么做的好处是保证了HAL的API都是以字母HAL开头。
更多NVIC相关知识需要大家看第xx章节(注:后面做到这个章节时补上)。
更多SysTick相关知识需要大家看第xx章节。
更多MPU相关知识需要大家看第xx章节。
函数原型:
void HAL_NVIC_SetPriorityGrouping(uint32_t PriorityGroup) void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority, uint32_t SubPriority) void HAL_NVIC_EnableIRQ(IRQn_Type IRQn) void HAL_NVIC_DisableIRQ(IRQn_Type IRQn) void HAL_NVIC_SystemReset(void) uint32_t HAL_NVIC_GetPriorityGrouping(void) void HAL_NVIC_GetPriority(IRQn_Type IRQn, uint32_t PriorityGroup, uint32_t *pPreemptPriority, uint32_t *pSubPriority) void HAL_NVIC_SetPendingIRQ(IRQn_Type IRQn) uint32_t HAL_NVIC_GetPendingIRQ(IRQn_Type IRQn) void HAL_NVIC_ClearPendingIRQ(IRQn_Type IRQn) uint32_t HAL_NVIC_GetActive(IRQn_Type IRQn)
这些函数的操作就比较容易了,这里就不做讲解了,用到时调用即可。
函数原型:
uint32_t HAL_SYSTICK_Config(uint32_t TicksNumb) void HAL_SYSTICK_CLKSourceConfig(uint32_t CLKSource) void HAL_SYSTICK_IRQHandler(void) __weak void HAL_SYSTICK_Callback(void)
这些函数的操作就比较容易了,这里就不做讲解了,用到时调用即可。
函数原型:
void HAL_MPU_ConfigRegion(MPU_Region_InitTypeDef *MPU_Init) { /* 部分省略未写 */ /* Set the Region number */ MPU->RNR = MPU_Init->Number; if ((MPU_Init->Enable) != RESET) { /* 部分省略未写 */ MPU->RBAR = MPU_Init->BaseAddress; MPU->RASR = ((uint32_t)MPU_Init->DisableExec << MPU_RASR_XN_Pos) | ((uint32_t)MPU_Init->AccessPermission << MPU_RASR_AP_Pos) | ((uint32_t)MPU_Init->TypeExtField << MPU_RASR_TEX_Pos) | ((uint32_t)MPU_Init->IsShareable << MPU_RASR_S_Pos) | ((uint32_t)MPU_Init->IsCacheable << MPU_RASR_C_Pos) | ((uint32_t)MPU_Init->IsBufferable << MPU_RASR_B_Pos) | ((uint32_t)MPU_Init->SubRegionDisable << MPU_RASR_SRD_Pos) | ((uint32_t)MPU_Init->Size << MPU_RASR_SIZE_Pos) | ((uint32_t)MPU_Init->Enable << MPU_RASR_ENABLE_Pos); } else { MPU->RBAR = 0x00; MPU->RASR = 0x00; } }
此函数在本教程第xx章有专门的讲解(注:后面做到这个章节时补上)。
本章节就为大家讲解这么多,对于一些常用的函数,望大家务必要掌握。