stm32.cube(三)——HAL.GPIO

发布者:tyloo820最新更新时间:2018-05-01 来源: eefocus关键字:stm32  cube  HAL  GPIO 手机看文章 扫描二维码
随时随地手机看文章

一、总览

HAL层里的库函数和大家熟悉的stm32lib基本上是一致的。一般来说一个cpu外设的操作函数分成两部分,分别是init和operation。

来看一个cube里的例子:

static GPIO_InitTypeDef  GPIO_InitStruct;


int main(void)

{

  /* This sample code shows how to use GPIO HAL API to toggle LED2 IO

    in an infinite loop. */


  /* STM32F103xB HAL library initialization:

       - Configure the Flash prefetch

       - Systick timer is configured by default as source of time base, but user 

         can eventually implement his proper time base source (a general purpose 

         timer for example or other time source), keeping in mind that Time base 

         duration should be kept 1ms since PPP_TIMEOUT_VALUEs are defined and 

         handled in milliseconds basis.

       - Set NVIC Group Priority to 4

       - Low Level Initialization

     */

  HAL_Init();


  /* Configure the system clock to 64 MHz */

  SystemClock_Config();


  /* -1- Enable GPIO Clock (to be able to program the configuration registers) */

  LED2_GPIO_CLK_ENABLE();


  /* -2- Configure IO in output push-pull mode to drive external LEDs */

  GPIO_InitStruct.Mode  = GPIO_MODE_OUTPUT_PP;

  GPIO_InitStruct.Pull  = GPIO_PULLUP;

  GPIO_InitStruct.Speed = GPIO_SPEED_HIGH;


  GPIO_InitStruct.Pin = LED2_PIN;

  HAL_GPIO_Init(LED2_GPIO_PORT, &GPIO_InitStruct);


  /* -3- Toggle IO in an infinite loop */

  while (1)

  {

    HAL_GPIO_TogglePin(LED2_GPIO_PORT, LED2_PIN);

    /* Insert delay 100 ms */

    HAL_Delay(100);

  }

}

注释里的内容说明了之前一篇文章里的提到的HAL_INIT()函数的作用。


在使用GPIO前要先初始化GPIO的时钟,因为许多外设都被设计成时序逻辑电路,所以必须要为外设提供时钟源,否则外设的电路状态就不会改变。


GPIO时钟源设定后,通过GPIO_initTypeDef结构体,初始化指定Pin脚的模式、速度、初始输出。


最后调用 HAL_GPIO_TogglePin()和HAL_Delay()将GPIO定时取反以达到使驱动的led闪烁的效果。


二、GPIO的初始化


GPIO_InitTypeDef结构体用于对一个或者一组GPIO来进行初始化,它的四个成员分别是,Pin、Mode、Pull、Speed。


GPIO_InitTypeDef原型:

/** 

  * @brief   GPIO Init structure definition  

  */ 

typedef struct

{

  uint32_t Pin;       /*!< Specifies the GPIO pins to be configured.

                           This parameter can be any value of @ref GPIO_pins_define */


  uint32_t Mode;      /*!< Specifies the operating mode for the selected pins.

                           This parameter can be a value of @ref GPIO_mode_define */


  uint32_t Pull;      /*!< Specifies the Pull-up or Pull-Down activation for the selected pins.

                           This parameter can be a value of @ref GPIO_pull_define */


  uint32_t Speed;     /*!< Specifies the speed for the selected pins.

                           This parameter can be a value of @ref GPIO_speed_define */

}GPIO_InitTypeDef;


Pin

#define GPIO_PIN_0                 ((uint16_t)0x0001)  /* Pin 0 selected    */

#define GPIO_PIN_1                 ((uint16_t)0x0002)  /* Pin 1 selected    */

#define GPIO_PIN_2                 ((uint16_t)0x0004)  /* Pin 2 selected    */

#define GPIO_PIN_3                 ((uint16_t)0x0008)  /* Pin 3 selected    */

#define GPIO_PIN_4                 ((uint16_t)0x0010)  /* Pin 4 selected    */

