summaryrefslogtreecommitdiff
path: root/src/diff.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/diff.c')
-rw-r--r--src/diff.c166
1 files changed, 92 insertions, 74 deletions
diff --git a/src/diff.c b/src/diff.c
index f3dfdd9dc..015c77edd 100644
--- a/src/diff.c
+++ b/src/diff.c
@@ -327,24 +327,39 @@ void git_diff_list_addref(git_diff_list *diff)
GIT_REFCOUNT_INC(diff);
}
-static int oid_for_workdir_item(
+int git_diff__oid_for_file(
git_repository *repo,
- const git_index_entry *item,
+ const char *path,
+ uint16_t mode,
+ git_off_t size,
git_oid *oid)
{
int result = 0;
git_buf full_path = GIT_BUF_INIT;
if (git_buf_joinpath(
- &full_path, git_repository_workdir(repo), item->path) < 0)
+ &full_path, git_repository_workdir(repo), path) < 0)
return -1;
+ if (!mode) {
+ struct stat st;
+
+ if (p_stat(path, &st) < 0) {
+ giterr_set(GITERR_OS, "Could not stat '%s'", path);
+ result = -1;
+ goto cleanup;
+ }
+
+ mode = st.st_mode;
+ size = st.st_size;
+ }
+
/* calculate OID for file if possible */
- if (S_ISGITLINK(item->mode)) {
+ if (S_ISGITLINK(mode)) {
git_submodule *sm;
const git_oid *sm_oid;
- if (!git_submodule_lookup(&sm, repo, item->path) &&
+ if (!git_submodule_lookup(&sm, repo, path) &&
(sm_oid = git_submodule_wd_oid(sm)) != NULL)
git_oid_cpy(oid, sm_oid);
else {
@@ -354,23 +369,22 @@ static int oid_for_workdir_item(
giterr_clear();
memset(oid, 0, sizeof(*oid));
}
- } else if (S_ISLNK(item->mode))
+ } else if (S_ISLNK(mode)) {
result = git_odb__hashlink(oid, full_path.ptr);
- else if (!git__is_sizet(item->file_size)) {
- giterr_set(GITERR_OS, "File size overflow for 32-bit systems");
+ } else if (!git__is_sizet(size)) {
+ giterr_set(GITERR_OS, "File size overflow (for 32-bits) on '%s'", path);
result = -1;
} else {
git_vector filters = GIT_VECTOR_INIT;
- result = git_filters_load(
- &filters, repo, item->path, GIT_FILTER_TO_ODB);
+ result = git_filters_load(&filters, repo, path, GIT_FILTER_TO_ODB);
if (result >= 0) {
int fd = git_futils_open_ro(full_path.ptr);
if (fd < 0)
result = fd;
else {
result = git_odb__hashfd_filtered(
- oid, fd, (size_t)item->file_size, GIT_OBJ_BLOB, &filters);
+ oid, fd, (size_t)size, GIT_OBJ_BLOB, &filters);
p_close(fd);
}
}
@@ -378,8 +392,8 @@ static int oid_for_workdir_item(
git_filters_free(&filters);
}
+cleanup:
git_buf_free(&full_path);
-
return result;
}
@@ -439,8 +453,7 @@ static int maybe_modified(
}
/* if oids and modes match, then file is unmodified */
- else if (git_oid_cmp(&oitem->oid, &nitem->oid) == 0 &&
- omode == nmode)
+ else if (git_oid_equal(&oitem->oid, &nitem->oid) && omode == nmode)
status = GIT_DELTA_UNMODIFIED;
/* if we have an unknown OID and a workdir iterator, then check some
@@ -493,12 +506,14 @@ static int maybe_modified(
/* if we got here and decided that the files are modified, but we
* haven't calculated the OID of the new item, then calculate it now
*/
- if (status != GIT_DELTA_UNMODIFIED && git_oid_iszero(&nitem->oid)) {
- if (oid_for_workdir_item(diff->repo, nitem, &noid) < 0)
+ if (status != GIT_DELTA_UNMODIFIED &&
+ git_oid_iszero(&nitem->oid) && !use_noid)
+ {
+ if (git_diff__oid_for_file(diff->repo,
+ nitem->path, nitem->mode, nitem->file_size, &noid) < 0)
return -1;
- else if (omode == nmode && git_oid_equal(&oitem->oid, &noid))
+ if (omode == nmode && git_oid_equal(&oitem->oid, &noid))
status = GIT_DELTA_UNMODIFIED;
-
/* store calculated oid so we don't have to recalc later */
use_noid = &noid;
}
@@ -507,31 +522,14 @@ static int maybe_modified(
diff, status, oitem, omode, nitem, nmode, use_noid);
}
-static int git_index_entry_cmp_case(const void *a, const void *b)
-{
- const git_index_entry *entry_a = a;
- const git_index_entry *entry_b = b;
-
- return strcmp(entry_a->path, entry_b->path);
-}
-
-static int git_index_entry_cmp_icase(const void *a, const void *b)
-{
- const git_index_entry *entry_a = a;
- const git_index_entry *entry_b = b;
-
- return strcasecmp(entry_a->path, entry_b->path);
-}
-
static bool entry_is_prefixed(
+ git_diff_list *diff,
const git_index_entry *item,
- git_iterator *prefix_iterator,
const git_index_entry *prefix_item)
{
size_t pathlen;
- if (!prefix_item ||
- ITERATOR_PREFIXCMP(*prefix_iterator, prefix_item->path, item->path))
+ if (!prefix_item || diff->prefixcmp(prefix_item->path, item->path))
return false;
pathlen = strlen(item->path);
@@ -541,6 +539,35 @@ static bool entry_is_prefixed(
prefix_item->path[pathlen] == '/');
}
+static int diff_list_init_from_iterators(
+ git_diff_list *diff,
+ git_iterator *old_iter,
+ git_iterator *new_iter)
+{
+ diff->old_src = old_iter->type;
+ diff->new_src = new_iter->type;
+
+ /* Use case-insensitive compare if either iterator has
+ * the ignore_case bit set */
+ if (!old_iter->ignore_case && !new_iter->ignore_case) {
+ diff->opts.flags &= ~GIT_DIFF_DELTAS_ARE_ICASE;
+
+ diff->strcmp = strcmp;
+ diff->strncmp = strncmp;
+ diff->prefixcmp = git__prefixcmp;
+ diff->entrycmp = git_index_entry__cmp;
+ } else {
+ diff->opts.flags |= GIT_DIFF_DELTAS_ARE_ICASE;
+
+ diff->strcmp = strcasecmp;
+ diff->strncmp = strncasecmp;
+ diff->prefixcmp = git__prefixcmp_icase;
+ diff->entrycmp = git_index_entry__cmp_icase;
+ }
+
+ return 0;
+}
+
static int diff_from_iterators(
git_repository *repo,
const git_diff_options *opts, /**< can be NULL for defaults */
@@ -548,37 +575,31 @@ static int diff_from_iterators(
git_iterator *new_iter,
git_diff_list **diff_ptr)
{
+ int error = 0;
const git_index_entry *oitem, *nitem;
git_buf ignore_prefix = GIT_BUF_INIT;
git_diff_list *diff = git_diff_list_alloc(repo, opts);
- git_vector_cmp entry_compare;
- if (!diff)
- goto fail;
-
- diff->old_src = old_iter->type;
- diff->new_src = new_iter->type;
+ *diff_ptr = NULL;
- /* Use case-insensitive compare if either iterator has
- * the ignore_case bit set */
- if (!old_iter->ignore_case && !new_iter->ignore_case) {
- entry_compare = git_index_entry_cmp_case;
- diff->opts.flags &= ~GIT_DIFF_DELTAS_ARE_ICASE;
- } else {
- entry_compare = git_index_entry_cmp_icase;
- diff->opts.flags |= GIT_DIFF_DELTAS_ARE_ICASE;
+ if (!diff ||
+ diff_list_init_from_iterators(diff, old_iter, new_iter) < 0)
+ goto fail;
+ if (diff->opts.flags & GIT_DIFF_DELTAS_ARE_ICASE) {
/* If one of the iterators doesn't have ignore_case set,
* then that's unfortunate because we'll have to spool
* its data, sort it icase, and then use that for our
* merge join to the other iterator that is icase sorted */
- if (!old_iter->ignore_case) {
- if (git_iterator_spoolandsort(&old_iter, old_iter, git_index_entry_cmp_icase, true) < 0)
- goto fail;
- } else if (!new_iter->ignore_case) {
- if (git_iterator_spoolandsort(&new_iter, new_iter, git_index_entry_cmp_icase, true) < 0)
- goto fail;
- }
+ if (!old_iter->ignore_case &&
+ git_iterator_spoolandsort(
+ &old_iter, old_iter, diff->entrycmp, true) < 0)
+ goto fail;
+
+ if (!new_iter->ignore_case &&
+ git_iterator_spoolandsort(
+ &new_iter, new_iter, diff->entrycmp, true) < 0)
+ goto fail;
}
if (git_iterator_current(old_iter, &oitem) < 0 ||
@@ -589,7 +610,7 @@ static int diff_from_iterators(
while (oitem || nitem) {
/* create DELETED records for old items not matched in new */
- if (oitem && (!nitem || entry_compare(oitem, nitem) < 0)) {
+ if (oitem && (!nitem || diff->entrycmp(oitem, nitem) < 0)) {
if (diff_delta__from_one(diff, GIT_DELTA_DELETED, oitem) < 0)
goto fail;
@@ -597,7 +618,7 @@ static int diff_from_iterators(
* instead of just generating a DELETE record
*/
if ((diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE_TREES) != 0 &&
- entry_is_prefixed(oitem, new_iter, nitem))
+ entry_is_prefixed(diff, oitem, nitem))
{
/* this entry has become a tree! convert to TYPECHANGE */
git_diff_delta *last = diff_delta__last_for_item(diff, oitem);
@@ -614,13 +635,12 @@ static int diff_from_iterators(
/* create ADDED, TRACKED, or IGNORED records for new items not
* matched in old (and/or descend into directories as needed)
*/
- else if (nitem && (!oitem || entry_compare(oitem, nitem) > 0)) {
+ else if (nitem && (!oitem || diff->entrycmp(oitem, nitem) > 0)) {
git_delta_t delta_type = GIT_DELTA_UNTRACKED;
/* check if contained in ignored parent directory */
if (git_buf_len(&ignore_prefix) &&
- ITERATOR_PREFIXCMP(*old_iter, nitem->path,
- git_buf_cstr(&ignore_prefix)) == 0)
+ diff->prefixcmp(nitem->path, git_buf_cstr(&ignore_prefix)) == 0)
delta_type = GIT_DELTA_IGNORED;
if (S_ISDIR(nitem->mode)) {
@@ -629,7 +649,7 @@ static int diff_from_iterators(
* directories and it is not under an ignored directory.
*/
bool contains_tracked =
- entry_is_prefixed(nitem, old_iter, oitem);
+ entry_is_prefixed(diff, nitem, oitem);
bool recurse_untracked =
(delta_type == GIT_DELTA_UNTRACKED &&
(diff->opts.flags & GIT_DIFF_RECURSE_UNTRACKED_DIRS) != 0);
@@ -693,7 +713,7 @@ static int diff_from_iterators(
*/
if (delta_type != GIT_DELTA_IGNORED &&
(diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE_TREES) != 0 &&
- entry_is_prefixed(nitem, old_iter, oitem))
+ entry_is_prefixed(diff, nitem, oitem))
{
/* this entry was a tree! convert to TYPECHANGE */
git_diff_delta *last = diff_delta__last_for_item(diff, oitem);
@@ -711,7 +731,7 @@ static int diff_from_iterators(
* (or ADDED and DELETED pair if type changed)
*/
else {
- assert(oitem && nitem && entry_compare(oitem, nitem) == 0);
+ assert(oitem && nitem && diff->entrycmp(oitem, nitem) == 0);
if (maybe_modified(old_iter, oitem, new_iter, nitem, diff) < 0 ||
git_iterator_advance(old_iter, &oitem) < 0 ||
@@ -720,21 +740,19 @@ static int diff_from_iterators(
}
}
- git_iterator_free(old_iter);
- git_iterator_free(new_iter);
- git_buf_free(&ignore_prefix);
-
*diff_ptr = diff;
- return 0;
fail:
+ if (!*diff_ptr) {
+ git_diff_list_free(diff);
+ error = -1;
+ }
+
git_iterator_free(old_iter);
git_iterator_free(new_iter);
git_buf_free(&ignore_prefix);
- git_diff_list_free(diff);
- *diff_ptr = NULL;
- return -1;
+ return error;
}