Junio C Hamano | 8502357 | 2006-12-19 14:34:12 -0800 | [diff] [blame] | 1 | #include "builtin.h" |
Brandon Williams | b2141fc | 2017-06-14 11:07:36 -0700 | [diff] [blame] | 2 | #include "config.h" |
Thiago Farina | fd5c363 | 2010-08-31 23:29:08 -0300 | [diff] [blame] | 3 | #include "exec_cmd.h" |
| 4 | #include "help.h" |
Jeff King | d8e96fd | 2009-01-28 02:38:14 -0500 | [diff] [blame] | 5 | #include "run-command.h" |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 6 | |
Ramsay Allan Jones | 822a7d5 | 2006-07-30 22:42:25 +0100 | [diff] [blame] | 7 | const char git_usage_string[] = |
Nazri Ramliy | 44e1e4d | 2013-09-09 21:47:43 +0800 | [diff] [blame] | 8 | "git [--version] [--help] [-C <path>] [-c name=value]\n" |
Kevin Bracey | 03a0fb0 | 2013-03-11 21:44:15 +0200 | [diff] [blame] | 9 | " [--exec-path[=<path>]] [--html-path] [--man-path] [--info-path]\n" |
Alex Henrie | 9c9b4f2 | 2015-01-13 00:44:47 -0700 | [diff] [blame] | 10 | " [-p | --paginate | --no-pager] [--no-replace-objects] [--bare]\n" |
Josh Triplett | a1bea2c | 2011-07-05 10:54:44 -0700 | [diff] [blame] | 11 | " [--git-dir=<path>] [--work-tree=<path>] [--namespace=<name>]\n" |
Štěpán Němec | 62b4698 | 2010-10-08 19:31:15 +0200 | [diff] [blame] | 12 | " <command> [<args>]"; |
Ramsay Allan Jones | 822a7d5 | 2006-07-30 22:42:25 +0100 | [diff] [blame] | 13 | |
Teemu Likonen | b7d9681 | 2008-06-06 00:15:36 +0300 | [diff] [blame] | 14 | const char git_more_info_string[] = |
Alex Henrie | ad5fe37 | 2014-08-30 13:56:01 -0600 | [diff] [blame] | 15 | N_("'git help -a' and 'git help -g' list available subcommands and some\n" |
Philip Oakley | 73903d0 | 2013-04-02 23:39:48 +0100 | [diff] [blame] | 16 | "concept guides. See 'git help <command>' or 'git help <concept>'\n" |
| 17 | "to read about a specific subcommand or concept."); |
Teemu Likonen | b7d9681 | 2008-06-06 00:15:36 +0300 | [diff] [blame] | 18 | |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 19 | static int use_pager = -1; |
Nguyễn Thái Ngọc Duy | c056261 | 2014-06-08 16:37:10 +0700 | [diff] [blame] | 20 | |
Jeff King | 8893fd9 | 2017-05-30 01:18:43 -0400 | [diff] [blame] | 21 | static void list_builtins(void); |
| 22 | |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 23 | static void commit_pager_choice(void) { |
| 24 | switch (use_pager) { |
| 25 | case 0: |
| 26 | setenv("GIT_PAGER", "cat", 1); |
| 27 | break; |
| 28 | case 1: |
| 29 | setup_pager(); |
| 30 | break; |
| 31 | default: |
| 32 | break; |
| 33 | } |
| 34 | } |
| 35 | |
Martin Ågren | 033fe3d | 2017-08-02 21:40:51 +0200 | [diff] [blame] | 36 | void setup_auto_pager(const char *cmd, int def) |
| 37 | { |
| 38 | if (use_pager != -1 || pager_in_use()) |
| 39 | return; |
| 40 | use_pager = check_pager_config(cmd); |
| 41 | if (use_pager == -1) |
| 42 | use_pager = def; |
| 43 | commit_pager_choice(); |
| 44 | } |
| 45 | |
Felipe Contreras | 4b25d09 | 2009-05-01 12:06:36 +0300 | [diff] [blame] | 46 | static int handle_options(const char ***argv, int *argc, int *envchanged) |
Johannes Schindelin | 4ab243a | 2006-07-24 14:10:45 +0200 | [diff] [blame] | 47 | { |
Junio C Hamano | 73546c0 | 2011-05-24 18:50:35 -0400 | [diff] [blame] | 48 | const char **orig_argv = *argv; |
Johannes Schindelin | 4ab243a | 2006-07-24 14:10:45 +0200 | [diff] [blame] | 49 | |
| 50 | while (*argc > 0) { |
| 51 | const char *cmd = (*argv)[0]; |
| 52 | if (cmd[0] != '-') |
| 53 | break; |
| 54 | |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 55 | /* |
| 56 | * For legacy reasons, the "version" and "help" |
| 57 | * commands can be written with "--" prepended |
| 58 | * to make them look like flags. |
| 59 | */ |
| 60 | if (!strcmp(cmd, "--help") || !strcmp(cmd, "--version")) |
| 61 | break; |
| 62 | |
| 63 | /* |
| 64 | * Check remaining flags. |
| 65 | */ |
Jeff King | ae021d8 | 2014-06-18 15:47:50 -0400 | [diff] [blame] | 66 | if (skip_prefix(cmd, "--exec-path", &cmd)) { |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 67 | if (*cmd == '=') |
Scott R Parish | 384df83 | 2007-10-27 01:36:51 -0700 | [diff] [blame] | 68 | git_set_argv_exec_path(cmd + 1); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 69 | else { |
| 70 | puts(git_exec_path()); |
| 71 | exit(0); |
| 72 | } |
Markus Heidelberg | 89a56bf | 2009-04-05 04:15:16 +0200 | [diff] [blame] | 73 | } else if (!strcmp(cmd, "--html-path")) { |
| 74 | puts(system_path(GIT_HTML_PATH)); |
| 75 | exit(0); |
Jon Seymour | f2dd8c3 | 2011-05-01 18:16:25 +1000 | [diff] [blame] | 76 | } else if (!strcmp(cmd, "--man-path")) { |
| 77 | puts(system_path(GIT_MAN_PATH)); |
| 78 | exit(0); |
| 79 | } else if (!strcmp(cmd, "--info-path")) { |
| 80 | puts(system_path(GIT_INFO_PATH)); |
| 81 | exit(0); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 82 | } else if (!strcmp(cmd, "-p") || !strcmp(cmd, "--paginate")) { |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 83 | use_pager = 1; |
Matthieu Moy | 463a849 | 2007-08-19 19:24:36 +0200 | [diff] [blame] | 84 | } else if (!strcmp(cmd, "--no-pager")) { |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 85 | use_pager = 0; |
Matthieu Moy | 463a849 | 2007-08-19 19:24:36 +0200 | [diff] [blame] | 86 | if (envchanged) |
| 87 | *envchanged = 1; |
Christian Couder | b0fa7ab | 2009-10-12 22:30:32 +0200 | [diff] [blame] | 88 | } else if (!strcmp(cmd, "--no-replace-objects")) { |
Michael Haggerty | afc711b | 2014-02-18 12:24:55 +0100 | [diff] [blame] | 89 | check_replace_refs = 0; |
Christian Couder | 6476b38 | 2009-11-18 07:50:58 +0100 | [diff] [blame] | 90 | setenv(NO_REPLACE_OBJECTS_ENVIRONMENT, "1", 1); |
| 91 | if (envchanged) |
| 92 | *envchanged = 1; |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 93 | } else if (!strcmp(cmd, "--git-dir")) { |
Brian Gernhardt | c321f00 | 2006-12-22 08:56:25 -0500 | [diff] [blame] | 94 | if (*argc < 2) { |
| 95 | fprintf(stderr, "No directory given for --git-dir.\n" ); |
| 96 | usage(git_usage_string); |
| 97 | } |
Shawn O. Pearce | 45b0979 | 2006-12-30 23:29:11 -0500 | [diff] [blame] | 98 | setenv(GIT_DIR_ENVIRONMENT, (*argv)[1], 1); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 99 | if (envchanged) |
| 100 | *envchanged = 1; |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 101 | (*argv)++; |
| 102 | (*argc)--; |
Jeff King | ae021d8 | 2014-06-18 15:47:50 -0400 | [diff] [blame] | 103 | } else if (skip_prefix(cmd, "--git-dir=", &cmd)) { |
| 104 | setenv(GIT_DIR_ENVIRONMENT, cmd, 1); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 105 | if (envchanged) |
| 106 | *envchanged = 1; |
Josh Triplett | a1bea2c | 2011-07-05 10:54:44 -0700 | [diff] [blame] | 107 | } else if (!strcmp(cmd, "--namespace")) { |
| 108 | if (*argc < 2) { |
| 109 | fprintf(stderr, "No namespace given for --namespace.\n" ); |
| 110 | usage(git_usage_string); |
| 111 | } |
| 112 | setenv(GIT_NAMESPACE_ENVIRONMENT, (*argv)[1], 1); |
| 113 | if (envchanged) |
| 114 | *envchanged = 1; |
| 115 | (*argv)++; |
| 116 | (*argc)--; |
Jeff King | ae021d8 | 2014-06-18 15:47:50 -0400 | [diff] [blame] | 117 | } else if (skip_prefix(cmd, "--namespace=", &cmd)) { |
| 118 | setenv(GIT_NAMESPACE_ENVIRONMENT, cmd, 1); |
Josh Triplett | a1bea2c | 2011-07-05 10:54:44 -0700 | [diff] [blame] | 119 | if (envchanged) |
| 120 | *envchanged = 1; |
Matthias Lederhofer | 892c41b | 2007-06-06 09:10:42 +0200 | [diff] [blame] | 121 | } else if (!strcmp(cmd, "--work-tree")) { |
| 122 | if (*argc < 2) { |
| 123 | fprintf(stderr, "No directory given for --work-tree.\n" ); |
| 124 | usage(git_usage_string); |
| 125 | } |
| 126 | setenv(GIT_WORK_TREE_ENVIRONMENT, (*argv)[1], 1); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 127 | if (envchanged) |
| 128 | *envchanged = 1; |
Matthias Lederhofer | 892c41b | 2007-06-06 09:10:42 +0200 | [diff] [blame] | 129 | (*argv)++; |
| 130 | (*argc)--; |
Jeff King | ae021d8 | 2014-06-18 15:47:50 -0400 | [diff] [blame] | 131 | } else if (skip_prefix(cmd, "--work-tree=", &cmd)) { |
| 132 | setenv(GIT_WORK_TREE_ENVIRONMENT, cmd, 1); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 133 | if (envchanged) |
| 134 | *envchanged = 1; |
Brandon Williams | 74866d7 | 2016-10-07 11:18:48 -0700 | [diff] [blame] | 135 | } else if (!strcmp(cmd, "--super-prefix")) { |
| 136 | if (*argc < 2) { |
| 137 | fprintf(stderr, "No prefix given for --super-prefix.\n" ); |
| 138 | usage(git_usage_string); |
| 139 | } |
| 140 | setenv(GIT_SUPER_PREFIX_ENVIRONMENT, (*argv)[1], 1); |
| 141 | if (envchanged) |
| 142 | *envchanged = 1; |
| 143 | (*argv)++; |
| 144 | (*argc)--; |
| 145 | } else if (skip_prefix(cmd, "--super-prefix=", &cmd)) { |
| 146 | setenv(GIT_SUPER_PREFIX_ENVIRONMENT, cmd, 1); |
| 147 | if (envchanged) |
| 148 | *envchanged = 1; |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 149 | } else if (!strcmp(cmd, "--bare")) { |
René Scharfe | 4d3ab44 | 2014-07-28 20:31:57 +0200 | [diff] [blame] | 150 | char *cwd = xgetcwd(); |
Junio C Hamano | 6adcca3 | 2007-08-27 00:58:06 -0700 | [diff] [blame] | 151 | is_bare_repository_cfg = 1; |
René Scharfe | 4d3ab44 | 2014-07-28 20:31:57 +0200 | [diff] [blame] | 152 | setenv(GIT_DIR_ENVIRONMENT, cwd, 0); |
| 153 | free(cwd); |
Jeff King | 2cd83d1 | 2013-03-08 04:32:22 -0500 | [diff] [blame] | 154 | setenv(GIT_IMPLICIT_WORK_TREE_ENVIRONMENT, "0", 1); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 155 | if (envchanged) |
| 156 | *envchanged = 1; |
Alex Riesen | 8b1fa77 | 2010-03-26 23:53:57 +0100 | [diff] [blame] | 157 | } else if (!strcmp(cmd, "-c")) { |
| 158 | if (*argc < 2) { |
| 159 | fprintf(stderr, "-c expects a configuration string\n" ); |
| 160 | usage(git_usage_string); |
| 161 | } |
Jeff King | 2b64fc8 | 2010-08-23 15:16:00 -0400 | [diff] [blame] | 162 | git_config_push_parameter((*argv)[1]); |
Alex Riesen | 8b1fa77 | 2010-03-26 23:53:57 +0100 | [diff] [blame] | 163 | (*argv)++; |
| 164 | (*argc)--; |
Jeff King | 823ab40 | 2012-12-19 17:37:30 -0500 | [diff] [blame] | 165 | } else if (!strcmp(cmd, "--literal-pathspecs")) { |
| 166 | setenv(GIT_LITERAL_PATHSPECS_ENVIRONMENT, "1", 1); |
| 167 | if (envchanged) |
| 168 | *envchanged = 1; |
| 169 | } else if (!strcmp(cmd, "--no-literal-pathspecs")) { |
| 170 | setenv(GIT_LITERAL_PATHSPECS_ENVIRONMENT, "0", 1); |
| 171 | if (envchanged) |
| 172 | *envchanged = 1; |
Nguyễn Thái Ngọc Duy | bd30c2e | 2013-07-14 15:36:08 +0700 | [diff] [blame] | 173 | } else if (!strcmp(cmd, "--glob-pathspecs")) { |
| 174 | setenv(GIT_GLOB_PATHSPECS_ENVIRONMENT, "1", 1); |
| 175 | if (envchanged) |
| 176 | *envchanged = 1; |
| 177 | } else if (!strcmp(cmd, "--noglob-pathspecs")) { |
| 178 | setenv(GIT_NOGLOB_PATHSPECS_ENVIRONMENT, "1", 1); |
| 179 | if (envchanged) |
| 180 | *envchanged = 1; |
Nguyễn Thái Ngọc Duy | 93d9353 | 2013-07-14 15:36:09 +0700 | [diff] [blame] | 181 | } else if (!strcmp(cmd, "--icase-pathspecs")) { |
| 182 | setenv(GIT_ICASE_PATHSPECS_ENVIRONMENT, "1", 1); |
| 183 | if (envchanged) |
| 184 | *envchanged = 1; |
Jeff King | 27344d6 | 2017-09-27 02:54:30 -0400 | [diff] [blame] | 185 | } else if (!strcmp(cmd, "--no-optional-locks")) { |
| 186 | setenv(GIT_OPTIONAL_LOCKS_ENVIRONMENT, "0", 1); |
| 187 | if (envchanged) |
| 188 | *envchanged = 1; |
Nguyễn Thái Ngọc Duy | 6035d6a | 2013-05-26 08:16:15 +0700 | [diff] [blame] | 189 | } else if (!strcmp(cmd, "--shallow-file")) { |
| 190 | (*argv)++; |
| 191 | (*argc)--; |
Nguyễn Thái Ngọc Duy | 069c053 | 2013-12-05 20:02:45 +0700 | [diff] [blame] | 192 | set_alternate_shallow_file((*argv)[0], 1); |
Nguyễn Thái Ngọc Duy | 6035d6a | 2013-05-26 08:16:15 +0700 | [diff] [blame] | 193 | if (envchanged) |
| 194 | *envchanged = 1; |
Nazri Ramliy | 44e1e4d | 2013-09-09 21:47:43 +0800 | [diff] [blame] | 195 | } else if (!strcmp(cmd, "-C")) { |
| 196 | if (*argc < 2) { |
| 197 | fprintf(stderr, "No directory given for -C.\n" ); |
| 198 | usage(git_usage_string); |
| 199 | } |
Karthik Nayak | 6a536e2 | 2015-03-06 16:48:08 +0530 | [diff] [blame] | 200 | if ((*argv)[1][0]) { |
| 201 | if (chdir((*argv)[1])) |
| 202 | die_errno("Cannot change to '%s'", (*argv)[1]); |
| 203 | if (envchanged) |
| 204 | *envchanged = 1; |
| 205 | } |
Nazri Ramliy | 44e1e4d | 2013-09-09 21:47:43 +0800 | [diff] [blame] | 206 | (*argv)++; |
| 207 | (*argc)--; |
Jeff King | 8893fd9 | 2017-05-30 01:18:43 -0400 | [diff] [blame] | 208 | } else if (!strcmp(cmd, "--list-builtins")) { |
| 209 | list_builtins(); |
| 210 | exit(0); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 211 | } else { |
| 212 | fprintf(stderr, "Unknown option: %s\n", cmd); |
Ramsay Allan Jones | 822a7d5 | 2006-07-30 22:42:25 +0100 | [diff] [blame] | 213 | usage(git_usage_string); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 214 | } |
Johannes Schindelin | 4ab243a | 2006-07-24 14:10:45 +0200 | [diff] [blame] | 215 | |
| 216 | (*argv)++; |
| 217 | (*argc)--; |
Johannes Schindelin | 4ab243a | 2006-07-24 14:10:45 +0200 | [diff] [blame] | 218 | } |
Junio C Hamano | 73546c0 | 2011-05-24 18:50:35 -0400 | [diff] [blame] | 219 | return (*argv) - orig_argv; |
Johannes Schindelin | 4ab243a | 2006-07-24 14:10:45 +0200 | [diff] [blame] | 220 | } |
| 221 | |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 222 | static int handle_alias(int *argcp, const char ***argv) |
| 223 | { |
SZEDER Gábor | af05d67 | 2008-03-25 22:06:26 +0100 | [diff] [blame] | 224 | int envchanged = 0, ret = 0, saved_errno = errno; |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 225 | int count, option_count; |
Felipe Contreras | 4b25d09 | 2009-05-01 12:06:36 +0300 | [diff] [blame] | 226 | const char **new_argv; |
Jeff King | 9435111 | 2008-02-24 17:17:14 -0500 | [diff] [blame] | 227 | const char *alias_command; |
| 228 | char *alias_string; |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 229 | |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 230 | alias_command = (*argv)[0]; |
Jeff King | 9435111 | 2008-02-24 17:17:14 -0500 | [diff] [blame] | 231 | alias_string = alias_lookup(alias_command); |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 232 | if (alias_string) { |
Theodore Ts'o | dfd42a3 | 2007-02-10 19:33:58 -0500 | [diff] [blame] | 233 | if (alias_string[0] == '!') { |
Jeff King | 850d2fe | 2016-02-22 17:44:21 -0500 | [diff] [blame] | 234 | struct child_process child = CHILD_PROCESS_INIT; |
Johannes Schindelin | a9bcf65 | 2017-06-14 13:36:00 +0200 | [diff] [blame] | 235 | int nongit_ok; |
| 236 | |
| 237 | /* Aliases expect GIT_PREFIX, GIT_DIR etc to be set */ |
| 238 | setup_git_directory_gently(&nongit_ok); |
Johannes Schindelin | a2f8028 | 2007-07-01 22:51:58 +0100 | [diff] [blame] | 239 | |
Erik Faye-Lund | 7f51f8b | 2011-01-07 00:00:38 +0100 | [diff] [blame] | 240 | commit_pager_choice(); |
| 241 | |
Jeff King | 850d2fe | 2016-02-22 17:44:21 -0500 | [diff] [blame] | 242 | child.use_shell = 1; |
| 243 | argv_array_push(&child.args, alias_string + 1); |
| 244 | argv_array_pushv(&child.args, (*argv) + 1); |
Erik Faye-Lund | 7f51f8b | 2011-01-07 00:00:38 +0100 | [diff] [blame] | 245 | |
Jeff King | 850d2fe | 2016-02-22 17:44:21 -0500 | [diff] [blame] | 246 | ret = run_command(&child); |
Erik Faye-Lund | 7f51f8b | 2011-01-07 00:00:38 +0100 | [diff] [blame] | 247 | if (ret >= 0) /* normal exit */ |
| 248 | exit(ret); |
| 249 | |
| 250 | die_errno("While expanding alias '%s': '%s'", |
| 251 | alias_command, alias_string + 1); |
Theodore Ts'o | dfd42a3 | 2007-02-10 19:33:58 -0500 | [diff] [blame] | 252 | } |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 253 | count = split_cmdline(alias_string, &new_argv); |
Deskin Miller | dc4179f | 2008-09-22 11:06:41 -0400 | [diff] [blame] | 254 | if (count < 0) |
Greg Brockman | ad9ac6d | 2010-08-07 01:13:39 -0400 | [diff] [blame] | 255 | die("Bad alias.%s string: %s", alias_command, |
| 256 | split_cmdline_strerror(count)); |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 257 | option_count = handle_options(&new_argv, &count, &envchanged); |
| 258 | if (envchanged) |
| 259 | die("alias '%s' changes environment variables\n" |
| 260 | "You can use '!git' in the alias to do this.", |
| 261 | alias_command); |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 262 | memmove(new_argv - option_count, new_argv, |
| 263 | count * sizeof(char *)); |
| 264 | new_argv -= option_count; |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 265 | |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 266 | if (count < 1) |
| 267 | die("empty alias for %s", alias_command); |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 268 | |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 269 | if (!strcmp(alias_command, new_argv[0])) |
| 270 | die("recursive alias: %s", alias_command); |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 271 | |
Christian Couder | b319ce4 | 2007-12-03 05:51:50 +0100 | [diff] [blame] | 272 | trace_argv_printf(new_argv, |
Christian Couder | 6ce4e61 | 2006-09-02 18:23:48 +0200 | [diff] [blame] | 273 | "trace: alias expansion: %s =>", |
| 274 | alias_command); |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 275 | |
René Scharfe | 2756ca4 | 2014-09-16 20:56:57 +0200 | [diff] [blame] | 276 | REALLOC_ARRAY(new_argv, count + *argcp); |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 277 | /* insert after command name */ |
Felipe Contreras | 4b25d09 | 2009-05-01 12:06:36 +0300 | [diff] [blame] | 278 | memcpy(new_argv + count, *argv + 1, sizeof(char *) * *argcp); |
Matthias Lederhofer | 0347a8c | 2006-07-30 03:30:29 +0200 | [diff] [blame] | 279 | |
| 280 | *argv = new_argv; |
| 281 | *argcp += count - 1; |
| 282 | |
| 283 | ret = 1; |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 284 | } |
| 285 | |
Johannes Schindelin | 47e5c0c | 2006-06-28 12:45:27 +0200 | [diff] [blame] | 286 | errno = saved_errno; |
| 287 | |
Johannes Schindelin | 2b11e31 | 2006-06-05 19:43:52 +0200 | [diff] [blame] | 288 | return ret; |
| 289 | } |
| 290 | |
Nguyễn Thái Ngọc Duy | ee38dfb | 2010-08-05 21:52:16 -0500 | [diff] [blame] | 291 | #define RUN_SETUP (1<<0) |
| 292 | #define RUN_SETUP_GENTLY (1<<1) |
| 293 | #define USE_PAGER (1<<2) |
Shawn O. Pearce | 7eff28a | 2006-12-30 23:32:38 -0500 | [diff] [blame] | 294 | /* |
| 295 | * require working tree to be present -- anything uses this needs |
| 296 | * RUN_SETUP for reading from the configuration file. |
| 297 | */ |
Nguyễn Thái Ngọc Duy | ee38dfb | 2010-08-05 21:52:16 -0500 | [diff] [blame] | 298 | #define NEED_WORK_TREE (1<<3) |
Brandon Williams | 74866d7 | 2016-10-07 11:18:48 -0700 | [diff] [blame] | 299 | #define SUPPORT_SUPER_PREFIX (1<<4) |
Martin Ågren | c409824 | 2017-08-02 21:40:50 +0200 | [diff] [blame] | 300 | #define DELAY_PAGER_CONFIG (1<<5) |
Linus Torvalds | a633fca | 2006-07-28 22:44:25 -0700 | [diff] [blame] | 301 | |
Linus Torvalds | 47d0b4f | 2007-06-24 10:10:40 -0700 | [diff] [blame] | 302 | struct cmd_struct { |
| 303 | const char *cmd; |
| 304 | int (*fn)(int, const char **, const char *); |
| 305 | int option; |
| 306 | }; |
| 307 | |
Jeff King | f172f33 | 2009-01-28 02:33:53 -0500 | [diff] [blame] | 308 | static int run_builtin(struct cmd_struct *p, int argc, const char **argv) |
Linus Torvalds | 47d0b4f | 2007-06-24 10:10:40 -0700 | [diff] [blame] | 309 | { |
Jonathan Nieder | 99caeed | 2009-11-09 09:05:01 -0600 | [diff] [blame] | 310 | int status, help; |
Linus Torvalds | 0f15731 | 2007-06-24 10:29:33 -0700 | [diff] [blame] | 311 | struct stat st; |
Linus Torvalds | 47d0b4f | 2007-06-24 10:10:40 -0700 | [diff] [blame] | 312 | const char *prefix; |
| 313 | |
| 314 | prefix = NULL; |
Jonathan Nieder | 99caeed | 2009-11-09 09:05:01 -0600 | [diff] [blame] | 315 | help = argc == 2 && !strcmp(argv[1], "-h"); |
| 316 | if (!help) { |
| 317 | if (p->option & RUN_SETUP) |
| 318 | prefix = setup_git_directory(); |
Luis R. Rodriguez | 27bd38d | 2014-04-21 17:47:56 -0700 | [diff] [blame] | 319 | else if (p->option & RUN_SETUP_GENTLY) { |
Nguyễn Thái Ngọc Duy | ee38dfb | 2010-08-05 21:52:16 -0500 | [diff] [blame] | 320 | int nongit_ok; |
| 321 | prefix = setup_git_directory_gently(&nongit_ok); |
| 322 | } |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 323 | |
Martin Ågren | c409824 | 2017-08-02 21:40:50 +0200 | [diff] [blame] | 324 | if (use_pager == -1 && p->option & (RUN_SETUP | RUN_SETUP_GENTLY) && |
| 325 | !(p->option & DELAY_PAGER_CONFIG)) |
Jonathan Nieder | 99caeed | 2009-11-09 09:05:01 -0600 | [diff] [blame] | 326 | use_pager = check_pager_config(p->cmd); |
| 327 | if (use_pager == -1 && p->option & USE_PAGER) |
| 328 | use_pager = 1; |
Nguyễn Thái Ngọc Duy | a9ca8a8 | 2010-11-26 22:31:57 +0700 | [diff] [blame] | 329 | |
| 330 | if ((p->option & (RUN_SETUP | RUN_SETUP_GENTLY)) && |
| 331 | startup_info->have_repository) /* get_git_dir() may set up repo, avoid that */ |
| 332 | trace_repo_setup(prefix); |
Jonathan Nieder | 99caeed | 2009-11-09 09:05:01 -0600 | [diff] [blame] | 333 | } |
Jeff King | 4e10738 | 2008-07-03 07:46:57 -0400 | [diff] [blame] | 334 | commit_pager_choice(); |
| 335 | |
Brandon Williams | 74866d7 | 2016-10-07 11:18:48 -0700 | [diff] [blame] | 336 | if (!help && get_super_prefix()) { |
| 337 | if (!(p->option & SUPPORT_SUPER_PREFIX)) |
| 338 | die("%s doesn't support --super-prefix", p->cmd); |
Brandon Williams | 74866d7 | 2016-10-07 11:18:48 -0700 | [diff] [blame] | 339 | } |
| 340 | |
Jonathan Nieder | 99caeed | 2009-11-09 09:05:01 -0600 | [diff] [blame] | 341 | if (!help && p->option & NEED_WORK_TREE) |
Mike Hommey | 59f0f2f | 2007-11-03 12:23:11 +0100 | [diff] [blame] | 342 | setup_work_tree(); |
| 343 | |
Christian Couder | b319ce4 | 2007-12-03 05:51:50 +0100 | [diff] [blame] | 344 | trace_argv_printf(argv, "trace: built-in: git"); |
Linus Torvalds | 47d0b4f | 2007-06-24 10:10:40 -0700 | [diff] [blame] | 345 | |
Linus Torvalds | 0f15731 | 2007-06-24 10:29:33 -0700 | [diff] [blame] | 346 | status = p->fn(argc, argv, prefix); |
| 347 | if (status) |
Johannes Sixt | 47e3de0 | 2009-07-05 20:57:46 +0200 | [diff] [blame] | 348 | return status; |
Linus Torvalds | 0f15731 | 2007-06-24 10:29:33 -0700 | [diff] [blame] | 349 | |
| 350 | /* Somebody closed stdout? */ |
| 351 | if (fstat(fileno(stdout), &st)) |
| 352 | return 0; |
| 353 | /* Ignore write errors for pipes and sockets.. */ |
| 354 | if (S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode)) |
| 355 | return 0; |
| 356 | |
| 357 | /* Check for ENOSPC and EIO errors.. */ |
Linus Torvalds | 0227f98 | 2007-06-30 11:44:20 -0700 | [diff] [blame] | 358 | if (fflush(stdout)) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 359 | die_errno("write failure on standard output"); |
Linus Torvalds | 0227f98 | 2007-06-30 11:44:20 -0700 | [diff] [blame] | 360 | if (ferror(stdout)) |
| 361 | die("unknown write failure on standard output"); |
| 362 | if (fclose(stdout)) |
Thomas Rast | d824cbb | 2009-06-27 17:58:46 +0200 | [diff] [blame] | 363 | die_errno("close failed on standard output"); |
Linus Torvalds | 0f15731 | 2007-06-24 10:29:33 -0700 | [diff] [blame] | 364 | return 0; |
Linus Torvalds | 47d0b4f | 2007-06-24 10:10:40 -0700 | [diff] [blame] | 365 | } |
| 366 | |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 367 | static struct cmd_struct commands[] = { |
| 368 | { "add", cmd_add, RUN_SETUP | NEED_WORK_TREE }, |
Paul Tan | 783d7e8 | 2015-08-04 21:52:06 +0800 | [diff] [blame] | 369 | { "am", cmd_am, RUN_SETUP | NEED_WORK_TREE }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 370 | { "annotate", cmd_annotate, RUN_SETUP }, |
| 371 | { "apply", cmd_apply, RUN_SETUP_GENTLY }, |
Junio C Hamano | eb0224c | 2016-11-22 13:37:04 -0800 | [diff] [blame] | 372 | { "archive", cmd_archive, RUN_SETUP_GENTLY }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 373 | { "bisect--helper", cmd_bisect__helper, RUN_SETUP }, |
| 374 | { "blame", cmd_blame, RUN_SETUP }, |
Martin Ågren | d74b541 | 2017-11-19 16:03:49 +0100 | [diff] [blame] | 375 | { "branch", cmd_branch, RUN_SETUP | DELAY_PAGER_CONFIG }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 376 | { "bundle", cmd_bundle, RUN_SETUP_GENTLY }, |
| 377 | { "cat-file", cmd_cat_file, RUN_SETUP }, |
| 378 | { "check-attr", cmd_check_attr, RUN_SETUP }, |
| 379 | { "check-ignore", cmd_check_ignore, RUN_SETUP | NEED_WORK_TREE }, |
| 380 | { "check-mailmap", cmd_check_mailmap, RUN_SETUP }, |
| 381 | { "check-ref-format", cmd_check_ref_format }, |
Eric Sunshine | 0ca560c | 2015-07-06 13:30:56 -0400 | [diff] [blame] | 382 | { "checkout", cmd_checkout, RUN_SETUP | NEED_WORK_TREE }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 383 | { "checkout-index", cmd_checkout_index, |
| 384 | RUN_SETUP | NEED_WORK_TREE}, |
| 385 | { "cherry", cmd_cherry, RUN_SETUP }, |
| 386 | { "cherry-pick", cmd_cherry_pick, RUN_SETUP | NEED_WORK_TREE }, |
| 387 | { "clean", cmd_clean, RUN_SETUP | NEED_WORK_TREE }, |
Nguyễn Thái Ngọc Duy | 86d26f2 | 2015-12-20 14:50:18 +0700 | [diff] [blame] | 388 | { "clone", cmd_clone }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 389 | { "column", cmd_column, RUN_SETUP_GENTLY }, |
| 390 | { "commit", cmd_commit, RUN_SETUP | NEED_WORK_TREE }, |
| 391 | { "commit-tree", cmd_commit_tree, RUN_SETUP }, |
| 392 | { "config", cmd_config, RUN_SETUP_GENTLY }, |
| 393 | { "count-objects", cmd_count_objects, RUN_SETUP }, |
| 394 | { "credential", cmd_credential, RUN_SETUP_GENTLY }, |
| 395 | { "describe", cmd_describe, RUN_SETUP }, |
| 396 | { "diff", cmd_diff }, |
| 397 | { "diff-files", cmd_diff_files, RUN_SETUP | NEED_WORK_TREE }, |
| 398 | { "diff-index", cmd_diff_index, RUN_SETUP }, |
| 399 | { "diff-tree", cmd_diff_tree, RUN_SETUP }, |
Johannes Schindelin | 019678d | 2017-01-19 21:30:40 +0100 | [diff] [blame] | 400 | { "difftool", cmd_difftool, RUN_SETUP | NEED_WORK_TREE }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 401 | { "fast-export", cmd_fast_export, RUN_SETUP }, |
| 402 | { "fetch", cmd_fetch, RUN_SETUP }, |
| 403 | { "fetch-pack", cmd_fetch_pack, RUN_SETUP }, |
| 404 | { "fmt-merge-msg", cmd_fmt_merge_msg, RUN_SETUP }, |
| 405 | { "for-each-ref", cmd_for_each_ref, RUN_SETUP }, |
| 406 | { "format-patch", cmd_format_patch, RUN_SETUP }, |
| 407 | { "fsck", cmd_fsck, RUN_SETUP }, |
| 408 | { "fsck-objects", cmd_fsck, RUN_SETUP }, |
| 409 | { "gc", cmd_gc, RUN_SETUP }, |
| 410 | { "get-tar-commit-id", cmd_get_tar_commit_id }, |
Brandon Williams | f9ee2fc | 2017-08-02 12:49:23 -0700 | [diff] [blame] | 411 | { "grep", cmd_grep, RUN_SETUP_GENTLY }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 412 | { "hash-object", cmd_hash_object }, |
| 413 | { "help", cmd_help }, |
| 414 | { "index-pack", cmd_index_pack, RUN_SETUP_GENTLY }, |
Nguyễn Thái Ngọc Duy | 86d26f2 | 2015-12-20 14:50:18 +0700 | [diff] [blame] | 415 | { "init", cmd_init_db }, |
| 416 | { "init-db", cmd_init_db }, |
John Keeping | cbd9fc2 | 2015-09-05 14:39:24 +0100 | [diff] [blame] | 417 | { "interpret-trailers", cmd_interpret_trailers, RUN_SETUP_GENTLY }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 418 | { "log", cmd_log, RUN_SETUP }, |
Brandon Williams | 188dce1 | 2017-06-22 11:43:48 -0700 | [diff] [blame] | 419 | { "ls-files", cmd_ls_files, RUN_SETUP }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 420 | { "ls-remote", cmd_ls_remote, RUN_SETUP_GENTLY }, |
| 421 | { "ls-tree", cmd_ls_tree, RUN_SETUP }, |
Junio C Hamano | 3f0ec06 | 2016-11-22 13:13:16 -0800 | [diff] [blame] | 422 | { "mailinfo", cmd_mailinfo, RUN_SETUP_GENTLY }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 423 | { "mailsplit", cmd_mailsplit }, |
| 424 | { "merge", cmd_merge, RUN_SETUP | NEED_WORK_TREE }, |
| 425 | { "merge-base", cmd_merge_base, RUN_SETUP }, |
| 426 | { "merge-file", cmd_merge_file, RUN_SETUP_GENTLY }, |
| 427 | { "merge-index", cmd_merge_index, RUN_SETUP }, |
| 428 | { "merge-ours", cmd_merge_ours, RUN_SETUP }, |
| 429 | { "merge-recursive", cmd_merge_recursive, RUN_SETUP | NEED_WORK_TREE }, |
| 430 | { "merge-recursive-ours", cmd_merge_recursive, RUN_SETUP | NEED_WORK_TREE }, |
| 431 | { "merge-recursive-theirs", cmd_merge_recursive, RUN_SETUP | NEED_WORK_TREE }, |
| 432 | { "merge-subtree", cmd_merge_recursive, RUN_SETUP | NEED_WORK_TREE }, |
| 433 | { "merge-tree", cmd_merge_tree, RUN_SETUP }, |
| 434 | { "mktag", cmd_mktag, RUN_SETUP }, |
| 435 | { "mktree", cmd_mktree, RUN_SETUP }, |
| 436 | { "mv", cmd_mv, RUN_SETUP | NEED_WORK_TREE }, |
| 437 | { "name-rev", cmd_name_rev, RUN_SETUP }, |
| 438 | { "notes", cmd_notes, RUN_SETUP }, |
| 439 | { "pack-objects", cmd_pack_objects, RUN_SETUP }, |
| 440 | { "pack-redundant", cmd_pack_redundant, RUN_SETUP }, |
| 441 | { "pack-refs", cmd_pack_refs, RUN_SETUP }, |
Jeff King | 4a73aaa | 2016-09-12 20:23:22 -0700 | [diff] [blame] | 442 | { "patch-id", cmd_patch_id, RUN_SETUP_GENTLY }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 443 | { "pickaxe", cmd_blame, RUN_SETUP }, |
| 444 | { "prune", cmd_prune, RUN_SETUP }, |
| 445 | { "prune-packed", cmd_prune_packed, RUN_SETUP }, |
Paul Tan | 1e1ea69 | 2015-06-14 16:41:51 +0800 | [diff] [blame] | 446 | { "pull", cmd_pull, RUN_SETUP | NEED_WORK_TREE }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 447 | { "push", cmd_push, RUN_SETUP }, |
Stefan Beller | 3d41542 | 2017-01-17 17:05:20 -0800 | [diff] [blame] | 448 | { "read-tree", cmd_read_tree, RUN_SETUP | SUPPORT_SUPER_PREFIX}, |
Johannes Schindelin | 4557f1a | 2017-02-09 23:23:06 +0100 | [diff] [blame] | 449 | { "rebase--helper", cmd_rebase__helper, RUN_SETUP | NEED_WORK_TREE }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 450 | { "receive-pack", cmd_receive_pack }, |
| 451 | { "reflog", cmd_reflog, RUN_SETUP }, |
| 452 | { "remote", cmd_remote, RUN_SETUP }, |
| 453 | { "remote-ext", cmd_remote_ext }, |
| 454 | { "remote-fd", cmd_remote_fd }, |
| 455 | { "repack", cmd_repack, RUN_SETUP }, |
| 456 | { "replace", cmd_replace, RUN_SETUP }, |
| 457 | { "rerere", cmd_rerere, RUN_SETUP }, |
| 458 | { "reset", cmd_reset, RUN_SETUP }, |
| 459 | { "rev-list", cmd_rev_list, RUN_SETUP }, |
| 460 | { "rev-parse", cmd_rev_parse }, |
| 461 | { "revert", cmd_revert, RUN_SETUP | NEED_WORK_TREE }, |
| 462 | { "rm", cmd_rm, RUN_SETUP }, |
| 463 | { "send-pack", cmd_send_pack, RUN_SETUP }, |
| 464 | { "shortlog", cmd_shortlog, RUN_SETUP_GENTLY | USE_PAGER }, |
| 465 | { "show", cmd_show, RUN_SETUP }, |
| 466 | { "show-branch", cmd_show_branch, RUN_SETUP }, |
| 467 | { "show-ref", cmd_show_ref, RUN_SETUP }, |
| 468 | { "stage", cmd_add, RUN_SETUP | NEED_WORK_TREE }, |
| 469 | { "status", cmd_status, RUN_SETUP | NEED_WORK_TREE }, |
| 470 | { "stripspace", cmd_stripspace }, |
Stefan Beller | 89c8626 | 2016-12-08 13:03:25 -0800 | [diff] [blame] | 471 | { "submodule--helper", cmd_submodule__helper, RUN_SETUP | SUPPORT_SUPER_PREFIX}, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 472 | { "symbolic-ref", cmd_symbolic_ref, RUN_SETUP }, |
Martin Ågren | de121ff | 2017-08-02 21:40:53 +0200 | [diff] [blame] | 473 | { "tag", cmd_tag, RUN_SETUP | DELAY_PAGER_CONFIG }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 474 | { "unpack-file", cmd_unpack_file, RUN_SETUP }, |
| 475 | { "unpack-objects", cmd_unpack_objects, RUN_SETUP }, |
| 476 | { "update-index", cmd_update_index, RUN_SETUP }, |
| 477 | { "update-ref", cmd_update_ref, RUN_SETUP }, |
| 478 | { "update-server-info", cmd_update_server_info, RUN_SETUP }, |
| 479 | { "upload-archive", cmd_upload_archive }, |
| 480 | { "upload-archive--writer", cmd_upload_archive_writer }, |
| 481 | { "var", cmd_var, RUN_SETUP_GENTLY }, |
Michael J Gruber | d07b00b | 2014-06-23 09:05:49 +0200 | [diff] [blame] | 482 | { "verify-commit", cmd_verify_commit, RUN_SETUP }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 483 | { "verify-pack", cmd_verify_pack }, |
| 484 | { "verify-tag", cmd_verify_tag, RUN_SETUP }, |
| 485 | { "version", cmd_version }, |
| 486 | { "whatchanged", cmd_whatchanged, RUN_SETUP }, |
Nguyễn Thái Ngọc Duy | df0b6cf | 2015-06-29 19:51:18 +0700 | [diff] [blame] | 487 | { "worktree", cmd_worktree, RUN_SETUP }, |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 488 | { "write-tree", cmd_write_tree, RUN_SETUP }, |
| 489 | }; |
| 490 | |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 491 | static struct cmd_struct *get_builtin(const char *s) |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 492 | { |
| 493 | int i; |
| 494 | for (i = 0; i < ARRAY_SIZE(commands); i++) { |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 495 | struct cmd_struct *p = commands + i; |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 496 | if (!strcmp(s, p->cmd)) |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 497 | return p; |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 498 | } |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 499 | return NULL; |
| 500 | } |
| 501 | |
| 502 | int is_builtin(const char *s) |
| 503 | { |
| 504 | return !!get_builtin(s); |
Sebastian Schuberth | c6127fa | 2014-01-02 17:17:11 +0100 | [diff] [blame] | 505 | } |
| 506 | |
Jeff King | 8893fd9 | 2017-05-30 01:18:43 -0400 | [diff] [blame] | 507 | static void list_builtins(void) |
| 508 | { |
| 509 | int i; |
| 510 | for (i = 0; i < ARRAY_SIZE(commands); i++) |
| 511 | printf("%s\n", commands[i].cmd); |
| 512 | } |
| 513 | |
Alexander Kuleshov | 63ca1c0 | 2016-02-22 13:18:29 +0600 | [diff] [blame] | 514 | #ifdef STRIP_EXTENSION |
| 515 | static void strip_extension(const char **argv) |
| 516 | { |
| 517 | size_t len; |
| 518 | |
| 519 | if (strip_suffix(argv[0], STRIP_EXTENSION, &len)) |
| 520 | argv[0] = xmemdupz(argv[0], len); |
| 521 | } |
| 522 | #else |
| 523 | #define strip_extension(cmd) |
| 524 | #endif |
| 525 | |
Sebastian Schuberth | 3f784a4 | 2014-01-02 17:15:44 +0100 | [diff] [blame] | 526 | static void handle_builtin(int argc, const char **argv) |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 527 | { |
Ralf Thielow | 2c6b6d9 | 2016-08-26 19:58:36 +0200 | [diff] [blame] | 528 | struct argv_array args = ARGV_ARRAY_INIT; |
Alexander Kuleshov | 63ca1c0 | 2016-02-22 13:18:29 +0600 | [diff] [blame] | 529 | const char *cmd; |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 530 | struct cmd_struct *builtin; |
Johannes Sixt | 23326d1 | 2007-12-08 20:57:25 +0100 | [diff] [blame] | 531 | |
Alexander Kuleshov | 63ca1c0 | 2016-02-22 13:18:29 +0600 | [diff] [blame] | 532 | strip_extension(argv); |
| 533 | cmd = argv[0]; |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 534 | |
Ralf Thielow | 2c6b6d9 | 2016-08-26 19:58:36 +0200 | [diff] [blame] | 535 | /* Turn "git cmd --help" into "git help --exclude-guides cmd" */ |
Linus Torvalds | 1cd9508 | 2006-04-15 11:13:49 -0700 | [diff] [blame] | 536 | if (argc > 1 && !strcmp(argv[1], "--help")) { |
Ralf Thielow | 2c6b6d9 | 2016-08-26 19:58:36 +0200 | [diff] [blame] | 537 | int i; |
| 538 | |
Linus Torvalds | 1cd9508 | 2006-04-15 11:13:49 -0700 | [diff] [blame] | 539 | argv[1] = argv[0]; |
| 540 | argv[0] = cmd = "help"; |
Ralf Thielow | 2c6b6d9 | 2016-08-26 19:58:36 +0200 | [diff] [blame] | 541 | |
| 542 | for (i = 0; i < argc; i++) { |
| 543 | argv_array_push(&args, argv[i]); |
| 544 | if (!i) |
| 545 | argv_array_push(&args, "--exclude-guides"); |
| 546 | } |
| 547 | |
| 548 | argc++; |
| 549 | argv = args.argv; |
Linus Torvalds | 1cd9508 | 2006-04-15 11:13:49 -0700 | [diff] [blame] | 550 | } |
| 551 | |
Slavomir Vlcek | c4f901d | 2014-11-12 14:10:22 +0100 | [diff] [blame] | 552 | builtin = get_builtin(cmd); |
Junio C Hamano | 441981b | 2016-01-26 22:52:02 -0800 | [diff] [blame] | 553 | if (builtin) |
| 554 | exit(run_builtin(builtin, argc, argv)); |
Ralf Thielow | 2c6b6d9 | 2016-08-26 19:58:36 +0200 | [diff] [blame] | 555 | argv_array_clear(&args); |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 556 | } |
| 557 | |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 558 | static void execv_dashed_external(const char **argv) |
| 559 | { |
Jeff King | 2b296c9 | 2017-01-06 20:16:24 -0500 | [diff] [blame] | 560 | struct child_process cmd = CHILD_PROCESS_INIT; |
Jeff King | d8e96fd | 2009-01-28 02:38:14 -0500 | [diff] [blame] | 561 | int status; |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 562 | |
Brandon Williams | 74866d7 | 2016-10-07 11:18:48 -0700 | [diff] [blame] | 563 | if (get_super_prefix()) |
| 564 | die("%s doesn't support --super-prefix", argv[0]); |
| 565 | |
Martin Ågren | 595d59e | 2017-08-02 21:40:55 +0200 | [diff] [blame] | 566 | if (use_pager == -1 && !is_builtin(argv[0])) |
Jeff King | 92058e4 | 2011-08-18 15:01:32 -0700 | [diff] [blame] | 567 | use_pager = check_pager_config(argv[0]); |
Jonathan Nieder | 030149a | 2010-07-14 17:55:12 -0500 | [diff] [blame] | 568 | commit_pager_choice(); |
| 569 | |
Jeff King | 2b296c9 | 2017-01-06 20:16:24 -0500 | [diff] [blame] | 570 | argv_array_pushf(&cmd.args, "git-%s", argv[0]); |
| 571 | argv_array_pushv(&cmd.args, argv + 1); |
| 572 | cmd.clean_on_exit = 1; |
Jeff King | 46df690 | 2017-01-06 20:22:23 -0500 | [diff] [blame] | 573 | cmd.wait_after_clean = 1; |
Jeff King | 2b296c9 | 2017-01-06 20:16:24 -0500 | [diff] [blame] | 574 | cmd.silent_exec_failure = 1; |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 575 | |
Jeff King | 2b296c9 | 2017-01-06 20:16:24 -0500 | [diff] [blame] | 576 | trace_argv_printf(cmd.args.argv, "trace: exec:"); |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 577 | |
| 578 | /* |
Jeff King | 246f0ed | 2017-01-06 20:17:48 -0500 | [diff] [blame] | 579 | * If we fail because the command is not found, it is |
| 580 | * OK to return. Otherwise, we just pass along the status code, |
| 581 | * or our usual generic code if we were not even able to exec |
| 582 | * the program. |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 583 | */ |
Jeff King | 2b296c9 | 2017-01-06 20:16:24 -0500 | [diff] [blame] | 584 | status = run_command(&cmd); |
Jeff King | 246f0ed | 2017-01-06 20:17:48 -0500 | [diff] [blame] | 585 | if (status >= 0) |
Johannes Sixt | 5709e03 | 2009-07-04 21:26:39 +0200 | [diff] [blame] | 586 | exit(status); |
Jeff King | 246f0ed | 2017-01-06 20:17:48 -0500 | [diff] [blame] | 587 | else if (errno != ENOENT) |
| 588 | exit(128); |
Junio C Hamano | 7550be0 | 2007-12-01 22:09:22 -0800 | [diff] [blame] | 589 | } |
| 590 | |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 591 | static int run_argv(int *argcp, const char ***argv) |
| 592 | { |
| 593 | int done_alias = 0; |
| 594 | |
| 595 | while (1) { |
Junio C Hamano | 441981b | 2016-01-26 22:52:02 -0800 | [diff] [blame] | 596 | /* |
| 597 | * If we tried alias and futzed with our environment, |
| 598 | * it no longer is safe to invoke builtins directly in |
| 599 | * general. We have to spawn them as dashed externals. |
| 600 | * |
| 601 | * NEEDSWORK: if we can figure out cases |
| 602 | * where it is safe to do, we can avoid spawning a new |
| 603 | * process. |
| 604 | */ |
| 605 | if (!done_alias) |
| 606 | handle_builtin(*argcp, *argv); |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 607 | |
| 608 | /* .. then try the external ones */ |
| 609 | execv_dashed_external(*argv); |
| 610 | |
| 611 | /* It could be an alias -- this works around the insanity |
| 612 | * of overriding "git log" with "git show" by having |
| 613 | * alias.log = show |
| 614 | */ |
Nguyễn Thái Ngọc Duy | c056261 | 2014-06-08 16:37:10 +0700 | [diff] [blame] | 615 | if (done_alias) |
| 616 | break; |
Nguyễn Thái Ngọc Duy | c056261 | 2014-06-08 16:37:10 +0700 | [diff] [blame] | 617 | if (!handle_alias(argcp, argv)) |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 618 | break; |
| 619 | done_alias = 1; |
| 620 | } |
| 621 | |
| 622 | return done_alias; |
| 623 | } |
| 624 | |
Jeff King | 3f2e229 | 2016-07-01 01:58:58 -0400 | [diff] [blame] | 625 | int cmd_main(int argc, const char **argv) |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 626 | { |
Steve Haslam | 4dd47c3 | 2009-01-18 13:00:10 +0100 | [diff] [blame] | 627 | const char *cmd; |
Alexander Kuleshov | 8fa7975 | 2015-03-02 18:02:37 +0600 | [diff] [blame] | 628 | int done_help = 0; |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 629 | |
Jeff King | 650c449 | 2016-07-01 02:04:04 -0400 | [diff] [blame] | 630 | cmd = argv[0]; |
Steffen Prohaska | 2cd72b0 | 2009-01-18 13:00:11 +0100 | [diff] [blame] | 631 | if (!cmd) |
Steve Haslam | 4dd47c3 | 2009-01-18 13:00:10 +0100 | [diff] [blame] | 632 | cmd = "git-help"; |
Jeff King | 6854a8f | 2016-11-26 23:31:13 -0500 | [diff] [blame] | 633 | else { |
| 634 | const char *slash = find_last_dir_sep(cmd); |
| 635 | if (slash) |
| 636 | cmd = slash + 1; |
| 637 | } |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 638 | |
Karsten Blees | 578da03 | 2014-07-12 02:07:01 +0200 | [diff] [blame] | 639 | trace_command_performance(argv); |
| 640 | |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 641 | /* |
| 642 | * "git-xxxx" is the same as "git xxxx", but we obviously: |
| 643 | * |
| 644 | * - cannot take flags in between the "git" and the "xxxx". |
| 645 | * - cannot execute it externally (since it would just do |
| 646 | * the same thing over again) |
| 647 | * |
Sebastian Schuberth | 3f784a4 | 2014-01-02 17:15:44 +0100 | [diff] [blame] | 648 | * So we just directly call the builtin handler, and die if |
| 649 | * that one cannot handle it. |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 650 | */ |
Jeff King | ae021d8 | 2014-06-18 15:47:50 -0400 | [diff] [blame] | 651 | if (skip_prefix(cmd, "git-", &cmd)) { |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 652 | argv[0] = cmd; |
Sebastian Schuberth | 3f784a4 | 2014-01-02 17:15:44 +0100 | [diff] [blame] | 653 | handle_builtin(argc, argv); |
| 654 | die("cannot handle %s as a builtin", cmd); |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 655 | } |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 656 | |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 657 | /* Look for flags.. */ |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 658 | argv++; |
| 659 | argc--; |
Matthias Lederhofer | 4394efe | 2007-06-08 22:57:55 +0200 | [diff] [blame] | 660 | handle_options(&argv, &argc, NULL); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 661 | if (argc > 0) { |
Jeff King | 6d87780 | 2014-06-18 15:56:48 -0400 | [diff] [blame] | 662 | /* translate --help and --version into commands */ |
| 663 | skip_prefix(argv[0], "--", &argv[0]); |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 664 | } else { |
Scott R Parish | 3d7e2d8 | 2007-10-27 01:36:49 -0700 | [diff] [blame] | 665 | /* The user didn't specify a command; give them help */ |
Nguyễn Thái Ngọc Duy | 73e25e7 | 2010-06-26 14:26:37 -0500 | [diff] [blame] | 666 | commit_pager_choice(); |
Scott R Parish | 3d7e2d8 | 2007-10-27 01:36:49 -0700 | [diff] [blame] | 667 | printf("usage: %s\n\n", git_usage_string); |
| 668 | list_common_cmds_help(); |
Kevin Bracey | 421a976 | 2013-03-10 17:10:20 +0200 | [diff] [blame] | 669 | printf("\n%s\n", _(git_more_info_string)); |
Scott R Parish | 3d7e2d8 | 2007-10-27 01:36:49 -0700 | [diff] [blame] | 670 | exit(1); |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 671 | } |
Johannes Schindelin | 6acbcb9 | 2006-07-25 20:24:22 +0200 | [diff] [blame] | 672 | cmd = argv[0]; |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 673 | |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 674 | /* |
Scott R Parish | 511707d | 2007-10-28 04:17:20 -0700 | [diff] [blame] | 675 | * We use PATH to find git commands, but we prepend some higher |
Mike Ralphson | 3ea3c21 | 2009-04-17 19:13:30 +0100 | [diff] [blame] | 676 | * precedence paths: the "--exec-path" option, the GIT_EXEC_PATH |
Scott R Parish | 511707d | 2007-10-28 04:17:20 -0700 | [diff] [blame] | 677 | * environment, and the $(gitexecdir) from the Makefile at build |
| 678 | * time. |
Linus Torvalds | 231af83 | 2006-02-26 12:34:51 -0800 | [diff] [blame] | 679 | */ |
Johannes Sixt | e1464ca | 2008-07-21 21:19:52 +0200 | [diff] [blame] | 680 | setup_path(); |
Junio C Hamano | 7dbc2c0 | 2005-11-15 23:13:30 -0800 | [diff] [blame] | 681 | |
Junio C Hamano | a025463 | 2006-06-05 18:09:40 -0700 | [diff] [blame] | 682 | while (1) { |
Alexander Kuleshov | 8fa7975 | 2015-03-02 18:02:37 +0600 | [diff] [blame] | 683 | int was_alias = run_argv(&argc, &argv); |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 684 | if (errno != ENOENT) |
Junio C Hamano | a025463 | 2006-06-05 18:09:40 -0700 | [diff] [blame] | 685 | break; |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 686 | if (was_alias) { |
Theodore Ts'o | f3dd015 | 2007-02-10 19:33:57 -0500 | [diff] [blame] | 687 | fprintf(stderr, "Expansion of alias '%s' failed; " |
Pete Harlan | 7283bbc | 2010-02-15 15:33:18 -0800 | [diff] [blame] | 688 | "'%s' is not a git command\n", |
Theodore Ts'o | f3dd015 | 2007-02-10 19:33:57 -0500 | [diff] [blame] | 689 | cmd, argv[0]); |
| 690 | exit(1); |
| 691 | } |
Adeodato Simó | a907e1b | 2009-01-04 18:16:01 +0100 | [diff] [blame] | 692 | if (!done_help) { |
| 693 | cmd = argv[0] = help_unknown_cmd(cmd); |
| 694 | done_help = 1; |
| 695 | } else |
| 696 | break; |
Theodore Ts'o | f3dd015 | 2007-02-10 19:33:57 -0500 | [diff] [blame] | 697 | } |
Alex Riesen | 10b15b8 | 2005-12-01 13:48:35 +0100 | [diff] [blame] | 698 | |
| 699 | fprintf(stderr, "Failed to run command '%s': %s\n", |
Andreas Ericsson | 33ebb87 | 2006-06-28 02:17:21 -0700 | [diff] [blame] | 700 | cmd, strerror(errno)); |
Andreas Ericsson | 8e49d50 | 2005-11-16 00:31:25 +0100 | [diff] [blame] | 701 | |
| 702 | return 1; |
| 703 | } |