【问题标题】:Using termios() to replace EOF?使用 termios() 替换 EOF?
【发布时间】:2017-01-09 15:31:39
【问题描述】:

我有一个在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);

}

当从管道读取时,我可以创建一个非阻塞的 read(),它会在例如 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 的完整值,即使管道上没有任何内容,然后返回。

【问题讨论】:

  • .c_cc[VEOF] 特殊字符仅用于“规范”模式(ICANON 位在.c_lflags 中设置),但cfmakeraw 设置为“原始”模式(ICANON 位未设置)。

标签: c linux terminal


【解决方案1】:

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)),则认为文件描述符已准备就绪。

【讨论】:

  • 这就是我所担心的。我正在通过 select() 使用非阻塞 read()。有没有办法让 read() 返回 0 而无需等待 timeval 中定义的值传递给 read() 文件描述符上的 select() ?如果程序在此之前返回,我想避免等待传递给 select 的 timeval 的完整值。感谢您的帮助。
  • 您可以设置.c_cc[VTIME] 以在“原始”(非规范)模式下设置以十进制为单位的超时。
  • 我在另一个问题上看到 VTIME 对非阻塞套接字没有影响。
  • 关于您对 select()/poll() 的编辑,问题是如果 select() 没有找到可供读取的数据,它将等待 timeval,如果我正在使用错了,如果输入完成,你能告诉我让它返回的正确方法吗?
  • 你可以在网上找到很多select()的例子。粘贴您的代码,以便人们可以帮助了解它为什么不适合您。如有必要,请提出新问题。
猜你喜欢
  • 1970-01-01
  • 2011-02-24
  • 2011-08-15
  • 2016-03-16
  • 2017-07-03
  • 1970-01-01
  • 2021-12-01
  • 2013-05-21
  • 1970-01-01
相关资源
最近更新 更多