话不多说,先看运行效果:

>./term
input flag 0x00006d02
BRKINT
ICRNL
IMAXBEL
IXANY
IXON
output flag 0x00000005
ONLCR
OPOST
control flag 0x000004bf
CREAD
CSIZE
CS6
CS7
CS8
HUPCL
local flag 0x00008a3b
ECHO
ECHOE
ECHOK
ICANON
IEXTEN
ISIG
input control char array size 32
cc[VDISCARD=13] = 15 (CTRL+O)
VDSUSP not defined
cc[VEOF=4] = 4 (CTRL+D)
cc[VEOL=11] = 255 (CTRL+?)
cc[VEOL2=16] = 255 (CTRL+?)
cc[VERASE=2] = 127 (CTRL+�)
VERASE2 not defined
cc[VINTR=0] = 3 (CTRL+C)
cc[VKILL=3] = 21 (CTRL+U)
cc[VLNEXT=15] = 22 (CTRL+V)
cc[VQUIT=1] = 28 (CTRL+\)
cc[VREPRINT=12] = 18 (CTRL+R)
cc[VSTART=8] = 17 (CTRL+Q)
VSTATUS not defined
cc[VSTOP=9] = 19 (CTRL+S)
cc[VSUSP=10] = 26 (CTRL+Z)
cc[VWERASE=14] = 23 (CTRL+W)

众所周知,通过 tcgetattr 接口与 termios 结构体,我们可以获取一个终端设备的设置信息:

struct termios
{
tcflag_t c_iflag; /* input mode flags */
tcflag_t c_oflag; /* output mode flags */
tcflag_t c_cflag; /* control mode flags */
tcflag_t c_lflag; /* local mode flags */
cc_t c_cc[NCCS]; /* control characters */
};

主要是各种类型的标志位,虽然你可以将它们打印出来,但是一眼望去,这些数字是什么意思,还要查对应平台的 man 手册。

这个工具可以将二进制的标志位,翻译为人类可以读懂的常量宏,例如上面的输出中,可以看到输入标志位打开了 ICRNL 与 IXON 两个标志位,

对应的含义分别是“将输入的CR转换为NL”、“使启动/停止输出控制流起作用”。

看这段输出也许你已经想到了代码的实现,就是挨个常量宏尝试呗,这有啥难的。

不错,但是考虑到不同平台上定义的宏不一致,有时增加一两个宏可能还需要修改源代码,这是多么痛苦的事啊!

这个小工具就解决了这个痛点,你可以在配置文件中指定要测试的宏名称,然后 make 一下就可以啦~~~

iflag.sym

BRKINT
ICRNL
IGNBRK
IGNCR
IGNPAR
IMAXBEL
INLCR
INPCK
ISTRIP
IUCLC
IXANY
IXOFF
IXON
PARMRK

oflag.sym

BSDLY
CMSPAR
CRDLY
FFDLY
NLDLY
OCRNL
OFDEL
OFILL
OLCUC
ONLCR
ONLRET
ONOCR
ONOEOT
OPOST
OXTABS
TABDLY
VTDLY

cflag.sym

CBAUDEXT
CCAR_OFLOW
CCTS_OFLOW
CDSR_OFLOW
CDTR_IFLOW
CIBAUDEXT
CIGNORE
CLOCAL
CREAD
CRTSCTS
CRTS_IFLOW
CRTSXOFF
CSIZE
CSTOPB
HUPCL
MDMBUF
PARENB
PAREXT
PARODD

lflag.sym

ALTWERASE
ECHO
ECHOCTL
ECHOE
ECHOK
ECHOKE
ECHONL
ECHOPRT
EXTPROC
FLUSHO
ICANON
IEXTEN
ISIG
NOFLSH
NOKERNINFO
PENDIN
TOSTOP
XCASE

其实这里是用 awk 读取配置文件自动生成 c 语言的代码来实现的:

print_flag.awk

 #! /bin/awk -f
# usage: print_flag.awk -v FUNC_NAME=xxx -v MACRO_FILE=xxx
# i.e.: print_flag.awk -v FUNC_NAME=output -v MACRO_FILE=oflag.sym
BEGIN {
printf("#include \"../apue.h\"\n")
printf("#include <termios.h>\n")
printf("\n")
printf("void print_%s_flag (tcflag_t flag)\n", FUNC_NAME)
printf("{\n")
printf(" printf (\"%s flag 0x%%08x\\n\", flag); \n", FUNC_NAME)
FS=":"
while (getline < MACRO_FILE > ) {
printf("#ifdef %s\n", $)
printf(" if (flag & %s)\n", $)
printf(" printf (\" %s\\n\"); \n", $)
printf(" else\n")
printf(" printf (\" %s not in\\n\"); \n", $)
printf("#else\n")
printf(" printf (\" %s not defined\\n\"); \n", $)
printf("#endif\n")
}
close (MACRO_FILE)
exit
}
END {
printf("}")
}

生成的 c 文件类似这样:

 #include "../apue.h"
