am --skip/--abort: merge HEAD/ORIG_HEAD tree into index
After running "git am --abort", and then running "git reset --hard", files that were not modified would still be re-checked out. This is because clean_index() in builtin/am.c mistakenly called the read_tree() function, which overwrites all entries in the index, including the stat info. "git am --skip" did not seem to have this issue because am_skip() called am_run(), which called refresh_cache() to update the stat info. However, there's still a performance penalty as the lack of stat info meant that refresh_cache() would have to scan all files for changes. Fix this by using unpack_trees() instead to merge the tree into the index, so that the stat info from the index is kept. Reported-by: Linus Torvalds <torvalds@linux-foundation.org> Helped-by: Junio C Hamano <gitster@pobox.com> Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> Signed-off-by: Paul Tan <pyokagan@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
e97a5e765d
commit
3ecc7040ef
49
builtin/am.c
49
builtin/am.c
@ -1939,16 +1939,49 @@ static int fast_forward_to(struct tree *head, struct tree *remote, int reset)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Merges a tree into the index. The index's stat info will take precedence
|
||||||
|
* over the merged tree's. Returns 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
|
static int merge_tree(struct tree *tree)
|
||||||
|
{
|
||||||
|
struct lock_file *lock_file;
|
||||||
|
struct unpack_trees_options opts;
|
||||||
|
struct tree_desc t[1];
|
||||||
|
|
||||||
|
if (parse_tree(tree))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
lock_file = xcalloc(1, sizeof(struct lock_file));
|
||||||
|
hold_locked_index(lock_file, 1);
|
||||||
|
|
||||||
|
memset(&opts, 0, sizeof(opts));
|
||||||
|
opts.head_idx = 1;
|
||||||
|
opts.src_index = &the_index;
|
||||||
|
opts.dst_index = &the_index;
|
||||||
|
opts.merge = 1;
|
||||||
|
opts.fn = oneway_merge;
|
||||||
|
init_tree_desc(&t[0], tree->buffer, tree->size);
|
||||||
|
|
||||||
|
if (unpack_trees(1, t, &opts)) {
|
||||||
|
rollback_lock_file(lock_file);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (write_locked_index(&the_index, lock_file, COMMIT_LOCK))
|
||||||
|
die(_("unable to write new index file"));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Clean the index without touching entries that are not modified between
|
* Clean the index without touching entries that are not modified between
|
||||||
* `head` and `remote`.
|
* `head` and `remote`.
|
||||||
*/
|
*/
|
||||||
static int clean_index(const unsigned char *head, const unsigned char *remote)
|
static int clean_index(const unsigned char *head, const unsigned char *remote)
|
||||||
{
|
{
|
||||||
struct lock_file *lock_file;
|
|
||||||
struct tree *head_tree, *remote_tree, *index_tree;
|
struct tree *head_tree, *remote_tree, *index_tree;
|
||||||
unsigned char index[GIT_SHA1_RAWSZ];
|
unsigned char index[GIT_SHA1_RAWSZ];
|
||||||
struct pathspec pathspec;
|
|
||||||
|
|
||||||
head_tree = parse_tree_indirect(head);
|
head_tree = parse_tree_indirect(head);
|
||||||
if (!head_tree)
|
if (!head_tree)
|
||||||
@ -1973,18 +2006,8 @@ static int clean_index(const unsigned char *head, const unsigned char *remote)
|
|||||||
if (fast_forward_to(index_tree, remote_tree, 0))
|
if (fast_forward_to(index_tree, remote_tree, 0))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
memset(&pathspec, 0, sizeof(pathspec));
|
if (merge_tree(remote_tree))
|
||||||
|
|
||||||
lock_file = xcalloc(1, sizeof(struct lock_file));
|
|
||||||
hold_locked_index(lock_file, 1);
|
|
||||||
|
|
||||||
if (read_tree(remote_tree, 0, &pathspec)) {
|
|
||||||
rollback_lock_file(lock_file);
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
|
||||||
|
|
||||||
if (write_locked_index(&the_index, lock_file, COMMIT_LOCK))
|
|
||||||
die(_("unable to write new index file"));
|
|
||||||
|
|
||||||
remove_branch_state();
|
remove_branch_state();
|
||||||
|
|
||||||
|
@ -168,4 +168,28 @@ test_expect_success 'am --abort on unborn branch will keep local commits intact'
|
|||||||
test_cmp expect actual
|
test_cmp expect actual
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'am --skip leaves index stat info alone' '
|
||||||
|
git checkout -f --orphan skip-stat-info &&
|
||||||
|
git reset &&
|
||||||
|
test_commit skip-should-be-untouched &&
|
||||||
|
test-chmtime =0 skip-should-be-untouched.t &&
|
||||||
|
git update-index --refresh &&
|
||||||
|
git diff-files --exit-code --quiet &&
|
||||||
|
test_must_fail git am 0001-*.patch &&
|
||||||
|
git am --skip &&
|
||||||
|
git diff-files --exit-code --quiet
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'am --abort leaves index stat info alone' '
|
||||||
|
git checkout -f --orphan abort-stat-info &&
|
||||||
|
git reset &&
|
||||||
|
test_commit abort-should-be-untouched &&
|
||||||
|
test-chmtime =0 abort-should-be-untouched.t &&
|
||||||
|
git update-index --refresh &&
|
||||||
|
git diff-files --exit-code --quiet &&
|
||||||
|
test_must_fail git am 0001-*.patch &&
|
||||||
|
git am --abort &&
|
||||||
|
git diff-files --exit-code --quiet
|
||||||
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
Loading…
Reference in New Issue
Block a user