STM32_GPIO配置及庫函數講解——獨立按鍵

2021-01-10 電子產品世界

UserButton硬體連接如下圖所示:當按鍵被按下,PB9檢測到低電平,相反PB9被3.3V電源拉高。

要想將PB9管腳配置成輸入模式,程序所需如下步驟:(必須的)

第一步:配置系統時鐘。見STM32F103xRCC寄存器配置

除此之外,還需將GPIO外設時鐘打開。

/* Enable GPIOC and GPIOB clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);

第二步:配置中斷向量表。決定將程序下載到RAM中還是FLASH中。以後講。

voidNVIC_Configuration(void)
{
#ifdefVECT_TAB_RAM
/* Set the Vector Table base location at 0x20 */
NVIC_SetVectorTable(NVIC_VectTab_RAM,0x0);
#else/* VECT_TAB_FLASH */
/* Set the Vector Table base location at 0x08 */
NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x0);
#endif
}

第三步:配置GPIO的模式。輸入模式還是輸出模式。STM32_GPIO配置及庫函數講解——LED跑馬燈已講過。

void GPIO_Configuration(void){GPIO_InitTypeDef GPIO_InitStructure;/* Configure PC.06, PC.07, PC.08 and PC.09 as Output push-pull */GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 GPIO_Pin_7 GPIO_Pin_8 GPIO_Pin_9;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOC, &GPIO_InitStructure);/* Configure PB.09 as Input pull-up */GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉輸入GPIO_Init(GPIOB, &GPIO_InitStructure);}

第四步:讀該管腳上的電平狀態。需要介紹一個庫函數

vGPIO_ReadInputDataBit從指定Port指定Pin,讀該管腳上的電平狀態:

u8 GPIO_ReadInputDataBit(GPIO_TypeDef*GPIOx,u16 GPIO_Pin)
{
u8 bitstatus=0x00;

/* Check the parameters */
assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
assert_param(IS_GET_GPIO_PIN(GPIO_Pin));

if((GPIOx->IDR&GPIO_Pin)!=(u32)Bit_RESET)
{
bitstatus=(u8)Bit_SET;
}
else
{
bitstatus=(u8)Bit_RESET;
}
returnbitstatus;
}

涉及到GPIO_IDR寄存器,如下所示

經過上面4步,就可以檢測PB9管腳的電平狀態。

按鍵按下,PB9管腳應該是低電平,怎麼才能驗證,最簡單的方法是:當按鍵被按下,點亮所有LED。

下面給出完整程序:

/* Includes */
#include"stm32f10x_lib.h"

/* Private function prototypes --*/
voidRCC_Configuration(void);
voidNVIC_Configuration(void);
voidGPIO_Configuration(void);
voidDelay(vu32 nCount);

/*******************************************************************************
* Function Name : main
* Description : Main program.
* Input : None
* Return : None
*******************************************************************************/
intmain(void)
{
#ifdefDEBUG
debug();
#endif

/* Configure the system clocks */
RCC_Configuration();

/* NVIC Configuration */
NVIC_Configuration();

/* Configure the GPIO ports */
GPIO_Configuration();

/* Infinite loop */
while(1)
{
if(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_9)==0)//檢測USR鍵是否被按下,若按下,則點亮全部LED
{
GPIO_SetBits(GPIOC,GPIO_Pin_6GPIO_Pin_7GPIO_Pin_8GPIO_Pin_9);
}
else
{
GPIO_ResetBits(GPIOC,GPIO_Pin_6GPIO_Pin_7GPIO_Pin_8GPIO_Pin_9);
}
}
}

