Unix C - 便携式WEXITSTATUS

Ish*_*eck 6 c unix openbsd portability freebsd

我正在尝试获取子进程的退出代码.在Linux和FreeBSD上,我可以这样:

[0] [ishpeck@kiyoshi /tmp]$ uname
FreeBSD
[0] [ishpeck@kiyoshi /tmp]$ cat tinker.c 
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    FILE *proc = popen("ls", "r");
    printf("Exit code: %d\n", WEXITSTATUS(pclose(proc)));
    return 0;
}
[0] [ishpeck@kiyoshi /tmp]$ gcc tinker.c -o tinker
[0] [ishpeck@kiyoshi /tmp]$ ./tinker
Exit code: 0
[0] [ishpeck@kiyoshi /tmp]$ grep WEXITSTATUS /usr/include/sys/wait.h 
#define WEXITSTATUS(x)  (_W_INT(x) >> 8)
Run Code Online (Sandbox Code Playgroud)

但是,在OpenBSD上,我收到编译器的投诉......

[0] [ishpeck@ishberk-00 /tmp]$ uname   
OpenBSD
[0] [ishpeck@ishberk-00 /tmp]$ cat tinker.c                                    
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    FILE *proc = popen("ls", "r");
    printf("Exit code: %d\n", WEXITSTATUS(pclose(proc)));
    return 0;
}
[0] [ishpeck@ishberk-00 /tmp]$ gcc tinker.c -o tinker                          
tinker.c: In function 'main':
tinker.c:7: error: lvalue required as unary '&' operand
[1] [ishpeck@ishberk-00 /tmp]$ grep WEXITSTATUS /usr/include/sys/wait.h        
#define WEXITSTATUS(x)  (int)(((unsigned)_W_INT(x) >> 8) & 0xff)
Run Code Online (Sandbox Code Playgroud)

我真的不在乎它是如何完成的,我只需要退出代码.

这让我相信我在Mac上也会遇到这个问题:http: //web.archiveorange.com/archive/v/8XiUWJBLMIKYSCRJnZK5#F4GgyRGRAgSCEG1

是否有更便携的方式来使用WEXITSTATUS宏?还是有更便携的替代方案?

use*_*342 9

OpenBSD的实现在其参数上WEXITSTATUS使用了address-of运算符(一元&),有效地要求其参数具有存储空间.您正在使用函数的返回值调用它,该函数没有存储空间,因此编译器会抱怨.

目前还不清楚OpenBSD是否WEXITSTATUS符合POSIX标准,但通过将返回值pclose()赋给变量可以轻松解决问题:

    int status = pclose(proc);
    printf("Exit code: %d\n", WEXITSTATUS(status));
Run Code Online (Sandbox Code Playgroud)