blob: cfbad74d145145944352c568064cc2f8c0d4c5cb [file] [log] [blame]
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +02001#include "cache.h"
2#include "run-command.h"
Michal Ostrowski77cb17e2006-01-10 21:12:17 -05003#include "exec_cmd.h"
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +02004
Shawn O. Pearce9b0b5092006-12-30 21:55:15 -05005int run_command_v_opt(const char **argv, int flags)
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +02006{
7 pid_t pid = fork();
8
9 if (pid < 0)
10 return -ERR_RUN_COMMAND_FORK;
11 if (!pid) {
Shawn O. Pearce95d3c4f2006-12-30 21:55:22 -050012 if (flags & RUN_COMMAND_NO_STDIN) {
Daniel Barkalow128aed62005-12-07 21:04:38 -050013 int fd = open("/dev/null", O_RDWR);
14 dup2(fd, 0);
Michal Ostrowski77cb17e2006-01-10 21:12:17 -050015 close(fd);
Shawn O. Pearce95d3c4f2006-12-30 21:55:22 -050016 }
17 if (flags & RUN_COMMAND_STDOUT_TO_STDERR)
Shawn O. Pearcecd83c742006-12-30 21:55:19 -050018 dup2(2, 1);
Michal Ostrowski77cb17e2006-01-10 21:12:17 -050019 if (flags & RUN_GIT_CMD) {
20 execv_git_cmd(argv);
21 } else {
22 execvp(argv[0], (char *const*) argv);
23 }
Junio C Hamano19614332005-08-02 14:24:22 -070024 die("exec %s failed.", argv[0]);
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020025 }
26 for (;;) {
27 int status, code;
David Rientjes6f002f92006-08-15 10:40:06 -070028 pid_t waiting = waitpid(pid, &status, 0);
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020029
David Rientjes6f002f92006-08-15 10:40:06 -070030 if (waiting < 0) {
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020031 if (errno == EINTR)
32 continue;
David Rientjes6f002f92006-08-15 10:40:06 -070033 error("waitpid failed (%s)", strerror(errno));
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020034 return -ERR_RUN_COMMAND_WAITPID;
35 }
David Rientjes6f002f92006-08-15 10:40:06 -070036 if (waiting != pid)
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020037 return -ERR_RUN_COMMAND_WAITPID_WRONG_PID;
38 if (WIFSIGNALED(status))
39 return -ERR_RUN_COMMAND_WAITPID_SIGNAL;
40
41 if (!WIFEXITED(status))
42 return -ERR_RUN_COMMAND_WAITPID_NOEXIT;
43 code = WEXITSTATUS(status);
44 if (code)
45 return -code;
46 return 0;
47 }
48}
49
Shawn O. Pearce9b0b5092006-12-30 21:55:15 -050050int run_command_v(const char **argv)
Daniel Barkalow128aed62005-12-07 21:04:38 -050051{
Shawn O. Pearce9b0b5092006-12-30 21:55:15 -050052 return run_command_v_opt(argv, 0);
Daniel Barkalow128aed62005-12-07 21:04:38 -050053}
54
Shawn O. Pearcecd83c742006-12-30 21:55:19 -050055static int run_command_va_opt(int opt, const char *cmd, va_list param)
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020056{
57 int argc;
Junio C Hamano9201c702006-03-05 02:47:29 -080058 const char *argv[MAX_RUN_COMMAND_ARGS];
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020059 const char *arg;
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020060
Junio C Hamano19614332005-08-02 14:24:22 -070061 argv[0] = (char*) cmd;
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020062 argc = 1;
63 while (argc < MAX_RUN_COMMAND_ARGS) {
64 arg = argv[argc++] = va_arg(param, char *);
65 if (!arg)
66 break;
67 }
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020068 if (MAX_RUN_COMMAND_ARGS <= argc)
69 return error("too many args to run %s", cmd);
Shawn O. Pearcecd83c742006-12-30 21:55:19 -050070 return run_command_v_opt(argv, opt);
71}
72
73int run_command_opt(int opt, const char *cmd, ...)
74{
75 va_list params;
76 int r;
77
78 va_start(params, cmd);
79 r = run_command_va_opt(opt, cmd, params);
80 va_end(params);
81 return r;
82}
83
84int run_command(const char *cmd, ...)
85{
86 va_list params;
87 int r;
88
89 va_start(params, cmd);
90 r = run_command_va_opt(0, cmd, params);
91 va_end(params);
92 return r;
Josef Weidendorferb1bf95b2005-07-31 21:17:43 +020093}