#include <termios.h> void print_input_flag (tcflag_t flag)
{
printf ("input flag 0x%08x\n", flag);
#ifdef BRKINT
if (flag & BRKINT)
printf (" BRKINT\n");
else
printf (" BRKINT not in\n");
#else
printf (" BRKINT not defined\n");
#endif
#ifdef ICRNL
if (flag & ICRNL)
printf (" ICRNL\n");
else
printf (" ICRNL not in\n");
#else
printf (" ICRNL not defined\n");
#endif
}

再看下 Makefile 的生成规则就更清楚啦:

Makefile

 all: term 

 term: term.o print_iflag.o print_oflag.o print_cflag.o print_lflag.o print_cchar.o apue.o
gcc -Wall -g $^ -o $@ term.o: term.c ../apue.h
gcc -Wall -g -c $< -o $@ print_iflag.o: print_iflag.c ../apue.h
gcc -Wall -g -c $< -o $@ print_iflag.c: print_flag.awk iflag.sym
./print_flag.awk -v FUNC_NAME=input -v MACRO_FILE=iflag.sym > print_iflag.c print_oflag.o: print_oflag.c ../apue.h
gcc -Wall -g -c $< -o $@ print_oflag.c: print_flag.awk oflag.sym
./print_flag.awk -v FUNC_NAME=output -v MACRO_FILE=oflag.sym > print_oflag.c print_cflag.o: print_cflag.c ../apue.h
gcc -Wall -g -c $< -o $@ print_cflag.c: print_flag.awk cflag.sym
./print_flag.awk -v FUNC_NAME=control -v MACRO_FILE=cflag.sym > print_cflag.c print_lflag.o: print_lflag.c ../apue.h
gcc -Wall -g -c $< -o $@ print_lflag.c: print_flag.awk lflag.sym
./print_flag.awk -v FUNC_NAME=local -v MACRO_FILE=lflag.sym > print_lflag.c print_cchar.o: print_cchar.c ../apue.h
gcc -Wall -g -c $< -o $@ print_cchar.c: print_char.awk cchar.sym
./print_char.awk -v FUNC_NAME=control -v MACRO_FILE=cchar.sym > print_cchar.c log.o: ../log.c ../log.h
gcc -Wall -g -c $< -o $@ apue.o: ../apue.c ../apue.h
gcc -Wall -g -c $< -o $@ -D__USE_BSD clean:
@echo "start clean..."
-rm -f *.o core.* *.log *~ *.swp term
@echo "end clean" .PHONY: clean

具体分析下生成过程:

1.通过 print_flag.awk 分别生成 print_iflag.c / print_oflag.c / print_cflag.c / print_lflag.c

2.分别将生成的 .c 编译为 .o 文件

3.在生成 term 工具时链接上述 .o 文件生成最终的可执行文件

当然了,除了各种标志位外,这里还处理了 cc_t cc 字段,它打印每个特殊输入字符,原理和上面相仿,就不再赘述了。

检查打印的特殊字符,发现少了下标为 5 / 6 / 7 的字符,查看头文件定义,原来是 linux 上面增加了三个新的定义:

cchar.sym

VTIME
VMIN
VSWTC

将它们添加到 sym 文件中,重新编译、运行,果然新的输出里有了:

    cc[VTIME=5] = 0 (CTRL+@)
cc[VMIN=6] = 1 (CTRL+A)
cc[VSWTC=7] = 255 (CTRL+?)

这对于在不同平台上进行测试有很大的帮助。

后记

其实有一个现成的命令:stty,这个用来查看、修改终端设置,例如显示当前终端所有设置:

>stty -a
speed 38400 baud; rows 24; columns 80; line = 0;
intr = ^C; quit = ^\; erase = ^?; kill = ^U; eof = ^D; eol = M-^?; eol2 = M-^?;
swtch = M-^?; start = ^Q; stop = ^S; susp = ^Z; rprnt = ^R; werase = ^W;
lnext = ^V; flush = ^O; min = 1; time = 0;
-parenb -parodd cs8 hupcl -cstopb cread -clocal -crtscts -cdtrdsr
-ignbrk brkint -ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl ixon -ixoff
-iuclc ixany imaxbel iutf8
opost -olcuc -ocrnl onlcr -onocr -onlret -ofill -ofdel nl0 cr0 tab0 bs0 vt0 ff0
isig icanon iexten echo echoe echok -echonl -noflsh -xcase -tostop -echoprt
echoctl echoke

不过这个命令是将所有配置打印出来,一不对它们进行分组、二不存在的标志也会显示出来(前面有个-),相比而言还是我们的工具更友好一些。

