GPIO之推挽输出和开漏输出

疑问

GPIO配置为输出时会有两种模式,一种叫推挽输出,一种叫开漏模式。那什么是推挽输出,什么又是开漏输出呢?

三种输出状态

如下图所示为将GPIO配置为输出时的内部示意图:
在这里插入图片描述
由上图可以看出,GPIO的输出状态完全取决于两个MOS管Q1和Q2的导通状态:

  1. Q1导通、Q2关断,此时输出接VCC,输出高电平;
  2. Q1关断、Q2导通,此时输出接地,输出低电平;
  3. Q1关断、Q2关断,此时输出浮空,相对与其它点的电阻无穷大,呈现高阻态;
  4. Q1导通、Q2导通,此时VCC直接对地短路,会烧毁MOS管,所以此状态不会也不允许存在。

所以,将GPIO配置为输出时,其只会处于以上1、2、3三种状态(即高电平、低电平、高阻态)之一。

推挽输出

首先,我们把上文中提到的1、2两种状态(即高电平、低电平)拿出来,组成一个组合。这便是我们对GPIO输出最基本的认知,要么输出高电平,要么输出低电平。

当输出高电平时,电流按下图箭头所示流出去,我们称之为,把电流推出去。
在这里插入图片描述
当输出低电平时,电流按下图箭头所示流进来,我们称之为,把电流挽回来。
在这里插入图片描述
所以,所谓的推挽,其实描述的是GPIO输出高低电平时电流的一个动作而已。

开漏输出

然后我们将上文中的2、3两种状态(即低电平、高阻态)拿出来,组成一个组合。

不难发现,在2、3两种状态中Q1都是关断的,所以可以认为它就是不存在,如下图:
在这里插入图片描述
此时,MOS管的漏极等于啥也没接,处于一个开路状态,所以这个模式称之为开漏模式

那开漏模式有什么用呢?

  1. 改变高电平的电压
    比如说现在想用这个GPIO去控制一个芯片的enable引脚,如下图所示:
    在这里插入图片描述
    后面芯片的enable引脚只支持3.3V输入,但是GPIO输出的高电平却由5V。如果用推挽模式输出5V的话就可能会把后面芯片烧毁。所以我们就需要使用开漏模式,同时外接一个上拉电阻。此时,将Q2关断时(即高阻态),enable引脚就被上拉电阻拉到了3.3V,也就是高电平,如下图所示:
    在这里插入图片描述
    同样的,当Q2导通时,enable引脚就接地被拉到了低电平。
    这样便可实现5V单片机对3.3V芯片的控制。
  2. 多个GPIO控制同一个输入
    如下图所示用两个GPIO去控制一个芯片的enable:
    在这里插入图片描述
    如果用推挽模式的话,当上面的GPIO输出高电平,下面的GPIO输出低电平时,两个GPIO之间就短路了,会烧毁MOS管。
    此时,我门将两个GPIO设置为开漏模式,再外接一个上拉电阻,如图:
    在这里插入图片描述
    这样的话,只要有任意一个GPIO输出为低电平时,enable都会被拉低。只有当两个GPIO为高阻态时,enable才会被拉高。

所以,开漏输出一般都会配置一颗外部的上拉电阻。


参考:推挽 开漏 高阻 这都是谁想出来的词??

