提交 234587fc 编写于 作者: J Jeff King 提交者: Junio C Hamano

gc: use argv-array for sub-commands

git-gc executes many sub-commands. The argument list for
some of these is constant, but for others we add more
arguments at runtime. The latter is implemented by allocating
a constant extra number of NULLs, and either using a custom
append function, or just referencing unused slots by number.

As of commit 7e52f566, which added two new arguments, it is
possible to exceed the constant number of slots for "repack"
by running "git gc --aggressive", causing "git gc" to die.

This patch converts all of the static argv lists to use
argv-array. In addition to fixing the overflow caused by
7e52f566, it has a few advantages:

  1. We can drop the custom append function (which,
     incidentally, had an off-by-one error exacerbating the
     static limit).

  2. We can drop the ugly magic numbers used when adding
     arguments to "prune".

  3. Adding further arguments will be easier; you can just
     add new "push" calls without worrying about increasing
     any static limits.
Signed-off-by: NJeff King <peff@peff.net>
Signed-off-by: NJunio C Hamano <gitster@pobox.com>
上级 d15bbe13
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include "cache.h" #include "cache.h"
#include "parse-options.h" #include "parse-options.h"
#include "run-command.h" #include "run-command.h"
#include "argv-array.h"
#define FAILED_RUN "failed to run %s" #define FAILED_RUN "failed to run %s"
...@@ -28,12 +29,11 @@ static int gc_auto_threshold = 6700; ...@@ -28,12 +29,11 @@ static int gc_auto_threshold = 6700;
static int gc_auto_pack_limit = 50; static int gc_auto_pack_limit = 50;
static const char *prune_expire = "2.weeks.ago"; static const char *prune_expire = "2.weeks.ago";
#define MAX_ADD 10 static struct argv_array pack_refs_cmd = ARGV_ARRAY_INIT;
static const char *argv_pack_refs[] = {"pack-refs", "--all", "--prune", NULL}; static struct argv_array reflog = ARGV_ARRAY_INIT;
static const char *argv_reflog[] = {"reflog", "expire", "--all", NULL}; static struct argv_array repack = ARGV_ARRAY_INIT;
static const char *argv_repack[MAX_ADD] = {"repack", "-d", "-l", NULL}; static struct argv_array prune = ARGV_ARRAY_INIT;
static const char *argv_prune[] = {"prune", "--expire", NULL, NULL, NULL}; static struct argv_array rerere = ARGV_ARRAY_INIT;
static const char *argv_rerere[] = {"rerere", "gc", NULL};
static int gc_config(const char *var, const char *value, void *cb) static int gc_config(const char *var, const char *value, void *cb)
{ {
...@@ -67,19 +67,6 @@ static int gc_config(const char *var, const char *value, void *cb) ...@@ -67,19 +67,6 @@ static int gc_config(const char *var, const char *value, void *cb)
return git_default_config(var, value, cb); return git_default_config(var, value, cb);
} }
static void append_option(const char **cmd, const char *opt, int max_length)
{
int i;
for (i = 0; cmd[i]; i++)
;
if (i + 2 >= max_length)
die(_("Too many options specified"));
cmd[i++] = opt;
cmd[i] = NULL;
}
static int too_many_loose_objects(void) static int too_many_loose_objects(void)
{ {
/* /*
...@@ -147,13 +134,11 @@ static int too_many_packs(void) ...@@ -147,13 +134,11 @@ static int too_many_packs(void)
static void add_repack_all_option(void) static void add_repack_all_option(void)
{ {
if (prune_expire && !strcmp(prune_expire, "now")) if (prune_expire && !strcmp(prune_expire, "now"))
append_option(argv_repack, "-a", MAX_ADD); argv_array_push(&repack, "-a");
else { else {
append_option(argv_repack, "-A", MAX_ADD); argv_array_push(&repack, "-A");
if (prune_expire) { if (prune_expire)
append_option(argv_repack, "--unpack-unreachable", MAX_ADD); argv_array_pushf(&repack, "--unpack-unreachable=%s", prune_expire);
append_option(argv_repack, prune_expire, MAX_ADD);
}
} }
} }
...@@ -187,7 +172,6 @@ int cmd_gc(int argc, const char **argv, const char *prefix) ...@@ -187,7 +172,6 @@ int cmd_gc(int argc, const char **argv, const char *prefix)
int aggressive = 0; int aggressive = 0;
int auto_gc = 0; int auto_gc = 0;
int quiet = 0; int quiet = 0;
char buf[80];
struct option builtin_gc_options[] = { struct option builtin_gc_options[] = {
OPT__QUIET(&quiet, "suppress progress reporting"), OPT__QUIET(&quiet, "suppress progress reporting"),
...@@ -202,6 +186,12 @@ int cmd_gc(int argc, const char **argv, const char *prefix) ...@@ -202,6 +186,12 @@ int cmd_gc(int argc, const char **argv, const char *prefix)
if (argc == 2 && !strcmp(argv[1], "-h")) if (argc == 2 && !strcmp(argv[1], "-h"))
usage_with_options(builtin_gc_usage, builtin_gc_options); usage_with_options(builtin_gc_usage, builtin_gc_options);
argv_array_pushl(&pack_refs_cmd, "pack-refs", "--all", "--prune", NULL);
argv_array_pushl(&reflog, "reflog", "expire", "--all", NULL);
argv_array_pushl(&repack, "repack", "-d", "-l", NULL);
argv_array_pushl(&prune, "prune", "--expire", NULL );
argv_array_pushl(&rerere, "rerere", "gc", NULL);
git_config(gc_config, NULL); git_config(gc_config, NULL);
if (pack_refs < 0) if (pack_refs < 0)
...@@ -213,15 +203,13 @@ int cmd_gc(int argc, const char **argv, const char *prefix) ...@@ -213,15 +203,13 @@ int cmd_gc(int argc, const char **argv, const char *prefix)
usage_with_options(builtin_gc_usage, builtin_gc_options); usage_with_options(builtin_gc_usage, builtin_gc_options);
if (aggressive) { if (aggressive) {
append_option(argv_repack, "-f", MAX_ADD); argv_array_push(&repack, "-f");
append_option(argv_repack, "--depth=250", MAX_ADD); argv_array_push(&repack, "--depth=250");
if (aggressive_window > 0) { if (aggressive_window > 0)
sprintf(buf, "--window=%d", aggressive_window); argv_array_pushf(&repack, "--window=%d", aggressive_window);
append_option(argv_repack, buf, MAX_ADD);
}
} }
if (quiet) if (quiet)
append_option(argv_repack, "-q", MAX_ADD); argv_array_push(&repack, "-q");
if (auto_gc) { if (auto_gc) {
/* /*
...@@ -239,25 +227,25 @@ int cmd_gc(int argc, const char **argv, const char *prefix) ...@@ -239,25 +227,25 @@ int cmd_gc(int argc, const char **argv, const char *prefix)
} else } else
add_repack_all_option(); add_repack_all_option();
if (pack_refs && run_command_v_opt(argv_pack_refs, RUN_GIT_CMD)) if (pack_refs && run_command_v_opt(pack_refs_cmd.argv, RUN_GIT_CMD))
return error(FAILED_RUN, argv_pack_refs[0]); return error(FAILED_RUN, pack_refs_cmd.argv[0]);
if (run_command_v_opt(argv_reflog, RUN_GIT_CMD)) if (run_command_v_opt(reflog.argv, RUN_GIT_CMD))
return error(FAILED_RUN, argv_reflog[0]); return error(FAILED_RUN, reflog.argv[0]);
if (run_command_v_opt(argv_repack, RUN_GIT_CMD)) if (run_command_v_opt(repack.argv, RUN_GIT_CMD))
return error(FAILED_RUN, argv_repack[0]); return error(FAILED_RUN, repack.argv[0]);
if (prune_expire) { if (prune_expire) {
argv_prune[2] = prune_expire; argv_array_push(&prune, prune_expire);
if (quiet) if (quiet)
argv_prune[3] = "--no-progress"; argv_array_push(&prune, "--no-progress");
if (run_command_v_opt(argv_prune, RUN_GIT_CMD)) if (run_command_v_opt(prune.argv, RUN_GIT_CMD))
return error(FAILED_RUN, argv_prune[0]); return error(FAILED_RUN, prune.argv[0]);
} }
if (run_command_v_opt(argv_rerere, RUN_GIT_CMD)) if (run_command_v_opt(rerere.argv, RUN_GIT_CMD))
return error(FAILED_RUN, argv_rerere[0]); return error(FAILED_RUN, rerere.argv[0]);
if (auto_gc && too_many_loose_objects()) if (auto_gc && too_many_loose_objects())
warning(_("There are too many unreachable loose objects; " warning(_("There are too many unreachable loose objects; "
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册