打印
[其他]

MM32F0140 GPIO 学习笔记

[复制链接]
501|3
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
本帖最后由 MM灵灵灵 于 2022-9-29 16:34 编辑

MM32F0140 GPIO 学习笔记
一.GPIO简介

GPIO作为通用输入输出端口,其全称为General Purpose Input Output,能够通过软件自由配置引脚状态及工作模式。除通用输入输出功能外,部分GPIO端口还可被用作第二功能的配置,即为复用功能。

每个通用I/O端口都可以通过软件自由配置为4种输入模式与4种输出模式。

1.输入模式

通过配置GPIOx_CRL寄存器或GPIOx_CRH寄存器中的MODEx[1:0]为00,即配置为输入模式,配置寄存器中的CNFx[1:0]选择工作模式(MM32F0140 的GPIOx中“x”的范围为A到D)。


输入浮空:
浮空就是输入引脚即不接高电平也不接低电平,如图1所示,I/O端口的数据在每个AHB时钟被采样到输入数据寄存器,通过读访问输入数据寄存器获取当前I/O状态,输入浮空常用于复用功能。
上拉输入:
电阻与VDD相连,形成上拉电阻,I/O端口在空闲时为高电平,能够用于检测由高到低的电平变化,常用于按键检测。
下拉输入:
电阻与VSS相连,形成下拉电阻,I/O端口在空闲时为低电平,能够用于检测由低到高的电平变化。
模拟输入:
模拟输入是模拟信号的输入,在模拟输入模式下,上拉电阻、下拉电阻及斯密特触发器均被禁止。

2.输出模式
通过配置GPIOx_CRL寄存器或GPIOx_CRH寄存器中的CNFx[1:0]选择输出模式,配置GPIOx_CRL中的MODEx[1:0]选择输出速度(MODEx[1:0]不为00)。当I/O端口使用复用输出功能时,端口必须配置为复用功能输出模式(推挽或开漏),输出配置如图2所示。


开漏输出:
在开漏输出模式中,输出控制寄存器配置为0时,数据经过输出控制模块,MOS管的栅极接收到高电平,MOS管导通,此时I/O端口接通在GND上,即对应引脚输出低电平;当输出控制寄存器配置为1,数据经过控制模块,给予MOS管的栅极一个低电平,此时MOS管不导通,对应的管脚处于高阻态。因此,开漏输出通常输出低电平,若要输出高电平则需外加上拉电阻。
推挽输出:
推挽输出一般指两个MOS管分别受两个互补信号的控制,总是在一个MOS管导通时另一个MOS管截止。当输出控制寄存器配置为0时,数据经过输出控制模块,MOS管的栅极接收到高电平,N-MOS管导通,P-MOS管不导通,此时I/O端口接通在VSS上,即对应引脚输出为低电平;当输出控制寄存器配置为1时,数据经过输出控制模块,MOS管的栅极接收到低电平,P-MOS管导通,N-MOS管不导通,此时I/O端口接通在VDD上,即对应引脚输出为高电平。因此,推挽输出可以输出高低电平。


复用开漏输出:
配置GPIOx_AFRH与GPIOx_AFRL寄存器的AFRLx[3:0]与AFRHx[3:0]选择复用功能。当GPIO被作为第二功能使用时,模式配置为复用模式,复用功能配置如图3所示。以图4为例,若要配置PB10引脚作为I2C_SCL,则在配置GPIO模式时要选择复用模式。通过片上外设复用功能,使用MOS管实现输出。通常I2C使用GPIO的复用功能时会使用复用开漏输出模式,由于I2C的一个主设备可挂载多个从设备,若不使用复用开漏输出,而使用复用推挽输出,数据传输时,两个从设备一个拉高,一个拉低,可能会造成短路。因此I2C大多使用GPIO的复用开漏输出模式。


复用推挽输出:
当GPIO被作为第二功能配置使用时,模式需配置为复用模式,通过配置GPIO_AFRH与GPIOx_AFRL寄存器的AFRLx[3:0]与AFRHx[3:0]选择复用功能,通过片上外设复用功能,使用两个MOS管来实现输出。与推挽输出不同,复用推挽输出模式下端口的I/O操作由对应复用的功能模块控制,而推挽输出模式下控制I/O端口需对GPIO内部的寄存器进行操作。

二.配置GPIO


