一尘不染

* nix伪终端如何工作?什么是主/从通道?

linux

我想在Linux系统上用C编写一个简单的X终端仿真器。

一开始,我只是想我必须弹出一个shell并显示其输出。我检查了xterm和rxvt代码,它看起来更加复杂。

首先,我必须使用openpty打开一个伪终端。因此,我看了手册页,发现openpty填充了两个文件描述符,即主文件和从文件。由于这些特殊文件的系统相关性,因此xterm和rxvt代码都很混乱。

我了解termios的内容:这只是关于终端的转义代码的一堆信息。我真正不明白的是:我应该如何处理主/从文件描述符?

打开终端,登录并在shell上执行“ ls”的示例程序非常棒。

(英语不是我的母语,请原谅我的最终错误)

编辑:这是我想出的示例代码:

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <pty.h>
#include <utmp.h>
#include <ctype.h>

void
safe_print (char* s)
{
    while(*s) { 
        if(*s == '\n')
            putchar("\n");
        else if(iscntrl(*s))
            printf("\\e(%d)", *s);
        else
            putchar(*s);
        s++;
    }
}


int
main (int argc, char** argv)
{
    char buf[BUFSIZ] = {0};
    int master;
    int ret = forkpty(&master, NULL, NULL, NULL);

    if(ret == -1)
        puts("no fork"), exit(0);

    if(!ret) { 
        execl("/bin/sh", "sh", NULL);
        exit(0);
    }

    sleep(1); /* let the shell run */


    if(argc >= 2) {
        write(master, argv[1], strlen(argv[1]));
        write(master, "\n", 1);
    } else {
        write(master, "date\n", sizeof "date\n");
    }


    while(1) {
        switch(ret = read(master, buf, BUFSIZ)) {
        case -1:
            puts("error!"); 
            exit(1);
            break;
        case 0:
            puts("nothing.."), sleep(1);
            break;
        default:
            buf[ret] = '\0';
            safe_print(buf);

        }
    }

    close(master);

    return 0;
}

阅读 276

收藏
2020-06-02

共1个答案

一尘不染

关于您问题的主/从部分,从pty(4)手册页(从我系统上的openpty(3)手册页引用):

伪终端是一对字符设备,一个主设备和一个从设备。从设备向进程提供与tty(4)中描述的接口相同的接口。但是,尽管提供tty(4)中描述的接口的所有其他设备在其后均具有某种硬件设备,但从设备却具有通过伪终端的上半部分操纵它的另一个过程。
即,写在主设备上的任何东西都作为输入提供给从设备,写在从设备上的任何东西都作为输入显示在主设备上。

手册页是您的朋友。

2020-06-02