#define GPIO_PIN_5                 ((uint16_t)0x0020)  /* Pin 5 selected    */

#define GPIO_PIN_6                 ((uint16_t)0x0040)  /* Pin 6 selected    */

#define GPIO_PIN_7                 ((uint16_t)0x0080)  /* Pin 7 selected    */

#define GPIO_PIN_8                 ((uint16_t)0x0100)  /* Pin 8 selected    */

#define GPIO_PIN_9                 ((uint16_t)0x0200)  /* Pin 9 selected    */

#define GPIO_PIN_10                ((uint16_t)0x0400)  /* Pin 10 selected   */

#define GPIO_PIN_11                ((uint16_t)0x0800)  /* Pin 11 selected   */

#define GPIO_PIN_12                ((uint16_t)0x1000)  /* Pin 12 selected   */

#define GPIO_PIN_13                ((uint16_t)0x2000)  /* Pin 13 selected   */

#define GPIO_PIN_14                ((uint16_t)0x4000)  /* Pin 14 selected   */

#define GPIO_PIN_15                ((uint16_t)0x8000)  /* Pin 15 selected   */

#define GPIO_PIN_All               ((uint16_t)0xFFFF)  /* All pins selected */


选择多个Pin的时候,可以用位选的宏定义相或得到。

Mode

#define  GPIO_MODE_INPUT                        ((uint32_t)0x00000000)   /*!< Input Floating Mode                   */

#define  GPIO_MODE_OUTPUT_PP                    ((uint32_t)0x00000001)   /*!< Output Push Pull Mode                 */

#define  GPIO_MODE_OUTPUT_OD                    ((uint32_t)0x00000011)   /*!< Output Open Drain Mode                */

#define  GPIO_MODE_AF_PP                        ((uint32_t)0x00000002)   /*!< Alternate Function Push Pull Mode     */

#define  GPIO_MODE_AF_OD                        ((uint32_t)0x00000012)   /*!< Alternate Function Open Drain Mode    */

#define  GPIO_MODE_AF_INPUT                     GPIO_MODE_INPUT          /*!< Alternate Function Input Mode         */


#define  GPIO_MODE_ANALOG                       ((uint32_t)0x00000003)   /*!< Analog Mode  */


#define  GPIO_MODE_IT_RISING                    ((uint32_t)0x10110000)   /*!< External Interrupt Mode with Rising edge trigger detection          */

#define  GPIO_MODE_IT_FALLING                   ((uint32_t)0x10210000)   /*!< External Interrupt Mode with Falling edge trigger detection         */

#define  GPIO_MODE_IT_RISING_FALLING            ((uint32_t)0x10310000)   /*!< External Interrupt Mode with Rising/Falling edge trigger detection  */


#define  GPIO_MODE_EVT_RISING                   ((uint32_t)0x10120000)   /*!< External Event Mode with Rising edge trigger detection               */

#define  GPIO_MODE_EVT_FALLING                  ((uint32_t)0x10220000)   /*!< External Event Mode with Falling edge trigger detection              */

#define  GPIO_MODE_EVT_RISING_FALLING           ((uint32_t)0x10320000)   /*!< External Event Mode with Rising/Falling edge trigger detection       */


GPIO的模式理解起来有点复杂。

GPIO可以被用作中断源(GPIO_MODE_IT)和事件源(GPIO_MODE_EVT),这以后再叙述。

GPIO也可以用来接收模拟量(GPIO_MODE_ANALOG)。

如果GPIO原本是一个复用I/O,比如可以被作为串口模块使用,要用GPIO_MODE_AF_x来设置。

GPIO可以被设置成输入或者输出。

当被设置成输出时,有open-drain和push-pull两者模式。我觉得理解这两种模式对于像我一样的模电知识较薄弱的人来说有点困难。而且此类知识的理解需要做实验来辅助,在没有实验环境的情况下,还要学习电路仿真软件的使用,学习成本较高,就不做叙述,直接说结论了:

