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