diff options
author | Junio C Hamano <gitster@pobox.com> | 2015-08-26 15:45:32 -0700 |
---|---|---|
committer | Junio C Hamano <gitster@pobox.com> | 2015-08-26 15:45:32 -0700 |
commit | b7d2a15b9f5cdbed7e8dbcc3fff0ea1ae92ed764 (patch) | |
tree | af0c7b8de03f8144f2b020e3bd273cb3cb88893a /builtin | |
parent | 788f2119666f644eb8477ea04253f6abfb3a35a4 (diff) | |
parent | 3ecc7040eff29fea0051df9faf21b0a73ee6d911 (diff) | |
download | git-b7d2a15b9f5cdbed7e8dbcc3fff0ea1ae92ed764.tar.gz |
Merge branch 'pt/am-builtin-abort-fix'
"git am" that was recently reimplemented in C had a performance
regression in "git am --abort" that goes back to the version before
an attempted (and failed) patch application.
* pt/am-builtin-abort-fix:
am --skip/--abort: merge HEAD/ORIG_HEAD tree into index
Diffstat (limited to 'builtin')
-rw-r--r-- | builtin/am.c | 49 |
1 files changed, 36 insertions, 13 deletions
diff --git a/builtin/am.c b/builtin/am.c index b9c62e3de3..5d34696097 100644 --- a/builtin/am.c +++ b/builtin/am.c @@ -1963,15 +1963,48 @@ static int fast_forward_to(struct tree *head, struct tree *remote, int reset) } /** + * 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 * `head` and `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; unsigned char index[GIT_SHA1_RAWSZ]; - struct pathspec pathspec; head_tree = parse_tree_indirect(head); if (!head_tree) @@ -1996,18 +2029,8 @@ static int clean_index(const unsigned char *head, const unsigned char *remote) if (fast_forward_to(index_tree, remote_tree, 0)) return -1; - memset(&pathspec, 0, sizeof(pathspec)); - - 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); + if (merge_tree(remote_tree)) return -1; - } - - if (write_locked_index(&the_index, lock_file, COMMIT_LOCK)) - die(_("unable to write new index file")); remove_branch_state(); |