我在linux上运行了一个psuedo终端,打开后如下所示:
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);
}当从管道读取时,我可以做一个非阻塞的read(),它将在例如5秒之后返回。问题是,如果我执行一个像"ls“这样的命令,它以几毫秒为单位返回,它将在返回之前等待整整5秒。
我理解使用物理GUI终端,而不是像这样的psuedo终端,用户可以按下CTRL^D并在每个输出的末尾发送EOF。
我想要这种效果,所以我的程序可以说:“好的,我已经收到了这个字节,这意味着终端已经完成输出,现在我可以将输出返回给调用者,并且我知道没有更多的输出可以读取”。
我知道这是使用termios()函数(或者它的包装器stty)完成的,但是我无法让它工作。我希望终端在输出完成后向我发送‘\“字符而不是ctrl^d (\x04),但是它不能工作。
为了清晰起见,在输出完成后(通过read()),我希望接收‘byte’字节。因此,如果我发送'pwd‘命令,我将得到以下输出:
$ pwd
/home/user/dir
$|有人能告诉我如何编辑这段代码来达到这个效果吗?
谢谢你的帮助。
编辑-选择()代码
while ( (bytes = select(pipe + 1, &fd_in, NULL, NULL, &tv)) )
{
if (FD_ISSET(pipe, &fd_in))
{
read (pipe...)正如我所描述的,上述代码的行为是在返回之前等待tv的全部值,即使管道上没有任何内容。
发布于 2017-01-09 15:51:52
c_cc[VEOF]的值永远不会发送给您。例如,c_cc[VEOF]的默认值是CTRL-D,但是当用户按CTRL-D时,您将不会在文件末尾看到CTRL-D。相反,您将被通知(例如,通过read()返回0),您将到达文件的末尾。
通常,当子程序完成它的工作时,它将显式地close() pty从端,因此pty主端将得到EOF。
关于
我可以做一个非阻塞的read(),它将在例如5秒之后返回。问题是,如果我执行像"ls“这样的命令(以几毫秒为单位返回),将在返回之前等待整整5秒。
不确定如何编写代码,但可以使用select()或poll(),可以在有可读取的数据时立即通知您。
根据select()的手册页:
select()和pselect()允许程序监视多个文件描述符,等待直到一个或多个文件描述符“就绪”才能执行某些I/O操作(例如,输入可能)。如果可以执行相应的I/O操作(例如,没有阻塞的read(2)或足够小的write(2)),则认为文件描述符已经就绪。
https://stackoverflow.com/questions/41551394
复制相似问题