如何解决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;
}
我们拥有的ksh
,csh
和zsh
的版本不受此问题的影响。
要进行调查,我在bash
下运行了csh
和truss
(类似于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 举报,一经查实,本站将立刻删除。