git
1/*
2* LibXDiff by Davide Libenzi ( File Differential Library )
3* Copyright (C) 2003 Davide Libenzi
4*
5* This library is free software; you can redistribute it and/or
6* modify it under the terms of the GNU Lesser General Public
7* License as published by the Free Software Foundation; either
8* version 2.1 of the License, or (at your option) any later version.
9*
10* This library is distributed in the hope that it will be useful,
11* but WITHOUT ANY WARRANTY; without even the implied warranty of
12* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13* Lesser General Public License for more details.
14*
15* You should have received a copy of the GNU Lesser General Public
16* License along with this library; if not, see
17* <http://www.gnu.org/licenses/>.
18*
19* Davide Libenzi <davidel@xmailserver.org>
20*
21*/
22
23#include "xinclude.h"24
25#define XDL_MAX_COST_MIN 25626#define XDL_HEUR_MIN_COST 25627#define XDL_LINE_MAX (long)((1UL << (CHAR_BIT * sizeof(long) - 1)) - 1)28#define XDL_SNAKE_CNT 2029#define XDL_K_HEUR 430
31typedef struct s_xdpsplit {32long i1, i2;33int min_lo, min_hi;34} xdpsplit_t;35
36/*
37* See "An O(ND) Difference Algorithm and its Variations", by Eugene Myers.
38* Basically considers a "box" (off1, off2, lim1, lim2) and scan from both
39* the forward diagonal starting from (off1, off2) and the backward diagonal
40* starting from (lim1, lim2). If the K values on the same diagonal crosses
41* returns the furthest point of reach. We might encounter expensive edge cases
42* using this algorithm, so a little bit of heuristic is needed to cut the
43* search and to return a suboptimal point.
44*/
45static long xdl_split(unsigned long const *ha1, long off1, long lim1,46unsigned long const *ha2, long off2, long lim2,47long *kvdf, long *kvdb, int need_min, xdpsplit_t *spl,48xdalgoenv_t *xenv) {49long dmin = off1 - lim2, dmax = lim1 - off2;50long fmid = off1 - off2, bmid = lim1 - lim2;51long odd = (fmid - bmid) & 1;52long fmin = fmid, fmax = fmid;53long bmin = bmid, bmax = bmid;54long ec, d, i1, i2, prev1, best, dd, v, k;55
56/*57* Set initial diagonal values for both forward and backward path.
58*/
59kvdf[fmid] = off1;60kvdb[bmid] = lim1;61
62for (ec = 1;; ec++) {63int got_snake = 0;64
65/*66* We need to extend the diagonal "domain" by one. If the next
67* values exits the box boundaries we need to change it in the
68* opposite direction because (max - min) must be a power of
69* two.
70*
71* Also we initialize the external K value to -1 so that we can
72* avoid extra conditions in the check inside the core loop.
73*/
74if (fmin > dmin)75kvdf[--fmin - 1] = -1;76else77++fmin;78if (fmax < dmax)79kvdf[++fmax + 1] = -1;80else81--fmax;82
83for (d = fmax; d >= fmin; d -= 2) {84if (kvdf[d - 1] >= kvdf[d + 1])85i1 = kvdf[d - 1] + 1;86else87i1 = kvdf[d + 1];88prev1 = i1;89i2 = i1 - d;90for (; i1 < lim1 && i2 < lim2 && ha1[i1] == ha2[i2]; i1++, i2++);91if (i1 - prev1 > xenv->snake_cnt)92got_snake = 1;93kvdf[d] = i1;94if (odd && bmin <= d && d <= bmax && kvdb[d] <= i1) {95spl->i1 = i1;96spl->i2 = i2;97spl->min_lo = spl->min_hi = 1;98return ec;99}100}101
102/*103* We need to extend the diagonal "domain" by one. If the next
104* values exits the box boundaries we need to change it in the
105* opposite direction because (max - min) must be a power of
106* two.
107*
108* Also we initialize the external K value to -1 so that we can
109* avoid extra conditions in the check inside the core loop.
110*/
111if (bmin > dmin)112kvdb[--bmin - 1] = XDL_LINE_MAX;113else114++bmin;115if (bmax < dmax)116kvdb[++bmax + 1] = XDL_LINE_MAX;117else118--bmax;119
120for (d = bmax; d >= bmin; d -= 2) {121if (kvdb[d - 1] < kvdb[d + 1])122i1 = kvdb[d - 1];123else124i1 = kvdb[d + 1] - 1;125prev1 = i1;126i2 = i1 - d;127for (; i1 > off1 && i2 > off2 && ha1[i1 - 1] == ha2[i2 - 1]; i1--, i2--);128if (prev1 - i1 > xenv->snake_cnt)129got_snake = 1;130kvdb[d] = i1;131if (!odd && fmin <= d && d <= fmax && i1 <= kvdf[d]) {132spl->i1 = i1;133spl->i2 = i2;134spl->min_lo = spl->min_hi = 1;135return ec;136}137}138
139if (need_min)140continue;141
142/*143* If the edit cost is above the heuristic trigger and if
144* we got a good snake, we sample current diagonals to see
145* if some of them have reached an "interesting" path. Our
146* measure is a function of the distance from the diagonal
147* corner (i1 + i2) penalized with the distance from the
148* mid diagonal itself. If this value is above the current
149* edit cost times a magic factor (XDL_K_HEUR) we consider
150* it interesting.
151*/
152if (got_snake && ec > xenv->heur_min) {153for (best = 0, d = fmax; d >= fmin; d -= 2) {154dd = d > fmid ? d - fmid: fmid - d;155i1 = kvdf[d];156i2 = i1 - d;157v = (i1 - off1) + (i2 - off2) - dd;158
159if (v > XDL_K_HEUR * ec && v > best &&160off1 + xenv->snake_cnt <= i1 && i1 < lim1 &&161off2 + xenv->snake_cnt <= i2 && i2 < lim2) {162for (k = 1; ha1[i1 - k] == ha2[i2 - k]; k++)163if (k == xenv->snake_cnt) {164best = v;165spl->i1 = i1;166spl->i2 = i2;167break;168}169}170}171if (best > 0) {172spl->min_lo = 1;173spl->min_hi = 0;174return ec;175}176
177for (best = 0, d = bmax; d >= bmin; d -= 2) {178dd = d > bmid ? d - bmid: bmid - d;179i1 = kvdb[d];180i2 = i1 - d;181v = (lim1 - i1) + (lim2 - i2) - dd;182
183if (v > XDL_K_HEUR * ec && v > best &&184off1 < i1 && i1 <= lim1 - xenv->snake_cnt &&185off2 < i2 && i2 <= lim2 - xenv->snake_cnt) {186for (k = 0; ha1[i1 + k] == ha2[i2 + k]; k++)187if (k == xenv->snake_cnt - 1) {188best = v;189spl->i1 = i1;190spl->i2 = i2;191break;192}193}194}195if (best > 0) {196spl->min_lo = 0;197spl->min_hi = 1;198return ec;199}200}201
202/*203* Enough is enough. We spent too much time here and now we
204* collect the furthest reaching path using the (i1 + i2)
205* measure.
206*/
207if (ec >= xenv->mxcost) {208long fbest, fbest1, bbest, bbest1;209
210fbest = fbest1 = -1;211for (d = fmax; d >= fmin; d -= 2) {212i1 = XDL_MIN(kvdf[d], lim1);213i2 = i1 - d;214if (lim2 < i2)215i1 = lim2 + d, i2 = lim2;216if (fbest < i1 + i2) {217fbest = i1 + i2;218fbest1 = i1;219}220}221
222bbest = bbest1 = XDL_LINE_MAX;223for (d = bmax; d >= bmin; d -= 2) {224i1 = XDL_MAX(off1, kvdb[d]);225i2 = i1 - d;226if (i2 < off2)227i1 = off2 + d, i2 = off2;228if (i1 + i2 < bbest) {229bbest = i1 + i2;230bbest1 = i1;231}232}233
234if ((lim1 + lim2) - bbest < fbest - (off1 + off2)) {235spl->i1 = fbest1;236spl->i2 = fbest - fbest1;237spl->min_lo = 1;238spl->min_hi = 0;239} else {240spl->i1 = bbest1;241spl->i2 = bbest - bbest1;242spl->min_lo = 0;243spl->min_hi = 1;244}245return ec;246}247}248}
249
250
251/*
252* Rule: "Divide et Impera" (divide & conquer). Recursively split the box in
253* sub-boxes by calling the box splitting function. Note that the real job
254* (marking changed lines) is done in the two boundary reaching checks.
255*/
256int xdl_recs_cmp(diffdata_t *dd1, long off1, long lim1,257diffdata_t *dd2, long off2, long lim2,258long *kvdf, long *kvdb, int need_min, xdalgoenv_t *xenv) {259unsigned long const *ha1 = dd1->ha, *ha2 = dd2->ha;260
261/*262* Shrink the box by walking through each diagonal snake (SW and NE).
263*/
264for (; off1 < lim1 && off2 < lim2 && ha1[off1] == ha2[off2]; off1++, off2++);265for (; off1 < lim1 && off2 < lim2 && ha1[lim1 - 1] == ha2[lim2 - 1]; lim1--, lim2--);266
267/*268* If one dimension is empty, then all records on the other one must
269* be obviously changed.
270*/
271if (off1 == lim1) {272char *rchg2 = dd2->rchg;273long *rindex2 = dd2->rindex;274
275for (; off2 < lim2; off2++)276rchg2[rindex2[off2]] = 1;277} else if (off2 == lim2) {278char *rchg1 = dd1->rchg;279long *rindex1 = dd1->rindex;280
281for (; off1 < lim1; off1++)282rchg1[rindex1[off1]] = 1;283} else {284xdpsplit_t spl;285spl.i1 = spl.i2 = 0;286
287/*288* Divide ...
289*/
290if (xdl_split(ha1, off1, lim1, ha2, off2, lim2, kvdf, kvdb,291need_min, &spl, xenv) < 0) {292
293return -1;294}295
296/*297* ... et Impera.
298*/
299if (xdl_recs_cmp(dd1, off1, spl.i1, dd2, off2, spl.i2,300kvdf, kvdb, spl.min_lo, xenv) < 0 ||301xdl_recs_cmp(dd1, spl.i1, lim1, dd2, spl.i2, lim2,302kvdf, kvdb, spl.min_hi, xenv) < 0) {303
304return -1;305}306}307
308return 0;309}
310
311
312int xdl_do_diff(mmfile_t *mf1, mmfile_t *mf2, xpparam_t const *xpp,313xdfenv_t *xe) {314long ndiags;315long *kvd, *kvdf, *kvdb;316xdalgoenv_t xenv;317diffdata_t dd1, dd2;318int res;319
320if (xdl_prepare_env(mf1, mf2, xpp, xe) < 0)321return -1;322
323if (XDF_DIFF_ALG(xpp->flags) == XDF_PATIENCE_DIFF) {324res = xdl_do_patience_diff(xpp, xe);325goto out;326}327
328if (XDF_DIFF_ALG(xpp->flags) == XDF_HISTOGRAM_DIFF) {329res = xdl_do_histogram_diff(xpp, xe);330goto out;331}332
333/*334* Allocate and setup K vectors to be used by the differential
335* algorithm.
336*
337* One is to store the forward path and one to store the backward path.
338*/
339ndiags = xe->xdf1.nreff + xe->xdf2.nreff + 3;340if (!XDL_ALLOC_ARRAY(kvd, 2 * ndiags + 2)) {341
342xdl_free_env(xe);343return -1;344}345kvdf = kvd;346kvdb = kvdf + ndiags;347kvdf += xe->xdf2.nreff + 1;348kvdb += xe->xdf2.nreff + 1;349
350xenv.mxcost = xdl_bogosqrt(ndiags);351if (xenv.mxcost < XDL_MAX_COST_MIN)352xenv.mxcost = XDL_MAX_COST_MIN;353xenv.snake_cnt = XDL_SNAKE_CNT;354xenv.heur_min = XDL_HEUR_MIN_COST;355
356dd1.nrec = xe->xdf1.nreff;357dd1.ha = xe->xdf1.ha;358dd1.rchg = xe->xdf1.rchg;359dd1.rindex = xe->xdf1.rindex;360dd2.nrec = xe->xdf2.nreff;361dd2.ha = xe->xdf2.ha;362dd2.rchg = xe->xdf2.rchg;363dd2.rindex = xe->xdf2.rindex;364
365res = xdl_recs_cmp(&dd1, 0, dd1.nrec, &dd2, 0, dd2.nrec,366kvdf, kvdb, (xpp->flags & XDF_NEED_MINIMAL) != 0,367&xenv);368xdl_free(kvd);369out:370if (res < 0)371xdl_free_env(xe);372
373return res;374}
375
376
377static xdchange_t *xdl_add_change(xdchange_t *xscr, long i1, long i2, long chg1, long chg2) {378xdchange_t *xch;379
380if (!(xch = (xdchange_t *) xdl_malloc(sizeof(xdchange_t))))381return NULL;382
383xch->next = xscr;384xch->i1 = i1;385xch->i2 = i2;386xch->chg1 = chg1;387xch->chg2 = chg2;388xch->ignore = 0;389
390return xch;391}
392
393
394static int recs_match(xrecord_t *rec1, xrecord_t *rec2)395{
396return (rec1->ha == rec2->ha);397}
398
399/*
400* If a line is indented more than this, get_indent() just returns this value.
401* This avoids having to do absurd amounts of work for data that are not
402* human-readable text, and also ensures that the output of get_indent fits
403* within an int.
404*/
405#define MAX_INDENT 200406
407/*
408* Return the amount of indentation of the specified line, treating TAB as 8
409* columns. Return -1 if line is empty or contains only whitespace. Clamp the
410* output value at MAX_INDENT.
411*/
412static int get_indent(xrecord_t *rec)413{
414long i;415int ret = 0;416
417for (i = 0; i < rec->size; i++) {418char c = rec->ptr[i];419
420if (!XDL_ISSPACE(c))421return ret;422else if (c == ' ')423ret += 1;424else if (c == '\t')425ret += 8 - ret % 8;426/* ignore other whitespace characters */427
428if (ret >= MAX_INDENT)429return MAX_INDENT;430}431
432/* The line contains only whitespace. */433return -1;434}
435
436/*
437* If more than this number of consecutive blank rows are found, just return
438* this value. This avoids requiring O(N^2) work for pathological cases, and
439* also ensures that the output of score_split fits in an int.
440*/
441#define MAX_BLANKS 20442
443/* Characteristics measured about a hypothetical split position. */
444struct split_measurement {445/*446* Is the split at the end of the file (aside from any blank lines)?
447*/
448int end_of_file;449
450/*451* How much is the line immediately following the split indented (or -1
452* if the line is blank):
453*/
454int indent;455
456/*457* How many consecutive lines above the split are blank?
458*/
459int pre_blank;460
461/*462* How much is the nearest non-blank line above the split indented (or
463* -1 if there is no such line)?
464*/
465int pre_indent;466
467/*468* How many lines after the line following the split are blank?
469*/
470int post_blank;471
472/*473* How much is the nearest non-blank line after the line following the
474* split indented (or -1 if there is no such line)?
475*/
476int post_indent;477};478
479struct split_score {480/* The effective indent of this split (smaller is preferred). */481int effective_indent;482
483/* Penalty for this split (smaller is preferred). */484int penalty;485};486
487/*
488* Fill m with information about a hypothetical split of xdf above line split.
489*/
490static void measure_split(const xdfile_t *xdf, long split,491struct split_measurement *m)492{
493long i;494
495if (split >= xdf->nrec) {496m->end_of_file = 1;497m->indent = -1;498} else {499m->end_of_file = 0;500m->indent = get_indent(xdf->recs[split]);501}502
503m->pre_blank = 0;504m->pre_indent = -1;505for (i = split - 1; i >= 0; i--) {506m->pre_indent = get_indent(xdf->recs[i]);507if (m->pre_indent != -1)508break;509m->pre_blank += 1;510if (m->pre_blank == MAX_BLANKS) {511m->pre_indent = 0;512break;513}514}515
516m->post_blank = 0;517m->post_indent = -1;518for (i = split + 1; i < xdf->nrec; i++) {519m->post_indent = get_indent(xdf->recs[i]);520if (m->post_indent != -1)521break;522m->post_blank += 1;523if (m->post_blank == MAX_BLANKS) {524m->post_indent = 0;525break;526}527}528}
529
530/*
531* The empirically-determined weight factors used by score_split() below.
532* Larger values means that the position is a less favorable place to split.
533*
534* Note that scores are only ever compared against each other, so multiplying
535* all of these weight/penalty values by the same factor wouldn't change the
536* heuristic's behavior. Still, we need to set that arbitrary scale *somehow*.
537* In practice, these numbers are chosen to be large enough that they can be
538* adjusted relative to each other with sufficient precision despite using
539* integer math.
540*/
541
542/* Penalty if there are no non-blank lines before the split */
543#define START_OF_FILE_PENALTY 1544
545/* Penalty if there are no non-blank lines after the split */
546#define END_OF_FILE_PENALTY 21547
548/* Multiplier for the number of blank lines around the split */
549#define TOTAL_BLANK_WEIGHT (-30)550
551/* Multiplier for the number of blank lines after the split */
552#define POST_BLANK_WEIGHT 6553
554/*
555* Penalties applied if the line is indented more than its predecessor
556*/
557#define RELATIVE_INDENT_PENALTY (-4)558#define RELATIVE_INDENT_WITH_BLANK_PENALTY 10559
560/*
561* Penalties applied if the line is indented less than both its predecessor and
562* its successor
563*/
564#define RELATIVE_OUTDENT_PENALTY 24565#define RELATIVE_OUTDENT_WITH_BLANK_PENALTY 17566
567/*
568* Penalties applied if the line is indented less than its predecessor but not
569* less than its successor
570*/
571#define RELATIVE_DEDENT_PENALTY 23572#define RELATIVE_DEDENT_WITH_BLANK_PENALTY 17573
574/*
575* We only consider whether the sum of the effective indents for splits are
576* less than (-1), equal to (0), or greater than (+1) each other. The resulting
577* value is multiplied by the following weight and combined with the penalty to
578* determine the better of two scores.
579*/
580#define INDENT_WEIGHT 60581
582/*
583* How far do we slide a hunk at most?
584*/
585#define INDENT_HEURISTIC_MAX_SLIDING 100586
587/*
588* Compute a badness score for the hypothetical split whose measurements are
589* stored in m. The weight factors were determined empirically using the tools
590* and corpus described in
591*
592* https://github.com/mhagger/diff-slider-tools
593*
594* Also see that project if you want to improve the weights based on, for
595* example, a larger or more diverse corpus.
596*/
597static void score_add_split(const struct split_measurement *m, struct split_score *s)598{
599/*600* A place to accumulate penalty factors (positive makes this index more
601* favored):
602*/
603int post_blank, total_blank, indent, any_blanks;604
605if (m->pre_indent == -1 && m->pre_blank == 0)606s->penalty += START_OF_FILE_PENALTY;607
608if (m->end_of_file)609s->penalty += END_OF_FILE_PENALTY;610
611/*612* Set post_blank to the number of blank lines following the split,
613* including the line immediately after the split:
614*/
615post_blank = (m->indent == -1) ? 1 + m->post_blank : 0;616total_blank = m->pre_blank + post_blank;617
618/* Penalties based on nearby blank lines: */619s->penalty += TOTAL_BLANK_WEIGHT * total_blank;620s->penalty += POST_BLANK_WEIGHT * post_blank;621
622if (m->indent != -1)623indent = m->indent;624else625indent = m->post_indent;626
627any_blanks = (total_blank != 0);628
629/* Note that the effective indent is -1 at the end of the file: */630s->effective_indent += indent;631
632if (indent == -1) {633/* No additional adjustments needed. */634} else if (m->pre_indent == -1) {635/* No additional adjustments needed. */636} else if (indent > m->pre_indent) {637/*638* The line is indented more than its predecessor.
639*/
640s->penalty += any_blanks ?641RELATIVE_INDENT_WITH_BLANK_PENALTY :642RELATIVE_INDENT_PENALTY;643} else if (indent == m->pre_indent) {644/*645* The line has the same indentation level as its predecessor.
646* No additional adjustments needed.
647*/
648} else {649/*650* The line is indented less than its predecessor. It could be
651* the block terminator of the previous block, but it could
652* also be the start of a new block (e.g., an "else" block, or
653* maybe the previous block didn't have a block terminator).
654* Try to distinguish those cases based on what comes next:
655*/
656if (m->post_indent != -1 && m->post_indent > indent) {657/*658* The following line is indented more. So it is likely
659* that this line is the start of a block.
660*/
661s->penalty += any_blanks ?662RELATIVE_OUTDENT_WITH_BLANK_PENALTY :663RELATIVE_OUTDENT_PENALTY;664} else {665/*666* That was probably the end of a block.
667*/
668s->penalty += any_blanks ?669RELATIVE_DEDENT_WITH_BLANK_PENALTY :670RELATIVE_DEDENT_PENALTY;671}672}673}
674
675static int score_cmp(struct split_score *s1, struct split_score *s2)676{
677/* -1 if s1.effective_indent < s2->effective_indent, etc. */678int cmp_indents = ((s1->effective_indent > s2->effective_indent) -679(s1->effective_indent < s2->effective_indent));680
681return INDENT_WEIGHT * cmp_indents + (s1->penalty - s2->penalty);682}
683
684/*
685* Represent a group of changed lines in an xdfile_t (i.e., a contiguous group
686* of lines that was inserted or deleted from the corresponding version of the
687* file). We consider there to be such a group at the beginning of the file, at
688* the end of the file, and between any two unchanged lines, though most such
689* groups will usually be empty.
690*
691* If the first line in a group is equal to the line following the group, then
692* the group can be slid down. Similarly, if the last line in a group is equal
693* to the line preceding the group, then the group can be slid up. See
694* group_slide_down() and group_slide_up().
695*
696* Note that loops that are testing for changed lines in xdf->rchg do not need
697* index bounding since the array is prepared with a zero at position -1 and N.
698*/
699struct xdlgroup {700/*701* The index of the first changed line in the group, or the index of
702* the unchanged line above which the (empty) group is located.
703*/
704long start;705
706/*707* The index of the first unchanged line after the group. For an empty
708* group, end is equal to start.
709*/
710long end;711};712
713/*
714* Initialize g to point at the first group in xdf.
715*/
716static void group_init(xdfile_t *xdf, struct xdlgroup *g)717{
718g->start = g->end = 0;719while (xdf->rchg[g->end])720g->end++;721}
722
723/*
724* Move g to describe the next (possibly empty) group in xdf and return 0. If g
725* is already at the end of the file, do nothing and return -1.
726*/
727static inline int group_next(xdfile_t *xdf, struct xdlgroup *g)728{
729if (g->end == xdf->nrec)730return -1;731
732g->start = g->end + 1;733for (g->end = g->start; xdf->rchg[g->end]; g->end++)734;735
736return 0;737}
738
739/*
740* Move g to describe the previous (possibly empty) group in xdf and return 0.
741* If g is already at the beginning of the file, do nothing and return -1.
742*/
743static inline int group_previous(xdfile_t *xdf, struct xdlgroup *g)744{
745if (g->start == 0)746return -1;747
748g->end = g->start - 1;749for (g->start = g->end; xdf->rchg[g->start - 1]; g->start--)750;751
752return 0;753}
754
755/*
756* If g can be slid toward the end of the file, do so, and if it bumps into a
757* following group, expand this group to include it. Return 0 on success or -1
758* if g cannot be slid down.
759*/
760static int group_slide_down(xdfile_t *xdf, struct xdlgroup *g)761{
762if (g->end < xdf->nrec &&763recs_match(xdf->recs[g->start], xdf->recs[g->end])) {764xdf->rchg[g->start++] = 0;765xdf->rchg[g->end++] = 1;766
767while (xdf->rchg[g->end])768g->end++;769
770return 0;771} else {772return -1;773}774}
775
776/*
777* If g can be slid toward the beginning of the file, do so, and if it bumps
778* into a previous group, expand this group to include it. Return 0 on success
779* or -1 if g cannot be slid up.
780*/
781static int group_slide_up(xdfile_t *xdf, struct xdlgroup *g)782{
783if (g->start > 0 &&784recs_match(xdf->recs[g->start - 1], xdf->recs[g->end - 1])) {785xdf->rchg[--g->start] = 1;786xdf->rchg[--g->end] = 0;787
788while (xdf->rchg[g->start - 1])789g->start--;790
791return 0;792} else {793return -1;794}795}
796
797/*
798* Move back and forward change groups for a consistent and pretty diff output.
799* This also helps in finding joinable change groups and reducing the diff
800* size.
801*/
802int xdl_change_compact(xdfile_t *xdf, xdfile_t *xdfo, long flags) {803struct xdlgroup g, go;804long earliest_end, end_matching_other;805long groupsize;806
807group_init(xdf, &g);808group_init(xdfo, &go);809
810while (1) {811/*812* If the group is empty in the to-be-compacted file, skip it:
813*/
814if (g.end == g.start)815goto next;816
817/*818* Now shift the change up and then down as far as possible in
819* each direction. If it bumps into any other changes, merge
820* them.
821*/
822do {823groupsize = g.end - g.start;824
825/*826* Keep track of the last "end" index that causes this
827* group to align with a group of changed lines in the
828* other file. -1 indicates that we haven't found such
829* a match yet:
830*/
831end_matching_other = -1;832
833/* Shift the group backward as much as possible: */834while (!group_slide_up(xdf, &g))835if (group_previous(xdfo, &go))836BUG("group sync broken sliding up");837
838/*839* This is this highest that this group can be shifted.
840* Record its end index:
841*/
842earliest_end = g.end;843
844if (go.end > go.start)845end_matching_other = g.end;846
847/* Now shift the group forward as far as possible: */848while (1) {849if (group_slide_down(xdf, &g))850break;851if (group_next(xdfo, &go))852BUG("group sync broken sliding down");853
854if (go.end > go.start)855end_matching_other = g.end;856}857} while (groupsize != g.end - g.start);858
859/*860* If the group can be shifted, then we can possibly use this
861* freedom to produce a more intuitive diff.
862*
863* The group is currently shifted as far down as possible, so
864* the heuristics below only have to handle upwards shifts.
865*/
866
867if (g.end == earliest_end) {868/* no shifting was possible */869} else if (end_matching_other != -1) {870/*871* Move the possibly merged group of changes back to
872* line up with the last group of changes from the
873* other file that it can align with.
874*/
875while (go.end == go.start) {876if (group_slide_up(xdf, &g))877BUG("match disappeared");878if (group_previous(xdfo, &go))879BUG("group sync broken sliding to match");880}881} else if (flags & XDF_INDENT_HEURISTIC) {882/*883* Indent heuristic: a group of pure add/delete lines
884* implies two splits, one between the end of the
885* "before" context and the start of the group, and
886* another between the end of the group and the
887* beginning of the "after" context. Some splits are
888* aesthetically better and some are worse. We compute
889* a badness "score" for each split, and add the scores
890* for the two splits to define a "score" for each
891* position that the group can be shifted to. Then we
892* pick the shift with the lowest score.
893*/
894long shift, best_shift = -1;895struct split_score best_score;896
897shift = earliest_end;898if (g.end - groupsize - 1 > shift)899shift = g.end - groupsize - 1;900if (g.end - INDENT_HEURISTIC_MAX_SLIDING > shift)901shift = g.end - INDENT_HEURISTIC_MAX_SLIDING;902for (; shift <= g.end; shift++) {903struct split_measurement m;904struct split_score score = {0, 0};905
906measure_split(xdf, shift, &m);907score_add_split(&m, &score);908measure_split(xdf, shift - groupsize, &m);909score_add_split(&m, &score);910if (best_shift == -1 ||911score_cmp(&score, &best_score) <= 0) {912best_score.effective_indent = score.effective_indent;913best_score.penalty = score.penalty;914best_shift = shift;915}916}917
918while (g.end > best_shift) {919if (group_slide_up(xdf, &g))920BUG("best shift unreached");921if (group_previous(xdfo, &go))922BUG("group sync broken sliding to blank line");923}924}925
926next:927/* Move past the just-processed group: */928if (group_next(xdf, &g))929break;930if (group_next(xdfo, &go))931BUG("group sync broken moving to next group");932}933
934if (!group_next(xdfo, &go))935BUG("group sync broken at end of file");936
937return 0;938}
939
940
941int xdl_build_script(xdfenv_t *xe, xdchange_t **xscr) {942xdchange_t *cscr = NULL, *xch;943char *rchg1 = xe->xdf1.rchg, *rchg2 = xe->xdf2.rchg;944long i1, i2, l1, l2;945
946/*947* Trivial. Collects "groups" of changes and creates an edit script.
948*/
949for (i1 = xe->xdf1.nrec, i2 = xe->xdf2.nrec; i1 >= 0 || i2 >= 0; i1--, i2--)950if (rchg1[i1 - 1] || rchg2[i2 - 1]) {951for (l1 = i1; rchg1[i1 - 1]; i1--);952for (l2 = i2; rchg2[i2 - 1]; i2--);953
954if (!(xch = xdl_add_change(cscr, i1, i2, l1 - i1, l2 - i2))) {955xdl_free_script(cscr);956return -1;957}958cscr = xch;959}960
961*xscr = cscr;962
963return 0;964}
965
966
967void xdl_free_script(xdchange_t *xscr) {968xdchange_t *xch;969
970while ((xch = xscr) != NULL) {971xscr = xscr->next;972xdl_free(xch);973}974}
975
976static int xdl_call_hunk_func(xdfenv_t *xe UNUSED, xdchange_t *xscr, xdemitcb_t *ecb,977xdemitconf_t const *xecfg)978{
979xdchange_t *xch, *xche;980
981for (xch = xscr; xch; xch = xche->next) {982xche = xdl_get_hunk(&xch, xecfg);983if (!xch)984break;985if (xecfg->hunk_func(xch->i1, xche->i1 + xche->chg1 - xch->i1,986xch->i2, xche->i2 + xche->chg2 - xch->i2,987ecb->priv) < 0)988return -1;989}990return 0;991}
992
993static void xdl_mark_ignorable_lines(xdchange_t *xscr, xdfenv_t *xe, long flags)994{
995xdchange_t *xch;996
997for (xch = xscr; xch; xch = xch->next) {998int ignore = 1;999xrecord_t **rec;1000long i;1001
1002rec = &xe->xdf1.recs[xch->i1];1003for (i = 0; i < xch->chg1 && ignore; i++)1004ignore = xdl_blankline(rec[i]->ptr, rec[i]->size, flags);1005
1006rec = &xe->xdf2.recs[xch->i2];1007for (i = 0; i < xch->chg2 && ignore; i++)1008ignore = xdl_blankline(rec[i]->ptr, rec[i]->size, flags);1009
1010xch->ignore = ignore;1011}1012}
1013
1014static int record_matches_regex(xrecord_t *rec, xpparam_t const *xpp) {1015regmatch_t regmatch;1016int i;1017
1018for (i = 0; i < xpp->ignore_regex_nr; i++)1019if (!regexec_buf(xpp->ignore_regex[i], rec->ptr, rec->size, 1,1020®match, 0))1021return 1;1022
1023return 0;1024}
1025
1026static void xdl_mark_ignorable_regex(xdchange_t *xscr, const xdfenv_t *xe,1027xpparam_t const *xpp)1028{
1029xdchange_t *xch;1030
1031for (xch = xscr; xch; xch = xch->next) {1032xrecord_t **rec;1033int ignore = 1;1034long i;1035
1036/*1037* Do not override --ignore-blank-lines.
1038*/
1039if (xch->ignore)1040continue;1041
1042rec = &xe->xdf1.recs[xch->i1];1043for (i = 0; i < xch->chg1 && ignore; i++)1044ignore = record_matches_regex(rec[i], xpp);1045
1046rec = &xe->xdf2.recs[xch->i2];1047for (i = 0; i < xch->chg2 && ignore; i++)1048ignore = record_matches_regex(rec[i], xpp);1049
1050xch->ignore = ignore;1051}1052}
1053
1054int xdl_diff(mmfile_t *mf1, mmfile_t *mf2, xpparam_t const *xpp,1055xdemitconf_t const *xecfg, xdemitcb_t *ecb) {1056xdchange_t *xscr;1057xdfenv_t xe;1058emit_func_t ef = xecfg->hunk_func ? xdl_call_hunk_func : xdl_emit_diff;1059
1060if (xdl_do_diff(mf1, mf2, xpp, &xe) < 0) {1061
1062return -1;1063}1064if (xdl_change_compact(&xe.xdf1, &xe.xdf2, xpp->flags) < 0 ||1065xdl_change_compact(&xe.xdf2, &xe.xdf1, xpp->flags) < 0 ||1066xdl_build_script(&xe, &xscr) < 0) {1067
1068xdl_free_env(&xe);1069return -1;1070}1071if (xscr) {1072if (xpp->flags & XDF_IGNORE_BLANK_LINES)1073xdl_mark_ignorable_lines(xscr, &xe, xpp->flags);1074
1075if (xpp->ignore_regex)1076xdl_mark_ignorable_regex(xscr, &xe, xpp);1077
1078if (ef(&xe, xscr, ecb, xecfg) < 0) {1079
1080xdl_free_script(xscr);1081xdl_free_env(&xe);1082return -1;1083}1084xdl_free_script(xscr);1085}1086xdl_free_env(&xe);1087
1088return 0;1089}
1090