Signed-off-by: Elijah Newren <newren@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
		
			
				
	
	
		
			291 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include "builtin.h"
 | 
						|
#include "abspath.h"
 | 
						|
#include "cache.h"
 | 
						|
#include "config.h"
 | 
						|
#include "environment.h"
 | 
						|
#include "gettext.h"
 | 
						|
#include "parse-options.h"
 | 
						|
#include "midx.h"
 | 
						|
#include "trace2.h"
 | 
						|
#include "object-store.h"
 | 
						|
 | 
						|
#define BUILTIN_MIDX_WRITE_USAGE \
 | 
						|
	N_("git multi-pack-index [<options>] write [--preferred-pack=<pack>]" \
 | 
						|
	   "[--refs-snapshot=<path>]")
 | 
						|
 | 
						|
#define BUILTIN_MIDX_VERIFY_USAGE \
 | 
						|
	N_("git multi-pack-index [<options>] verify")
 | 
						|
 | 
						|
#define BUILTIN_MIDX_EXPIRE_USAGE \
 | 
						|
	N_("git multi-pack-index [<options>] expire")
 | 
						|
 | 
						|
#define BUILTIN_MIDX_REPACK_USAGE \
 | 
						|
	N_("git multi-pack-index [<options>] repack [--batch-size=<size>]")
 | 
						|
 | 
						|
static char const * const builtin_multi_pack_index_write_usage[] = {
 | 
						|
	BUILTIN_MIDX_WRITE_USAGE,
 | 
						|
	NULL
 | 
						|
};
 | 
						|
static char const * const builtin_multi_pack_index_verify_usage[] = {
 | 
						|
	BUILTIN_MIDX_VERIFY_USAGE,
 | 
						|
	NULL
 | 
						|
};
 | 
						|
static char const * const builtin_multi_pack_index_expire_usage[] = {
 | 
						|
	BUILTIN_MIDX_EXPIRE_USAGE,
 | 
						|
	NULL
 | 
						|
};
 | 
						|
static char const * const builtin_multi_pack_index_repack_usage[] = {
 | 
						|
	BUILTIN_MIDX_REPACK_USAGE,
 | 
						|
	NULL
 | 
						|
};
 | 
						|