[apue] 一个查看当前终端标志位设置的小工具的更多相关文章

  1. 韩顺刚-tcp报文头协议详细分析第一包数据:序号是0,发送数据的长度是0,因为没有收到对端的数据,所以确认号是0, Syn的标志位设置成1,这里没有发送的数据,只发送TCP的20个字节的头部

    TCP报文段首部格式 大部分TCP报文头部都是20个字节,有的数据包要加上选项. 上面一行代表4个字节,源端口和目的端口都是2个字节. TCP协议是面向字节流的协议 TCP是一段一段分块的发送数据的 ...

  2. 分享一个 Linux 环境下,强力的Python 小工具

    场景 Linux 用户,经常需要在终端查看一些数据,从文件里看 或者网络协议获取数据并查看. 比如,查看文件里的json数据:比如,查看etcd里存下的数据. 如果直接看cat 或者 curl 得到的 ...

  3. [apue] 一个快速确定新系统上各类限制值的工具

    对于在不同 Unix 系统之间移植程序,最重要的事情就是确定新系统的一些编译时.运行时固定或不固定的限制值了.例如文件路径最大长度 PATH_MAX.进程最大可打开文件句柄数 OPEN_MAX.用户可 ...

  4. 进程保护--CrossThreadFlags标志位

    原理: 1. 将进程的所有线程的线程CrossThreadFlags标志位设置成Terminated或者System. 效果:任务管理器,WSYSCheck,ICESWORD无法结束进程.. 但PCH ...

  5. InkImageDataSetGenerator-开源一个可用于机器学习的书写轨迹图片生成的小工具

    这是一个简单易用的图片数据集生成小工具,基于OpenCV和UWP Ink API,它可以根据指定的手写轨迹生成一系列各个角度的图片.每张图片的尺寸和总体数量都是可以指定的,均存放在统一的生成目录中.h ...

  6. 从点击Button到弹出一个MessageBox, 背后发生了什么(每个UI线程都有一个ThreadInfo结构, 里面包含4个队列和一些标志位)

    思考一个最简单的程序行为:我们的Dialog上有一个Button, 当用户用鼠标点击这个Button时, 我们弹出一个MessageBox.   这个看似简单的行为, 谁能说清楚它是如何运行起来的,背 ...

  7. C# 设置和获取一个字节的某一位的值的方法

    自己工作中遇到需要对单字节的高位.低位进行赋值,即一个字节byte,想要给每一位都赋值,这个值是动态来的,是0或是1. 好不容易收集到一些珍贵资料,整理一下: 一.设置 方法code: /// < ...

  8. 如何只修改EFLAGS寄存器中一个标志位的值?

    版权声明:本文为博主原创文章,2019-08-23,22:21:42转载请附上原文出处链接和本声明.作者By-----溺心与沉浮----博客园   1.写汇编指令只影响CF位的值(不能影响其他标志位 ...

  9. 登录操作(方法一:设置flag标志位)

    登录操作(方法一:设置flag标志位) user_name="star"passwoed='123'passed_authentication=Falsecount=0for i ...

随机推荐

  1. P1022 绵羊排序

    题目描述 聪聪想要给他家农场里的 \(n\) 只绵羊按照品质从高到低进行排序. 农场里的 \(n\) 只绵羊编号从 \(1\) 到 \(n\) ,第 \(i\) 只绵羊的体重为 \(w_i\) ,高度 ...

  2. CSS 兼容问题

    CSS常见兼容性问题总结 浏览器的兼容性问题通常是因为不同的浏览器对不同的代码有不同的解析造成页面显示不统一的情况,这里的浏览器通常指IE 6,7,8,9... Google Firefox Oper ...

  3. Nutch网页抓取速度优化

    Nutch网页抓取速度优化 Here are the things that could potentially slow down fetching 1) DNS setup 2) The numb ...

  4. Netty小结

    前言 在实际开发中,netty的开发使用相对较小,why?在企业中涉及网络编程的部分比重较小,在这大环境内,企业会优先使用简单的http,udp等基础的通讯协议工具,如果不能满足需求,会考虑基于rpc ...

  5. 从物联网设备生命周期理解Apple Homekit框架

    本次笔记主要基于苹果公开的HomeKit Accessory Protocol协议文档和IOS Security Guide中相关部分. 总体来讲,设备与App之间采用端到端的加密,密钥由设备和App ...

  6. 用VISA工具驱动继电器外设

    1.驱动方式:TCP 2.开发过程 第一步:外设识别 TCP方式将继电器插上网线后,并不能像串口一样自动识别到这个外设,需要手动连接.打开NI MAX后,右击设备与接口,然后点击新建,双击VISA T ...

  7. schema list validator --python cerberus

    工作中需要对如下json结构进行验证: "ActiveStatus" : [ { "effectiveDates" : { "effectiveFro ...

  8. 怎么彻底删除用友通T3财务软件?

    [问题现象]怎么彻底删除用友通T3财务软件? [原因分析]通过"添加或删除程序"无法正常卸载用友通T3,也尝试了360安全卫士强力卸载,都无法完全卸载,有没有办法可以彻底删除用友通 ...

  9. 日志管理-log4j与slf4j的使用

    一.概述 1.log4j: Log4j是Apache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台.文件.GUI组件,甚至是套接口服务器.NT的事件记录器.UNIX Sy ...

  10. 【题解】P4137 Rmq Problem(莫队)

    [题解]P4137 Rmq Problem(莫队) 其实这道题根本就不用离散化! 因为显然有\(mex\)值是\(\le 2\times 10^5\)的,所以对于大于\(2\times 10^5\)的 ...