Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 1 | #include "cache.h" |
| 2 | #include "run-command.h" |
Michal Ostrowski | 77cb17e | 2006-01-10 21:12:17 -0500 | [diff] [blame] | 3 | #include "exec_cmd.h" |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 4 | |
Shawn O. Pearce | 9b0b509 | 2006-12-30 21:55:15 -0500 | [diff] [blame] | 5 | int run_command_v_opt(const char **argv, int flags) |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 6 | { |
| 7 | pid_t pid = fork(); |
| 8 | |
| 9 | if (pid < 0) |
| 10 | return -ERR_RUN_COMMAND_FORK; |
| 11 | if (!pid) { |
Shawn O. Pearce | 95d3c4f | 2006-12-30 21:55:22 -0500 | [diff] [blame] | 12 | if (flags & RUN_COMMAND_NO_STDIN) { |
Daniel Barkalow | 128aed6 | 2005-12-07 21:04:38 -0500 | [diff] [blame] | 13 | int fd = open("/dev/null", O_RDWR); |
| 14 | dup2(fd, 0); |
Michal Ostrowski | 77cb17e | 2006-01-10 21:12:17 -0500 | [diff] [blame] | 15 | close(fd); |
Shawn O. Pearce | 95d3c4f | 2006-12-30 21:55:22 -0500 | [diff] [blame] | 16 | } |
| 17 | if (flags & RUN_COMMAND_STDOUT_TO_STDERR) |
Shawn O. Pearce | cd83c74 | 2006-12-30 21:55:19 -0500 | [diff] [blame] | 18 | dup2(2, 1); |
Michal Ostrowski | 77cb17e | 2006-01-10 21:12:17 -0500 | [diff] [blame] | 19 | if (flags & RUN_GIT_CMD) { |
| 20 | execv_git_cmd(argv); |
| 21 | } else { |
| 22 | execvp(argv[0], (char *const*) argv); |
| 23 | } |
Junio C Hamano | 1961433 | 2005-08-02 14:24:22 -0700 | [diff] [blame] | 24 | die("exec %s failed.", argv[0]); |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 25 | } |
| 26 | for (;;) { |
| 27 | int status, code; |
David Rientjes | 6f002f9 | 2006-08-15 10:40:06 -0700 | [diff] [blame] | 28 | pid_t waiting = waitpid(pid, &status, 0); |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 29 | |
David Rientjes | 6f002f9 | 2006-08-15 10:40:06 -0700 | [diff] [blame] | 30 | if (waiting < 0) { |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 31 | if (errno == EINTR) |
| 32 | continue; |
David Rientjes | 6f002f9 | 2006-08-15 10:40:06 -0700 | [diff] [blame] | 33 | error("waitpid failed (%s)", strerror(errno)); |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 34 | return -ERR_RUN_COMMAND_WAITPID; |
| 35 | } |
David Rientjes | 6f002f9 | 2006-08-15 10:40:06 -0700 | [diff] [blame] | 36 | if (waiting != pid) |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 37 | 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. Pearce | 9b0b509 | 2006-12-30 21:55:15 -0500 | [diff] [blame] | 50 | int run_command_v(const char **argv) |
Daniel Barkalow | 128aed6 | 2005-12-07 21:04:38 -0500 | [diff] [blame] | 51 | { |
Shawn O. Pearce | 9b0b509 | 2006-12-30 21:55:15 -0500 | [diff] [blame] | 52 | return run_command_v_opt(argv, 0); |
Daniel Barkalow | 128aed6 | 2005-12-07 21:04:38 -0500 | [diff] [blame] | 53 | } |
| 54 | |
Shawn O. Pearce | cd83c74 | 2006-12-30 21:55:19 -0500 | [diff] [blame] | 55 | static int run_command_va_opt(int opt, const char *cmd, va_list param) |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 56 | { |
| 57 | int argc; |
Junio C Hamano | 9201c70 | 2006-03-05 02:47:29 -0800 | [diff] [blame] | 58 | const char *argv[MAX_RUN_COMMAND_ARGS]; |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 59 | const char *arg; |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 60 | |
Junio C Hamano | 1961433 | 2005-08-02 14:24:22 -0700 | [diff] [blame] | 61 | argv[0] = (char*) cmd; |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 62 | argc = 1; |
| 63 | while (argc < MAX_RUN_COMMAND_ARGS) { |
| 64 | arg = argv[argc++] = va_arg(param, char *); |
| 65 | if (!arg) |
| 66 | break; |
| 67 | } |
Josef Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 68 | if (MAX_RUN_COMMAND_ARGS <= argc) |
| 69 | return error("too many args to run %s", cmd); |
Shawn O. Pearce | cd83c74 | 2006-12-30 21:55:19 -0500 | [diff] [blame] | 70 | return run_command_v_opt(argv, opt); |
| 71 | } |
| 72 | |
| 73 | int 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 | |
| 84 | int 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 Weidendorfer | b1bf95b | 2005-07-31 21:17:43 +0200 | [diff] [blame] | 93 | } |