ermios 结构是在POSIX规范中定义的标准接口,它相似于系统V中的termio接口,经过设置termios类型的数据结构中的值和使用一小ios
组函数调用,你就能够对终端接口进行控制。shell
能够被调整来影响终端的值按照不一样的模式被分为以下几组:数组
1.输入模式数据结构
2.输出模式函数
3.控制模式spa
4.本地模式命令行
5.特殊控制模式code
最小的termios结构的典型定义以下:blog
struct termios { tcflag_t c_iflag; tcflag_t c_oflag; tcflag_t c_cflag; tcflag_t c_lflag; cc_t c_cc[NCCS]; };
结构成员的名称与上面列出的5种参数类型相对应。接口
你能够调用函数tcgetattr来初始化一个终端对应的termios结构,该函数的原型以下:
#include<termios.h> int tcgetattr(int fd, struct termios *termios_p);
这个函数调用把当前终端接口变量的值写入termios_p参数指向的结构。若是这些值其后被修改了,你能够经过调用函数tcsetattr来从新配置终端接口。
#include<termios.h> int tcsetattr(int fd , int actions , const struct termios *termios_h);
参数actions控制修改方式,共有三种修改方式,以下所示。
1.TCSANOW:马上对值进行修改
2.TCSADRAIN:等当前的输出完成后再对值进行修改。
3.TCSAFLUSH:等当前的输出完成以后,再对值进行修改,但丢弃还未从read调用返回的当前的可用的任何输入。
接下来咱们将分别对五种模式进行介绍。
输入模式控制输入数据在传递给程序以前的处理方式。你经过设置termios结构中的c_iflag成员的标志对它们进行控制。全部的标志都被定义为
宏,并可经过按位或的方式结合起来。
可用于c_iflag成员的宏以下所示:
BRKINT:当在输入行中检测到一个终止状态时,产生一个中断。
TGNBRK:忽略输入行中的终止状态。
TCRNL:将接受到的回车符转换为新行符。
TGNCR:忽略接受到的新行符。
INLCR:将接受到的新行符转换为回车符。
IGNPAR:忽略奇偶校检错误的字符。
INPCK:对接收到的字符执行奇偶校检。
PARMRK:对奇偶校检错误做出标记。
ISTRIP:将全部接收的字符裁减为7比特。
IXOFF:对输入启用软件流控。
IXON:对输出启用软件流控。
若是BRKINT和TGNBRK标志都未被设置,则输入行中的终止状态就被读取为NULL(0X00)字符。
输出模式控制输出字符的处理方式,即由程序发出的字符在传递到串行口或屏幕以前如何处理.经过设置c_oflag成员的标识对输出模式进行控制.
OPSOT:打开输出处理功能
ONLCR:将输出中的换行符转换为回车符
OCRNL:将回车符转换为换行符
ONOCR:第0行不输出回车符
ONLRET:不输出回车符
NLDLY:换行符延时选择
CRDLY:回车符延时
TABDLY:制表符延时
...
输出模式用得也很少
控制模式控制终端的硬件特性,经过c_cflag成员标识配置.
CLOCAL:忽略全部调制解调器的状态行
CREAD:启用字符接收器
CS5/6/7/8:发送或接收字符时使用5/6/7/8比特
CSTOPB:每一个字符使用两中止位
HUPCL:关闭时挂断调制解调器
PARENB:启用奇偶校验码的生成和检测功能
PARODD:只使用奇检验而不用偶校验
通常也不用这种方式,一般直接修改终端配置文件来修改硬件特性要容易一些
五.本地模式
经过c_lflag成员控制终端的某些特性
ECHO:启用输入字符的本地回显功能
ECHONL:回显换行符
ICANON:启用标准输入处理
ISIG:启用信号
...
最经常使用的是ECHO和ICANON标志,前者抑制键入字符的回显(抑制??),后者如说明
标准模式和非标准模式下,c_cc数组的下标有不一样的值:
标准模式:
VEOF:EOF字符
VEOL:EOL字符
VERASE:ERASE字符
VINTR:INTR字符
VKILL:KILL字符
VQUIT:QUIT字符
VSTART:START字符VSTOP:STOP字符
非标准模式:
VINTR:INTR字符
VMIN:MIN值
VQUIT:QUIT字符
VSUSP:SUSP字符
VTIME:TIME值
VSTART:START字符
VSTOP:STOP字符
INTR:该字符使终端驱动程序向与终端相连的进程以送SIGINT信号
QUIT:该字符使终端驱动程序向与终端相连的进程发送SIGQUIT信号
EOF;该字符使终端驱动程序将输入行中的所有字符传递给正在读取输入的应用程序.若是输入行为空,read调用将返回0,就好像在文件尾调用read同样
...
这两个值只用于非标准模式,二者结合共同控制对输入的读取方式,还能控制在一个程序试图与一个终端关联的文件描述符时将发生的状况
MIN = 0, TIME = 0时:read当即返回,若是有待处理的字符,它们就会被返回,若是没有,read调用返回0,且不读取任何字符
MIN = 0, TIME > 0时:有字符处理或通过TIME个0.1秒后返回
MIN > 0, TIME = 0时:read一直等待,直到有MIN个字符能够读取,返回值是字符的数量.到达文件尾时返回0
MIN > 0, TIME > 0时:read调用时,它会等待接收一个字符.在接收到第一个字符及其后续的每一个字符后,启用一个字符间隔定时器.当有MIN个字符可读或两字符间的时间间隔超进TIME个0.1秒时,read返回
经过设置MIN和TIME值,咱们能够逐个字符地对输入进行处理
stty -a:这个命令用来查看当前终端的设置状况
stty sane:若是不当心设错了终端模式,可用这个命令恢复,另外一种恢复办法是在设置以前保存当前stty设置,在须要时再读出
stty -g > save_stty:将当前设置保存到文件save_atty中
stty $(cat save_stty):读出save_atty文件,恢复原终端设置
第三种恢复的办法是从新打下一个终端模拟器.查看死掉的终端进程,kill掉它
4.在命令行模式下设置终端模式
好比想让shell脚本读取单个字符,就须要关闭标准模式,同时将MIN设为1,TIME设为0:
stty -icanon min1 time 0
另外一个例子是关闭输入密码时的回显功能:
atty -echo
使用完这个命令后要执行atty echo,将回显功能再次恢复
termios结构中没有关于终端速度的成员和标识,但咱们能够经过一组函数来实现.注意输入和输出是分开的,应使用不一样的函数
#include <termios.h> speed_t cfgetispeed(const struct termios *); speed_t cfgetospeed(const struct termios *); int cfsetispeed(struct termios *, speed_t speed); int cfseospeed(struct termios *, speed_t speed);
这些函数只做用于termios结构,所以须要先调用tcgetattr()得到termios结构,再调用以上函数之一设置终端速度,最后调用tcsetattr()使设置生效
上面的speed参数可设的值,其中比较重要的几个:
B0:挂起终端
B1200:1200波特
B2400:2400波特
B9600:9600波特
B19200:19200波特
B38400:38400波特
这些函数直接做用于文件描述符,不须要读写termios结构:
#include <termios.h> int tcdrain(int fd);让调用程序一直等待,直到全部排队的输出都发送完毕 int tcflow(int, int flowtype);暂停或从新开始输出 int tcflush(int fd, int in_out_selector);清空输入,输出或二者都清空