From: "Elijah Newren via GitGitGadget" <gitgitgadget@gmail.com>
To: git@vger.kernel.org
Cc: Derrick Stolee <stolee@gmail.com>,
Elijah Newren <newren@gmail.com>,
Johannes Schindelin <Johannes.Schindelin@gmx.de>,
Elijah Newren <newren@gmail.com>
Subject: [PATCH v2 00/11] merge-ort: add basic rename detection
Date: Mon, 14 Dec 2020 16:21:29 +0000 [thread overview]
Message-ID: <pull.812.v2.git.1607962900.gitgitgadget@gmail.com> (raw)
In-Reply-To: <pull.812.git.1607542887.gitgitgadget@gmail.com>
This series builds on en/merge-ort-2 and adds basic rename detection to
merge-ort.
Changes since v1 (all due to feedback from Stolee's reviews):
* embedded struct rename_info directly in struct merge_options_internal (no
longer a pointer)
* expanded use of new enum merge_side and its new MERGE_BASE, MERGE_SIDE1,
MERGE_SIDE2 constants
* removed unnecessary secondary sort in compare_pairs()
* improved commit messages
* document p->score reuse with better comment(s)
* space around operators
Elijah Newren (11):
merge-ort: add basic data structures for handling renames
merge-ort: add initial outline for basic rename detection
merge-ort: implement detect_regular_renames()
merge-ort: implement compare_pairs() and collect_renames()
merge-ort: add basic outline for process_renames()
merge-ort: add implementation of both sides renaming identically
merge-ort: add implementation of both sides renaming differently
merge-ort: add implementation of rename collisions
merge-ort: add implementation of rename/delete conflicts
merge-ort: add implementation of normal rename handling
merge-ort: add implementation of type-changed rename handling
merge-ort.c | 445 ++++++++++++++++++++++++++++++++++++++++++++++++++--
1 file changed, 429 insertions(+), 16 deletions(-)
base-commit: c5a6f65527aa3b6f5d7cf25437a88d8727ab0646
Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-812%2Fnewren%2Fort-renames-v2
Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-812/newren/ort-renames-v2
Pull-Request: https://github.com/gitgitgadget/git/pull/812
Range-diff vs v1:
1: ef8f315f828 ! 1: 78621ca0788 merge-ort: add basic data structures for handling renames
@@ Commit message
Signed-off-by: Elijah Newren <newren@gmail.com>
## merge-ort.c ##
-@@
- #include "unpack-trees.h"
- #include "xdiff-interface.h"
+@@ merge-ort.c: enum merge_side {
+ MERGE_SIDE2 = 2
+ };
+struct rename_info {
+ /*
@@ merge-ort.c: struct merge_options_internal {
+ /*
+ * renames: various data relating to rename detection
+ */
-+ struct rename_info *renames;
++ struct rename_info renames;
+
/*
* current_dir_name: temporary var used in collect_merge_info_callback()
*
-@@ merge-ort.c: static void merge_start(struct merge_options *opt, struct merge_result *result)
-
- /* Initialization of opt->priv, our internal merge data */
- opt->priv = xcalloc(1, sizeof(*opt->priv));
-+ opt->priv->renames = xcalloc(1, sizeof(*opt->priv->renames));
-
- /*
- * Although we initialize opt->priv->paths with strdup_strings=0,
2: b9e0e1a60b9 ! 2: d846decf40b merge-ort: add initial outline for basic rename detection
@@ merge-ort.c: static int handle_content_merge(struct merge_options *opt,
{
- int clean = 1;
+ struct diff_queue_struct combined;
-+ struct rename_info *renames = opt->priv->renames;
++ struct rename_info *renames = &opt->priv->renames;
+ int s, clean = 1;
+
+ memset(&combined, 0, sizeof(combined));
+
-+ detect_regular_renames(opt, merge_base, side1, 1);
-+ detect_regular_renames(opt, merge_base, side2, 2);
++ detect_regular_renames(opt, merge_base, side1, MERGE_SIDE1);
++ detect_regular_renames(opt, merge_base, side2, MERGE_SIDE2);
+
+ ALLOC_GROW(combined.queue,
+ renames->pairs[1].nr + renames->pairs[2].nr,
+ combined.alloc);
-+ clean &= collect_renames(opt, &combined, 1);
-+ clean &= collect_renames(opt, &combined, 2);
++ clean &= collect_renames(opt, &combined, MERGE_SIDE1);
++ clean &= collect_renames(opt, &combined, MERGE_SIDE2);
+ QSORT(combined.queue, combined.nr, compare_pairs);
+
+ clean &= process_renames(opt, &combined);
+
+ /* Free memory for renames->pairs[] and combined */
-+ for (s = 1; s <= 2; s++) {
++ for (s = MERGE_SIDE1; s <= MERGE_SIDE2; s++) {
+ free(renames->pairs[s].queue);
+ DIFF_QUEUE_CLEAR(&renames->pairs[s]);
+ }
3: ba30bc8686e ! 3: 620fc64032d merge-ort: implement detect_regular_renames()
@@ Metadata
## Commit message ##
merge-ort: implement detect_regular_renames()
- Based heavily on merge-recursive's get_diffpairs() function.
+ Based heavily on merge-recursive's get_diffpairs() function, and also
+ includes the necessary paired call to diff_warn_rename_limit() so that
+ users will be warned if merge.renameLimit is not sufficiently large for
+ rename detection to run.
Signed-off-by: Elijah Newren <newren@gmail.com>
@@ merge-ort.c: static void detect_regular_renames(struct merge_options *opt,
{
- die("Not yet implemented.");
+ struct diff_options diff_opts;
-+ struct rename_info *renames = opt->priv->renames;
++ struct rename_info *renames = &opt->priv->renames;
+
+ repo_diff_setup(opt->repo, &diff_opts);
+ diff_opts.flags.recursive = 1;
@@ merge-ort.c: static void detect_regular_renames(struct merge_options *opt,
+ &diff_opts);
+ diffcore_std(&diff_opts);
+
-+ if (diff_opts.needed_rename_limit > opt->priv->renames->needed_limit)
-+ opt->priv->renames->needed_limit = diff_opts.needed_rename_limit;
++ if (diff_opts.needed_rename_limit > renames->needed_limit)
++ renames->needed_limit = diff_opts.needed_rename_limit;
+
+ renames->pairs[side_index] = diff_queued_diff;
+
@@ merge-ort.c: void merge_switch_to_result(struct merge_options *opt,
+
+ /* Also include needed rename limit adjustment now */
+ diff_warn_rename_limit("merge.renamelimit",
-+ opti->renames->needed_limit, 0);
++ opti->renames.needed_limit, 0);
}
merge_finalize(opt, result);
4: 207bb9a837c ! 4: 9382dc4d50b merge-ort: implement compare_pairs() and collect_renames()
@@ merge-ort.c: static int process_renames(struct merge_options *opt,
+ const struct diff_filepair *a = *((const struct diff_filepair **)a_);
+ const struct diff_filepair *b = *((const struct diff_filepair **)b_);
+
-+ int cmp = strcmp(a->one->path, b->one->path);
-+ if (cmp)
-+ return cmp;
-+ return a->score - b->score;
++ return strcmp(a->one->path, b->one->path);
}
/* Call diffcore_rename() to compute which files have changed on given side */
@@ merge-ort.c: static int collect_renames(struct merge_options *opt,
- die("Not yet implemented.");
+ int i, clean = 1;
+ struct diff_queue_struct *side_pairs;
-+ struct rename_info *renames = opt->priv->renames;
++ struct rename_info *renames = &opt->priv->renames;
+
+ side_pairs = &renames->pairs[side_index];
+
@@ merge-ort.c: static int collect_renames(struct merge_options *opt,
+ diff_free_filepair(p);
+ continue;
+ }
++
++ /*
++ * p->score comes back from diffcore_rename_extended() with
++ * the similarity of the renamed file. The similarity is
++ * was used to determine that the two files were related
++ * and are a rename, which we have already used, but beyond
++ * that we have no use for the similarity. So p->score is
++ * now irrelevant. However, process_renames() will need to
++ * know which side of the merge this rename was associated
++ * with, so overwrite p->score with that value.
++ */
+ p->score = side_index;
+ result->queue[result->nr++] = p;
+ }
5: 35b070b9b7c ! 5: d20fab8d403 merge-ort: add basic outline for process_renames()
@@ merge-ort.c: static int handle_content_merge(struct merge_options *opt,
+ * diff_filepairs have copies of pathnames, thus we have to
+ * use standard 'strcmp()' (negated) instead of '=='.
+ */
-+ if (i+1 < renames->nr &&
++ if (i + 1 < renames->nr &&
+ !strcmp(oldpath, renames->queue[i+1]->one->path)) {
+ /* Handle rename/rename(1to2) or rename/rename(1to1) */
+ const char *pathnames[3];
@@ merge-ort.c: static int handle_content_merge(struct merge_options *opt,
+
+ VERIFY_CI(oldinfo);
+ VERIFY_CI(newinfo);
-+ target_index = pair->score; /* from append_rename_pairs() */
++ target_index = pair->score; /* from collect_renames() */
+ assert(target_index == 1 || target_index == 2);
-+ other_source_index = 3-target_index;
++ other_source_index = 3 - target_index;
+ old_sidemask = (1 << other_source_index); /* 2 or 4 */
+ source_deleted = (oldinfo->filemask == 1);
+ collision = ((newinfo->filemask & old_sidemask) != 0);
6: 9c79b9f4a09 ! 6: 15fff3dd0c4 merge-ort: add implementation of both sides renaming identically
@@ merge-ort.c: static int process_renames(struct merge_options *opt,
+ assert(side1 == side2);
+ memcpy(&side1->stages[0], &base->stages[0],
+ sizeof(merged));
-+ side1->filemask |= (1 << 0);
++ side1->filemask |= (1 << MERGE_BASE);
+ /* Mark base as resolved by removal */
+ base->merged.is_null = 1;
+ base->merged.clean = 1;
7: d4595397052 ! 7: d00e26be784 merge-ort: add implementation of both sides renaming differently
@@ merge-ort.c: static int process_renames(struct merge_options *opt,
+ &merged);
+ if (!clean_merge &&
+ merged.mode == side1->stages[1].mode &&
-+ oideq(&merged.oid, &side1->stages[1].oid)) {
++ oideq(&merged.oid, &side1->stages[1].oid))
+ was_binary_blob = 1;
-+ }
+ memcpy(&side1->stages[1], &merged, sizeof(merged));
+ if (was_binary_blob) {
+ /*
8: ab15f85f698 = 8: edd610321a0 merge-ort: add implementation of rename collisions
9: c069d34b15f ! 9: f017534243c merge-ort: add implementation of rename/delete conflicts
@@ merge-ort.c: static int process_renames(struct merge_options *opt,
+ */
+ memcpy(&newinfo->stages[0], &oldinfo->stages[0],
+ sizeof(newinfo->stages[0]));
-+ newinfo->filemask |= (1 << 0);
++ newinfo->filemask |= (1 << MERGE_BASE);
+ newinfo->pathnames[0] = oldpath;
if (type_changed) {
/* rename vs. typechange */
10: 14baa5874af = 10: 22cb7110261 merge-ort: add implementation of normal rename handling
11: 476553e2b20 = 11: ff09ddb9caf merge-ort: add implementation of type-changed rename handling
--
gitgitgadget
next prev parent reply other threads:[~2020-12-14 16:23 UTC|newest]
Thread overview: 65+ messages / expand[flat|nested] mbox.gz Atom feed top
2020-12-09 19:41 [PATCH 00/11] merge-ort: add basic rename detection Elijah Newren via GitGitGadget
2020-12-09 19:41 ` [PATCH 01/11] merge-ort: add basic data structures for handling renames Elijah Newren via GitGitGadget
2020-12-11 2:03 ` Derrick Stolee
2020-12-11 9:41 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 02/11] merge-ort: add initial outline for basic rename detection Elijah Newren via GitGitGadget
2020-12-11 2:39 ` Derrick Stolee
2020-12-11 9:40 ` Elijah Newren
2020-12-13 7:47 ` Elijah Newren
2020-12-14 14:33 ` Derrick Stolee
2020-12-14 15:42 ` Johannes Schindelin
2020-12-14 16:11 ` Elijah Newren
2020-12-14 16:50 ` Johannes Schindelin
2020-12-14 17:35 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 03/11] merge-ort: implement detect_regular_renames() Elijah Newren via GitGitGadget
2020-12-11 2:54 ` Derrick Stolee
2020-12-11 17:38 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 04/11] merge-ort: implement compare_pairs() and collect_renames() Elijah Newren via GitGitGadget
2020-12-11 3:00 ` Derrick Stolee
2020-12-11 18:43 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 05/11] merge-ort: add basic outline for process_renames() Elijah Newren via GitGitGadget
2020-12-11 3:24 ` Derrick Stolee
2020-12-11 20:03 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 06/11] merge-ort: add implementation of both sides renaming identically Elijah Newren via GitGitGadget
2020-12-11 3:32 ` Derrick Stolee
2020-12-09 19:41 ` [PATCH 07/11] merge-ort: add implementation of both sides renaming differently Elijah Newren via GitGitGadget
2020-12-11 3:39 ` Derrick Stolee
2020-12-11 21:56 ` Elijah Newren
2020-12-09 19:41 ` [PATCH 08/11] merge-ort: add implementation of rename collisions Elijah Newren via GitGitGadget
2020-12-09 19:41 ` [PATCH 09/11] merge-ort: add implementation of rename/delete conflicts Elijah Newren via GitGitGadget
2020-12-09 19:41 ` [PATCH 10/11] merge-ort: add implementation of normal rename handling Elijah Newren via GitGitGadget
2020-12-09 19:41 ` [PATCH 11/11] merge-ort: add implementation of type-changed " Elijah Newren via GitGitGadget
2020-12-14 16:21 ` Elijah Newren via GitGitGadget [this message]
2020-12-14 16:21 ` [PATCH v2 01/11] merge-ort: add basic data structures for handling renames Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 02/11] merge-ort: add initial outline for basic rename detection Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 03/11] merge-ort: implement detect_regular_renames() Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 04/11] merge-ort: implement compare_pairs() and collect_renames() Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 05/11] merge-ort: add basic outline for process_renames() Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 06/11] merge-ort: add implementation of both sides renaming identically Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 07/11] merge-ort: add implementation of both sides renaming differently Elijah Newren via GitGitGadget
2020-12-14 16:21 ` [PATCH v2 08/11] merge-ort: add implementation of rename collisions Elijah Newren via GitGitGadget
2020-12-15 14:09 ` Derrick Stolee
2020-12-15 16:56 ` Elijah Newren
2020-12-14 16:21 ` [PATCH v2 09/11] merge-ort: add implementation of rename/delete conflicts Elijah Newren via GitGitGadget
2020-12-15 14:23 ` Derrick Stolee
2020-12-15 17:07 ` Elijah Newren
2020-12-15 14:27 ` Derrick Stolee
2020-12-14 16:21 ` [PATCH v2 10/11] merge-ort: add implementation of normal rename handling Elijah Newren via GitGitGadget
2020-12-15 14:27 ` Derrick Stolee
2020-12-14 16:21 ` [PATCH v2 11/11] merge-ort: add implementation of type-changed " Elijah Newren via GitGitGadget
2020-12-15 14:31 ` Derrick Stolee
2020-12-15 17:11 ` Elijah Newren
2020-12-15 14:34 ` [PATCH v2 00/11] merge-ort: add basic rename detection Derrick Stolee
2020-12-15 22:09 ` Junio C Hamano
2020-12-15 18:27 ` [PATCH v3 " Elijah Newren via GitGitGadget
2020-12-15 18:27 ` [PATCH v3 01/11] merge-ort: add basic data structures for handling renames Elijah Newren via GitGitGadget
2020-12-15 18:27 ` [PATCH v3 02/11] merge-ort: add initial outline for basic rename detection Elijah Newren via GitGitGadget
2020-12-15 18:27 ` [PATCH v3 03/11] merge-ort: implement detect_regular_renames() Elijah Newren via GitGitGadget
2020-12-15 18:27 ` [PATCH v3 04/11] merge-ort: implement compare_pairs() and collect_renames() Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 05/11] merge-ort: add basic outline for process_renames() Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 06/11] merge-ort: add implementation of both sides renaming identically Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 07/11] merge-ort: add implementation of both sides renaming differently Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 08/11] merge-ort: add implementation of rename/delete conflicts Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 09/11] merge-ort: add implementation of rename collisions Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 10/11] merge-ort: add implementation of normal rename handling Elijah Newren via GitGitGadget
2020-12-15 18:28 ` [PATCH v3 11/11] merge-ort: add implementation of type-changed " Elijah Newren via GitGitGadget
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=pull.812.v2.git.1607962900.gitgitgadget@gmail.com \
--to=gitgitgadget@gmail.com \
--cc=Johannes.Schindelin@gmx.de \
--cc=git@vger.kernel.org \
--cc=newren@gmail.com \
--cc=stolee@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).