微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

solaris:在stdin上设置了O_NDELAY:当进程退出时,shell退出

如何解决solaris:在stdin上设置了O_NDELAY:当进程退出时,shell退出

TLDR:在Solaris 中,如果{em}子进程在O_NDELAY上设置了stdin,则bash退出。为什么?

以下代码导致交互式bash(v4.3.33)或tcsh(6.19.00)shell在进程完成运行后退出

#include <fcntl.h>

int main() {
  fcntl( 0,F_SETFL,O_NDELAY );

//int x = fcntl( 0,F_GETFL );
//fcntl( 0,~(x ^ (~O_NDELAY)) );

  return 0;
}

我们拥有的kshcshzsh的版本不受此问题的影响。

要进行调查,我在bash下运行了cshtruss(类似于Linux上的strace

$ truss -eaf -o bash.txt -u'*' -{v,r,w}all bash --noprofile --norc
$ truss -eaf -o csh.txt -u'*' -{v,w}all csh -f

csh完成运行过程后,将执行以下操作:

fcntl( 0,F_GETFL ) = FWRITE|FNDELAY
fcntl( 0,FWRITE) = 0

...这给了我一个主意。我将程序更改为上面注释掉的代码,因此它将切换O_NDELAY的状态。如果我连续运行两次,bash不会退出

解决方法

This answer使我开始正确的道路。 read的手册页(在Solaris中)说:

When attempting to read a file associated with a terminal that has no data currently available:

* If O_NDELAY is set,read() returns 0
* If O_NONBLOCK is set,read() returns -1 and sets errno to EAGAIN

...因此,当bash尝试读取stdin时,它返回0,导致它假定已击中EOF。

This page指示不应再使用O_NDELAY,而建议使用O_NONBLOCK。对于各种风味的UNIX,我已经找到了关于O_NDELAY / FIONBIO的类似语句。

顺便说一句,在Linux O_NDELAY == FNDELAY == O_NONBLOCK中,因此无法在那种环境下重现这个问题也就不足为奇了。

不幸的是,尽管我通过实验发现了解决该问题的方法,但是执行此操作的工具并不是我的源代码。

如果没有别的什么我可以做一个简单的程序来删除O_NDELAY,然后将该程序的执行包装在一个shell脚本中,该脚本始终在另一个脚本之后运行“ fixer”程序。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。