首先,使能对应I/O口的时钟,根据所使用的外设对RCC的RCC_AHBENR寄存器进行赋值,将对应外设位置1即可使能时钟,详细外设如图5所示。


其次,配置所需的GPIO引脚、速度及工作模式。端口0到端口7使用GPIOx_CRL寄存器配置工作模式与速度,该寄存器中MODEx[1:0]位表示端口输入输出速度,CNFx[1:0]位表示端口工作模式(”MODEx”与”CNFx”的”x”表示指定端口号)。若配置GPIOx_CRL寄存器中的MODEx位等于00则端口为输入模式,此时CNFx位有四种配置方式,分别为:00(模拟输入模式),01(浮空输入模式),10(上拉/下拉输入模式);若MODEx位不为00,则对应端口为输出模式,此时CNFx具有四种配置方式:00(推挽输出模式),01(开漏输出模式),10(推挽复用模式),11(开漏复用模式)。端口8到端口15的配置使用GPIOx_CRH寄存器配置指定端口的工作模式与速度,详细配置方式与CPIOx_CRL寄存器相同。

若使用端口复用功能,需对GPIOx_AFRL(端口复用功能低位)寄存器与GPIOx_AFRH(端口复用功能高位)寄存器进行配置。端口号为0到7则使用GPIOx_AFRL寄存器,端口号为8到15则使用GPIOx_AFRH寄存器,根据端口号与复用功能表进行配置,例如若使用PA0引脚作为I2C1_SCL,则需将GPIOx_AFRL寄存器中AF3的对应位置1(GPIO的工作模式也要配置为复用模式),PA端口的复用功能表如图6所示。


三.实验

本实验通过使用GPIO获取按键状态控制LED亮灭,读取指定GPIO端口引脚的输入数据(读GPIOx_IDR寄存器)来获取当前的按键状态,通过对端口设置/清除寄存器(GPIOx_BSRR寄存器)与端口位清除寄存器(GPIOx_BRR寄存器)的对应端口赋值,使对应的LED亮灭。具体实验内容为配置PB3引脚对应LED2, PB4引脚对应LED3,PB2引脚对应的K2(如图7所示),若K2按下,则K2对应的端口输入低电平,设置实验现象为LED2灭、LED3亮,K2处于非按下时,K2对应的端口输入高电平,设置实验现象为LED2亮、LED3灭。


1.外设时钟初始化
GPIO在AHB线上,实验使用引脚均为GPIOB组的引脚,因此对RCC_AHBENR寄存器的GPIOB对应位置1。
RCC->AHB1ENR |= (1u << 18u);

2.按键初始化
实验使用引脚为PB2的K2按键,按键原理图如图8所示,若K2按键按下则与GND导通,因此在初始化按键时需配置该端口的工作模式为上拉输入。


GPIOx_CRL寄存器为端口配置低寄存器,用于配置指定端口的速度与工作模式;GPIOx_BSRR寄存器用于设置/清除对应端口,该寄存器低16位的对应端口位置1会产生高电平。由图9所示,K2所使用的端口2为GPIOx_CRL寄存器内第8~11位。


//对应端口的配置涉及到4位,后两位配置端口输入输出速度,前两位配置工作模式;清零端口2的配置位
GPIOB->CRL &= ~(0xf << 8u);

//速度配置位为0,端口为输入模式,上拉输入的工作模式位为10,因此使用0x08.
GPIOB->CRL |= (0x08 << 8u);

//配置PB2引脚为高电平
GPIOB->BSRR |= (1u << 2u);

3.LED初始化
实验使用PB3、PB4,引脚,为观察LED的高低电平,在配置工作模式时使用可以输出高低电平的推挽输出。因为PB3与PB4对应端口在端口0~7中,所以使用GPIOx_CRL寄存器对LED进行初始化配置(若使用端口为8~15则使用GPIOx_CRH寄存器)。

如图9所示,端口3为GPIOx_CRL寄存器内第12~15位,端口4为GPIOx_CRL寄存器内第16~19位。由于LED为低电平点亮,设置GPIOx_BSRR寄存器中LED2与LED3的对应位置1使LED初始状态为灭。
//复位将要使用的端口3与端口4的配置位
GPIOB->CRL &= ~( (0xf << 12u) | (0xf << 16u) );