push-pull模式,速度快,但功耗大。

open-drain模式,功耗低,同时有线与功能。如果此模式芯片内部没有上拉电路,需要外接上拉电阻。


Pull


#define  GPIO_NOPULL        ((uint32_t)0x00000000)   /*!< No Pull-up or Pull-down activation  */

#define  GPIO_PULLUP        ((uint32_t)0x00000001)   /*!< Pull-up activation                  */

#define  GPIO_PULLDOWN      ((uint32_t)0x00000002)   /*!< Pull-down activation                */


拉高或拉低。


Speed


#define  GPIO_SPEED_LOW              (GPIO_CRL_MODE0_1) /*!< Low speed */

#define  GPIO_SPEED_MEDIUM           (GPIO_CRL_MODE0_0) /*!< Medium speed */

#define  GPIO_SPEED_HIGH             (GPIO_CRL_MODE0)   /*!< High speed */


不同的速率会产生不同的功耗和噪声,设置Speed的时候要根据实际应用来选取GPIO的工作频率。


设置好属性以后,调用HAL_GPIO_Init来初始化GPIO,函数的第一个参数是一个指向GPIO寄存器组的指针,这个结构体是在芯片手册里定义的。其他对GPIO的操作也都需要传递这个指针,方面在函数里直接访问寄存器,也可以使语义更加清晰。


三、GPIO的操作


GPIO operation函数


GPIO_PinState HAL_GPIO_ReadPin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

void          HAL_GPIO_WritePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, GPIO_PinState PinState);

void          HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

HAL_StatusTypeDef HAL_GPIO_LockPin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

void          HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin);

void          HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin);


HAL里的GPIO操作函数,较之之前的stmlib库更加简单了,基础的操作是读、写、锁、取反,就不做叙述了。


HAL_GPIO_EXTI_x是GPIO用作外部中断时使用的函数。


四、总结


对任一外设,都要先进行初始化,然后再进行操作。通常初始化一个外设的方法就是建立一个描述该外设属性的结构体,用宏定义约定参数值代表的含义。然后将该外设的配置寄存器指针传给初始化函数,初始化函数解释各属性并写相应寄存器。





关键字:stm32  cube  HAL  GPIO 引用地址:stm32.cube(三)——HAL.GPIO

上一篇:stm32.cube(四)——HAL.ADC
下一篇:stm32.cube(二)——HAL结构及初始化

推荐阅读最新更新时间:2024-03-16 16:01

