pack-objects: run rev-list equivalent internally.
Instead of piping the rev-list output from its standard input, you can say: pack-objects --all --unpacked --revs pack and feed the rev parameters you would otherwise give the rev-list on its command line from the standard input. In other words: echo 'master..next' | pack-objects --revs pack and rev-list --objects master..next | pack-objects pack are equivalent. Signed-off-by: Junio C Hamano <junkio@cox.net>
This commit is contained in:
parent
c64ed70d25
commit
b5d97e6b0a
@ -9,10 +9,13 @@
|
|||||||
#include "pack.h"
|
#include "pack.h"
|
||||||
#include "csum-file.h"
|
#include "csum-file.h"
|
||||||
#include "tree-walk.h"
|
#include "tree-walk.h"
|
||||||
|
#include "diff.h"
|
||||||
|
#include "revision.h"
|
||||||
|
#include "list-objects.h"
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
|
||||||
static const char pack_usage[] = "git-pack-objects [-q] [--no-reuse-delta] [--non-empty] [--local] [--incremental] [--window=N] [--depth=N] {--stdout | base-name} < object-list";
|
static const char pack_usage[] = "git-pack-objects [-q] [--no-reuse-delta] [--non-empty] [--local] [--incremental] [--window=N] [--depth=N] {--stdout | base-name} [--revs [--unpacked | --all]* <ref-list | <object-list]";
|
||||||
|
|
||||||
struct object_entry {
|
struct object_entry {
|
||||||
unsigned char sha1[20];
|
unsigned char sha1[20];
|
||||||
@ -1326,13 +1329,137 @@ static int git_pack_config(const char *k, const char *v)
|
|||||||
return git_default_config(k, v);
|
return git_default_config(k, v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void read_object_list_from_stdin(void)
|
||||||
|
{
|
||||||
|
int num_preferred_base = 0;
|
||||||
|
char line[40 + 1 + PATH_MAX + 2];
|
||||||
|
unsigned char sha1[20];
|
||||||
|
unsigned hash;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
if (!fgets(line, sizeof(line), stdin)) {
|
||||||
|
if (feof(stdin))
|
||||||
|
break;
|
||||||
|
if (!ferror(stdin))
|
||||||
|
die("fgets returned NULL, not EOF, not error!");
|
||||||
|
if (errno != EINTR)
|
||||||
|
die("fgets: %s", strerror(errno));
|
||||||
|
clearerr(stdin);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (line[0] == '-') {
|
||||||
|
if (get_sha1_hex(line+1, sha1))
|
||||||
|
die("expected edge sha1, got garbage:\n %s",
|
||||||
|
line);
|
||||||
|
if (num_preferred_base++ < window)
|
||||||
|
add_preferred_base(sha1);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (get_sha1_hex(line, sha1))
|
||||||
|
die("expected sha1, got garbage:\n %s", line);
|
||||||
|
|
||||||
|
hash = name_hash(line+41);
|
||||||
|
add_preferred_base_object(line+41, hash);
|
||||||
|
add_object_entry(sha1, hash, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* copied from rev-list but needs to do things slightly differently */
|
||||||
|
static void mark_edge_parents_uninteresting(struct commit *commit)
|
||||||
|
{
|
||||||
|
struct commit_list *parents;
|
||||||
|
|
||||||
|
for (parents = commit->parents; parents; parents = parents->next) {
|
||||||
|
struct commit *parent = parents->item;
|
||||||
|
if (!(parent->object.flags & UNINTERESTING))
|
||||||
|
continue;
|
||||||
|
mark_tree_uninteresting(parent->tree);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mark_edges_uninteresting(struct commit_list *list)
|
||||||
|
{
|
||||||
|
for ( ; list; list = list->next) {
|
||||||
|
struct commit *commit = list->item;
|
||||||
|
|
||||||
|
if (commit->object.flags & UNINTERESTING) {
|
||||||
|
mark_tree_uninteresting(commit->tree);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
mark_edge_parents_uninteresting(commit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void show_commit(struct commit *commit)
|
||||||
|
{
|
||||||
|
unsigned hash = name_hash("");
|
||||||
|
add_object_entry(commit->object.sha1, hash, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void show_object(struct object_array_entry *p)
|
||||||
|
{
|
||||||
|
unsigned hash = name_hash(p->name);
|
||||||
|
add_object_entry(p->item->sha1, hash, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void get_object_list(int unpacked, int all)
|
||||||
|
{
|
||||||
|
struct rev_info revs;
|
||||||
|
char line[1000];
|
||||||
|
const char *av[6];
|
||||||
|
int ac;
|
||||||
|
int flags = 0;
|
||||||
|
|
||||||
|
av[0] = "pack-objects";
|
||||||
|
av[1] = "--objects";
|
||||||
|
ac = 2;
|
||||||
|
if (unpacked)
|
||||||
|
av[ac++] = "--unpacked";
|
||||||
|
if (all)
|
||||||
|
av[ac++] = "--all";
|
||||||
|
av[ac++] = "--stdin";
|
||||||
|
av[ac] = NULL;
|
||||||
|
|
||||||
|
init_revisions(&revs, NULL);
|
||||||
|
save_commit_buffer = 0;
|
||||||
|
track_object_refs = 0;
|
||||||
|
setup_revisions(ac, av, &revs, NULL);
|
||||||
|
|
||||||
|
/* make sure we did not get pathspecs */
|
||||||
|
if (revs.prune_data)
|
||||||
|
die("pathspec given");
|
||||||
|
|
||||||
|
while (fgets(line, sizeof(line), stdin) != NULL) {
|
||||||
|
int len = strlen(line);
|
||||||
|
if (line[len - 1] == '\n')
|
||||||
|
line[--len] = 0;
|
||||||
|
if (!len)
|
||||||
|
break;
|
||||||
|
if (*line == '-') {
|
||||||
|
if (!strcmp(line, "--not")) {
|
||||||
|
flags ^= UNINTERESTING;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
die("not a rev '%s'", line);
|
||||||
|
}
|
||||||
|
if (handle_revision_arg(line, &revs, flags, 1))
|
||||||
|
die("bad revision '%s'", line);
|
||||||
|
}
|
||||||
|
|
||||||
|
prepare_revision_walk(&revs);
|
||||||
|
mark_edges_uninteresting(revs.commits);
|
||||||
|
|
||||||
|
traverse_commit_list(&revs, show_commit, show_object);
|
||||||
|
}
|
||||||
|
|
||||||
int cmd_pack_objects(int argc, const char **argv, const char *prefix)
|
int cmd_pack_objects(int argc, const char **argv, const char *prefix)
|
||||||
{
|
{
|
||||||
SHA_CTX ctx;
|
SHA_CTX ctx;
|
||||||
char line[40 + 1 + PATH_MAX + 2];
|
|
||||||
int depth = 10;
|
int depth = 10;
|
||||||
struct object_entry **list;
|
struct object_entry **list;
|
||||||
int num_preferred_base = 0;
|
int use_internal_rev_list = 0;
|
||||||
|
int unpacked = 0;
|
||||||
|
int all = 0;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
git_config(git_pack_config);
|
git_config(git_pack_config);
|
||||||
@ -1341,7 +1468,9 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
|
|||||||
for (i = 1; i < argc; i++) {
|
for (i = 1; i < argc; i++) {
|
||||||
const char *arg = argv[i];
|
const char *arg = argv[i];
|
||||||
|
|
||||||
if (*arg == '-') {
|
if (*arg != '-')
|
||||||
|
break;
|
||||||
|
|
||||||
if (!strcmp("--non-empty", arg)) {
|
if (!strcmp("--non-empty", arg)) {
|
||||||
non_empty = 1;
|
non_empty = 1;
|
||||||
continue;
|
continue;
|
||||||
@ -1388,16 +1517,44 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
|
|||||||
pack_to_stdout = 1;
|
pack_to_stdout = 1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
usage(pack_usage);
|
if (!strcmp("--revs", arg)) {
|
||||||
|
use_internal_rev_list = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!strcmp("--unpacked", arg)) {
|
||||||
|
unpacked = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!strcmp("--all", arg)) {
|
||||||
|
all = 1;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
if (base_name)
|
|
||||||
usage(pack_usage);
|
usage(pack_usage);
|
||||||
base_name = arg;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Traditionally "pack-objects [options] base extra" failed;
|
||||||
|
* we would however want to take refs parameter that would
|
||||||
|
* have been given to upstream rev-list ourselves, which means
|
||||||
|
* we somehow want to say what the base name is. So the
|
||||||
|
* syntax would be:
|
||||||
|
*
|
||||||
|
* pack-objects [options] base <refs...>
|
||||||
|
*
|
||||||
|
* in other words, we would treat the first non-option as the
|
||||||
|
* base_name and send everything else to the internal revision
|
||||||
|
* walker.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (!pack_to_stdout)
|
||||||
|
base_name = argv[i++];
|
||||||
|
|
||||||
if (pack_to_stdout != !base_name)
|
if (pack_to_stdout != !base_name)
|
||||||
usage(pack_usage);
|
usage(pack_usage);
|
||||||
|
|
||||||
|
/* --unpacked and --all makes sense only with --revs */
|
||||||
|
if (!use_internal_rev_list && (unpacked || all))
|
||||||
|
usage(pack_usage);
|
||||||
|
|
||||||
prepare_packed_git();
|
prepare_packed_git();
|
||||||
|
|
||||||
if (progress) {
|
if (progress) {
|
||||||
@ -1405,35 +1562,11 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)
|
|||||||
setup_progress_signal();
|
setup_progress_signal();
|
||||||
}
|
}
|
||||||
|
|
||||||
for (;;) {
|
if (!use_internal_rev_list)
|
||||||
unsigned char sha1[20];
|
read_object_list_from_stdin();
|
||||||
unsigned hash;
|
else
|
||||||
|
get_object_list(unpacked, all);
|
||||||
|
|
||||||
if (!fgets(line, sizeof(line), stdin)) {
|
|
||||||
if (feof(stdin))
|
|
||||||
break;
|
|
||||||
if (!ferror(stdin))
|
|
||||||
die("fgets returned NULL, not EOF, not error!");
|
|
||||||
if (errno != EINTR)
|
|
||||||
die("fgets: %s", strerror(errno));
|
|
||||||
clearerr(stdin);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (line[0] == '-') {
|
|
||||||
if (get_sha1_hex(line+1, sha1))
|
|
||||||
die("expected edge sha1, got garbage:\n %s",
|
|
||||||
line+1);
|
|
||||||
if (num_preferred_base++ < window)
|
|
||||||
add_preferred_base(sha1);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (get_sha1_hex(line, sha1))
|
|
||||||
die("expected sha1, got garbage:\n %s", line);
|
|
||||||
hash = name_hash(line+41);
|
|
||||||
add_preferred_base_object(line+41, hash);
|
|
||||||
add_object_entry(sha1, hash, 0);
|
|
||||||
}
|
|
||||||
if (progress)
|
if (progress)
|
||||||
fprintf(stderr, "Done counting %d objects.\n", nr_objects);
|
fprintf(stderr, "Done counting %d objects.\n", nr_objects);
|
||||||
sorted_by_sha = create_final_object_list();
|
sorted_by_sha = create_final_object_list();
|
||||||
|
Loading…
Reference in New Issue
Block a user