免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 7587 | 回复: 5
打印 上一主题 下一主题

system()调用返回-1,错误码No child process [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-03-04 17:16 |只看该作者 |倒序浏览
red hat 9.0,C语言,同事的一段代码中调用system() 返回-1,错误码No child process,被调用的程序执行正常。
自己在同样环境下写了一个小程序,发现调用system()正常。

想麻烦各位,指点一下大概是哪方面的问题。查了一些资料没弄明白。
ps:有问题的那段代码,如果在程序开始的地方调用system(NULL),返回是0,而不是nonzero。
至少可以排除了不是被调用程序的问题。我调用类似date这样的命令也是返回-1。

谢谢

论坛徽章:
0
2 [报告]
发表于 2007-03-04 17:56 |只看该作者
strace $prog
ltrace $prog

贴跟踪结果和代码吧,如果方便。

论坛徽章:
0
3 [报告]
发表于 2007-03-05 15:14 |只看该作者
谢谢版主。
strace -p question_pro_pid:

write(3, "2007-3-5 15:00:42.639Command is "..., 83) = 83
rt_sigaction(SIGINT, {SIG_IGN}, {0xd92508, ~[ILL ABRT BUS FPE KILL SEGV STOP XCPU XFSZ SYS RTMIN RT_1], SA_RESTART|SA_SIGINFO}, = 0
rt_sigaction(SIGQUIT, {SIG_IGN}, {SIG_IGN}, = 0
rt_sigprocmask(SIG_BLOCK, [CHLD], [], = 0
clone(child_stack=0, flags=CLONE_PARENT_SETTID|SIGCHLD, parent_tidptr=0xbfb6bf24) = 21307
waitpid(21307, 0xbfb6bf28, 0)           = -1 ECHILD (No child processes)
rt_sigaction(SIGINT, {0xd92508, ~[ILL ABRT BUS FPE KILL SEGV STOP XCPU XFSZ SYS RTMIN RT_1], SA_RESTART|SA_SIGINFO}, NULL, = 0
rt_sigaction(SIGQUIT, {SIG_IGN}, NULL, = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, = 0


/**************************************************************/
strace -p mytest_pro_pid:

rt_sigaction(SIGINT, {SIG_IGN}, {SIG_DFL}, = 0
rt_sigaction(SIGQUIT, {SIG_IGN}, {SIG_DFL}, = 0
rt_sigprocmask(SIG_BLOCK, [CHLD], [], = 0
clone(child_stack=0, flags=CLONE_PARENT_SETTID|SIGCHLD, parent_tidptr=0xbfee1d94) = 21489
waitpid(21489, [{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0) = 21489
rt_sigaction(SIGINT, {SIG_DFL}, NULL, = 0
rt_sigaction(SIGQUIT, {SIG_DFL}, NULL, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
--- SIGCHLD (Child exited) @ 0 (0) ---


开始的是我跟踪问题代码的,后面的是我跟踪正常代码(就是几行代码调用一个system返回成功的)。

谢谢

论坛徽章:
0
4 [报告]
发表于 2007-03-05 16:31 |只看该作者
能不能贴一下源代码。

论坛徽章:
0
5 [报告]
发表于 2007-03-05 18:17 |只看该作者
测试程序是:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
int main(int argc, char * argv[])
{
        char command[60];
        int ret = -1;
        int i = 0;
        sleep(30);
        memset(command, 0x00, sizeof(command));
        strcpy(command, "ls -al");
        ret = system(command);
        if (ret)
        {
                perror("system");
                return -1;
        }
        return 0;
}


问题代码比较复杂,涉及到的调用太多,所以我也不清楚该怎么贴出来。
所以我跑来问问可能的解决方向。
即时我在问题代码的开始处使用system(),也同样出问题。
可能和问题代码被调用有关系,可能是信号量屏蔽之类有问题。

但可以保证的是问题代码在调用system()语法上没有问题。
有朋友告诉我说-1可能是wait/waitpid返回的。
我查了man,里面说到
ECHILD:if the process specified in pid does not exist or is not a child of the calling process.  (This can happen for one's own child if the action for SIGCHLD is set to SIG_IGN.

但是,记得SIGCHLD默认动作就是IGNORE。

不是很理解应该往那边去寻求解决的方式。

目前只有在代码中做了个小处理,判断返回的errno,如果是ECHILD,就不报错。程序按照正常流程继续进行。

论坛徽章:
0
6 [报告]
发表于 2007-03-05 19:08 |只看该作者
我似乎找到了原因,如下,还请各位看看,或者给解释明白些。

原因好像是因为SIGCHLD的默认动作好像不是IGNORE。我写了个测试代码(在最后,a.c)。
gcc a.c
执行a.out DFL正常,执行a.out IGN就会返回-1,错误码解释为“No child processes”。
分别执行 strace -o DFL.log a.out DFL 和 strace -o IGN.log a.out IGN
比较两个文件,
前者为 rt_sigaction(SIGCHLD, {SIG_DFL}, {SIG_DFL}, = 0
后者为 rt_sigaction(SIGCHLD, {SIG_IGN}, {SIG_DFL}, = 0
这似乎是唯一区别。似乎能解释返回-1的原因。
但记得SIGCHLD的默认动作好像就是忽略。
并且,看不懂问题代码的那段。
rt_sigaction(SIGINT, {SIG_IGN}, {0xd92508, ~[ILL ABRT BUS FPE KILL SEGV STOP XCPU XFSZ SYS RTMIN RT_1], SA_RESTART|SA_SIGINFO}, = 0
rt_sigaction(SIGQUIT, {SIG_IGN}, {SIG_IGN}, = 0


/* ************************************************ */
/*                                          test code a.c                                    */
/* ************************************************ */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <signal.h>

int main(int argc, char * argv[])
{
        static struct sigaction act;
        static struct sigaction old_act;
        char command[60];
        int ret = -1;
        int i = 0;
       
        sigemptyset(&act.sa_mask);
        sigaddset(&act.sa_mask, SIGCHLD);
        act.sa_flags = SA_RESTART;
       
        if (argc == 2)
        {
                if (!memcmp(argv[1], "IGN", 3))
                {
                        act.sa_handler = SIG_IGN;
                }
                else if (!memcmp(argv[1], "DFL", 3))
                {
                        act.sa_handler = SIG_DFL;
                }
                else
                {
                        printf("useage: %s IGN  DFL \n", argv[0]);
                        return -1;
                }
        }
        else
        {
                printf("useage: %s IGN | DFL \n", argv[0]);
                return -1;
        }
       
        if (sigaction(SIGCHLD, &act, &old_act) == -1)
        {
                perror("sigaction";
                return -1;
        }
       
        ret = sigismember(&act.sa_mask, SIGCHLD);
        printf("ret = %d\n",ret);
        memset(command, 0x00, sizeof(command));
        strcpy(command, "ls -al";
        ret = system(command);
        printf("ret = %d\n",ret);
        if (ret)
        {
                perror("system";
                return -1;
        }
        return 0;
}
/* ************************************************ */
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP