这几天遇到一个比较奇怪的故障。
一个程序从shell里执行是正常的,可以正常初始化之后转入守护程序然后正常运行。可是如果在守护程序里使用execvp重启动就不能正常工作。
今天仔细研究了一下,才发现了问题所在。
程序转为守护进程有这样一段代码:
for(int i=0;i<3;i++)close(i);
这段代码的目的是关闭标准输出、标准输入和标准错误。然后在守护程序使用execvp重新启动的时候,这几个已经被关闭了,所以再一次转为守护进程的时候关闭的不是标准输出这些,而是在初始化时打开的其它文件,这样就造成了后续程序执行异常。
把代码改成这样:
struct stat st;
for(int i=0;i<3;i++){
fstat(i,&st);
if(S_ISCHR(st.st_mode))close(i);
};
关闭之前先判断一下,看是不是字符设备,如果是再关闭。这下就正常了。当然,字符设备并不一定就是标准输出这些,如果你也有打开字符设备,那可能就要在打开这些文件和转为守护进程之间安排好顺序了。