diff options
| author | Philip Kelley <phkelley@hotmail.com> | 2013-03-19 14:56:45 -0400 | 
|---|---|---|
| committer | Philip Kelley <phkelley@hotmail.com> | 2013-03-19 14:56:45 -0400 | 
| commit | 799f9a04e38579a6340145440bb927d3dd83c642 (patch) | |
| tree | d9572ea41197d7dd242601e95a56c547b5f52c83 /src/push.c | |
| parent | 7dbf4039ae0881407fc9ead24c09c1d7cfd4103a (diff) | |
| download | libgit2-799f9a04e38579a6340145440bb927d3dd83c642.tar.gz | |
Reduce the number of unnecessary objects in pushed packs
Diffstat (limited to 'src/push.c')
| -rw-r--r-- | src/push.c | 177 | 
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;  } | 