//端口输入输出速度配置位不为00时,端口为输出模式,配置最大速度为50MHz,推挽输出模式的配置为00.
GPIOB->CRL |= ( (0x01 << 12u) | (0x01 << 16u) );

//PB3对应的LED2初始化状态为灭
GPIOB->BSRR = (1u << 3u);

//PB4对应的LED3初始化状态为灭
GPIOB->BSRR = (1u << 4u);

4.按键扫描
读GPIOx_IDR寄存器获取对应端口输入数据,本实验中K2配置为上拉输入,即按键未按下时为高电平,按下按键后,K2对应端口输入为低电平。若GPIOx_BSRR寄存器的低16位的对应端口位置1,则该端口为高电平;若GPIOx_BRR寄存器的对应端口位置1,则该端口为低电平。实验设置按键未按下时,LED2(PB3)亮、LED3(PB4)灭;按下按键时LED2灭、LED3亮。
while(1)
{
    //K2的引脚为PB2,  1u<<2u = 0100u,将PB2的对应位置1,若读出GPIOx_IDR的对应端口数据为高电平则按键未按下
    if ( 0u != ( GPIOB->IDR & (1u << 2u) ) )
    {
    //PB4引脚对应的LED3灭
        GPIOB->BSRR = (1u << 4u);

        //PB3引脚对应的LED2亮
        GPIOB->BRR = (1u << 3u);
    }
    else  //K2 按键按下
    {
        //PB4引脚对应的LED3亮
        GPIOB->BRR =(1u << 4u);

        //PB3引脚对应的LED2灭
        GPIOB->BSRR = (1u << 3u);
    }
}

5.Main()函数
综合上述寄存器配置到main函数中,图10为实验效果。
int main(void)
{
    //GPIOB时钟初始化
    RCC->AHB1ENR |= (1u << 18u);

    //清零端口2的配置位
    GPIOB->CRL &= ~(0xfu << 8u);
    //端口2配置为上拉输入
    GPIOB->CRL |= (0x08u << 8u);
    //端口2配置为高电平
    GPIOB->BSRR |= (1u << 2u);

    //端口3与端口4配置位清零
    GPIOB->CRL &= ~( (0xf << 12u) | (0xf << 16u) );
    //端口3与端口4配置为推挽输出,速度最大为50MHz
    GPIOB->CRL |= ( (0x01 << 12u) | (0x01 << 16u) );

    //PB3对应的LED2初始化状态为灭
    GPIOB->BSRR = (1u << 3u);
    //PB4对应的LED3初始化状态为灭
    GPIOB->BSRR = (1u << 4u);

    while(1)
    {
        //K2的引脚为PB2,  1u << 2u = 0100u,将PB2的对应位置1,若读出GPIOx_IDR的对应端口数据为高电平则按键未按下
        if ( 0u != ( GPIOB->IDR & (1u << 2u) ) )
        {
            // PB4引脚对应的LED3灭
            GPIOB->BSRR = (1u << 4u);

            //PB3引脚对应的LED2亮
            GPIOB->BRR = (1u << 3u);
        }
        else  //按下按键K2
        {
            // PB4引脚对应的LED3亮
            GPIOB->BRR = (1u << 4u);

            // PB3引脚对应的LED2灭
            GPIOB->BSRR = (1u << 3u);
        }
    }
}

使用特权

评论回复
沙发
海滨消消| | 2022-10-10 16:15 | 只看该作者
学习笔记是个不错的东西,感谢分享

使用特权

评论回复
板凳
哈悟哈悟| | 2022-10-11 10:58 | 只看该作者
感谢分享

使用特权

评论回复
地板
星辰大海不退缩| | 2022-10-13 10:41 | 只看该作者
您好!楼主,这个没有实际引脚嘛?我看您配置直接操作的寄存器呀,应该有封装吧,标准库的话

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

认证:上海灵动微电子股份有限公司
简介:上海灵动微电子股份有限公司成立于 2011 年,是中国本土通用 32 位 MCU 产品及解决方案供应商。 灵动股份的 MCU 产品以 MM32 为标识,基于 Arm Cortex-M 系列内核,自主研发软硬件和生态系统。目前已量产近 300 多款型号,累计交付超 4 亿颗,在本土通用 32 位 MCU 公司中位居前列。

85

主题

103

帖子

5

粉丝