1 /* Output all lines of a diff_result. */
3 * Copyright (c) 2020 Neels Hofmeyr <neels@hofmeyr.de>
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
24 #include <arraylist.h>
25 #include <diff_main.h>
26 #include <diff_output.h>
28 #include "diff_internal.h"
31 output_plain_chunk(struct diff_output_info
*outinfo
,
32 FILE *dest
, const struct diff_input_info
*info
,
33 const struct diff_result
*result
,
34 struct diff_chunk_context
*cc
, off_t
*outoff
, bool headers_only
)
37 int left_start
, left_len
, right_start
, right_len
;
41 left_len
= cc
->left
.end
- cc
->left
.start
;
44 else if (result
->left
->atoms
.len
== 0)
46 else if (left_len
== 0 && cc
->left
.start
> 0)
47 left_start
= cc
->left
.start
;
48 else if (cc
->left
.end
> 0)
49 left_start
= cc
->left
.start
+ 1;
51 left_start
= cc
->left
.start
;
53 right_len
= cc
->right
.end
- cc
->right
.start
;
56 else if (result
->right
->atoms
.len
== 0)
58 else if (right_len
== 0 && cc
->right
.start
> 0)
59 right_start
= cc
->right
.start
;
60 else if (cc
->right
.end
> 0)
61 right_start
= cc
->right
.start
+ 1;
63 right_start
= cc
->right
.start
;
68 rc
= fprintf(dest
, "%da%d\n", left_start
, right_start
);
70 rc
= fprintf(dest
, "%da%d,%d\n", left_start
,
71 right_start
, cc
->right
.end
);
73 } else if (right_len
== 0) {
76 rc
= fprintf(dest
, "%dd%d\n", left_start
,
79 rc
= fprintf(dest
, "%d,%dd%d\n", left_start
,
80 cc
->left
.end
, right_start
);
85 if (left_len
== 1 && right_len
== 1) {
86 rc
= fprintf(dest
, "%dc%d\n", left_start
, right_start
);
87 } else if (left_len
== 1) {
88 rc
= fprintf(dest
, "%dc%d,%d\n", left_start
,
89 right_start
, cc
->right
.end
);
90 } else if (right_len
== 1) {
91 rc
= fprintf(dest
, "%d,%dc%d\n", left_start
,
92 cc
->left
.end
, right_start
);
94 rc
= fprintf(dest
, "%d,%dc%d,%d\n", left_start
,
95 cc
->left
.end
, right_start
, cc
->right
.end
);
101 ARRAYLIST_ADD(offp
, outinfo
->line_offsets
);
109 * Now write out all the joined chunks.
111 * If the hunk denotes a change, it will come in the form of a deletion
112 * chunk followed by a addition chunk. Print a marker to break up the
113 * additions and deletions when this happens.
116 for (c_idx
= cc
->chunk
.start
; !headers_only
&& c_idx
< cc
->chunk
.end
;
118 const struct diff_chunk
*c
= &result
->chunks
.head
[c_idx
];
119 if (c
->left_count
&& !c
->right_count
)
120 rc
= diff_output_lines(outinfo
, dest
,
121 c
->solved
? "< " : "?",
122 c
->left_start
, c
->left_count
);
123 else if (c
->right_count
&& !c
->left_count
) {
125 rc
= fprintf(dest
, "---\n");
130 outinfo
->line_offsets
);
137 rc
= diff_output_lines(outinfo
, dest
,
138 c
->solved
? "> " : "?",
139 c
->right_start
, c
->right_count
);
143 if (cc
->chunk
.end
== result
->chunks
.len
) {
144 rc
= diff_output_trailing_newline_msg(outinfo
, dest
, c
);
145 if (rc
!= DIFF_RC_OK
)
154 diff_output_plain(struct diff_output_info
**output_info
,
155 FILE *dest
, const struct diff_input_info
*info
,
156 const struct diff_result
*result
, int hunk_headers_only
)
158 struct diff_output_info
*outinfo
= NULL
;
159 struct diff_chunk_context cc
= {};
160 int atomizer_flags
= (result
->left
->atomizer_flags
|
161 result
->right
->atomizer_flags
);
162 int flags
= (result
->left
->root
->diff_flags
|
163 result
->right
->root
->diff_flags
);
164 bool force_text
= (flags
& DIFF_FLAG_FORCE_TEXT_DATA
);
165 bool have_binary
= (atomizer_flags
& DIFF_ATOMIZER_FOUND_BINARY_DATA
);
167 off_t outoff
= 0, *offp
;
171 if (result
->rc
!= DIFF_RC_OK
)
175 *output_info
= diff_output_info_alloc();
176 if (*output_info
== NULL
)
178 outinfo
= *output_info
;
181 if (have_binary
&& !force_text
) {
182 for (i
= 0; i
< result
->chunks
.len
; i
++) {
183 struct diff_chunk
*c
= &result
->chunks
.head
[i
];
184 enum diff_chunk_type t
= diff_chunk_type(c
);
186 if (t
!= CHUNK_MINUS
&& t
!= CHUNK_PLUS
)
189 rc
= fprintf(dest
, "Binary files %s and %s differ\n",
190 diff_output_get_label_left(info
),
191 diff_output_get_label_right(info
));
195 ARRAYLIST_ADD(offp
, outinfo
->line_offsets
);
207 for (i
= 0; i
< result
->chunks
.len
; i
++) {
208 struct diff_chunk
*chunk
= &result
->chunks
.head
[i
];
209 enum diff_chunk_type t
= diff_chunk_type(chunk
);
210 struct diff_chunk_context next
;
212 if (t
!= CHUNK_MINUS
&& t
!= CHUNK_PLUS
)
215 if (diff_chunk_context_empty(&cc
)) {
216 /* Note down the start point, any number of subsequent
217 * chunks may be joined up to this chunk by being
218 * directly adjacent. */
219 diff_chunk_context_get(&cc
, result
, i
, 0);
223 /* There already is a previous chunk noted down for being
224 * printed. Does it join up with this one? */
225 diff_chunk_context_get(&next
, result
, i
, 0);
227 if (diff_chunk_contexts_touch(&cc
, &next
)) {
228 /* This next context touches or overlaps the previous
230 diff_chunk_contexts_merge(&cc
, &next
);
231 /* When we merge the last chunk we can end up with one
232 * hanging chunk and have to come back for it after the
236 rc
= output_plain_chunk(outinfo
, dest
, info
, result
, &cc
,
237 &outoff
, hunk_headers_only
);
238 if (rc
!= DIFF_RC_OK
)
242 if (!diff_chunk_context_empty(&cc
))
243 return output_plain_chunk(outinfo
, dest
, info
, result
, &cc
,
244 &outoff
, hunk_headers_only
);