1. system函数
#include <stdlib.h>
int system(const char *cmd);
如果cmd是一个空指针,则仅仅当命令处理程序可用时,system返回非0值。
因为system在其实现中调用了fork,exec和waitpid,因此有三种返回值:
1). 如果fork失败或者waitpid返回除EINTR之外的出错,则system返回-1,而且errno中设置了错误类型。
2). 如果exec失败(表示不能执行shell),则其返回值如同shell执行了exit(127)一样。
3). 否则所有三个函数(fork,exec和waitpid)都执行成功,并且system的返回值是shell的终止状态,其格式已经在waitpid中说明。
如果一个进程正以特殊的权限(设置用户ID和设置组ID)运行,它又想生成另一个进程执行另一个程序,则它应当直接使用fork和exec,而且在fork之后,exec之前要更改回普通权限。设置用户ID和设置组ID程序绝不应调用system函数,因为特殊权限会在system中执行了fork和exec后扔保持下来。
1.1. system函数原型(没有信号处理)
其中shell的-c选项是告诉shell程序取下一个命令行参数作为命令输入(而不是从标准输入或从一个给定的文件中读命令)。Shell对以null字符终止的命令字符串进行语法解析,将它们分成命令行参数。
如果不使用 shell执行此命令,而是试图由我们自己去执行它,则必须用execlp而不是execl,像shell那样使用PATH变量,必须将null结尾的命令字符串分成各个命令行参数,以便调用execlp。最后,我们也不能使用任何一个shell元字符。
注意,我们调用了_exit而不是exit,这是防止任一标准I/O缓冲区(这些缓冲区由父进程复制到子进程)在子进程中被冲洗。
#include <errno.h> #include <unistd.h> #include <sys/wait.h> int system(const char *command) { pid_t pid; int status; if (NULL == command) { return(1); } if((pid = fork()) < 0) { status = -1; } else if(pid == 0) { execl("/bin/sh", "sh", "-C", command,(char *)0); _exit(127); } else { while(waitpid(pid, &status, 0) < 0) { if(errno == EINTR) { status = -1; break; } } } return status; }
1.2. 完整的system函数原型
#include <sys/types.h> #include <signal.h> #include <stdlib.h> #include <unistd.h> #include <paths.h> #include <sys/wait.h> extern char **environ; int system(const char *command) { pid_t pid; sig_t intsave, quitsave; sigset_t mask, omask; int pstat; char *argp[] = {"sh", "-c", NULL, NULL}; /* just checking... */ if (!command) { return(1); } argp[2] = (char *)command; sigemptyset(&mask); sigaddset(&mask, SIGCHLD); sigprocmask(SIG_BLOCK, &mask, &omask); switch (pid = vfork()) { case -1: /* error */ sigprocmask(SIG_SETMASK, &omask, NULL); return(-1); case 0: /* child */ sigprocmask(SIG_SETMASK, &omask, NULL); execve(_PATH_BSHELL, argp, environ); _exit(127); } intsave = (sig_t) bsd_signal(SIGINT, SIG_IGN); quitsave = (sig_t) bsd_signal(SIGQUIT, SIG_IGN); pid = waitpid(pid, (int *)&pstat, 0); sigprocmask(SIG_SETMASK, &omask, NULL); (void)bsd_signal(SIGINT, intsave); (void)bsd_signal(SIGQUIT, quitsave); return (pid == -1 ? -1 : pstat); }