2 * wiggle - apply rejected patches
4 * Copyright (C) 2003 Neil Brown <neilb@cse.unsw.edu.au>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 * Email: <neilb@cse.unsw.edu.au>
24 * School of Computer Science and Engineering
25 * The University of New South Wales
32 * This file contains routines use to create a merge.
33 * The core process is to take two coincidence lists, A-B and B-C,
34 * which identify coincidences and, but ommission, changes, and
35 * to apply to replace every part of A that matches B with the
36 * part of C that aligns with that part of B. In the case where
37 * a B-C difference does not align completely with an A-B coincidence,
40 * Throught the processing of merges we need a concept of a position in the
41 * overall merge. This is represented by an index into one of the files, and
42 * and indicator as to which file.
44 * A - then it is an unmatched part of A, before a coincidence.
45 * B - then it is in a section where A matches B and B matches C.
46 * C - then it is in an unmatched part of C, but the corresponding part
47 * of B completely coincides with A.
48 * With each position we keep indexes into the coincidence lists for
49 * the containing or next coincidence in each.
52 * The first stage of merge processing is to identify conflicts.
53 * A conflict is identified by a start point and an end point.
54 * The first approximation for the start point is the end
55 * of the last A-B coincidence that starts before the B start
56 * of the B-C difference that causes the conflict.
59 * We have a concept of a 'point'
60 * The start and end of the file are each points.
61 * Also the start and end of any conflict is a point.
62 * Outside of a conflict, we can move points forwards or backwards
63 * through the merger. Inside a conflict movement is not well defined.
64 * Any point is 'forward looking' or 'backward looking'.
65 * A forward looking point can be moved forward but not backward.
66 * A backward looking point can be moved backward, not forward.
68 * If a forward looking point is a tri-point, in a double-coincidence,
69 * then c1/c2 will be set to the furthest forward double coincidence that is before
70 * or contains the point, thus it well-defines the start of a double coincidence
71 * or that end of a conflict.
72 * inversely, a BL point well-defines the end of a DC or start of a conflict.
74 * The start of a conflict is a backward looking point.
75 * The end of a conflict is a forward looking point.
77 * In normal (Word/line) mode, we move the boundaries of a conflict out
78 * until they are at end-of-line.
79 * When moving forward, this is until we cross over a newline word.
80 * When moving backward, this is until one step before crossing over
81 * a newline word, so we need to remember our last position.
83 * Away from a conflict, every point can be clearly defined as a
84 * location either in A or in C. The 'point' is immediately before
85 * the word at that location.
86 * At the end of a conflict, this is still well defined as the 'next word'
87 * is outside a conflict.
88 * At the start of a conflict this may not be well defined as there may not
89 * be a clear 'next' word. We choose the point the would be reached by
90 * the step-forward algorithm so that it is easy to test if at start-of-conflict.
92 * A conflict is always bounded by a double coincidence. i.e. the word before a conflict
93 * is the same in all 3 texts, and the word after a conflict is the same in all
94 * 3 texts. To allow for conflicts at start and end of file, we consider the
95 * start and end of the three texts to each be double co-incidences.
97 * Each double co-incidence has a start and an end. When we find a conflict, it
98 * is taken to extend from the end of the previous double coincidence to the
99 * start of the next double co-incidence.
100 * Between conflicts we can mergers which can be printed simply be advancing the start
101 * point and printing each word as we go.
103 * The double co-incidence at the start begins forward-looking A=0 or C=0,
104 * depending on which word is first, and ends at backward-looking A=0.
105 * The double co-incidence at the end begins at forward-looking
106 * C=max and ends at backward looking A=max or C=max depending on which
107 * would be the last word.
109 * Each point is defined by a flag "in_a" which is true if the point is in A,
110 * and index 'pos' which gives the position in A or C depending on "in_a", and
111 * an index into each co-incidence list, c1 and c2.
113 * For forward looking points:
115 * c1 is the first co-incidence that ends after pos. - or is tail co-incidence.
116 * c2 is the first co-incidence that ends at or after c1.b
118 * c2 is the first co-incidence that ends after pos - or is tail co-incidence.
119 * c1 is the first co-incidence that ends at or after c2.a
121 * For a backward looking point:
123 * c1 is the last co-incidence that starts before pos, or -1
124 * c2 is the last co-incidence that starts at or before c1.b
126 * c2 is the last co-incidence that starts before pos, or -1
127 * c1 is the last co-incidence that .. lines up properly.
129 * To advance a point we increment pos, then
130 * if in_a and at start of c1
131 * slide up to c and if at end of c2, advance c2, then c1 and repeat
132 * if in_c and within c2 and corresponding a at end of c1, and c1->len != 0
133 * slide down to a, increment c1 and advance c2, then repeat.
135 * To retreat a backward facing point
136 * if in_a and at end of c1 and c1!=-1,
137 * slide up to c and if at start of c2, retreat c2, thenc 1, and repeat
138 * if in_c and within c2 and corresponding a at start of c1
139 * slide down to a, decrement c1 and retreat c2, then repeat.
140 * Then decrement pos.
142 * We never actually compare points for ordering. We should 'know' the likely order
143 * and only compare equal or not. This can be tested independant of direction,
144 * and done by simply comparing in_a and pos.
148 /* Each point involves a location in each of A, B, and C.
149 * There is a conflict for each change in B-C where the B section
150 * is not wholey contained in an A-B co-incidence.
151 * The start point of a conflict is determined as:
152 * C is the end of the C side of the previous B-C coincidence (or
154 * B is the end of the B side of the matching A-B coincidence if
155 * the point is in an A-B coincidence, or the end of the previous
156 * A-B coincidence of not.
157 * As B moves backwards searching for an A-B coincidence, if it enters
158 * a B-C coincidence, C is moved backwards too.
159 * A is the matching point to B in the A-B coincidence that B is in.
161 * The end point of a conflict is determined in a similar way,
162 * except that B is in a coincidence that is at, or *follows* the
163 * end of the next B-C coincidence.
165 * Once these coincidences have been enumerated, the endpoints are
166 * optionally moved to be at start-of-line. The start point is moved
167 * backwards and the endpoint forwards. The endofline must be in an
168 * A-B coincidence and may be in C if there is also a B-C coincidence.
170 * The next step is to merge adjacent conflicts where the B point
171 * from one overlaps the next.
178 /* A point is somewhere either in_a or not in_a (in which case, in C).
179 * if in_a, c1 points to the next a-b coincidence strictly after pos
180 * c2 points to the b-c coincidence that contains (possibly as end point) or follows c1.b
181 * if !in_a, c2 points to the b-c coincidence that contains (possibly as endpoint) or follows pos
182 * c1 points to the a-b coincidence that contains c2.b
184 * A point is not well defined inside a conflict, Though it is at the
185 * 'start' and 'end' of a conflict.
187 * At the start of the file c1 and c2 will be the firsts match in A-B and B-C
188 * If [c1]->a is 0, then !in_a and pos is [c2]->b+x where x is
189 * chosen such that [c1]->b == [c2]->a+x and x < [c2]->len. If such choice
190 * is not possible, there is a conflict at the start of the file and so we choose
191 * a point as if [c1]->a were not 0.
193 * If [c1]->a is not 0, then in_a and pos == 0.
195 * To find the start of file, we set in_a and pos==-1, and advance one step.
197 * At the end of the file, c1 will be the EOF match in A-B, c2 will be the
198 * EOF match in B-C, !in_a and pos == [c2]->b
200 struct point
{ int pos
, in_a
; int c1
,c2
; };
203 static int tripoint(struct point
*here
,
204 struct csl
*c1
, struct csl
*c2
,
205 int *a
, int *b
, int *c
)
207 /* find a, b, and c for 'here'.
208 * If any are not well defined, return 0.
219 } else if (c1
->a
<= *a
&& c1
->a
+c1
->len
>= *a
)
220 *b
= c1
->b
+ (*a
- c1
->a
);
227 } else if (c2
->a
<= *b
&& c2
->a
+ c2
->len
>= *b
)
228 *c
= c2
->b
+ *b
- c2
->a
;
237 } else if (c2
->b
<= *c
&& c2
->b
+c2
->len
>= *c
)
238 *b
= c2
->a
+ *c
- c2
->b
;
246 } else if (c1
->b
<= *b
&& c1
->b
+ c1
->len
>= *b
)
247 *a
= c1
->a
+ *b
- c1
->b
;
254 static int retreat(struct csl
*c1
, struct csl
*c2
, struct point
*p
)
261 /* retreat c1 to first coincidence containing or after pos */
263 while ((p
->c1
== 0 && a
== 0) ||
264 (p
->c1
> 0 && c1
[p
->c1
-1].a
+ c1
[p
->c1
-1].len
>= a
)) {
266 if ( a
>= c1
[p
->c1
].a
)
271 /* if we aren't in a co-incidence, just return */
276 /* retreat c2 to first coincidence containing or after pos->b */
280 b
= c1
[p
->c1
].b
+ a
- c1
[p
->c1
].a
;
281 while ((p
->c2
== 0 && b
== 0) ||
282 (p
->c1
> 0 && c2
[p
->c2
-1].a
+ c2
[p
->c2
-1].len
>= b
)) {
284 if (b
>= c2
[p
->c2
].a
)
289 /* check if this is a conflict */
290 if ((p
->c2
>=0 && c2
[p
->c2
].a
> b
))
296 c
= c2
[p
->c2
].b
+ b
- c2
[p
->c2
].a
;
298 /* ok, this is the furthest backward double coincidence
299 * if we are not at the start of the A-B coincidence,
302 if (p
->c1
>= 0 && a
> c1
[p
->c1
].a
) {
309 /* retreat c2 to first coincidence containing or after pos */
311 while ((p
->c2
== 0 && c
== 0) ||
312 (p
->c2
> 0 && c2
[p
->c2
-1].b
+ c2
[p
->c2
-1].len
>= c
)) {
314 if (c
>= c2
[p
->c2
].b
)
319 /* if we aren't in a coincidence, return */
324 /* retreat c1 to first coincidence containing or afer pos->b */
328 b
= c2
[p
->c2
].a
+ c
- c2
[p
->c2
].b
;
329 while ((p
->c1
==0 && b
== 0) ||
330 (p
->c1
> 0 && c1
[p
->c1
-1].b
+ c1
[p
->c1
-1].len
>= b
)) {
332 if (b
>= c1
[p
->c1
].b
)
337 /* check if this is a conflict */
338 if ((p
->c1
>=0 && c1
[p
->c1
].b
> b
))
344 a
= c1
[p
->c1
].a
+ b
- c1
[p
->c1
].b
;
346 /* ok, this is the furthest backward double coincidence
347 * if we are at the start of the A-B coincidence, slide down to A
358 return 0; /* StartOfFile */
368 static int advance(struct csl
*c1
, struct csl
*c2
, struct point
*p
)
372 /* make next char at point is the 'right' one, either in a or c.
373 * This might involve move p->c1 and p->c2 forward
374 * and changing pos/in_a to an 'equivalent' point
377 if (!p->in_a && c2[p->c2].b == p->pos && c2[p->c2].len == 0)
378 return 0; / * at end of file * /
382 /* advance c1 to last coincidence containing or before pos */
384 while ((p
->c1
== -1 || c1
[p
->c1
].len
) &&
385 c1
[p
->c1
+1].a
<= a
) {
387 if ((p
->c1
== -1 && a
==0) ||
388 (p
->c1
>=0 && a
<= c1
[p
->c1
].a
+c1
[p
->c1
].len
))
393 /* if we aren't in a co-incidence, just return */
394 if (p
->c1
== -1 || c1
[p
->c1
].a
+c1
[p
->c1
].len
< a
)
397 /* advance c2 to last coincidence containing or before pos->b */
398 b
= c1
[p
->c1
].b
+ a
- c1
[p
->c1
].a
;
399 while ((p
->c2
== -1 || c2
[p
->c2
].len
) &&
400 c2
[p
->c2
+1].a
<= b
) {
402 if ((p
->c2
== -1 && b
== 0) ||
403 (p
->c2
>= 0 && b
<= c2
[p
->c2
].a
+c2
[p
->c2
].len
))
408 /* check if this is a conflict */
409 if ((p
->c2
== -1 && b
>0) ||
410 (p
->c2
>=0 && c2
[p
->c2
].a
+ c2
[p
->c2
].len
< b
))
416 c
= c2
[p
->c2
].b
+ b
- c2
[p
->c2
].a
;
418 /* Ok, this is the furthest forward double coincidence
419 * If we are at eof, or the next char is in the coincidence
422 if (c1
[p
->c1
].len
== 0 ||
423 a
< c1
[p
->c1
].a
+ c1
[p
->c1
].len
) {
426 * if we've slid, make sure not to skip over
429 if(slid
&& p
->c2
!= -1 && c2
[p
->c2
].a
== b
&&
430 c2
[p
->c2
].b
> c2
[p
->c2
].a
) {
431 c
-= c2
[p
->c2
].b
- c2
[p
->c2
].a
;
439 /* advance c2 to last coincidence containing or before pos */
441 while ((p
->c2
== -1 || c2
[p
->c2
].len
) &&
442 c2
[p
->c2
+1].b
<= c
) {
444 if ((p
->c2
== -1 && c
== 0) ||
445 (p
->c2
>= 0 && c
<= c2
[p
->c2
].b
+c2
[p
->c2
].len
))
450 /* if we aren't in a co-incidence then just return */
451 if (p
->c2
== -1 || c2
[p
->c2
].b
+c2
[p
->c2
].len
< c
)
454 /* advance c1 to last coincidence containing or before pos->b */
455 b
= c2
[p
->c2
].a
+ c
- c2
[p
->c2
].b
;
456 while ((p
->c1
== -1 || c1
[p
->c1
].len
) &&
457 c1
[p
->c1
+1].b
<= b
) {
459 if ((p
->c1
== -1 && b
==0) ||
460 (p
->c1
>= 0 && b
<= c1
[p
->c1
].b
+c1
[p
->c1
].len
))
465 /* check if this is a conflict */
466 if (p
->c1
== -1 || c1
[p
->c1
].b
+ c1
[p
->c1
].len
< b
)
469 a
= c1
[p
->c1
].a
+ b
- c1
[p
->c1
].b
;
471 /* ok, this is the furthest forward double coincidence
472 * If it is the end of an A-B coincidence but not EOF,
475 if (a
== c1
[p
->c1
].a
+ c1
[p
->c1
].len
&&
483 if (!p
->in_a
&& c2
[p
->c2
].b
== p
->pos
&& c2
[p
->c2
].len
== 0)
484 return 0; /* at end of file */
492 static int point_crossed(struct point first
, struct point second
,
493 struct csl
*cs1
, struct csl
*cs2
)
498 if (tripoint(&first
, cs1
,cs2
, &a1
,&b1
,&c1
) &&
499 tripoint(&second
, cs1
,cs2
, &a2
,&b2
,&c2
))
500 return a1
>=a2
&& b1
>=b2
&& c1
>=c2
;
503 return first.in_a == second.in_a &&
504 first.pos == second.pos;
509 static void print_merger(FILE *out
, struct file
*a
, struct file
*c
,
510 struct csl
*cs1
, struct csl
*cs2
,
511 struct point start
, struct point end
)
513 while (!point_crossed(start
, end
, cs1
,cs2
)) {
515 printf("%c %d (%d,%d)\n", start
.in_a
?'A':'C', start
.pos
, start
.c1
,start
.c2
);
518 printword(out
, a
->list
[start
.pos
]);
520 printword(out
, c
->list
[start
.pos
]);
521 fflush(out
); /* DEBUG */
524 if (point_crossed(start
, end
, cs1
,cs2
))
526 advance(cs1
, cs2
, &start
);
531 static int inline at_sol(struct file
*f
, int i
)
533 return i
== 0 || i
== f
->elcnt
||
534 ends_line(f
->list
[i
-1]);
537 static void print_range(FILE *out
, struct file
*f
, int start
, int end
)
539 for (; start
< end
; start
++)
540 printword(out
, f
->list
[start
]);
543 static int print_conflict(FILE *out
, struct file
*a
, struct file
*b
, struct file
*c
,
544 struct csl
*c1
, struct csl
*c2
,
545 struct point start
, struct point end
,
548 int astart
, bstart
, cstart
;
549 int aend
, bend
, cend
;
553 if (point_same(start
,end
))
554 return 0; /* no conflict here !! */
556 if (!tripoint(&start
, c1
,c2
, &astart
, &bstart
, &cstart
))
558 if (!tripoint(&end
, c1
,c2
, &aend
, &bend
, &cend
))
562 /* Now contract the conflict if possible, but insist on
563 * an end-of-line boundary unless 'words'.
565 /* first contract leading removed text.
566 * so <<<--- X 1 ||| X 2 === 3 --->>> becomes <<<--- 1 ||| 2 === 3 --->>>
569 while (bi
< bend
&& start
.c1
>= 0 && bi
>= c1
[start
.c1
].b
&& bi
< c1
[start
.c1
].b
+ c1
[start
.c1
].len
) {
571 if (words
|| at_sol(b
,bi
)) {
576 /* and contract trailing removed text */
578 while (bi
> bstart
&& bi
> c1
[end
.c1
].b
) {
580 if (words
|| at_sol(b
, bi
)) {
586 /* now contract leading unmatched text so
587 * <<<--- 1 ||| X 2 === X 3 --->>> becomes <<<--- 1 ||| 2 === 3 --->>>
590 while (bi
< bend
&& start
.c2
>= 0 && bi
>= c2
[start
.c2
].a
&& bi
< c2
[start
.c2
].a
+ c2
[start
.c2
].len
) {
592 if (words
|| at_sol(b
,bi
)) {
597 /* and trailing unmatched */
599 while (bi
> bstart
&& bi
> c2
[end
.c2
].a
) {
601 if (words
|| at_sol(b
,bi
)) {
606 if (astart
>= aend
&& bstart
>= bend
&& cstart
>= cend
)
609 fputs(words
?"<<<---":"<<<<<<<\n", out
);
610 print_range(out
, a
, astart
, aend
);
611 fputs(words
?"|||":"|||||||\n", out
);
612 print_range(out
, b
, bstart
, bend
);
613 fputs(words
?"===":"=======\n", out
);
614 print_range(out
, c
, cstart
, cend
);
615 fputs(words
?"--->>>":">>>>>>>\n", out
);
619 static int end_of_file(struct point p
, struct csl
*c1
, struct csl
*c2
)
621 return advance(c1
,c2
,&p
)==0;
624 static int next_conflict(struct point here
, struct csl
*start_c1
, struct csl
*start_c2
,
625 struct point
*start
, struct point
*end
,
626 struct file
*a
, struct file
*b
, struct file
*c
)
628 /* We want to find the start and end of the 'next' conflict.
629 * There may not be another conflict, in which case set start and
630 * end to the end of the files.
631 * The start and end of a conflict must be the end and start of
632 * regions where A matches B and B matches C - except for
633 * The start which might be the start of the file.
634 * 'here' is a potentially valid starting point. Any other starting
635 * point must be the end of a double coincidence.
637 * So we walk c1 and c2 looking for double coincidences and conflicts.
638 * When we find a conflict, we remember the fact.
639 * When we find a double coincidence we:
640 * Set 'end' to the start of the DC.
641 * If conflict-found - return.
642 * Set 'start' to the end of the DC.
643 * If the DC was EOF, start will == end == EOF, and we return.
645 * A double coincidence is easily detected by just looking at a single
646 * entry in c1 and c2. If
647 * c1->b+c1->len > c2->a && c2->a+c2->len > c1->b
648 * || c1->len == c2->len == 0
649 * then we have a double coincidence.
651 * A conflict is detected when stepping forward.
652 * If we step c2 forward and the new coincidence is beyond or at the
653 * end of c1, or we step forward c1 and it's start is beyond or at the end of c2,
654 * then that is a conflict.
655 * Also, we can detect a conflict at start-of-file (here.in_a, here.pos==0) if
656 * c2 doesn't start at 0.
658 * 'here' is significant only for its c1/c2 values. They will contain a
659 * double coincidence, though it might be start-of-file.
660 * start must be set to a backward-looking point at the end of a double-coincidence
661 * and end to a forward-looking point and the start of a double-coincidence
665 int conflict_found
= 0;
666 struct csl
*c1
= start_c1
;
667 struct csl
*c2
= start_c2
;
676 /* Step one of c1 or c2 forward
677 * depending on which ends earlier.
678 * Watch to see if we are stepping over a conflict.
682 * Move both c1 and c2 forward.
684 * We have a conflict iff new c1->b > 0 and c2->a > 0
685 * or c1->b >0 && c2->b > 0
689 (c2
->a
> 0 || c2
->b
> 0))
691 if (c2
->a
+c2
->len
< c1
->b
)
693 } else if (c1
->b
+c1
->len
== c2
->a
+c2
->len
) {
694 /* both coincidences end at same place. There is
695 * a conflict if there is a gap in c1->b or
696 * c2->a has no gap but c2->b does (implying insertion
697 * at undefined location
699 if (c1
->len
&& c2
->len
) {
700 if (c1
[1].b
> c1
->b
+ c1
->len
||
701 (c2
[1].a
== c2
->a
+ c2
->len
&&
702 c2
[1].b
> c2
->b
+ c2
->len
))
709 } else if (c2
->len
==0 || (c1
->len
&& c1
->b
+c1
->len
< c2
->a
+c2
->len
)) {
710 /* c1 ends earlier. If the new start of c1 is
711 * beyond the current end of c2, we have a conflict
714 if (c1
->b
> c2
->a
+c2
->len
)
717 /* c2 ends earlier. If the new start of c2 is
718 * beyond the end of c1, we have a conflict.
719 * Also if the new start of c2 is at the end of c1,
720 * and the old end of c2 is also at end of c1,
721 * then have a conflict, as long as there was actually
722 * something inserted there...
725 if (c2
->a
> c1
->b
+c1
->len
)
728 if ((c1
->len
== 0 && c2
->len
==0) ||
729 (c1
->b
+c1
->len
>= c2
->a
&& c2
->a
+c2
->len
>= c1
->b
)
731 /* double coincidence !
732 * It starts at max of c1->b and c2->a, in c
733 * and ends at min of c1->b+len (in a), c2->a+len (in c)
735 end
->c1
= c1
-start_c1
;
736 end
->c2
= c2
-start_c2
;
738 if (conflict_found
) {
739 /* end->c1/c2 holds the end of the conflict,
740 * and start->c1/c2 holds the start
741 * We need to set in_a and pos for each
742 * so that start is backward-looking and the end
743 * of a double-coincidence, and end is forward-looking
744 * at the start of a double-coincidence.
750 if (start
->c1
== -1) {
753 } else if (c1
[start
->c1
].b
+c1
[start
->c1
].len
<=
754 c2
[start
->c2
].a
+c2
[start
->c2
].len
) {
756 start
->pos
= c1
[start
->c1
].a
+c1
[start
->c1
].len
;
759 start
->pos
= c2
[start
->c2
].b
+c2
[start
->c2
].len
;
761 retreat(c1
,c2
, start
);
763 if (c1
[end
->c1
].b
<= c2
[end
->c2
].a
) {
765 end
->pos
= c2
[end
->c2
].b
;
768 end
->pos
= c2
[end
->c2
].b
+
769 c1
[end
->c1
].b
- c2
[end
->c2
].a
;
774 start
->c1
= c1
-start_c1
;
775 start
->c2
= c2
-start_c2
;
777 if (c1
->len
== 0 && c2
->len
== 0) {
778 /* eof and no conflict found.
779 * set start and end to eof
781 start
->in_a
= end
->in_a
= 0;
782 start
->pos
= end
->pos
= c2
->b
;
789 static int already_applied(struct csl
*cs1
, struct csl
*cs2
,
790 struct point start
, struct point end
,
791 struct file
*a
, struct file
*b
, struct file
*c
)
793 /* check if this conflict reflects and already-applied change
794 * i.e. the section in a matches the section in b
799 if (!tripoint(&start
,cs1
,cs2
,&a1
,&b1
,&c1
))
801 if (!tripoint(&end
,cs1
,cs2
,&a2
,&b2
,&c2
))
803 if (a1
==a2
&& b1
==b2
) return 0;
804 if ((a2
-a1
) != (c2
-c1
)) return 0;
807 if (!match(&a
->list
[a1
], &c
->list
[c1
]))
815 static int Startofline(struct point p
, struct csl
*cs1
, struct csl
*cs2
,
816 struct file
*a
, struct file
*b
, struct file
*c
)
820 tripoint(&p
,cs1
,cs2
,&a1
,&b1
,&c1
) &&
821 at_sol(a
,a1
) && at_sol(b
,b1
) && at_sol(c
,c1
);
825 struct ci
print_merge(FILE *out
, struct file
*a
, struct file
*b
, struct file
*c
,
826 struct csl
*c1
, struct csl
*c2
,
829 struct point start_last
, end_last
, start_next
, end_next
;
832 rv
.ignored
= rv
.conflicts
= 0;
836 for (i
=0; c1
[i
].len
; i
++) printf("%2d c1 %d:%d %d\n", i
, c1
[i
].a
,c1
[i
].b
,c1
[i
].len
);
837 printf("%2d c1 %d:%d END\n", i
, c1
[i
].a
,c1
[i
].b
);
838 for (i
=0; c2
[i
].len
; i
++) printf("%2d c2 %d:%d %d\n", i
, c2
[i
].a
,c2
[i
].b
,c2
[i
].len
);
839 printf("%2d c2 %d:%d END\n", i
, c2
[i
].a
,c2
[i
].b
);
842 /* end_last is a forward looking point */
845 end_last
.c1
= end_last
.c2
= -1;
846 advance(c1
,c2
, &end_last
);
848 /* start_last is a backward looking point */
851 start_last
.c1
= start_last
.c2
= 0;
852 retreat(c1
,c2
, &start_last
);
854 while (!end_of_file(end_last
, c1
, c2
)) {
855 next_conflict(end_last
, c1
, c2
, &start_next
, &end_next
, a
, b
, c
);
856 while (already_applied(c1
,c2
,start_next
,end_next
,a
,b
,c
)) {
858 next_conflict(end_next
, c1
,c2
,&start_next
,&end_next
,a
,b
,c
);
861 printf("start %d %d (%d,%d) end %d %d (%d,%d)\n",
862 start_next
.in_a
, start_next
.pos
, start_next
.c1
, start_next
.c2
,
863 end_next
.in_a
, end_next
.pos
, end_next
.c1
, end_next
.c2
);
865 while (!point_crossed(end_last
, start_next
,c1
,c2
) &&
866 !(words
|| Startofline(end_last
, c1
,c2
, a
,b
,c
))) {
868 advance(c1
,c2
, &end_last
);
871 while (!point_crossed(end_last
, start_next
, c1
,c2
) &&
872 !(words
|| Startofline(start_next
, c1
,c2
, a
,b
,c
))) {
874 retreat(c1
,c2
, &start_next
);
877 if (point_crossed(end_last
, start_next
, c1
,c2
)) {
881 if (print_conflict(out
, a
,b
,c
, c1
,c2
, start_last
, end_last
, words
))
884 print_merger(out
, a
,c
, c1
,c2
, end_last
, start_next
);
885 start_last
= start_next
;
888 if (print_conflict(out
,a
,b
,c
, c1
,c2
, start_last
, end_last
, words
))