我有一个在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的全部值,然后再返回。
答案 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)
),则认为文件描述符已准备好。