好的,以下是使用STM32F427II初始化所有GPIO口为复用推挽输出的示例代码: ```c #include "stm32f427xx.h" void GPIO_Init(void) { // 开启GPIO时钟 RCC->AHB1ENR |= RCC_AHB1ENR_GPIOAEN | RCC_AHB1ENR_GPIOBEN | RCC_AHB1ENR_GPIOCEN | RCC_AHB1ENR_GPIODEN | RCC_AHB1ENR_GPIOEEN | RCC_AHB1ENR_GPIOFEN | RCC_AHB1ENR_GPIOGEN | RCC_AHB1ENR_GPIOHEN | RCC_AHB1ENR_GPIOIEN | RCC_AHB1ENR_GPIOJEN | RCC_AHB1ENR_GPIOKEN; // 配置所有GPIO口为复用推挽输出 GPIOA->MODER = 0xAAAAAAAA; GPIOA->OTYPER = 0x00000000; GPIOA->OSPEEDR = 0xFFFFFFFF; GPIOA->PUPDR = 0x00000000; GPIOA->AFR[0] = 0x00000000; GPIOA->AFR[1] = 0x00000000; GPIOB->MODER = 0xAAAAAAAA; GPIOB->OTYPER = 0x00000000; GPIOB->OSPEEDR = 0xFFFFFFFF; GPIOB->PUPDR = 0x00000000; GPIOB->AFR[0] = 0x00000000; GPIOB->AFR[1] = 0x00000000; GPIOC->MODER = 0xAAAAAAAA; GPIOC->OTYPER = 0x00000000; GPIOC->OSPEEDR = 0xFFFFFFFF; GPIOC->PUPDR = 0x00000000; GPIOC->AFR[0] = 0x00000000; GPIOC->AFR[1] = 0x00000000; GPIOD->MODER = 0xAAAAAAAA; GPIOD->OTYPER = 0x00000000; GPIOD->OSPEEDR = 0xFFFFFFFF; GPIOD->PUPDR = 0x00000000; GPIOD->AFR[0] = 0x00000000; GPIOD->AFR[1] = 0x00000000; GPIOE->MODER = 0xAAAAAAAA; GPIOE->OTYPER = 0x00000000; GPIOE->OSPEEDR = 0xFFFFFFFF; GPIOE->PUPDR = 0x00000000; GPIOE->AFR[0] = 0x00000000; GPIOE->AFR[1] = 0x00000000; GPIOF->MODER = 0xAAAAAAAA; GPIOF->OTYPER = 0x00000000; GPIOF->OSPEEDR = 0xFFFFFFFF; GPIOF->PUPDR = 0x00000000; GPIOF->AFR[0] = 0x00000000; GPIOF->AFR[1] = 0x00000000; GPIOG->MODER = 0xAAAAAAAA; GPIOG->OTYPER = 0x00000000; GPIOG->OSPEEDR = 0xFFFFFFFF; GPIOG->PUPDR = 0x00000000; GPIOG->AFR[0] = 0x00000000; GPIOG->AFR[1] = 0x00000000; GPIOH->MODER = 0xAAAAAAAA; GPIOH->OTYPER = 0x00000000; GPIOH->OSPEEDR = 0xFFFFFFFF; GPIOH->PUPDR = 0x00000000; GPIOH->AFR[0] = 0x00000000; GPIOH->AFR[1] = 0x00000000; GPIOI->MODER = 0xAAAAAAAA; GPIOI->OTYPER = 0x00000000; GPIOI->OSPEEDR = 0xFFFFFFFF; GPIOI->PUPDR = 0x00000000; GPIOI->AFR[0] = 0x00000000; GPIOI->AFR[1] = 0x00000000; GPIOJ->MODER = 0xAAAAAAAA; GPIOJ->OTYPER = 0x00000000; GPIOJ->OSPEEDR = 0xFFFFFFFF; GPIOJ->PUPDR = 0x00000000; GPIOJ->AFR[0] = 0x00000000; GPIOJ->AFR[1] = 0x00000000; GPIOK->MODER = 0xAAAAAAAA; GPIOK->OTYPER = 0x00000000; GPIOK->OSPEEDR = 0xFFFFFFFF; GPIOK->PUPDR = 0x00000000; GPIOK->AFR[0] = 0x00000000; GPIOK->AFR[1] = 0x00000000; } ``` 在这个例子中,我们首先开启了所有GPIO口的时钟,然后使用一个简单的循环将每个GPIO口都配置为复用推挽输出。我们还需要使用GPIO的AFR寄存器来设置GPIO的复用功能。你可以根据你的具体需求修改这些参数。
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值