打印
[牛人杂谈]

准双向模式什么时候可以用什么时候不可以用?

[复制链接]
5353|11
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
gaoyang9992006|  楼主 | 2016-7-3 23:09 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
通用 I/O (GPIO)
NuMicro M05xxBN 和 M05xxDN/DE最多有40个通用I/O引脚,这些引脚和其它功能共享。40个引脚分为5个端口,分别命名为P0, P1, P2, P3 和 P4,每个端口最多有8个引脚。每个引脚都是独立的,都有相应的寄存器来控制引脚工作模式与数据。每个引脚的I/O类型可由软件独立地配置为输入,输出,开漏或准双向模式。复位后,M05xxBN默认所有的I/O引脚处于准双向模式,M05xxDN/DE默认I/O引脚的模式由CONFIG0的CIOINI位决定(准双向或者输入模式),端口数据寄存器Px_DOUT[7:0]的值为0x000_00FF。每个I/O引脚配有一个非常弱的独立的上拉电阻,VDD从5.0V 到 2.5V时,内部弱上拉电阻阻值大约为110KΩ~300KΩ。
特性
4种 I/O 模式:
 输入模式带高阻
 推挽输出
 开漏输出
 准双向
 准双向TTL/Schmitt 触发输入模式由Px_MFP[23:16]选择
 每个I/O 引脚都可以作为中断源,支持边沿/电平触发
 准双向模式下,I/O引脚内部上拉电阻被使能
 引脚中断功能使能后,引脚的唤醒功能也将被使能
 复位后,所有I/O引脚默认模式由CIOINI(CONFIG[10])决定(只有M05xxDN/DE)
 CIOINI = 0, 复位后所有GPIO引脚为输入三态模式
 CIOINI = 1, 复位后所有GPIO引脚为准双向模式
————————————————————————
那么问题来了什么时候能用准双向,什么时候不可以?
沙发
mintspring| | 2016-7-3 23:09 | 只看该作者
准双向口就是带上拉电阻功能的开漏输出口,需要大电流高电平输出能力的场合和高速场合不可以用这种结构

使用特权

评论回复
板凳
mintspring| | 2016-7-3 23:12 | 只看该作者
单片机I/O口推挽与开漏输出详解
推挽输出:可以输出高,低电平,连接数字器件;推挽结构一般是指两个三极管分别受两互补信号的控制,总是在一个三极管导通的时候另一个截止.
开漏输出:输出端相当于三极管的集电极. 要得到高电平状态需要上拉电阻才行. 适合于做电流型的驱动,其吸收电流的能力相对强(一般20ma以内).

我们先来说说集电极开路输出的结构。集电极开路输出的结构如图1所示,右边的那个三极管集电极什么都不接,所以叫做集电极开路(左边的三极管为反相之用,使输入为“0”时,输出也为“0”)。对于图1,当左端的输入为“0”时,前面的三极管截止(即集电极C跟发射极E之间相当于断开),所以5V电源通过1K电阻加到右边的三极管上,右边的三极管导通(即相当于一个开关闭合);当左端的输入为“1”时,前面的三极管导通,而后面的三极管截止(相当于开关断开)。

      我们将图1简化成图2的样子。图2中的开关受软件控制,“1”时断开,“0”时闭合。很明显可以看出,当开关闭合时,输出直接接地,所以输出电平为0。而当开关断开时,则输出端悬空了,即高阻态。这时电平状态未知,如果后面一个电阻负载(即使很轻的负载)到地,那么输出端的电平就被这个负载拉到低电平了,所以这个电路是不能输出高电平的。

再看图三。图三中那个1K的电阻即是上拉电阻。如果开关闭合,则有电流从1K电阻及开关上流过,但由于开关闭其它三个口带内部上拉),当我们要使用输入功能时,只要将输出口设置为1即可,这样就相当于那个开关断开,而对于P0口来说,就是高阻态了。

对于漏极开路(OD)输出,跟集电极开路输出是十分类似的。将上面的三极管换成场效应管即可。这样集电极就变成了漏极,OC就变成了OD,原理分析是一样的。

