第11章 GPIO的触手
树莓派可以通过很多接口来连接到其他设备。在各种各样的接口中,最有特色的就是一组GPIO(General Purpose Input/Output)接口。这组GPIO接口大大拓展了树莓派的能力。GPIO不仅能实现通信,还能直接控制电子元器件,从而让用户体验到硬件编程的乐趣。
11.1 GPIO简介
树莓派3上的GPIO接口由40个针脚(PIN)组成,如图11-1所示。
图11-1 树莓派3的GPIO针脚
每个针脚都可以用导线和外部设备相连。你可以通过焊接的方式把导线固定在PIN上,也可以用母型的跳线套接在PIN上。在40个PIN中,有固定输出的5V(2、4号PIN)、3.3V(1、17号PIN)和地线(Ground,6、9、14、20、25、30、34、39)。如果一个电路两端接在5V和地线之间,那么该电路就会获得5V的电压输入。27和28号PIN标着ID_SD和ID_SC。它们是两个特殊的PIN,用于和附加的电路板通信。其他的PIN大多编成GPIOX的编号,如GPIO14。树莓派的操作系统会用GPIO的编号14来指代这个PIN,而不是位置编号的8。
有一些PIN不仅有GPIO功能,还能充当其他形式的端口。比如,GPIO14和GPIO15除了作为GPIO接口,还可以充当UART端口。此外,GPIO上还能找到I2C和SPI接口。
计算机中用高、低两个电压来表示二进制的1和0。树莓派上的GPIO用相同的方式来表示数据。每个GPIO的PIN都能处于输入或输出状态。当处于输出状态时,系统可以把1或0传给该PIN。如果是1,那么对应的物理PIN向外输出3.3V的高电压,否则输出0V的低电压。相应的,处于输入状态的PIN可以探测物理PIN上的电压。如果是高电压,那么该PIN将向系统返回1,否则返回0。利用简单的二元机制树莓派实现了和物理电路的互动。
11.2 控制LED灯
我们先来看GPIO输出的一个例子。在GPIO21和地线之间接一个串联电路,电路上有一个LED灯,还有一个用于防止短路的330Ω电阻。当GPIO21位于高电平时,将有电流通过电路点亮LED灯,如图11-2所示。
图11-2 GPIO与LED灯连接
我们通过Shell来控制GPIO21。在Linux中,外部设备经常被表示成文件。向文件写入或读取字符,就相当于向设备输出或者从设备输入字符。树莓派上的GPIO端口也是如此,其代表文件位于/sys/class/gpio/下。
首先,激活GPIO21:
$echo 21 > /sys/class/gpio/export
这个命令把字符“21”输入/sys/class/gpio/export中。命令执行后,/sys/class/gpio/下面增加了代表GPIO21的一个目录,目录名就是gpio21。
其次,把GPIO21置于输出状态:
$echo out > /sys/class/gpio/gpio21/direction
文件/sys/class/gpio/gpio21/direction用于控制GPIO21的方向,向里面写入了代表输出的字符“out”。
最后,向GPIO21写入1,从而让PIN处于高电压:
$echo 1 > /sys/class/gpio/gpio21/value
可以看到,LED灯亮了起来。
如果想关掉LED灯,那么只需要向GPIO21写入0:
$echo 0 > /sys/class/gpio/gpio21/value
使用完GPIO21可以删除该端口:
$echo 21 > /sys/class/gpio/unexport
/sys/class/gpio/gpio21随即消失。
11.3 两个树莓派之间的GPIO
我们可以用GPIO的方式连接两个树莓派,如图11-3所示。一个树莓派的GPIO输出将成为另一个树莓派的GPIO输入。连接方式很简单,只需要两根导线。一根导线连接两个树莓派的地线,另一根导线连接树莓派的两个PIN。
图11-3 两个树莓派之间用GPIO连接
用左侧的树莓派来输出,用右侧树莓派来输入。输出过程和上面控制LED灯的例子相似。第一个树莓派中的GPIO21准备输出:
$echo 21 > /sys/class/gpio/export $echo out > /sys/class/gpio/gpio21/direction
在第二个树莓派中,准备好读取GPIO26:
$echo 26 > /sys/class/gpio/export $echo in > /sys/class/gpio/gpio26/direction
当我们向/sys/class/gpio/gpio26中写入“in”时,就是把GPIO26置于输入状态。
此后,在第一个树莓派中就可以更改输出值为1或0了:
$echo 1 > /sys/class/gpio/gpio21/value $echo 0 > /sys/class/gpio/gpio21/value
在第二个树莓派中,可以用cat命令来读取文件,获得输入值:
$cat /sys/class/gpio/gpio26/value
cat命令读完一次后会返回,为了持续读取,可以用bash中的while循环来反复调用cat:
$while true; do cat /sys/class/gpio/gpio26/value; done
这里的while是bash提供的循环结构,随后do和done之间的内容会重复执行。第二个树莓派将循环查看GPIO26的输入值。当第一个树莓派中的输出改变时,第二个树莓派获得的输入也随之改变。我们在两个树莓派之间实现了简单的通信。
最后,在使用完GPIO后,别忘了删除端口。
11.4 UART编程
计算机的数据是许多位的0和1构成的序列。尽管GPIO可以在0和1之间切换,但无法传输二进制序列。比如,把一个二进制序列11000111输出到GPIO端口,在输入端看来,只是输入了一段时间的1,然后变成0,最后又变成1。输入端无法准确说出,一段高电平输入究竟包含了几位1。
一个解决方案是用多个PIN同时通信,每个PIN表示一位。当输入端读取完成后,通知输出端,让输出端送来下一批的数据。这种通信方式被称为并口传输。和并口传输对应的是串口传输,传输时依然是用一个PIN,但输入方可以知道一位数据持续了多长时间。GPIO上的UART、I2C、SPI都是串口通信。
UART与其余两者的区别在于,通信双方通过事先约定的速率来发送或接收数据。这种通信方式称为异步通信。I2C和SPI这类同步通信方式会用额外的连线来保证双方速率相同。UART的连线和实现方式很简单,因而成为最流行的串口通信方式。但UART的缺点在于,如果发送方和接收方的速率不同,那么通信就会发生错误。通信速率称为波特率(Baudrate),单位是每秒通信的位数(bps)。
UART的端口至少有RX、TX和地线三个针脚。RX负责读取,TX负责输出。如果有两个UART端口,它们的连接方式如图11-4所示。
图11-4 UART连接
在树莓派3上,TX和RX就是GPIO14和GPIO15针脚。因此,我们可以把两个树莓派按照图11-4的方式连接起来,然后在两个树莓派之间实现UART通信。
在这里,我们要注意树莓派3发生了一点变化。树莓派1和2中都使用了标准的UART,在操作系统中的对应文件是/dev/ttyAMA0。在树莓派3中,新增的蓝牙模块占用了标准U A RT端口和树莓派沟通,外部的UART通信采用了简单的Mini UART,在操作系统中的对应文件是/dev/ttyS0。由于mini UART的波特率依赖于CPU时钟频率,而CPU时钟频率可能在运行过程中浮动,因此mini UART经常会带来意想不到的错误。一般有两种解决方案:一种是关闭蓝牙模块,让外部连接重新使用标准UART端口;另一种是固定CPU时钟频率,以便mini UART能以准确的波特率进行通信。
关闭蓝牙模块,需要修改/boot/config.txt。将dtoverlay键的值改为:
dtoverlay=pi3-disable-bt
修改后重启。此后的U A RT通信,就可以通过/dev/ttyAMA0进行。
如果采取第二种解决方案,那么要修改/boot/config.txt,把上面的修改变成:
core_freq=250 dtoverlay=pi3-miniuart-bt
修改后重启。此后的U A RT通信就可以通过/dev/ttyS0进行了。
我们以第一种解决方案为例,进行UART通信。首先,设定波特率:
$stty -F /dev/ttyAMA09600
然后,向UART端口输出文本:
$echo "hello" > /dev/ttyAMA0
在UART的另一端读取文本:
$cat /dev/ttyAMA0
可以看到,UART可以实现更加复杂的文本通信。如果使用第二种解决方案,即限定核心频率的办法,那么只需把/dev/ttyAMA0改为/dev/ttyS0即可。
11.5 用UART连接PC
一般的PC都没有暴露在外的UART针脚。为了通过UART来连接PC和树莓派,我们需要一个USB和UART的转换器。这个转换器的一端是USB接口,另一端是UART的针脚。我们把USB一端插入PC,另一端按照UART到UART的方式,连接到树莓派的UART针脚。
连接好之后,就可以在PC上利用串口操作软件来和树莓派通信了。在Linux下,USB连接表示为/dev/ttyUSB0。当然,当计算机上只有1个USB设备时,最后的编号才会是0。而在笔者的Mac OS X上,该USB连接被表示成/dev/cu.SLAB_USBtoUART。此后,就可以通过操作USB文件来进行UART通信了。
11.6 用UART登录树莓派
我们还可以用UART的方式连接并登录树莓派。进入树莓派设置:
$sudo raspi-config
在Interfacing Options→Serial中,允许开机时通过串口登录。
重启后,树莓派启动时会自动把开机信息以115200的波特率推到UART端口。在UART另一端的PC上,如果使用Mac OS X,那么可以用下面的命令连接:
$screen /dev/cu.SLAB_USBtoUART 115200
如果PC是Linux系统,则只需把USB设备文件改为对应的设备文件即可。如果是Windows系统,则可以使用Putty通过串口连接树莓派。首先在Windows的设备管理器中找到该USB设备。假如USB设备被识别为COM3,那么在Putty的设置页面中,把连接类型(Connection Type)设置成串口(Serial),然后在串口线路(Serial Line)中输入USB设备的名称,例如COM3。速度(Speed)选择115200。设置好后单击“打开(Open)”按钮即可连接。