使用termios()替换EOF?

时间:2017-01-09 15:31:39

标签: c linux terminal

我有一个在Linux C上运行的伪终端,如图所示打开:

else
{ 
     struct termios slave_orig_term_settings; // Saved terminal settings
     struct termios new_term_settings; // Current terminal settings

    // CHILD
    close(fdm);

    rc = tcgetattr(fds, &slave_orig_term_settings);

    new_term_settings = slave_orig_term_settings;

    new_term_settings.c_cc[VEOF] = '|';
    new_term_settings.c_oflag &= ~(ICANON | ECHO | ECHOE | ECHOK | ECHONL | ISIG | ICRNL);

    cfmakeraw (&new_term_settings);

    tcsetattr (fds, TCSANOW, &new_term_settings);

}

从管道读取时,我可以进行非阻塞读取(),它将在例如5秒后返回。问题是,如果我执行像" ls"这样的命令,它会在几毫秒后返回,它将在返回前等待整整5秒。

我理解使用物理GUI终端,而不是像这样的伪终端,用户可以按CTRL ^ D并在每个输出结束时发送EOF。

我想要这个效果,所以我的程序可以说"好的,我收到了这个特定的字节,这意味着终端已完成输出,我现在可以将此输出返回给调用者,我知道那里没有更多的输出阅读"。

我理解这是用termios()函数(或它的包装stty)完成的,但是我无法使它工作。我希望终端向我发送' |'输出完成后,char而不是ctrl ^ d(\ x04),但它不起作用。

为清楚起见,在输出完成后(通过read()),我希望收到' |'字节。所以,如果我发送' pwd'命令,我得到这个输出:

 $ pwd
 /home/user/dir
 $|

有人可以告诉我如何修改此代码以实现这种效果吗?

感谢您的帮助。

编辑 - 选择()代码

while ( (bytes = select(pipe + 1, &fd_in, NULL, NULL, &tv)) )
{       
            if (FD_ISSET(pipe, &fd_in))
            {

                read (pipe...)

上述代码的行为就像我所描述的那样,即使管道上没有任何东西,也要等到tv的全部值,然后再返回。

1 个答案:

答案 0 :(得分:1)

永远不会向您发送c_cc[VEOF]的值。例如,c_cc[VEOF]的默认值为CTRL-D,但当用户按下CTRL-D时,您不会在文件末尾看到CTRL-D。相反,我们会通知您(例如read()返回0)您已到达文件末尾。

通常当孩子完成其工作后,它会明确地close() pty奴隶方,以便pty主人将获得EOF

关于

  

我可以创建一个非阻塞的read(),它将在例如5秒之后返回。问题是,如果我执行像" ls"这样的命令,它会在几个ms内返回,它将在返回之前等待整整5秒。

不确定如何编写代码,但您可以使用select()poll(),当有可供阅读的数据时,可以立即通知您。

根据select()的手册页:

  

select()pselect()允许程序监控多个文件描述符,等待直到一个或多个文件描述符变为"准备好" I / O操作类(例如,输入可能)。如果可以执行相应的I / O操作(例如,read(2)没有阻塞,或者足够小的write(2)),则认为文件描述符已准备好。