上一講創(chuàng)建了一個stm32工程,從本講開始將深入stm32內核與外設講解。首先介紹stm32的GPIO,這是入門的起點,也是最容易上手的部分。
本文引用地址:http://www.ex-cimer.com/article/201611/322388.htm
一、GPIO的綜合描述
stm32每一個GPIO端口擁有2個32bits的configuration寄存器(GPIOx_CRL,GPIOx_CRH),2個32bits的數據寄存器(GPIOx_IDR,GPIOx_ODR),1個32bits的set/reset寄存器(GPIOx_BSRR),1個16bits的reset寄存器(GPIOx_BRR)和1個32bits的Lock寄存器(GPIOx_LCKR)。
(一)每一個IO引腳都可以使用軟件配置為以下幾種模式:
1.浮空輸入
2.帶上拉輸入
3.帶下拉輸入
4.模擬輸入
5.開漏輸出——(此模式可實現hotpower說的真雙向IO)
6.推挽輸出
7.復用功能的推挽輸出
8.復用功能的開漏輸出
模式7和模式8需根據具體的復用功能決定。
每一個IO引腳都可以單獨編程,但是每一個IO寄存器只能32bits訪問(半字或者字節(jié)訪問都被禁止)。
(二)專門的寄存器(GPIOx_BSRR和GPIOx_BRR)實現對GPIO口的原子操作,即回避了設置或清除I/O端口時的“讀-修改-寫”操作,使得設置或清除I/O端口的操作不會被中斷處理打斷而造成誤動作。
(三)每個GPIO口都可以作為外部中斷的輸入,便于系統靈活設計。
(四)I/O口的輸出模式下,有3種輸出速度可選(2MHz、10MHz和50MHz),這有利于噪聲控制。
(五)所有I/O口兼容CMOS和TTL,多數I/O口兼容5V電平。
(六)大電流驅動能力:GPIO口在高低電平分別為0.4V和VDD-0.4V時,可以提供或吸收8mA電流;如果把輸入輸出電平分別放寬到1.3V和VDD-1.3V時,可以提供或吸收20mA電流。
(七)具有獨立的喚醒I/O口。
(八)很多I/O口的復用功能可以重新映射。
(九)GPIO口的配置具有上鎖功能,當配置好GPIO口后,可以通過程序鎖住配置組合,直到下次芯片復位才能解鎖。此功能非常有利于在程序跑飛的情況下保護系統中其他的設備,不會因為某些I/O口的配置被改變而損壞——如一個輸入口變成輸出口并輸出電流。
二、GPIO的結構
三、GPIO的配置
(一)GPIO模式選擇和速度匹配
(1) 浮空輸入_IN_FLOATING ——浮空輸入,可以做KEY識別,RX1。
(2)帶上拉輸入_IPU——IO內部上拉電阻輸入。
(3)帶下拉輸入_IPD—— IO內部下拉電阻輸入。
(4) 模擬輸入_AIN ——應用ADC模擬輸入,或者低功耗下省電。
(5)開漏輸出_OUT_OD ——IO輸出0接GND,IO輸出1,懸空,需要外接上拉電阻,才能實現輸出高電平。當輸出為1時,IO口的狀態(tài)由上拉電阻拉高電平,但由于是開漏輸出模式,這樣IO口也就可以由外部電路改變?yōu)榈碗娖交虿蛔???梢宰xIO輸入電平變化,實現C51的IO雙向功能。
(6)推挽輸出_OUT_PP ——IO輸出0-接GND, IO輸出1 -接VCC,讀輸入值是未知的。
(7)復用功能的推挽輸出_AF_PP ——片內外設功能(I 2C的SCL,SDA)
(8)復用功能的開漏輸出_AF_OD——片內外設功能(TX1,MOSI,MISO.SCK)
GPIO輸出的速度匹配:
GPIO_Speed_10MHz 最高輸出速率10MHz
GPIO_Speed_2MHz 最高輸出速率2MHz
GPIO_Speed_50MHz 最高輸出速率50MHz
I/O口的輸出模式下,有3種輸出速度可選(2MHz、10MHz和50MHz),這個速度是指I/O口驅動電路的響應速度而不是輸出信號的速度,輸出信號的速度與程序有關(芯片內部在I/O口的輸出部分安排了多個不同響應速度的輸出驅動電路,用戶可以根據自己的需要選擇合適的驅動電路)。通過選擇速度來選擇不同的輸出驅動模塊,達到最佳的噪聲控制和降低功耗的目的。高頻的驅動電路,噪聲也高,當不需要高的輸出頻率時,請選用低頻驅動電路,這樣非常有利于提高系統的EMI性能。當然如果要輸出較高頻率的信號,但卻選用了較低頻率的驅動模塊,很可能會得到失真的輸出信號。
有一點是關鍵,即GPIO的引腳速度跟應用匹配(推薦10倍以上)。比如:
1)對于串口,假如最大波特率只需115.2k,那么用2M的GPIO的引腳速度就夠了,既省電也噪聲小。
2)對于I2C接口,假如使用400k波特率,若想把余量留大些,那么用2M的GPIO的引腳速度或許不夠,這時可以選用10M的GPIO引腳速度。
3)對于SPI接口,假如使用18M或9M波特率,用10M的GPIO的引腳速度顯然不夠了,需要選用50M的GPIO的引腳速度。
(二)在STM32中如何配置片內外設使用的IO端口
①配置輸入的時鐘;②初始化后即被激活(開啟);③如果使用該外設的輸入輸出管腳,則需要配置相應的GPIO端口(否則該外設對應的輸入輸出管腳可以做普通GPIO管腳使用);④再對外設進行詳細配置。
對應到外設的輸入輸出功能有下述三種情況:
①外設對應的管腳為輸出:需要根據外圍電路的配置選擇對應的管腳為復用功能的推挽輸出或復用功能的開漏輸出。
②外設對應的管腳為輸入:則根據外圍電路的配置可以選擇浮空輸入、帶上拉輸入或帶下拉輸入。
③ADC對應的管腳:配置管腳為模擬輸入。
如果把端口配置成復用輸出功能,則引腳和輸出寄存器斷開,并和片上外設的輸出信號連接。將管腳配置成復用輸出功能后,如果外設沒有被激活,那么它的輸出將不確定。
(三)通用IO端口(GPIO)初始化:
1、RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|B|C,ENABLE):使能APB2總線外設時鐘;
2、RCC_APB2PeriphResetCmd(RCC_APB2Periph_GPIOA|B|C,DISABLE):釋放GPIO復位;
3、配置各個PIN端口(模擬輸入_AIN、輸入浮空_IN_FLOATING、輸入上拉_IPU、輸入下拉_IPD、開漏輸出_OUT_OD、推挽式輸出_OUT_PP、推挽式復用輸出_AF_PP、開漏復用輸出_AF_OD)和匹配速度。
4、GPIO初始化完成
四、GPIO實例
#include "stm32f10x.h"
#include "stm32_eval.h"
GPIO_InitTypeDef GPIO_InitStructure;
void RCC_Configuration(void);
void Delay(__IO uint32_t nCount);
int main(void)
{
RCC_Configuration();
#if 0
// 配置所有未使用GPIO引腳為輸入模式(浮空輸入),這樣可以降低功耗,并且提高器件的抗EMI/EMC 的性能
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |
RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD |
RCC_APB2Periph_GPIOE, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_Init(GPIOC, &GPIO_InitStructure);
//armfly :注釋掉的原因是當代碼在外部存儲器運行時,GPIOD,E,F,G部分IO用于FSMC,因此對這些IO不能重置,否則導致取指異常
// GPIO_Init(GPIOD, &GPIO_InitStructure);
// GPIO_Init(GPIOE, &GPIO_InitStructure);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |
RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD |
RCC_APB2Periph_GPIOE, DISABLE);
#ifdef USE_STM3210E_EVAL
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF | RCC_APB2Periph_GPIOG, ENABLE);
// armfly :由于代碼在外部SRAM運行時,GPIOD,E,F,G部分IO用于FSMC,
因此對這些IO不能重置,否則導致取指異常
//GPIO_Init(GPIOF, &GPIO_InitStructure);
//GPIO_Init(GPIOG, &GPIO_InitStructure);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF | RCC_APB2Periph_GPIOG, DISABLE);
#endif
#endif
// 初始化STM3210X-EVAL板上的4個LED,即初始化LED對應的IO引腳
STM_EVAL_LEDInit(LED1);
STM_EVAL_LEDInit(LED2);
STM_EVAL_LEDInit(LED3);
STM_EVAL_LEDInit(LED4);
while (1)
{
STM_EVAL_LEDOn(LED1);
// 原始值 = 0xAFFFF,但是當代碼在外部RAM運行時,效率很低,會延遲10秒以上,而到代碼在內部RAM或內部Flash執(zhí)行時,速度很快,小于100ms
Delay(0xAFFFF);
// Turn on LD2 and LD3
STM_EVAL_LEDOn(LED2);
STM_EVAL_LEDOn(LED3);
// Turn off LD1
STM_EVAL_LEDOff(LED1);
// Insert delay
Delay(0xAFFFF);
// Turn on LD4
STM_EVAL_LEDOn(LED4);
//Turn off LD2 and LD3
STM_EVAL_LEDOff(LED2);
STM_EVAL_LEDOff(LED3);
//Insert delay
Delay(0xAFFFF);
//Turn off LD4
STM_EVAL_LEDOff(LED4);
}
}
void RCC_Configuration(void)
{
// 建立MCU系統,初始化嵌入式FLASH接口,初始化鎖相環(huán)(附注1)和系統頻率參數
SystemInit();
}
void Delay(__IO uint32_t nCount)
{
for(; nCount != 0; nCount--);
}
#ifdefUSE_FULL_ASSERT
// @briefReports the name of the source file and the source line number
//where the assert_param error has occurred.
//@paramfile: pointer to the source file name
// @paramline: assert_param error line source number
// @retval None
void assert_failed(uint8_t* file, uint32_t 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", file, line)
//Infinite loop
while (1)
{
}
}
#endif
評論