另一种输出结构是推挽输出。推挽输出的结构就是把上面的上拉电阻也换成一个开关,当要输出高电平时,上面的开关通,下面的开关断;而要输出低电平时,则刚好相反。比起OC或者OD来说,这样的推挽结构高、低电平驱动能力都很强。如果两个输出不同电平的输出口接在一起的话,就会产生很大的电流,有可能将输出口烧坏。而上面说的OC或OD输出则不会有这样的情况,因为上拉电阻提供的电流比较小。如果是推挽输出的要设置为高阻态时,则两个开关必须同时断开(或者在输出口上使用一个传输门),这样可作为输入状态,AVR单片机的一些IO口就是这种结构。

开漏电路特点及应用

在电路设计时我们常常遇到开漏(open drain)和开集(open collector)的概念。
所谓开漏电路概念中提到的“漏”就是指MOSFET的漏极。同理,开集电路中的“集”就是指三极管的集电极。开漏电路就是指以MOSFET的漏极为输出的电路。一般的用法是会在漏极外部的电路添加上拉电阻。完整的开漏电路应该由开漏器件和开漏上拉电阻组成。如图1所示:


组成开漏形式的电路有以下几个特点:
    1. 利用外部电路的驱动能力,减少IC内部的驱动(或驱动比芯片电源电压高的负载)。当IC内部MOSFET导通时,驱动电流是从外部的VCC流经R pull-up ,MOSFET到GND。IC内部仅需很下的栅极驱动电流。如图1。
    2. 可以将多个开漏输出的Pin,连接到一条线上。形成 “与逻辑” 关系。如图1,当PIN_A、PIN_B、PIN_C任意一个变低后,开漏线上的逻辑就为0了。这也是I2C,SMBus等总线判断总线占用状态的原理。如果作为输出必须接上拉电阻。接容性负载时,下降延是芯片内的晶体管,是有源驱动,速度较快;上升延是无源的外接电阻,速度慢。如果要求速度高电阻选择要小,功耗会大。所以负载电阻的选择要兼顾功耗和速度。
    3. 可以利用改变上拉电源的电压,改变传输电平。如图2, IC的逻辑电平由电源Vcc1决定,而输出高电平则由Vcc2(上拉电阻的电源电压)决定。这样我们就可以用低电平逻辑控制输出高电平逻辑了(这样你就可以进行任意电平的转换)。(例如加上上拉电阻就可以提供TTL/CMOS电平输出等。)


4. 开漏Pin不连接外部的上拉电阻,则只能输出低电平(因此对于经典的51单片机的P0口而言,要想做输入输出功能必须加外部上拉电阻,否则无法输出高电平逻辑)。一般来说,开漏是用来连接不同电平的器件,匹配电平用的。
    5. 标准的开漏脚一般只有输出的能力。添加其它的判断电路,才能具备双向输入、输出的能力。

6.正常的CMOS输出级是上、下两个管子,把上面的管子去掉就是OPEN-DRAIN了。这种输出的主要目的有两个:电平转换、线与。

7.线与功能主要用于有多个电路对同一信号进行拉低操作的场合,如果本电路不想拉低,就输出高电平,因为OPEN-DRAIN上面的管子被拿掉,高电平是靠外接的上拉电阻实现的。(而正常的CMOS输出级,如果出现一个输出为高另外一个为低时,等于电源短路。)

8.OPEN-DRAIN提供了灵活的输出方式,但是也有其弱点,就是带来上升沿的延时。因为上升沿是通过外接上拉无源电阻对负载充电,所以当电阻选择小时延时就小,但功耗大;反之延时大功耗小。所以如果对延时有要求,则建议用下降沿输出。