/*******************************************************************************
* Function Name : RCC_Configuration
* Description : Configures the different system clocks.
* Input : None
* Return : None
*******************************************************************************/
voidRCC_Configuration(void)
{
ErrorStatusHSEStartUpStatus;

/* RCC system reset(for debug purpose) */
RCC_DeInit();

/* Enable HSE */
RCC_HSEConfig(RCC_HSE_ON);

/* Wait till HSE is ready */
HSEStartUpStatus=RCC_WaitForHSEStartUp();

if(HSEStartUpStatus==SUCCESS)
{
/* Enable Prefetch Buffer */
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

/* Flash 2 wait state */
FLASH_SetLatency(FLASH_Latency_2);

/* HCLK = SYSCLK */
RCC_HCLKConfig(RCC_SYSCLK_Div1);

/* PCLK2 = HCLK */
RCC_PCLK2Config(RCC_HCLK_Div1);

/* PCLK1 = HCLK/2 */
RCC_PCLK1Config(RCC_HCLK_Div2);

/* PLLCLK = 8MHz * 9 = 72 MHz */
RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9);

/* Enable PLL */
RCC_PLLCmd(ENABLE);

/* Wait till PLL is ready */
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET){}

/* Select PLL as system clock source */
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

/* Wait till PLL is used as system clock source */
while(RCC_GetSYSCLKSource()!=0x08){}
}

/* Enable GPIOC and GPIOB clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
}

/*******************************************************************************
* Function Name : NVIC_Configuration
* Description : Configures Vector Table base location.
* Input : None
* Return : None
*******************************************************************************/
voidNVIC_Configuration(void)
{
#ifdefVECT_TAB_RAM
/* Set the Vector Table base location at 0x20 */
NVIC_SetVectorTable(NVIC_VectTab_RAM,0x0);
#else/* VECT_TAB_FLASH */
/* Set the Vector Table base location at 0x08 */
NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x0);
#endif
}

/*******************************************************************************
* Function Name : GPIO_Configuration
* Description : Configures the different GPIO ports.
* Input : None
* Return : None
*******************************************************************************/
voidGPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

/* Configure PC.06, PC.07, PC.08 and PC.09 as Output push-pull */
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6GPIO_Pin_7GPIO_Pin_8GPIO_Pin_9;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
GPIO_Init(GPIOC,&GPIO_InitStructure);

/* Configure PB.09 as Input pull-up */
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉輸入
GPIO_Init(GPIOB,&GPIO_InitStructure);
}

/*******************************************************************************
* Function Name : Delay
* Description : Inserts a delay time.
* Input : nCount: specifies the delay time length.
* Return : None
*******************************************************************************/
voidDelay(vu32 nCount)
{
for(;nCount!=0;nCount--);
}

#ifdefDEBUG
/*******************************************************************************
* Function Name : assert_failed
* Description : Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* Input : - file: pointer to the source file name
* - line: assert_param error line source number
* Return : None
*******************************************************************************/
voidassert_failed(u8*file,u32 line)
{
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */

/* Infinite loop */
while(1)
{
}
}
#endif

還有個問題:PB9的初始狀態是什麼?或者說GPIO_Configuration函數後PB9管腳上是高電平還是低電平?

這要看GPIO_InitStructure結構體的GPIO_Mode成員變量初始化為什麼,如果為上拉,則PB9管腳為高電平;如果為下拉,則PB9管腳為低電平。

GPIO管腳內部電路設計如圖:

當GPIO初始化為輸入上拉模式,由上圖可知:該埠向外輸出高電平,即:ODR對應位為1

當GPIO初始化為輸入下拉模式,由上圖可知:該埠向外輸出低電平,即:ODR對應位為0

如何調試:在下面兩處處設個斷點。

1. GPIO初始化函數GPIO_Configuration中

可以看到:當GPIO設置成輸入上拉模式時,等待GPIO初始化完畢,該管腳ODR9為1

2. 在main函數中,設置一個斷點。

全速執行(F5),按住UserButton不放,可以看到GPIOB_IDR的IDR9=0。當放開按鍵時,再單步調試(F10),

GPIOB_IDR的IDR9=1。

總結:

1. GPIO配置成輸入模式時,最好配置成浮空輸入(ODR對應位為0)。上拉、下拉只是該管腳初始化完對外表現的電平狀態。

2. GPIO配置成輸入模式,只關心GPIO_IDR寄存器。檢測該管腳外部輸入的是高電平還是低電平。

3. GPIO配置成輸出模式,只關心GPIO_ODR寄存器。通過該管腳向外部輸出高電平還是低電平。


