# HG changeset patch # User Matt Mackall # Date 2016-06-02 22:09:06 # Node ID 66dbdd3cc2b92fc6a36a5d38793571027a48204b # Parent 9b4f0ad02f5182498f196fe9c603b4e0e39664ad bdiff: extend matches across popular lines For very large diffs that have large numbers of identical lines (JSON dumps) that also have large blocks of identical text, bdiff could become confused about which block matches which because it can only match very limited regions. The result is very large diffs for small sets of edits. The earlier recursion rebalancing fix made this behavior more frequent because it's now more prone to match block 1 to block 2. One frequent user of large JSON files reported being unable to pass the resulting diffs through their code review system. Prior to this change, bdiff would calculate the length of a match at (i, j) as 1 + length found at (i-1, j-1). With large number of popular (ignored) lines, this often meant matches couldn't be extended backwards at all and thus all matching regions were very small. Disabling the popularity threshold is not an option because it brings back quadratic behavior. Instead, we extend a match backwards until we either found a previously discovered match or we find a mismatching line. This thus successfully bridges over any popular lines inside and before a matching region. The larger regions then significant reduce the probability of confusion. diff --git a/mercurial/bdiff.c b/mercurial/bdiff.c --- a/mercurial/bdiff.c +++ b/mercurial/bdiff.c @@ -166,10 +166,17 @@ static int longest_match(struct line *a, /* loop through all lines match a[i] in b */ for (; j >= b1; j = b[j].n) { /* does this extend an earlier match? */ - if (i > a1 && j > b1 && pos[j - 1].pos == i - 1) - k = pos[j - 1].len + 1; - else - k = 1; + for (k = 1; j - k >= b1 && i - k >= a1; k++) { + /* reached an earlier match? */ + if (pos[j - k].pos == i - k) { + k += pos[j - k].len; + break; + } + /* previous line mismatch? */ + if (a[i - k].e != b[j - k].e) + break; + } + pos[j].pos = i; pos[j].len = k;