STM32模拟SPI接口
做单片机开发的时候经常需要用到模拟spi接口,这种写法不错的,网上很多类似的,我也拿来主义了。 #define MOSI_H GPIO_SetBits(GPIOB, GPIO_Pin_10) #define MOSI_L GPIO_ResetBits(GPIOB, GPIO_Pin_10) #define SCLK_H GPIO_SetBits(GPIOB, GPIO_Pin_13) #define SCLK_L GPIO_ResetBits(GPIOB, GPIO_Pin_13) #define MISO GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11) unsigned char SPI_
[单片机]
IAR EW6.30版本下建立STM32工程(芯片型号STM32F105VC)
参考博客及文档有: 1,http://www.cnblogs.com/ele-eye/archive/2011/11/28/2266229.html 2,http://www.cnblogs.com/king-77024128/articles/2512997.html(CMSIS介绍) 3, http://wenku.baidu.com/link?url=2blE__GxI1lmJglihBi5xJQPolza-NMxcZqGThxwysiY80ZOe6hLl5k6mowbTN2j30WsPftFZsBe1pRt8WlCDIT7Ce5EIReo4uGCEyegh7S(IAR 新建 STM32工程) 4,
[单片机]
IAR EW6.30版本下建立<font color='red'>STM32</font>工程(芯片型号STM32F105VC)
STM32 nvic 解释
朋友们,如果你需要在STM32上移植RTOS,那么首先必须深入理解它的中断系统。 什么是NVIC?即嵌套向量中断控制器(Nested Vectored Interrupt Controller)。STM32的中有一个强大而方便的NVIC,它是属于Cortex内核的器件,不可屏蔽中断 (NMI)和外部中断都由它来处理,而SYSTICK不是由 NVIC来控制的。 特性: ● 60个可屏蔽中断通道(不包含16个Cortex™-M3的中断线); ● 16个可编程的优先等级(使用了4位中断优先级); ● 低延迟的异常和中断处理; ● 电源管理控制; ● 系统控制寄存器的实现; 1.中断优先级分组 STM32(Cortex-M3)
[单片机]
<font color='red'>STM32</font> nvic 解释
STM32定时器时钟源问题
STM32中有多达8个定时器,其中TIM1和TIM8是能够产生三对PWM互补输出的高级定时器,常用于三相电机的驱动,它们的时钟由APB2的输出产生。其它6个为普通定时器,时钟由APB1的输出产生。 下图是STM32参考手册上时钟分配图中,有关定时器时钟部分的截图: 从图中可以看出,定时器的时钟不是直接来自APB1或APB2,而是来自于输入为APB1或APB2的一个倍频器,图中的蓝色部分。 下面以定时器2~7的时钟说明这个倍频器的作用:当APB1的预分频系数为1时,这个倍频器不起作用,定时器的时钟频率等于APB1的频率;当APB1的预分频系数为其它数值(即预分频系数为2、4、8或16)时,这个倍频器起作用,定时器的时钟
[单片机]
<font color='red'>STM32</font>定时器时钟源问题
ST推出STM32无线微控制器模块,提升物联网产品开发效率
半导体供应商意法半导体推出一个新的加快物联网产品上市的解决方案,该方案可利用现成的微型STM32无线微控制器(MCU)模块加快基于Bluetooth® LE和802.15.4新物联网设备的开发周期。 这个7mm x 11.3mm的 STM32WB5MMG模块让缺少无线设计能力的产品研发团队也能开发物联网产品。为开发层数最少的低成本PCB电路板而设计,新模块集成了直到天线的整个射频子系统。用户还可以免费使用意法半导体的STM32Cube MCU开发生态系统工具、设计向导、射频协议栈和完整软件库,快速高效地完成开发项目。 意法半导体部门副总裁兼微控制器产品总经理Ricardo de Sa Earp表示:“我们的首个基于S
[单片机]
ST推出<font color='red'>STM32</font>无线微控制器模块,提升物联网产品开发效率
stm32学习笔记(三)串口通讯实验
#include sys.h #include usart.h ////////////////////////////////////////////////////////////////////////////////// //如果使用ucos,则包括下面的头文件即可. #if SYSTEM_SUPPORT_OS #include includes.h //ucos 使用 #endif ////////////////////////////////////////////////////////////////////////////////// //本程序只供学习使用,未经作者许可,不得用于其它
[单片机]
<font color='red'>stm32</font>学习笔记(三)串口通讯实验
STM32中使用标准库重定义printf()函数
//重定义函数1 PUTCHAR_PROTOTYPE { /* Place your implementation of fputc here */ /* e.g. write a character to the USART */ USART_SendData(USART1, (uint8_t) ch); /* 循环等待直到发送结束*/ while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET) {} return ch; } //重定义函数2 int fputc(int ch, FILE *f) { USART_SendData(USART
[单片机]
STM32串口实验中重定向c库中的printf()函数和scanf()函数
在STM 32 的串口实验中,往往需要把内容打印输出在串口调试助手中,这时就需要重定向c库中的printf()函数,重定向后可使用。 注意下面的FILE的定义在头文件stdio.h中,所以在写程序时需要包含此头文件。 例: int fputc(int ch,FILE *f) { USART_SendData(USART_TypeDef* USARTx, uint8_t Data); //发送一个字节数到串口 while(USART_GetFlagStatus(USART_TypeDef* USARTx, USART_FLAG_TC)==RESET); //等待发送完毕 return(ch);
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
热门活动
换一批
更多
设计资源 培训 开发板 精华推荐

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

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

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