static char const * const builtin_multi_pack_index_usage[] = {
 | 
						|
	BUILTIN_MIDX_WRITE_USAGE,
 | 
						|
	BUILTIN_MIDX_VERIFY_USAGE,
 | 
						|
	BUILTIN_MIDX_EXPIRE_USAGE,
 | 
						|
	BUILTIN_MIDX_REPACK_USAGE,
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
static struct opts_multi_pack_index {
 | 
						|
	char *object_dir;
 | 
						|
	const char *preferred_pack;
 | 
						|
	const char *refs_snapshot;
 | 
						|
	unsigned long batch_size;
 | 
						|
	unsigned flags;
 | 
						|
	int stdin_packs;
 | 
						|
} opts;
 | 
						|
 | 
						|
 | 
						|
static int parse_object_dir(const struct option *opt, const char *arg,
 | 
						|
			    int unset)
 | 
						|
{
 | 
						|
	char **value = opt->value;
 | 
						|
	free(*value);
 | 
						|
	if (unset)
 | 
						|
		*value = xstrdup(get_object_directory());
 | 
						|
	else
 | 
						|
		*value = real_pathdup(arg, 1);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static struct option common_opts[] = {
 | 
						|
	OPT_CALLBACK(0, "object-dir", &opts.object_dir,
 | 
						|
	  N_("directory"),
 | 
						|
	  N_("object directory containing set of packfile and pack-index pairs"),
 | 
						|
	  parse_object_dir),
 | 
						|
	OPT_END(),
 | 
						|
};
 | 
						|
 | 
						|
static struct option *add_common_options(struct option *prev)
 | 
						|
{
 | 
						|
	return parse_options_concat(common_opts, prev);
 | 
						|
}
 | 
						|
 | 
						|
static int git_multi_pack_index_write_config(const char *var, const char *value,
 | 
						|
					     void *cb UNUSED)
 | 
						|
{
 | 
						|
	if (!strcmp(var, "pack.writebitmaphashcache")) {
 | 
						|
		if (git_config_bool(var, value))
 | 
						|
			opts.flags |= MIDX_WRITE_BITMAP_HASH_CACHE;
 | 
						|
		else
 | 
						|
			opts.flags &= ~MIDX_WRITE_BITMAP_HASH_CACHE;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!strcmp(var, "pack.writebitmaplookuptable")) {
 | 
						|
		if (git_config_bool(var, value))
 | 
						|
			opts.flags |= MIDX_WRITE_BITMAP_LOOKUP_TABLE;
 | 
						|
		else
 | 
						|
			opts.flags &= ~MIDX_WRITE_BITMAP_LOOKUP_TABLE;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * We should never make a fall-back call to 'git_default_config', since
 | 
						|
	 * this was already called in 'cmd_multi_pack_index()'.
 | 
						|
	 */
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void read_packs_from_stdin(struct string_list *to)
 | 
						|
{
 | 
						|
	struct strbuf buf = STRBUF_INIT;
 | 
						|
	while (strbuf_getline(&buf, stdin) != EOF)
 | 
						|
		string_list_append(to, buf.buf);
 | 
						|
	string_list_sort(to);
 | 
						|
 | 
						|
	strbuf_release(&buf);
 | 
						|
}
 | 
						|
 | 
						|
static int cmd_multi_pack_index_write(int argc, const char **argv,
 | 
						|
				      const char *prefix)
 | 
						|
{
 | 
						|
	struct option *options;
 | 
						|
	static struct option builtin_multi_pack_index_write_options[] = {
 | 
						|
		OPT_STRING(0, "preferred-pack", &opts.preferred_pack,
 | 
						|
			   N_("preferred-pack"),
 | 
						|
			   N_("pack for reuse when computing a multi-pack bitmap")),
 | 
						|
		OPT_BIT(0, "bitmap", &opts.flags, N_("write multi-pack bitmap"),
 | 
						|
			MIDX_WRITE_BITMAP | MIDX_WRITE_REV_INDEX),
 | 
						|
		OPT_BIT(0, "progress", &opts.flags,
 | 
						|
			N_("force progress reporting"), MIDX_PROGRESS),
 | 
						|
		OPT_BOOL(0, "stdin-packs", &opts.stdin_packs,
 | 
						|
			 N_("write multi-pack index containing only given indexes")),
 | 
						|
		OPT_FILENAME(0, "refs-snapshot", &opts.refs_snapshot,
 | 
						|
			     N_("refs snapshot for selecting bitmap commits")),
 | 
						|
		OPT_END(),
 | 
						|
	};
 | 
						|
 | 
						|
	opts.flags |= MIDX_WRITE_BITMAP_HASH_CACHE;
 | 
						|
 | 
						|
	git_config(git_multi_pack_index_write_config, NULL);
 | 
						|
 | 
						|
	options = add_common_options(builtin_multi_pack_index_write_options);
 | 
						|
 | 
						|
	trace2_cmd_mode(argv[0]);
 | 
						|
 | 
						|
	if (isatty(2))
 | 
						|
		opts.flags |= MIDX_PROGRESS;
 | 
						|
	argc = parse_options(argc, argv, prefix,
 | 
						|
			     options, builtin_multi_pack_index_write_usage,
 | 
						|
			     0);
 | 
						|
	if (argc)
 | 
						|
		usage_with_options(builtin_multi_pack_index_write_usage,
 | 
						|
				   options);
 | 
						|
 | 
						|
	FREE_AND_NULL(options);
 | 
						|
 | 
						|
	if (opts.stdin_packs) {
 | 
						|
		struct string_list packs = STRING_LIST_INIT_DUP;
 | 
						|
		int ret;
 | 
						|
 | 
						|
		read_packs_from_stdin(&packs);
 | 
						|
 | 
						|
		ret = write_midx_file_only(opts.object_dir, &packs,
 | 
						|
					   opts.preferred_pack,
 | 
						|
					   opts.refs_snapshot, opts.flags);
 | 
						|
 | 
						|
		string_list_clear(&packs, 0);
 | 
						|
 | 
						|
		return ret;
 | 
						|
 | 
						|
	}
 | 
						|
	return write_midx_file(opts.object_dir, opts.preferred_pack,
 | 
						|
			       opts.refs_snapshot, opts.flags);
 | 
						|
}
 | 
						|
 | 
						|
static int cmd_multi_pack_index_verify(int argc, const char **argv,
 | 
						|
				       const char *prefix)
 | 
						|
{
 | 
						|
	struct option *options;
 | 
						|
	static struct option builtin_multi_pack_index_verify_options[] = {
 | 
						|
		OPT_BIT(0, "progress", &opts.flags,
 | 
						|
			N_("force progress reporting"), MIDX_PROGRESS),
 | 
						|
		OPT_END(),
 | 
						|
	};
 | 
						|
	options = add_common_options(builtin_multi_pack_index_verify_options);
 | 
						|
 | 
						|
	trace2_cmd_mode(argv[0]);
 | 
						|
 | 
						|
	if (isatty(2))
 | 
						|
		opts.flags |= MIDX_PROGRESS;
 | 
						|
	argc = parse_options(argc, argv, prefix,
 | 
						|
			     options, builtin_multi_pack_index_verify_usage,
 | 
						|
			     0);
 | 
						|
	if (argc)
 | 
						|
		usage_with_options(builtin_multi_pack_index_verify_usage,
 | 
						|
				   options);
 | 
						|
 | 
						|
	FREE_AND_NULL(options);
 | 
						|
 | 
						|
	return verify_midx_file(the_repository, opts.object_dir, opts.flags);
 | 
						|
}
 | 
						|
 | 
						|
static int cmd_multi_pack_index_expire(int argc, const char **argv,
 | 
						|
				       const char *prefix)
 | 
						|
{
 | 
						|
	struct option *options;
 | 
						|
	static struct option builtin_multi_pack_index_expire_options[] = {
 | 
						|
		OPT_BIT(0, "progress", &opts.flags,
 | 
						|
			N_("force progress reporting"), MIDX_PROGRESS),
 | 
						|
		OPT_END(),
 | 
						|
	};
 | 
						|
	options = add_common_options(builtin_multi_pack_index_expire_options);
 | 
						|
 | 
						|
	trace2_cmd_mode(argv[0]);
 | 
						|
 | 
						|
	if (isatty(2))
 | 
						|
		opts.flags |= MIDX_PROGRESS;
 | 
						|
	argc = parse_options(argc, argv, prefix,
 | 
						|
			     options, builtin_multi_pack_index_expire_usage,
 | 
						|
			     0);
 | 
						|
	if (argc)
 | 
						|
		usage_with_options(builtin_multi_pack_index_expire_usage,
 | 
						|
				   options);
 | 
						|
 | 
						|
	FREE_AND_NULL(options);
 | 
						|
 | 
						|
	return expire_midx_packs(the_repository, opts.object_dir, opts.flags);
 | 
						|
}
 | 
						|
 | 
						|
static int cmd_multi_pack_index_repack(int argc, const char **argv,
 | 
						|
				       const char *prefix)
 | 
						|
{
 | 
						|
	struct option *options;
 | 
						|
	static struct option builtin_multi_pack_index_repack_options[] = {
 | 
						|
		OPT_MAGNITUDE(0, "batch-size", &opts.batch_size,
 | 
						|
		  N_("during repack, collect pack-files of smaller size into a batch that is larger than this size")),
 | 
						|
		OPT_BIT(0, "progress", &opts.flags,
 | 
						|
		  N_("force progress reporting"), MIDX_PROGRESS),
 | 
						|
		OPT_END(),
 | 
						|
	};
 | 
						|
 | 
						|
	options = add_common_options(builtin_multi_pack_index_repack_options);
 | 
						|
 | 
						|
	trace2_cmd_mode(argv[0]);
 | 
						|
 | 
						|
	if (isatty(2))
 | 
						|
		opts.flags |= MIDX_PROGRESS;
 | 
						|
	argc = parse_options(argc, argv, prefix,
 | 
						|
			     options,
 | 
						|
			     builtin_multi_pack_index_repack_usage,
 | 
						|
			     0);
 | 
						|
	if (argc)
 | 
						|
		usage_with_options(builtin_multi_pack_index_repack_usage,
 | 
						|
				   options);
 | 
						|
 | 
						|
	FREE_AND_NULL(options);
 | 
						|
 | 
						|
	return midx_repack(the_repository, opts.object_dir,
 | 
						|
			   (size_t)opts.batch_size, opts.flags);
 | 
						|
}
 | 
						|
 | 
						|
int cmd_multi_pack_index(int argc, const char **argv,
 | 
						|
			 const char *prefix)
 | 
						|
{
 | 
						|
	int res;
 | 
						|
	parse_opt_subcommand_fn *fn = NULL;
 | 
						|
	struct option builtin_multi_pack_index_options[] = {
 | 
						|
		OPT_SUBCOMMAND("repack", &fn, cmd_multi_pack_index_repack),
 | 
						|
		OPT_SUBCOMMAND("write", &fn, cmd_multi_pack_index_write),
 | 
						|
		OPT_SUBCOMMAND("verify", &fn, cmd_multi_pack_index_verify),
 | 
						|
		OPT_SUBCOMMAND("expire", &fn, cmd_multi_pack_index_expire),
 | 
						|
		OPT_END(),
 | 
						|
	};
 | 
						|
	struct option *options = parse_options_concat(builtin_multi_pack_index_options, common_opts);
 | 
						|
 | 
						|
	git_config(git_default_config, NULL);
 | 
						|
 | 
						|
	if (the_repository &&
 | 
						|
	    the_repository->objects &&
 | 
						|
	    the_repository->objects->odb)
 | 
						|
		opts.object_dir = xstrdup(the_repository->objects->odb->path);
 | 
						|
 | 
						|
	argc = parse_options(argc, argv, prefix, options,
 | 
						|
			     builtin_multi_pack_index_usage, 0);
 | 
						|
	FREE_AND_NULL(options);
 | 
						|
 | 
						|
	res = fn(argc, argv, prefix);
 | 
						|
 | 
						|
	free(opts.object_dir);
 | 
						|
	return res;
 | 
						|
}
 |