summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPhilip Kelley <phkelley@hotmail.com>2013-03-19 14:56:45 -0400
committerPhilip Kelley <phkelley@hotmail.com>2013-03-19 14:56:45 -0400
commit799f9a04e38579a6340145440bb927d3dd83c642 (patch)
treed9572ea41197d7dd242601e95a56c547b5f52c83
parent7dbf4039ae0881407fc9ead24c09c1d7cfd4103a (diff)
downloadlibgit2-799f9a04e38579a6340145440bb927d3dd83c642.tar.gz
Reduce the number of unnecessary objects in pushed packs
-rw-r--r--src/push.c177
1 files changed, 142 insertions, 35 deletions
diff --git a/src/push.c b/src/push.c
index 37f641812..ee21bd9d1 100644
--- a/src/push.c
+++ b/src/push.c
@@ -13,6 +13,7 @@
#include "remote.h"
#include "vector.h"
#include "push.h"
+#include "tree.h"
static int push_spec_rref_cmp(const void *a, const void *b)
{
@@ -346,60 +347,166 @@ on_error:
return error == GIT_ITEROVER ? 0 : error;
}
-static int queue_objects(git_push *push)
+static int queue_differences(
+ git_tree *base,
+ git_tree *delta,
+ git_packbuilder *pb)
{
- git_vector commits;
- git_oid *o;
- unsigned int i;
+ git_tree *b_child = NULL, *d_child = NULL;
+ size_t b_length = git_tree_entrycount(base);
+ size_t d_length = git_tree_entrycount(delta);
+ size_t i = 0, j = 0;
int error;
- if (git_vector_init(&commits, 0, NULL) < 0)
- return -1;
+#define _enqueue_object(ENTRY) do { \
+ switch (git_tree_entry_type((ENTRY))) { \
+ case GIT_OBJ_COMMIT: \
+ break; \
+ case GIT_OBJ_TREE: \
+ if ((error = git_packbuilder_insert_tree(pb, &(ENTRY)->oid)) < 0) \
+ goto on_error; \
+ break; \
+ default: \
+ if ((error = git_packbuilder_insert(pb, &(ENTRY)->oid, \
+ (ENTRY)->filename)) < 0) \
+ goto on_error; \
+ break; \
+ } \
+} while (0)
+
+ while (i < b_length && j < d_length) {
+ const git_tree_entry *b_entry = git_tree_entry_byindex(base, i);
+ const git_tree_entry *d_entry = git_tree_entry_byindex(delta, j);
+ int cmp = 0;
+
+ if (!git_oid_cmp(&b_entry->oid, &d_entry->oid))
+ goto loop;
+
+ cmp = memcmp(b_entry->filename,
+ d_entry->filename,
+ b_entry->filename_len);
+
+ /* If the entries are both trees and they have the same name but are
+ * different, then we'll recurse after adding the right-hand entry */
+ if (!cmp &&
+ git_tree_entry__is_tree(b_entry) &&
+ git_tree_entry__is_tree(d_entry)) {
+ /* Add the right-hand entry */
+ if ((error = git_packbuilder_insert(pb, &d_entry->oid,
+ d_entry->filename)) < 0)
+ goto on_error;
+
+ /* Acquire the subtrees and recurse */
+ if ((error = git_tree_lookup(&b_child,
+ git_tree_owner(base), &b_entry->oid)) < 0 ||
+ (error = git_tree_lookup(&d_child,
+ git_tree_owner(delta), &d_entry->oid)) < 0 ||
+ (error = queue_differences(b_child, d_child, pb)) < 0)
+ goto on_error;
+
+ git_tree_free(b_child); b_child = NULL;
+ git_tree_free(d_child); d_child = NULL;
+ }
+ /* If the object is new or different in the right-hand tree,
+ * then enumerate it */
+ else if (cmp >= 0)
+ _enqueue_object(d_entry);
+
+ loop:
+ if (cmp <= 0) i++;
+ if (cmp >= 0) j++;
+ }
+
+ /* Drain the right-hand tree of entries */
+ for (; j < d_length; j++)
+ _enqueue_object(git_tree_entry_byindex(delta, j));
+
+#undef _enqueue_object
+
+ error = 0;
+
+on_error:
+ if (b_child)
+ git_tree_free(b_child);
+
+ if (d_child)
+ git_tree_free(d_child);
+
+ return error;
+}
+
+static int queue_objects(git_push *push)
+{
+ git_vector commits = GIT_VECTOR_INIT;
+ git_oid *oid;
+ size_t i;
+ unsigned j;
+ int error;
if ((error = revwalk(&commits, push)) < 0)
goto on_error;
- if (!commits.length) {
- git_vector_free(&commits);
- return 0; /* nothing to do */
- }
+ git_vector_foreach(&commits, i, oid) {
+ git_commit *parent = NULL, *commit;
+ git_tree *tree = NULL, *ptree = NULL;
+ size_t parentcount;
- git_vector_foreach(&commits, i, o) {
- if ((error = git_packbuilder_insert(push->pb, o, NULL)) < 0)
+ if ((error = git_commit_lookup(&commit, push->repo, oid)) < 0)
goto on_error;
- }
- git_vector_foreach(&commits, i, o) {
- git_object *obj;
+ /* Insert the commit */
+ if ((error = git_packbuilder_insert(push->pb, oid, NULL)) < 0)
+ goto loop_error;
- if ((error = git_object_lookup(&obj, push->repo, o, GIT_OBJ_ANY)) < 0)
- goto on_error;
+ parentcount = git_commit_parentcount(commit);
- switch (git_object_type(obj)) {
- case GIT_OBJ_TAG: /* TODO: expect tags */
- case GIT_OBJ_COMMIT:
+ if (!parentcount) {
if ((error = git_packbuilder_insert_tree(push->pb,
- git_commit_tree_id((git_commit *)obj))) < 0) {
- git_object_free(obj);
- goto on_error;
+ git_commit_tree_id(commit))) < 0)
+ goto loop_error;
+ } else {
+ if ((error = git_tree_lookup(&tree, push->repo,
+ git_commit_tree_id(commit))) < 0 ||
+ (error = git_packbuilder_insert(push->pb,
+ git_commit_tree_id(commit), NULL)) < 0)
+ goto loop_error;
+
+ /* For each parent, add the items which are different */
+ for (j = 0; j < parentcount; j++) {
+ if ((error = git_commit_parent(&parent, commit, j)) < 0 ||
+ (error = git_commit_tree(&ptree, parent)) < 0 ||
+ (error = queue_differences(ptree, tree, push->pb)) < 0)
+ goto loop_error;
+
+ git_tree_free(ptree); ptree = NULL;
+ git_commit_free(parent); parent = NULL;
}
- break;
- case GIT_OBJ_TREE:
- case GIT_OBJ_BLOB:
- default:
- git_object_free(obj);
- giterr_set(GITERR_INVALID, "Given object type invalid");
- error = -1;
- goto on_error;
}
- git_object_free(obj);
+
+ error = 0;
+
+ loop_error:
+ if (tree)
+ git_tree_free(tree);
+
+ if (ptree)
+ git_tree_free(ptree);
+
+ if (parent)
+ git_commit_free(parent);
+
+ git_commit_free(commit);
+
+ if (error < 0)
+ goto on_error;
}
+
error = 0;
on_error:
- git_vector_foreach(&commits, i, o) {
- git__free(o);
- }
+ git_vector_foreach(&commits, i, oid)
+ git__free(oid);
+
git_vector_free(&commits);
return error;
}