相關焦點

  • 再造STM32---第十部分:GPIO輸入—按鍵檢測
    按鍵檢測使用到 GPIO 外設的基本輸入功能,本章中不再贅述 GPIO 外設的概念,如您忘記了,可重讀前面「GPIO 框圖剖析」小節, STM32 標準庫中 GPIO 初始化結構體GPIO_TypeDef 的定義與「定義引腳模式的枚舉類型」小節中講解的相同。
  • STM32_GPIO配置及庫函數講解——LED跑馬燈
    gpiogeneral-purposeinput/output通用輸入/輸出埠GPIO寄存器縮寫列表本文引用地址:http://www.eepw.com.cn/article/201611/317025.htm
  • STM32教程(五)HAL庫之GPIO函數及實例詳細介紹!
    今天介紹STM32 HAL庫的GPIO函數庫,首先打開stm32f4xx_hal_gpio.h文件,我們能看到一些結構體定義以及GPIO引腳的地址定義等等,這些都很好理解,我們需要重點看HAL庫中GPIO的函數庫有哪些?以及這些函數庫有什麼作用以及怎麼使用這些函數?
  • STM32的GPIO使用的函數剖析
    該文是自己學習了一段STM32後所寫,是對STM32使用固件庫編程最簡單的一段程序,是對固件庫函數的一部分進行解析。如有錯誤之處請指正,不勝感激。一、GPIO_Init函數解析1本文引用地址:http://www.eepw.com.cn/article/201611/317503.htm1、參數GPIO_TypeDef12、參數GPIO_InitStruct23、函數代碼詳解44、備註6一、GPIO_Init函數解析首先來看一下GPIO_Init函數的原型
  • STM32庫函數配置
    使用V3.0以上版本固件庫的方法如下:本文引用地址:http://www.eepw.com.cn/article/201612/325031.htm1.選擇device(配置函數STM32F10x.h,具體配置方法如下)在STM32F10x.h中有如下代碼:#if !
  • stm32庫函數學習篇通用定時器輸入捕獲功能
    本文引用地址:http://www.eepw.com.cn/article/201611/318488.htm先有必要了解stm32定時器的輸入觸發模塊,如下圖:也就是說,每個通用定時器都只有四個獨立通道,當某一通道作為了輸入觸發功能那就不能再作為輸出匹配功能。這一點我們也可以從其他地方找到印證。
  • STM32 CubeMX按鍵中斷
    按鍵需要上拉電阻,當按鍵按下時接到地,CPU讀取出低電平知道按鍵按下。EXTI0~EXTI4是獨立的,9~5共用中斷源,15~10共用中斷源,9~5和15~10的中斷需要在中斷函數中判斷是哪個中斷引腳。
  • stm32USART使用(庫函數)
    關鍵字:stm32  USART  庫函數 編輯:什麼魚 引用地址:http://news.eeworld.com.cn/mcu/ic475448.html 推薦閱讀
  • STM32F0單片機快速入門五: GPIO中斷
    按鍵如何觸發中斷我們從一個例程看一下,如何用一個GPIO引腳來觸發中斷,來控制另一個引腳的翻轉去控制LED指示燈。Main()函數裡調用了一個外部中斷模塊(EXTI)初始化後就進入while循環了。我們在下面可以找到這個初始化函數的實現。它調用HAL庫函數對按鍵連接的引腳PC13做了初始化,把此引腳初始化為能觸發中斷的模式。
  • stm32通過spi連接esp8266的hspi 開發
    1,stm32 spi驅動開發我使用了stm32的標準庫,並沒用hal庫,因為之前stm32的大部分其它的程序用的都是標準庫開發的,所以沒有改為hal庫。對於spi協議,可以參考另外的文檔,我已經將其傳到了CSDN上可以到我的CSDN的資源頁中下載。如果下載不到可以email 聯繫我,abc_123_ok at 163.com。
  • STM32單片機編程printf()函數重定向
    在stm32單片機編程中,可以重定向printf()函數來輸出調試信息。編寫fputc()函數其實printf()函數在頭文件中被定義成了一個宏,會調用到fputc()函數,但在stm32單片機編程中不可以直接調用標準庫的fputc()函數進行輸出信息,需要自己編寫fputc()函數供printf()函數來調用。編譯器進行編譯時首先會調用自己編寫的fputc()函數,而不是庫函數。
  • STM32關於GPIO的問題
    進行重映射4、配置IO口的頻率問題問:配置IO口的頻率時,為什麼只給一個IO口配置?答:GPIO_InitStructure.GPIO_Speed 這個結構體的成員始終為GPIO_Speed_50MHz 配置其他io口時也默認用50MHz 。5、請問怎樣開始STM32W108CBU64晶片的PB5引腳外設輸入中斷?問:有沒有這麼一個例程,有開啟外設中斷和中斷函數的?
  • stm32的USART如何發送字符、字符串
    一、stm32有自己的字符發送函數。void PC_SendChar(uint8_t DataToSend){USART_SendData (USART1 ,DataToSend);while(USART_GetFlagStatus (USART1 ,USART_FLAG_TC )!
  • 小用stm32f4-CAN控制器(使用庫函數)
    nr");  while(1);}下面是CAN_Config函數首先需要配置使用的GPIO的復用功能,然後配置CAN控制器的收發模式以及過濾器,並打開接收中斷。配置過程可以參照STM32F4的函數庫文件stm32f4xx_can.c中前面注釋中的介紹「How to use this 。。。」
  • 通過stm32 cubemx 配置 串口中斷
    首先選擇好串口①  第三項的配置:上極限,下極限,零位,左極限,右極限,馬達循環上極限,馬達循環下極限,馬達循環次數,馬達循環等待時間;保存至少30款產品的配置,配置界面需要有添加,讀取,查找,刪除功能。
  • STM32開發板例程講解之二:GPIO的描述和配置
    上一講創建了一個stm32工程,從本講開始將深入stm32內核與外設講解。    首先介紹stm32的GPIO,這是入門的起點,也是最容易上手的部分。(七)具有獨立的喚醒I/O口。(八)很多I/O口的復用功能可以重新映射。(九)GPIO口的配置具有上鎖功能,當配置好GPIO口後,可以通過程序鎖住配置組合,直到下次晶片復位才能解鎖。此功能非常有利於在程序跑飛的情況下保護系統中其他的設備,不會因為某些I/O口的配置被改變而損壞——如一個輸入口變成輸出口並輸出電流。
  • stm32晶片被鎖,使用jlink就能輕輕鬆鬆解鎖
    如果一不小心將stm32開發板的晶片給鎖了,那原因可能是忽略了一個比較低級的問題,就是將stm32中的一些不能獨立使用的IO口給了高電平,stm32中有很多口是不能獨立使用的,如圖(以stm32103rct6  正點原子的mini板為例):(N:表示不能獨立使用,Y:表示能獨立使用)
  • 樹莓派用gpiozero庫控制LED燈
    如果你用的不是 noobs,那麼建議使用 Etcher這個鏡像燒錄軟體:安裝gpiozero庫gpiozero庫介紹和一些廢話zero是零的意思,也有初級入門的含義,所以gpiozero庫的宗旨就是為了降低使用樹莓派gpio的門檻,相比於其他操縱樹莓派引腳gpio的庫,gpiozero用戶體驗更加友好,更加直觀和簡單。
  • STM32 printf函數詳解
    本實驗所用的硬體:STM32F103RTB6實驗所用的晶振: 8M本文引用地址:http://www.eepw.com.cn/article/201611/319241.htm實驗所用的ST官方庫:3.5版C語言中的標準庫中所用的標準輸出函數,默認的輸出設備是顯示器,要實現串口或LCD的輸出,必須重新定義標準庫函數裡與輸出函數相關的函數
  • 關於stm32的GPIO的操作
    (5)GPIO_Mode_Out_OD 開漏輸出(6)GPIO_Mode_Out_PP 推輓輸出(7)GPIO_Mode_AF_OD 復用開漏輸出(8)GPIO_Mode_AF_PP 復用推輓輸出 首先簡述一下stm3的gpio