应用中需注意: 1.   开漏和开集的原理类似,在许多应用中我们利用开集电路代替开漏电路。例如,某输入Pin要求由开漏电路驱动。则我们常见的驱动方式是利用一个三极管组成开集电路来驱动它,即方便又节省成本。如图4。


  2. 上拉电阻R pull-up的阻值决定了逻辑电平转换的沿的速度。阻值越大,速度越低功耗越小。反之亦然。
     Push-Pull输出就是一般所说的推挽输出,在CMOS电路里面应该较CMOS输出更合适,因为在CMOS里面的push-pull输出能力不可能做得双极那么大。输出能力看IC内部输出极N管P管的面积。和开漏输出相比,push-pull的高低电平由IC的电源低定,不能简单的做逻辑操作等。push-pull是现在CMOS电路里面用得最多的输出级设计方式。
    当然open drain也不是没有代价,这就是输出的驱动能力很差。输出的驱动能力很差的说法不准确,驱动能力取决于IC中的末级晶体管功率。OD只是带来上升沿的延时,因为上升沿是通过外接上拉无源电阻对负载充电的,当电阻选择小时延时就小、但功耗大,反之延时大功耗小。OPEN DRAIN提供了灵活的输出方式,但也是有代价的,如果对延时有要求,建议用下降沿输出。
    电阻小延时小的前提条件是电阻选择的原则应在末级晶体管功耗允许范围内,有经验的设计者在使用逻辑芯片时,不会选择1欧姆的电阻作为上拉电阻。在脉冲的上升沿电源通过上拉无源电阻对负载充电,显然电阻越小上升时间越短,在脉冲的下降沿,除了负载通过有源晶体管放电外,电源也通过上拉电阻和导通的晶体管对地 形成通路,带来的问题是芯片的功耗和耗电问题。电阻影响上升沿,不影响下降沿。如果使用中不关心上升沿,上拉电阻就可选择尽可能的大点,以减少对地通路的 电流。如果对上升沿时间要求较高,电阻大小的选择应以芯片功耗为参考。



使用特权

评论回复
地板
mintspring| | 2016-7-3 23:16 | 只看该作者
有人讨论过这个问题,如下,很好。

  最近公司在进行一个项目,需要用到超声波测距的功能,于是在做好硬件电路,但在写控制程序时,却遇上了令我费解的事情。

当在单片机最小系统上调好输出频率40kHz,占空比50%的方波输出信号后,将程序烧至超声波应用电路中的主控中。实验时,却发现输出波形变成了频率40kHz,占空比小于1%的尖波输出信号。

于是我怀疑自己的电路有问题,遂拿了市面上销售的超声波模块成品来学习一下,发现单片机端口的输出波形很正常,没有变形有情况。在网上搜索良久,一直不明白是怎么回事。当某天在**论坛看到一位网友的一句话,提到GPIO端口的“强推挽工作模式”,对我启发很大,开始怀疑是不是端口工作模式设置的问题。

很多年前就学单片机了,一直以来也经常写一些小程序,但是使用GPIO端口时,一直是按默认的配置来使用的(即“准双向IO”)。

超声波传感器好比是一个电容,当它两个引脚分别接单片机的两个端口时,需要一个端口输出高电平的同时,另一个端口输出低电平,如此反复,才能像晶振一样“起振”,从而输出超声波。但是,如果A端口输出高电平,然后B端口输出低电平,按单片机GPIO的“准双向模式”来分析,此时A端口的状态必然就被改变为与B端口一致了。

那么如何实现“A端口输出高电平,B端口输出低电平,两个端口的状态却不相互影响”的效果呢?

答案是:强推换输出模式。

设置GPIO的工作模式为“强推换输出模式”后,就不会输入电路外部的连接状态了,就是说变成一个纯粹的输出端口了,没有了输出的功能,这样端口的状态只有MCU自己靠程序来改变,外部电路不能改变其状态。


使用特权

评论回复
5
mintspring| | 2016-7-3 23:31 | 只看该作者
在头文件中四中模式的定义宏如下所示。
/*---------------------------------------------------------------------------------------------------------*/
/*  PMD Constant Definitions                                                                               */
/*---------------------------------------------------------------------------------------------------------*/
#define GPIO_PMD_INPUT          0x0UL /*!< Input Mode */
#define GPIO_PMD_OUTPUT         0x1UL /*!< Output Mode */
#define GPIO_PMD_OPEN_DRAIN     0x2UL /*!< Open-Drain Mode */
#define GPIO_PMD_QUASI          0x3UL /*!< Quasi-bidirectional Mode */


使用特权

评论回复
6
落叶行健ywm| | 2016-7-4 17:09 | 只看该作者
IO引脚可以配置为四种模式:

1,高阻输入模式,这种方式只用于输入数据的读取。
2,开漏模式,这种方式若不接上拉电阻时,则可以输出高阻或低电平两种状态,作为输出是一般需要外接上拉电阻的。
3,准双向模式,这种方式具有输入和输出的功能。当IO输出为高电平时,其驱动能力很弱,外部负载很荣誉将其拉至低电平。当IO输出为低时,其驱动能力很强,可吸收相当大的电流。准双向IO有三个上拉晶体管“极弱上拉”、“弱上拉”、“强上拉”。当IO作为输入时具有一个施密特触发器,用于抑制输入抖动和干扰,此时锁存器必须输入为1。
4,推挽模式,这种方式具有很强的驱动电流能力。
       以上四种IO配置各有特点。高阻模式只能做输入。开漏模式具有很好的电气兼容性,外部上拉电阻接3V电源就能和3V逻辑器件接口;若上拉电阻接5V电源,则可以接5V逻辑器件接口,因为内部没有上拉电阻所以对于单片机的功耗是很小的。准双向IO的最大特点是即可做输入又可做输出不需要通过控制字切换方向;推挽输出的特点是驱动电流能力强,无论是上拉还是下拉,但推挽模式只能用于输出模式不能用于输入模式。

使用特权

评论回复
7
捉虫天师| | 2016-7-4 19:48 | 只看该作者
/* Configure P1.2 as Output mode and P4.1 as Input mode */
    GPIO_SetMode(P1, BIT2, GPIO_PMD_OUTPUT);
    GPIO_SetMode(P4, BIT1, GPIO_PMD_INPUT);
这个是例程里的配置为输入和输出的方式。那么其他的方式如何配置呢,我们就要找找都在头文件怎么说的了。
/*---------------------------------------------------------------------------------------------------------*/
/*  PMD Constant Definitions                                                                               */
/*---------------------------------------------------------------------------------------------------------*/
#define GPIO_PMD_INPUT          0x0UL /*!< Input Mode */
#define GPIO_PMD_OUTPUT         0x1UL /*!< Output Mode */
#define GPIO_PMD_OPEN_DRAIN     0x2UL /*!< Open-Drain Mode */
#define GPIO_PMD_QUASI          0x3UL /*!< Quasi-bidirectional Mode */


void GPIO_SetMode(GPIO_T *port, uint32_t u32PinMask, uint32_t u32Mode);
对比发现了吧,Mode的变量其实就是GPIO_PMD_
好变态,也不弄个一致命名的变量。

使用特权

评论回复
8
捉虫天师| | 2016-7-4 19:56 | 只看该作者
因此如果设置为开漏模式那就是
GPIO_SetMode(P1, BIT2, GPIO_PMD_OPEN_DRAIN);
设置为准双向模式那就是
GPIO_SetMode(P1, BIT2, GPIO_PMD_QUASI);

使用特权

评论回复
9
捉虫天师| | 2016-7-4 20:20 | 只看该作者
/* Configure P1.2 and P4.1 to default Quasi-bidirectional mode */
    GPIO_SetMode(P1, BIT2, GPIO_PMD_QUASI);
    GPIO_SetMode(P4, BIT1, GPIO_PMD_QUASI);
这就是配置为准双向模式。

使用特权

评论回复
10
Jessicakjdsl| | 2016-7-5 14:31 | 只看该作者
什么叫准双向模式呢,就是能做输出的时候不能作为输入?

使用特权

评论回复
11
gaoyang9992006|  楼主 | 2016-7-7 14:19 | 只看该作者
Jessicakjdsl 发表于 2016-7-5 14:31
什么叫准双向模式呢,就是能做输出的时候不能作为输入?

准双向就是可以像51单片机一样,端口既可以输入也可以当输出。

使用特权

评论回复
12
Micachl| | 2016-7-9 20:58 | 只看该作者
用作数据线的时候属于准双向模式吗

使用特权

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

本版积分规则

个人签名:如果你觉得我的分享或者答复还可以,请给我点赞,谢谢。

2005

主题

16120

帖子

214

粉丝