我应该在SIGINT / SIGTERM上重置termios设置吗?

时间:2017-12-16 21:43:22

标签: c signals termios

我正在玩termios并且我很快发现,如果我更改终端设置并退出,我的更改将会持续存在并且会破坏我的环境。所以我设置我的程序以使用tcgetattr保存初始设置并在退出之前重置它们。

但是,我预测,如果我在程序运行时点击Ctrl-C发送SIGINT,那么它会导致终端仍然有我修改过的设置,因为我的程序没有执行代码来重置它们回到旧设置。

但这并没有发生。在Ubuntu和macOS Sierra中,我的终端设置都被还原了,好像我已经在程序中重置它们一样。

所以问题是:这种行为一般是我可以依赖的吗?或者在退出之前注册信号处理程序以捕获SIGINT / SIGTERM并恢复终端设置是否有意义?

代码

回答这个问题可能不需要查看代码,但这是我的例子,以防你好奇:

#include <stdio.h>
#include <string.h>
#include <termios.h>

int main() {
        // put terminal into non-canonical mode
        struct termios old;
        struct termios new;
        tcgetattr(0, &old);
        new = old;
        new.c_lflag &= ~(ICANON | ECHO);
        tcsetattr(0, TCSANOW, &new);

        // loop: get keypress and display (exit via 'x')
        char key;
        printf("Enter a key to see the ASCII value; press x to exit.\n");
        while (1) {
                key = getchar();
                printf("%i\n", (int)key);
                if (key == 'x') { break; }
        }

        // set terminal back to canonical
        tcsetattr(0, TCSANOW, &old);
        return 0;
}

1 个答案:

答案 0 :(得分:1)

我有点惊讶地发现在我的Arch Linux终端设置中也“被还原”了。但实际上它们保持不变。当我更改你的代码时,我设法跟踪了一些异常情况。

//...
new.c_lflag &= ~(ICANON | ECHO);
new.c_cc[VMIN]  = 0;
new.c_cc[VTIME] = 0;
//...

因此,如果您不按任何按钮,则输出为-1。如果您点击Ctrl-C,重新编译并启动原始程序(来自同一终端),它也会打印-1,因此没有自动重置。

我不知道为什么ECHO被“隐藏”了,我想知道,但我建议您手动恢复所有终端设置。