diff options
author | Bernhard Reutner-Fischer <rep.dot.nop@gmail.com> | 2006-05-29 12:12:45 +0000 |
---|---|---|
committer | Bernhard Reutner-Fischer <rep.dot.nop@gmail.com> | 2006-05-29 12:12:45 +0000 |
commit | bbc225e13d0b4388a095c98c86caff6dd675962b (patch) | |
tree | 5e1eb47b2c43663d199bb185af6ee85424df39db | |
parent | d2c306e862abf49dd4b1ff1d1bd1a789317b7905 (diff) | |
download | busybox-w32-bbc225e13d0b4388a095c98c86caff6dd675962b.tar.gz busybox-w32-bbc225e13d0b4388a095c98c86caff6dd675962b.tar.bz2 busybox-w32-bbc225e13d0b4388a095c98c86caff6dd675962b.zip |
- just whitespace
-rw-r--r-- | coreutils/diff.c | 1472 |
1 files changed, 741 insertions, 731 deletions
diff --git a/coreutils/diff.c b/coreutils/diff.c index a1a74d51e..57b32eb78 100644 --- a/coreutils/diff.c +++ b/coreutils/diff.c | |||
@@ -36,9 +36,9 @@ | |||
36 | /* | 36 | /* |
37 | * Output flags | 37 | * Output flags |
38 | */ | 38 | */ |
39 | #define D_HEADER 1 /* Print a header/footer between files */ | 39 | #define D_HEADER 1 /* Print a header/footer between files */ |
40 | #define D_EMPTY1 2 /* Treat first file as empty (/dev/null) */ | 40 | #define D_EMPTY1 2 /* Treat first file as empty (/dev/null) */ |
41 | #define D_EMPTY2 4 /* Treat second file as empty (/dev/null) */ | 41 | #define D_EMPTY2 4 /* Treat second file as empty (/dev/null) */ |
42 | 42 | ||
43 | /* | 43 | /* |
44 | * Status values for print_status() and diffreg() return values | 44 | * Status values for print_status() and diffreg() return values |
@@ -68,6 +68,7 @@ | |||
68 | 68 | ||
69 | /* Command line options */ | 69 | /* Command line options */ |
70 | static unsigned long cmd_flags; | 70 | static unsigned long cmd_flags; |
71 | |||
71 | #define FLAG_a (1<<0) | 72 | #define FLAG_a (1<<0) |
72 | #define FLAG_b (1<<1) | 73 | #define FLAG_b (1<<1) |
73 | #define FLAG_d (1<<2) | 74 | #define FLAG_d (1<<2) |
@@ -90,14 +91,14 @@ char **dl; | |||
90 | int dl_count = 0; | 91 | int dl_count = 0; |
91 | 92 | ||
92 | struct cand { | 93 | struct cand { |
93 | int x; | 94 | int x; |
94 | int y; | 95 | int y; |
95 | int pred; | 96 | int pred; |
96 | }; | 97 | }; |
97 | 98 | ||
98 | struct line { | 99 | struct line { |
99 | int serial; | 100 | int serial; |
100 | int value; | 101 | int value; |
101 | } *file[2]; | 102 | } *file[2]; |
102 | 103 | ||
103 | /* | 104 | /* |
@@ -106,145 +107,145 @@ struct line { | |||
106 | * understand the highly mnemonic field names) | 107 | * understand the highly mnemonic field names) |
107 | */ | 108 | */ |
108 | struct context_vec { | 109 | struct context_vec { |
109 | int a; /* start line in old file */ | 110 | int a; /* start line in old file */ |
110 | int b; /* end line in old file */ | 111 | int b; /* end line in old file */ |
111 | int c; /* start line in new file */ | 112 | int c; /* start line in new file */ |
112 | int d; /* end line in new file */ | 113 | int d; /* end line in new file */ |
113 | }; | 114 | }; |
114 | 115 | ||
115 | static int *J; /* will be overlaid on class */ | 116 | static int *J; /* will be overlaid on class */ |
116 | static int *class; /* will be overlaid on file[0] */ | 117 | static int *class; /* will be overlaid on file[0] */ |
117 | static int *klist; /* will be overlaid on file[0] after class */ | 118 | static int *klist; /* will be overlaid on file[0] after class */ |
118 | static int *member; /* will be overlaid on file[1] */ | 119 | static int *member; /* will be overlaid on file[1] */ |
119 | static int clen; | 120 | static int clen; |
120 | static int len[2]; | 121 | static int len[2]; |
121 | static int pref, suff; /* length of prefix and suffix */ | 122 | static int pref, suff; /* length of prefix and suffix */ |
122 | static int slen[2]; | 123 | static int slen[2]; |
123 | static int anychange; | 124 | static int anychange; |
124 | static long *ixnew; /* will be overlaid on file[1] */ | 125 | static long *ixnew; /* will be overlaid on file[1] */ |
125 | static long *ixold; /* will be overlaid on klist */ | 126 | static long *ixold; /* will be overlaid on klist */ |
126 | static struct cand *clist; /* merely a free storage pot for candidates */ | 127 | static struct cand *clist; /* merely a free storage pot for candidates */ |
127 | static int clistlen; /* the length of clist */ | 128 | static int clistlen; /* the length of clist */ |
128 | static struct line *sfile[2]; /* shortened by pruning common prefix/suffix */ | 129 | static struct line *sfile[2]; /* shortened by pruning common prefix/suffix */ |
129 | static struct context_vec *context_vec_start; | 130 | static struct context_vec *context_vec_start; |
130 | static struct context_vec *context_vec_end; | 131 | static struct context_vec *context_vec_end; |
131 | static struct context_vec *context_vec_ptr; | 132 | static struct context_vec *context_vec_ptr; |
132 | 133 | ||
133 | static void print_only(const char *path, size_t dirlen, const char *entry) | 134 | static void print_only(const char *path, size_t dirlen, const char *entry) |
134 | { | 135 | { |
135 | if (dirlen > 1) | 136 | if (dirlen > 1) |
136 | dirlen--; | 137 | dirlen--; |
137 | printf("Only in %.*s: %s\n", (int)dirlen, path, entry); | 138 | printf("Only in %.*s: %s\n", (int) dirlen, path, entry); |
138 | } | 139 | } |
139 | 140 | ||
140 | static void print_status(int val, char *path1, char *path2, char *entry) | 141 | static void print_status(int val, char *path1, char *path2, char *entry) |
141 | { | 142 | { |
142 | const char * const _entry = entry ? entry : ""; | 143 | const char *const _entry = entry ? entry : ""; |
143 | char *_path1 = entry ? concat_path_file(path1, _entry) : path1; | 144 | char *_path1 = entry ? concat_path_file(path1, _entry) : path1; |
144 | char *_path2 = entry ? concat_path_file(path2, _entry) : path2; | 145 | char *_path2 = entry ? concat_path_file(path2, _entry) : path2; |
145 | switch (val) { | 146 | |
146 | case D_ONLY: | 147 | switch (val) { |
147 | print_only(path1, strlen(path1), entry); | 148 | case D_ONLY: |
148 | break; | 149 | print_only(path1, strlen(path1), entry); |
149 | case D_COMMON: | 150 | break; |
150 | printf("Common subdirectories: %s and %s\n", _path1, _path2); | 151 | case D_COMMON: |
151 | break; | 152 | printf("Common subdirectories: %s and %s\n", _path1, _path2); |
152 | case D_BINARY: | 153 | break; |
153 | printf("Binary files %s and %s differ\n", _path1, _path2); | 154 | case D_BINARY: |
154 | break; | 155 | printf("Binary files %s and %s differ\n", _path1, _path2); |
155 | case D_DIFFER: | 156 | break; |
156 | if (cmd_flags & FLAG_q) | 157 | case D_DIFFER: |
157 | printf("Files %s and %s differ\n", _path1, _path2); | 158 | if (cmd_flags & FLAG_q) |
158 | break; | 159 | printf("Files %s and %s differ\n", _path1, _path2); |
159 | case D_SAME: | 160 | break; |
160 | if (cmd_flags & FLAG_s) | 161 | case D_SAME: |
161 | printf("Files %s and %s are identical\n", _path1, _path2); | 162 | if (cmd_flags & FLAG_s) |
162 | break; | 163 | printf("Files %s and %s are identical\n", _path1, _path2); |
163 | case D_MISMATCH1: | 164 | break; |
164 | printf("File %s is a directory while file %s is a regular file\n", | 165 | case D_MISMATCH1: |
165 | _path1, _path2); | 166 | printf("File %s is a directory while file %s is a regular file\n", |
166 | break; | 167 | _path1, _path2); |
167 | case D_MISMATCH2: | 168 | break; |
168 | printf("File %s is a regular file while file %s is a directory\n", | 169 | case D_MISMATCH2: |
169 | _path1, _path2); | 170 | printf("File %s is a regular file while file %s is a directory\n", |
170 | break; | 171 | _path1, _path2); |
171 | case D_SKIPPED1: | 172 | break; |
172 | printf("File %s is not a regular file or directory and was skipped\n", | 173 | case D_SKIPPED1: |
173 | _path1); | 174 | printf("File %s is not a regular file or directory and was skipped\n", |
174 | break; | 175 | _path1); |
175 | case D_SKIPPED2: | 176 | break; |
176 | printf("File %s is not a regular file or directory and was skipped\n", | 177 | case D_SKIPPED2: |
177 | _path2); | 178 | printf("File %s is not a regular file or directory and was skipped\n", |
178 | break; | 179 | _path2); |
179 | } | 180 | break; |
180 | if (entry) { | 181 | } |
181 | free(_path1); | 182 | if (entry) { |
182 | free(_path2); | 183 | free(_path1); |
183 | } | 184 | free(_path2); |
185 | } | ||
184 | } | 186 | } |
185 | 187 | ||
186 | /* | 188 | /* |
187 | * Hash function taken from Robert Sedgewick, Algorithms in C, 3d ed., p 578. | 189 | * Hash function taken from Robert Sedgewick, Algorithms in C, 3d ed., p 578. |
188 | */ | 190 | */ |
189 | static int readhash(FILE *f) | 191 | static int readhash(FILE * f) |
190 | { | 192 | { |
191 | int i, t, space; | 193 | int i, t, space; |
192 | int sum; | 194 | int sum; |
193 | 195 | ||
194 | sum = 1; | 196 | sum = 1; |
195 | space = 0; | 197 | space = 0; |
196 | if (!(cmd_flags & FLAG_b) && !(cmd_flags & FLAG_w)) { | 198 | if (!(cmd_flags & FLAG_b) && !(cmd_flags & FLAG_w)) { |
197 | if (FLAG_i) | 199 | if (FLAG_i) |
198 | for (i = 0; (t = getc(f)) != '\n'; i++) { | 200 | for (i = 0; (t = getc(f)) != '\n'; i++) { |
199 | if (t == EOF) { | 201 | if (t == EOF) { |
200 | if (i == 0) | 202 | if (i == 0) |
201 | return (0); | 203 | return (0); |
202 | break; | 204 | break; |
203 | } | 205 | } |
204 | sum = sum * 127 + t; | 206 | sum = sum * 127 + t; |
205 | } | 207 | } else |
206 | else | 208 | for (i = 0; (t = getc(f)) != '\n'; i++) { |
207 | for (i = 0; (t = getc(f)) != '\n'; i++) { | 209 | if (t == EOF) { |
208 | if (t == EOF) { | 210 | if (i == 0) |
209 | if (i == 0) | 211 | return (0); |
210 | return (0); | 212 | break; |
211 | break; | 213 | } |
212 | } | 214 | sum = sum * 127 + t; |
213 | sum = sum * 127 + t; | 215 | } |
214 | } | 216 | } else { |
215 | } else { | 217 | for (i = 0;;) { |
216 | for (i = 0;;) { | 218 | switch (t = getc(f)) { |
217 | switch (t = getc(f)) { | 219 | case '\t': |
218 | case '\t': | 220 | case '\r': |
219 | case '\r': | 221 | case '\v': |
220 | case '\v': | 222 | case '\f': |
221 | case '\f': | 223 | case ' ': |
222 | case ' ': | 224 | space++; |
223 | space++; | 225 | continue; |
224 | continue; | 226 | default: |
225 | default: | 227 | if (space && !(cmd_flags & FLAG_w)) { |
226 | if (space && !(cmd_flags & FLAG_w)) { | 228 | i++; |
227 | i++; | 229 | space = 0; |
228 | space = 0; | 230 | } |
229 | } | 231 | sum = sum * 127 + t; |
230 | sum = sum * 127 + t; | 232 | i++; |
231 | i++; | 233 | continue; |
232 | continue; | 234 | case EOF: |
233 | case EOF: | 235 | if (i == 0) |
234 | if (i == 0) | 236 | return (0); |
235 | return (0); | 237 | /* FALLTHROUGH */ |
236 | /* FALLTHROUGH */ | 238 | case '\n': |
237 | case '\n': | 239 | break; |
238 | break; | 240 | } |
239 | } | 241 | break; |
240 | break; | 242 | } |
241 | } | 243 | } |
242 | } | 244 | /* |
243 | /* | 245 | * There is a remote possibility that we end up with a zero sum. |
244 | * There is a remote possibility that we end up with a zero sum. | 246 | * Zero is used as an EOF marker, so return 1 instead. |
245 | * Zero is used as an EOF marker, so return 1 instead. | 247 | */ |
246 | */ | 248 | return (sum == 0 ? 1 : sum); |
247 | return (sum == 0 ? 1 : sum); | ||
248 | } | 249 | } |
249 | 250 | ||
250 | 251 | ||
@@ -253,103 +254,103 @@ static int readhash(FILE *f) | |||
253 | * Check to see if the given files differ. | 254 | * Check to see if the given files differ. |
254 | * Returns 0 if they are the same, 1 if different, and -1 on error. | 255 | * Returns 0 if they are the same, 1 if different, and -1 on error. |
255 | */ | 256 | */ |
256 | static int files_differ(FILE *f1, FILE *f2, int flags) | 257 | static int files_differ(FILE * f1, FILE * f2, int flags) |
257 | { | 258 | { |
258 | char buf1[BUFSIZ], buf2[BUFSIZ]; | 259 | char buf1[BUFSIZ], buf2[BUFSIZ]; |
259 | size_t i, j; | 260 | size_t i, j; |
260 | 261 | ||
261 | if ((flags & (D_EMPTY1|D_EMPTY2)) || stb1.st_size != stb2.st_size || | 262 | if ((flags & (D_EMPTY1 | D_EMPTY2)) || stb1.st_size != stb2.st_size || |
262 | (stb1.st_mode & S_IFMT) != (stb2.st_mode & S_IFMT)) | 263 | (stb1.st_mode & S_IFMT) != (stb2.st_mode & S_IFMT)) |
263 | return (1); | 264 | return (1); |
264 | while(1) { | 265 | while (1) { |
265 | i = fread(buf1, 1, sizeof(buf1), f1); | 266 | i = fread(buf1, 1, sizeof(buf1), f1); |
266 | j = fread(buf2, 1, sizeof(buf2), f2); | 267 | j = fread(buf2, 1, sizeof(buf2), f2); |
267 | if (i != j) | 268 | if (i != j) |
268 | return (1); | 269 | return (1); |
269 | if (i == 0 && j == 0) { | 270 | if (i == 0 && j == 0) { |
270 | if (ferror(f1) || ferror(f2)) | 271 | if (ferror(f1) || ferror(f2)) |
271 | return (1); | 272 | return (1); |
272 | return (0); | 273 | return (0); |
273 | } | 274 | } |
274 | if (memcmp(buf1, buf2, i) != 0) | 275 | if (memcmp(buf1, buf2, i) != 0) |
275 | return (1); | 276 | return (1); |
276 | } | 277 | } |
277 | } | 278 | } |
278 | 279 | ||
279 | static void prepare(int i, FILE *fd, off_t filesize) | 280 | static void prepare(int i, FILE * fd, off_t filesize) |
280 | { | 281 | { |
281 | struct line *p; | 282 | struct line *p; |
282 | int h; | 283 | int h; |
283 | size_t j, sz; | 284 | size_t j, sz; |
284 | 285 | ||
285 | rewind(fd); | 286 | rewind(fd); |
286 | 287 | ||
287 | sz = (filesize <= FSIZE_MAX ? filesize : FSIZE_MAX) / 25; | 288 | sz = (filesize <= FSIZE_MAX ? filesize : FSIZE_MAX) / 25; |
288 | if (sz < 100) | 289 | if (sz < 100) |
289 | sz = 100; | 290 | sz = 100; |
290 | 291 | ||
291 | p = xmalloc((sz + 3) * sizeof(struct line)); | 292 | p = xmalloc((sz + 3) * sizeof(struct line)); |
292 | for (j = 0; (h = readhash(fd));) { | 293 | for (j = 0; (h = readhash(fd));) { |
293 | if (j == sz) { | 294 | if (j == sz) { |
294 | sz = sz * 3 / 2; | 295 | sz = sz * 3 / 2; |
295 | p = xrealloc(p, (sz + 3) * sizeof(struct line)); | 296 | p = xrealloc(p, (sz + 3) * sizeof(struct line)); |
296 | } | 297 | } |
297 | p[++j].value = h; | 298 | p[++j].value = h; |
298 | } | 299 | } |
299 | len[i] = j; | 300 | len[i] = j; |
300 | file[i] = p; | 301 | file[i] = p; |
301 | } | 302 | } |
302 | 303 | ||
303 | static void prune(void) | 304 | static void prune(void) |
304 | { | 305 | { |
305 | int i, j; | 306 | int i, j; |
306 | 307 | ||
307 | for (pref = 0; pref < len[0] && pref < len[1] && | 308 | for (pref = 0; pref < len[0] && pref < len[1] && |
308 | file[0][pref + 1].value == file[1][pref + 1].value; | 309 | file[0][pref + 1].value == file[1][pref + 1].value; pref++); |
309 | pref++) | 310 | for (suff = 0; suff < len[0] - pref && suff < len[1] - pref && |
310 | ; | 311 | file[0][len[0] - suff].value == file[1][len[1] - suff].value; |
311 | for (suff = 0; suff < len[0] - pref && suff < len[1] - pref && | 312 | suff++); |
312 | file[0][len[0] - suff].value == file[1][len[1] - suff].value; | 313 | for (j = 0; j < 2; j++) { |
313 | suff++) | 314 | sfile[j] = file[j] + pref; |
314 | ; | 315 | slen[j] = len[j] - pref - suff; |
315 | for (j = 0; j < 2; j++) { | 316 | for (i = 0; i <= slen[j]; i++) |
316 | sfile[j] = file[j] + pref; | 317 | sfile[j][i].serial = i; |
317 | slen[j] = len[j] - pref - suff; | 318 | } |
318 | for (i = 0; i <= slen[j]; i++) | ||
319 | sfile[j][i].serial = i; | ||
320 | } | ||
321 | } | 319 | } |
322 | 320 | ||
323 | static void equiv(struct line *a, int n, struct line *b, int m, int *c) | 321 | static void equiv(struct line *a, int n, struct line *b, int m, int *c) |
324 | { | 322 | { |
325 | int i, j; | 323 | int i, j; |
326 | 324 | ||
327 | i = j = 1; | 325 | i = j = 1; |
328 | while (i <= n && j <= m) { | 326 | while (i <= n && j <= m) { |
329 | if (a[i].value < b[j].value) | 327 | if (a[i].value < b[j].value) |
330 | a[i++].value = 0; | 328 | a[i++].value = 0; |
331 | else if (a[i].value == b[j].value) | 329 | else if (a[i].value == b[j].value) |
332 | a[i++].value = j; | 330 | a[i++].value = j; |
333 | else | 331 | else |
334 | j++; | 332 | j++; |
335 | } | 333 | } |
336 | while (i <= n) | 334 | while (i <= n) |
337 | a[i++].value = 0; | 335 | a[i++].value = 0; |
338 | b[m + 1].value = 0; | 336 | b[m + 1].value = 0; |
339 | j = 0; | 337 | j = 0; |
340 | while (++j <= m) { | 338 | while (++j <= m) { |
341 | c[j] = -b[j].serial; | 339 | c[j] = -b[j].serial; |
342 | while (b[j + 1].value == b[j].value) { | 340 | while (b[j + 1].value == b[j].value) { |
343 | j++; | 341 | j++; |
344 | c[j] = b[j].serial; | 342 | c[j] = b[j].serial; |
345 | } | 343 | } |
346 | } | 344 | } |
347 | c[j] = -1; | 345 | c[j] = -1; |
348 | } | 346 | } |
349 | 347 | ||
350 | static int isqrt(int n) { | 348 | static int isqrt(int n) |
349 | { | ||
351 | int y, x = 1; | 350 | int y, x = 1; |
352 | if (n == 0) return(0); | 351 | |
352 | if (n == 0) | ||
353 | return (0); | ||
353 | 354 | ||
354 | do { | 355 | do { |
355 | y = x; | 356 | y = x; |
@@ -363,120 +364,121 @@ static int isqrt(int n) { | |||
363 | 364 | ||
364 | static int newcand(int x, int y, int pred) | 365 | static int newcand(int x, int y, int pred) |
365 | { | 366 | { |
366 | struct cand *q; | 367 | struct cand *q; |
367 | 368 | ||
368 | if (clen == clistlen) { | 369 | if (clen == clistlen) { |
369 | clistlen = clistlen * 11 / 10; | 370 | clistlen = clistlen * 11 / 10; |
370 | clist = xrealloc(clist, clistlen * sizeof(struct cand)); | 371 | clist = xrealloc(clist, clistlen * sizeof(struct cand)); |
371 | } | 372 | } |
372 | q = clist + clen; | 373 | q = clist + clen; |
373 | q->x = x; | 374 | q->x = x; |
374 | q->y = y; | 375 | q->y = y; |
375 | q->pred = pred; | 376 | q->pred = pred; |
376 | return (clen++); | 377 | return (clen++); |
377 | } | 378 | } |
378 | 379 | ||
379 | 380 | ||
380 | static int search(int *c, int k, int y) | 381 | static int search(int *c, int k, int y) |
381 | { | 382 | { |
382 | int i, j, l, t; | 383 | int i, j, l, t; |
383 | 384 | ||
384 | if (clist[c[k]].y < y) /* quick look for typical case */ | 385 | if (clist[c[k]].y < y) /* quick look for typical case */ |
385 | return (k + 1); | 386 | return (k + 1); |
386 | i = 0; | 387 | i = 0; |
387 | j = k + 1; | 388 | j = k + 1; |
388 | while (1) { | 389 | while (1) { |
389 | l = i + j; | 390 | l = i + j; |
390 | if ((l >>= 1) <= i) | 391 | if ((l >>= 1) <= i) |
391 | break; | 392 | break; |
392 | t = clist[c[l]].y; | 393 | t = clist[c[l]].y; |
393 | if (t > y) | 394 | if (t > y) |
394 | j = l; | 395 | j = l; |
395 | else if (t < y) | 396 | else if (t < y) |
396 | i = l; | 397 | i = l; |
397 | else | 398 | else |
398 | return (l); | 399 | return (l); |
399 | } | 400 | } |
400 | return (l + 1); | 401 | return (l + 1); |
401 | } | 402 | } |
402 | 403 | ||
403 | 404 | ||
404 | static int stone(int *a, int n, int *b, int *c) | 405 | static int stone(int *a, int n, int *b, int *c) |
405 | { | 406 | { |
406 | int i, k, y, j, l; | 407 | int i, k, y, j, l; |
407 | int oldc, tc, oldl; | 408 | int oldc, tc, oldl; |
408 | unsigned int numtries; | 409 | unsigned int numtries; |
410 | |||
409 | #if ENABLE_FEATURE_DIFF_MINIMAL | 411 | #if ENABLE_FEATURE_DIFF_MINIMAL |
410 | const unsigned int bound = (cmd_flags & FLAG_d) ? UINT_MAX : MAX(256, isqrt(n)); | 412 | const unsigned int bound = |
413 | (cmd_flags & FLAG_d) ? UINT_MAX : MAX(256, isqrt(n)); | ||
411 | #else | 414 | #else |
412 | const unsigned int bound = MAX(256, isqrt(n)); | 415 | const unsigned int bound = MAX(256, isqrt(n)); |
413 | #endif | 416 | #endif |
414 | k = 0; | 417 | k = 0; |
415 | c[0] = newcand(0, 0, 0); | 418 | c[0] = newcand(0, 0, 0); |
416 | for (i = 1; i <= n; i++) { | 419 | for (i = 1; i <= n; i++) { |
417 | j = a[i]; | 420 | j = a[i]; |
418 | if (j == 0) | 421 | if (j == 0) |
419 | continue; | 422 | continue; |
420 | y = -b[j]; | 423 | y = -b[j]; |
421 | oldl = 0; | 424 | oldl = 0; |
422 | oldc = c[0]; | 425 | oldc = c[0]; |
423 | numtries = 0; | 426 | numtries = 0; |
424 | do { | 427 | do { |
425 | if (y <= clist[oldc].y) | 428 | if (y <= clist[oldc].y) |
426 | continue; | 429 | continue; |
427 | l = search(c, k, y); | 430 | l = search(c, k, y); |
428 | if (l != oldl + 1) | 431 | if (l != oldl + 1) |
429 | oldc = c[l - 1]; | 432 | oldc = c[l - 1]; |
430 | if (l <= k) { | 433 | if (l <= k) { |
431 | if (clist[c[l]].y <= y) | 434 | if (clist[c[l]].y <= y) |
432 | continue; | 435 | continue; |
433 | tc = c[l]; | 436 | tc = c[l]; |
434 | c[l] = newcand(i, y, oldc); | 437 | c[l] = newcand(i, y, oldc); |
435 | oldc = tc; | 438 | oldc = tc; |
436 | oldl = l; | 439 | oldl = l; |
437 | numtries++; | 440 | numtries++; |
438 | } else { | 441 | } else { |
439 | c[l] = newcand(i, y, oldc); | 442 | c[l] = newcand(i, y, oldc); |
440 | k++; | 443 | k++; |
441 | break; | 444 | break; |
442 | } | 445 | } |
443 | } while ((y = b[++j]) > 0 && numtries < bound); | 446 | } while ((y = b[++j]) > 0 && numtries < bound); |
444 | } | 447 | } |
445 | return (k); | 448 | return (k); |
446 | } | 449 | } |
447 | 450 | ||
448 | static void unravel(int p) | 451 | static void unravel(int p) |
449 | { | 452 | { |
450 | struct cand *q; | 453 | struct cand *q; |
451 | int i; | 454 | int i; |
452 | 455 | ||
453 | for (i = 0; i <= len[0]; i++) | 456 | for (i = 0; i <= len[0]; i++) |
454 | J[i] = i <= pref ? i : | 457 | J[i] = i <= pref ? i : i > len[0] - suff ? i + len[1] - len[0] : 0; |
455 | i > len[0] - suff ? i + len[1] - len[0] : 0; | 458 | for (q = clist + p; q->y != 0; q = clist + q->pred) |
456 | for (q = clist + p; q->y != 0; q = clist + q->pred) | 459 | J[q->x + pref] = q->y + pref; |
457 | J[q->x + pref] = q->y + pref; | ||
458 | } | 460 | } |
459 | 461 | ||
460 | 462 | ||
461 | static void unsort(struct line *f, int l, int *b) | 463 | static void unsort(struct line *f, int l, int *b) |
462 | { | 464 | { |
463 | int *a, i; | 465 | int *a, i; |
464 | 466 | ||
465 | a = xmalloc((l + 1) * sizeof(int)); | 467 | a = xmalloc((l + 1) * sizeof(int)); |
466 | for (i = 1; i <= l; i++) | 468 | for (i = 1; i <= l; i++) |
467 | a[f[i].serial] = f[i].value; | 469 | a[f[i].serial] = f[i].value; |
468 | for (i = 1; i <= l; i++) | 470 | for (i = 1; i <= l; i++) |
469 | b[i] = a[i]; | 471 | b[i] = a[i]; |
470 | free(a); | 472 | free(a); |
471 | } | 473 | } |
472 | 474 | ||
473 | static int skipline(FILE *f) | 475 | static int skipline(FILE * f) |
474 | { | 476 | { |
475 | int i, c; | 477 | int i, c; |
476 | 478 | ||
477 | for (i = 1; (c = getc(f)) != '\n' && c != EOF; i++) | 479 | for (i = 1; (c = getc(f)) != '\n' && c != EOF; i++) |
478 | continue; | 480 | continue; |
479 | return (i); | 481 | return (i); |
480 | } | 482 | } |
481 | 483 | ||
482 | 484 | ||
@@ -486,175 +488,174 @@ static int skipline(FILE *f) | |||
486 | * to confounding by hashing (which result in "jackpot") | 488 | * to confounding by hashing (which result in "jackpot") |
487 | * 2. collect random access indexes to the two files | 489 | * 2. collect random access indexes to the two files |
488 | */ | 490 | */ |
489 | static void check(FILE *f1, FILE *f2) | 491 | static void check(FILE * f1, FILE * f2) |
490 | { | 492 | { |
491 | int i, j, jackpot, c, d; | 493 | int i, j, jackpot, c, d; |
492 | long ctold, ctnew; | 494 | long ctold, ctnew; |
493 | 495 | ||
494 | rewind(f1); | 496 | rewind(f1); |
495 | rewind(f2); | 497 | rewind(f2); |
496 | j = 1; | 498 | j = 1; |
497 | ixold[0] = ixnew[0] = 0; | 499 | ixold[0] = ixnew[0] = 0; |
498 | jackpot = 0; | 500 | jackpot = 0; |
499 | ctold = ctnew = 0; | 501 | ctold = ctnew = 0; |
500 | for (i = 1; i <= len[0]; i++) { | 502 | for (i = 1; i <= len[0]; i++) { |
501 | if (J[i] == 0) { | 503 | if (J[i] == 0) { |
502 | ixold[i] = ctold += skipline(f1); | 504 | ixold[i] = ctold += skipline(f1); |
503 | continue; | 505 | continue; |
504 | } | 506 | } |
505 | while (j < J[i]) { | 507 | while (j < J[i]) { |
506 | ixnew[j] = ctnew += skipline(f2); | 508 | ixnew[j] = ctnew += skipline(f2); |
507 | j++; | 509 | j++; |
508 | } | 510 | } |
509 | if ((cmd_flags & FLAG_b) || (cmd_flags & FLAG_w) || (cmd_flags & FLAG_i)) { | 511 | if ((cmd_flags & FLAG_b) || (cmd_flags & FLAG_w) |
510 | while (1) { | 512 | || (cmd_flags & FLAG_i)) { |
511 | c = getc(f1); | 513 | while (1) { |
512 | d = getc(f2); | 514 | c = getc(f1); |
513 | /* | 515 | d = getc(f2); |
514 | * GNU diff ignores a missing newline | 516 | /* |
515 | * in one file if bflag || wflag. | 517 | * GNU diff ignores a missing newline |
516 | */ | 518 | * in one file if bflag || wflag. |
517 | if (((cmd_flags & FLAG_b) || (cmd_flags & FLAG_w)) && | 519 | */ |
518 | ((c == EOF && d == '\n') || | 520 | if (((cmd_flags & FLAG_b) || (cmd_flags & FLAG_w)) && |
519 | (c == '\n' && d == EOF))) { | 521 | ((c == EOF && d == '\n') || (c == '\n' && d == EOF))) { |
520 | break; | 522 | break; |
521 | } | 523 | } |
522 | ctold++; | 524 | ctold++; |
523 | ctnew++; | 525 | ctnew++; |
524 | if ((cmd_flags & FLAG_b) && isspace(c) && isspace(d)) { | 526 | if ((cmd_flags & FLAG_b) && isspace(c) && isspace(d)) { |
525 | do { | 527 | do { |
526 | if (c == '\n') | 528 | if (c == '\n') |
527 | break; | 529 | break; |
528 | ctold++; | 530 | ctold++; |
529 | } while (isspace(c = getc(f1))); | 531 | } while (isspace(c = getc(f1))); |
530 | do { | 532 | do { |
531 | if (d == '\n') | 533 | if (d == '\n') |
532 | break; | 534 | break; |
533 | ctnew++; | 535 | ctnew++; |
534 | } while (isspace(d = getc(f2))); | 536 | } while (isspace(d = getc(f2))); |
535 | } else if (cmd_flags & FLAG_w) { | 537 | } else if (cmd_flags & FLAG_w) { |
536 | while (isspace(c) && c != '\n') { | 538 | while (isspace(c) && c != '\n') { |
537 | c = getc(f1); | 539 | c = getc(f1); |
538 | ctold++; | 540 | ctold++; |
539 | } | 541 | } |
540 | while (isspace(d) && d != '\n') { | 542 | while (isspace(d) && d != '\n') { |
541 | d = getc(f2); | 543 | d = getc(f2); |
542 | ctnew++; | 544 | ctnew++; |
543 | } | 545 | } |
544 | } | 546 | } |
545 | if (c != d) { | 547 | if (c != d) { |
546 | jackpot++; | 548 | jackpot++; |
547 | J[i] = 0; | 549 | J[i] = 0; |
548 | if (c != '\n' && c != EOF) | 550 | if (c != '\n' && c != EOF) |
549 | ctold += skipline(f1); | 551 | ctold += skipline(f1); |
550 | if (d != '\n' && c != EOF) | 552 | if (d != '\n' && c != EOF) |
551 | ctnew += skipline(f2); | 553 | ctnew += skipline(f2); |
552 | break; | 554 | break; |
553 | } | 555 | } |
554 | if (c == '\n' || c == EOF) | 556 | if (c == '\n' || c == EOF) |
555 | break; | 557 | break; |
556 | } | 558 | } |
557 | } else { | 559 | } else { |
558 | while (1) { | 560 | while (1) { |
559 | ctold++; | 561 | ctold++; |
560 | ctnew++; | 562 | ctnew++; |
561 | if ((c = getc(f1)) != (d = getc(f2))) { | 563 | if ((c = getc(f1)) != (d = getc(f2))) { |
562 | J[i] = 0; | 564 | J[i] = 0; |
563 | if (c != '\n' && c != EOF) | 565 | if (c != '\n' && c != EOF) |
564 | ctold += skipline(f1); | 566 | ctold += skipline(f1); |
565 | if (d != '\n' && c != EOF) | 567 | if (d != '\n' && c != EOF) |
566 | ctnew += skipline(f2); | 568 | ctnew += skipline(f2); |
567 | break; | 569 | break; |
568 | } | 570 | } |
569 | if (c == '\n' || c == EOF) | 571 | if (c == '\n' || c == EOF) |
570 | break; | 572 | break; |
571 | } | 573 | } |
572 | } | 574 | } |
573 | ixold[i] = ctold; | 575 | ixold[i] = ctold; |
574 | ixnew[j] = ctnew; | 576 | ixnew[j] = ctnew; |
575 | j++; | 577 | j++; |
576 | } | 578 | } |
577 | for (; j <= len[1]; j++) | 579 | for (; j <= len[1]; j++) |
578 | ixnew[j] = ctnew += skipline(f2); | 580 | ixnew[j] = ctnew += skipline(f2); |
579 | } | 581 | } |
580 | 582 | ||
581 | /* shellsort CACM #201 */ | 583 | /* shellsort CACM #201 */ |
582 | static void sort(struct line *a, int n) | 584 | static void sort(struct line *a, int n) |
583 | { | 585 | { |
584 | struct line *ai, *aim, w; | 586 | struct line *ai, *aim, w; |
585 | int j, m = 0, k; | 587 | int j, m = 0, k; |
586 | 588 | ||
587 | if (n == 0) | 589 | if (n == 0) |
588 | return; | 590 | return; |
589 | for (j = 1; j <= n; j *= 2) | 591 | for (j = 1; j <= n; j *= 2) |
590 | m = 2 * j - 1; | 592 | m = 2 * j - 1; |
591 | for (m /= 2; m != 0; m /= 2) { | 593 | for (m /= 2; m != 0; m /= 2) { |
592 | k = n - m; | 594 | k = n - m; |
593 | for (j = 1; j <= k; j++) { | 595 | for (j = 1; j <= k; j++) { |
594 | for (ai = &a[j]; ai > a; ai -= m) { | 596 | for (ai = &a[j]; ai > a; ai -= m) { |
595 | aim = &ai[m]; | 597 | aim = &ai[m]; |
596 | if (aim < ai) | 598 | if (aim < ai) |
597 | break; /* wraparound */ | 599 | break; /* wraparound */ |
598 | if (aim->value > ai[0].value || | 600 | if (aim->value > ai[0].value || |
599 | (aim->value == ai[0].value && | 601 | (aim->value == ai[0].value && aim->serial > ai[0].serial)) |
600 | aim->serial > ai[0].serial)) | 602 | break; |
601 | break; | 603 | w.value = ai[0].value; |
602 | w.value = ai[0].value; | 604 | ai[0].value = aim->value; |
603 | ai[0].value = aim->value; | 605 | aim->value = w.value; |
604 | aim->value = w.value; | 606 | w.serial = ai[0].serial; |
605 | w.serial = ai[0].serial; | 607 | ai[0].serial = aim->serial; |
606 | ai[0].serial = aim->serial; | 608 | aim->serial = w.serial; |
607 | aim->serial = w.serial; | 609 | } |
608 | } | 610 | } |
609 | } | 611 | } |
610 | } | ||
611 | } | 612 | } |
612 | 613 | ||
613 | 614 | ||
614 | static void uni_range(int a, int b) | 615 | static void uni_range(int a, int b) |
615 | { | 616 | { |
616 | if (a < b) | 617 | if (a < b) |
617 | printf("%d,%d", a, b - a + 1); | 618 | printf("%d,%d", a, b - a + 1); |
618 | else if (a == b) | 619 | else if (a == b) |
619 | printf("%d", b); | 620 | printf("%d", b); |
620 | else | 621 | else |
621 | printf("%d,0", b); | 622 | printf("%d,0", b); |
622 | } | 623 | } |
623 | 624 | ||
624 | static int fetch(long *f, int a, int b, FILE *lb, int ch) | 625 | static int fetch(long *f, int a, int b, FILE * lb, int ch) |
625 | { | 626 | { |
626 | int i, j, c, lastc, col, nc; | 627 | int i, j, c, lastc, col, nc; |
627 | 628 | ||
628 | if (a > b) | 629 | if (a > b) |
629 | return (0); | 630 | return (0); |
630 | for (i = a; i <= b; i++) { | 631 | for (i = a; i <= b; i++) { |
631 | fseek(lb, f[i - 1], SEEK_SET); | 632 | fseek(lb, f[i - 1], SEEK_SET); |
632 | nc = f[i] - f[i - 1]; | 633 | nc = f[i] - f[i - 1]; |
633 | if (ch != '\0') { | 634 | if (ch != '\0') { |
634 | putchar(ch); | 635 | putchar(ch); |
635 | if (cmd_flags & FLAG_T) | 636 | if (cmd_flags & FLAG_T) |
636 | putchar('\t'); | 637 | putchar('\t'); |
637 | } | 638 | } |
638 | col = 0; | 639 | col = 0; |
639 | for (j = 0, lastc = '\0'; j < nc; j++, lastc = c) { | 640 | for (j = 0, lastc = '\0'; j < nc; j++, lastc = c) { |
640 | if ((c = getc(lb)) == EOF) { | 641 | if ((c = getc(lb)) == EOF) { |
641 | puts("\n\\ No newline at end of file"); | 642 | puts("\n\\ No newline at end of file"); |
642 | return (0); | 643 | return (0); |
643 | } | 644 | } |
644 | if (c == '\t' && (cmd_flags & FLAG_t)) { | 645 | if (c == '\t' && (cmd_flags & FLAG_t)) { |
645 | do { | 646 | do { |
646 | putchar(' '); | 647 | putchar(' '); |
647 | } while (++col & 7); | 648 | } while (++col & 7); |
648 | } else { | 649 | } else { |
649 | putchar(c); | 650 | putchar(c); |
650 | col++; | 651 | col++; |
651 | } | 652 | } |
652 | } | 653 | } |
653 | } | 654 | } |
654 | return (0); | 655 | return (0); |
655 | } | 656 | } |
656 | 657 | ||
657 | static int asciifile(FILE *f) | 658 | static int asciifile(FILE * f) |
658 | { | 659 | { |
659 | #if ENABLE_FEATURE_DIFF_BINARY | 660 | #if ENABLE_FEATURE_DIFF_BINARY |
660 | unsigned char buf[BUFSIZ]; | 661 | unsigned char buf[BUFSIZ]; |
@@ -677,97 +678,93 @@ static int asciifile(FILE *f) | |||
677 | } | 678 | } |
678 | 679 | ||
679 | /* dump accumulated "unified" diff changes */ | 680 | /* dump accumulated "unified" diff changes */ |
680 | static void dump_unified_vec(FILE *f1, FILE *f2) | 681 | static void dump_unified_vec(FILE * f1, FILE * f2) |
681 | { | 682 | { |
682 | struct context_vec *cvp = context_vec_start; | 683 | struct context_vec *cvp = context_vec_start; |
683 | int lowa, upb, lowc, upd; | 684 | int lowa, upb, lowc, upd; |
684 | int a, b, c, d; | 685 | int a, b, c, d; |
685 | char ch; | 686 | char ch; |
686 | 687 | ||
687 | if (context_vec_start > context_vec_ptr) | 688 | if (context_vec_start > context_vec_ptr) |
688 | return; | 689 | return; |
689 | 690 | ||
690 | b = d = 0; /* gcc */ | 691 | b = d = 0; /* gcc */ |
691 | lowa = MAX(1, cvp->a - context); | 692 | lowa = MAX(1, cvp->a - context); |
692 | upb = MIN(len[0], context_vec_ptr->b + context); | 693 | upb = MIN(len[0], context_vec_ptr->b + context); |
693 | lowc = MAX(1, cvp->c - context); | 694 | lowc = MAX(1, cvp->c - context); |
694 | upd = MIN(len[1], context_vec_ptr->d + context); | 695 | upd = MIN(len[1], context_vec_ptr->d + context); |
695 | 696 | ||
696 | fputs("@@ -", stdout); | 697 | fputs("@@ -", stdout); |
697 | uni_range(lowa, upb); | 698 | uni_range(lowa, upb); |
698 | fputs(" +", stdout); | 699 | fputs(" +", stdout); |
699 | uni_range(lowc, upd); | 700 | uni_range(lowc, upd); |
700 | fputs(" @@", stdout); | 701 | fputs(" @@", stdout); |
701 | putchar('\n'); | 702 | putchar('\n'); |
702 | 703 | ||
703 | /* | 704 | /* |
704 | * Output changes in "unified" diff format--the old and new lines | 705 | * Output changes in "unified" diff format--the old and new lines |
705 | * are printed together. | 706 | * are printed together. |
706 | */ | 707 | */ |
707 | for (; cvp <= context_vec_ptr; cvp++) { | 708 | for (; cvp <= context_vec_ptr; cvp++) { |
708 | a = cvp->a; | 709 | a = cvp->a; |
709 | b = cvp->b; | 710 | b = cvp->b; |
710 | c = cvp->c; | 711 | c = cvp->c; |
711 | d = cvp->d; | 712 | d = cvp->d; |
712 | 713 | ||
713 | /* | 714 | /* |
714 | * c: both new and old changes | 715 | * c: both new and old changes |
715 | * d: only changes in the old file | 716 | * d: only changes in the old file |
716 | * a: only changes in the new file | 717 | * a: only changes in the new file |
717 | */ | 718 | */ |
718 | if (a <= b && c <= d) | 719 | if (a <= b && c <= d) |
719 | ch = 'c'; | 720 | ch = 'c'; |
720 | else | 721 | else |
721 | ch = (a <= b) ? 'd' : 'a'; | 722 | ch = (a <= b) ? 'd' : 'a'; |
722 | #if 0 | 723 | #if 0 |
723 | switch (ch) { | 724 | switch (ch) { |
724 | case 'c': | 725 | case 'c': |
725 | fetch(ixold, lowa, a - 1, f1, ' '); | 726 | fetch(ixold, lowa, a - 1, f1, ' '); |
726 | fetch(ixold, a, b, f1, '-'); | 727 | fetch(ixold, a, b, f1, '-'); |
727 | fetch(ixnew, c, d, f2, '+'); | 728 | fetch(ixnew, c, d, f2, '+'); |
728 | break; | 729 | break; |
729 | case 'd': | 730 | case 'd': |
730 | fetch(ixold, lowa, a - 1, f1, ' '); | 731 | fetch(ixold, lowa, a - 1, f1, ' '); |
731 | fetch(ixold, a, b, f1, '-'); | 732 | fetch(ixold, a, b, f1, '-'); |
732 | break; | 733 | break; |
733 | case 'a': | 734 | case 'a': |
734 | fetch(ixnew, lowc, c - 1, f2, ' '); | 735 | fetch(ixnew, lowc, c - 1, f2, ' '); |
735 | fetch(ixnew, c, d, f2, '+'); | 736 | fetch(ixnew, c, d, f2, '+'); |
736 | break; | 737 | break; |
737 | } | 738 | } |
738 | #else | 739 | #else |
739 | if (ch == 'c' || ch == 'd') { | 740 | if (ch == 'c' || ch == 'd') { |
740 | fetch(ixold, lowa, a - 1, f1, ' '); | 741 | fetch(ixold, lowa, a - 1, f1, ' '); |
741 | fetch(ixold, a, b, f1, '-'); | 742 | fetch(ixold, a, b, f1, '-'); |
742 | } | 743 | } |
743 | if (ch == 'a') | 744 | if (ch == 'a') |
744 | fetch(ixnew, lowc, c - 1, f2, ' '); | 745 | fetch(ixnew, lowc, c - 1, f2, ' '); |
745 | if (ch == 'c' || ch == 'a') | 746 | if (ch == 'c' || ch == 'a') |
746 | fetch(ixnew, c, d, f2, '+'); | 747 | fetch(ixnew, c, d, f2, '+'); |
747 | #endif | 748 | #endif |
748 | lowa = b + 1; | 749 | lowa = b + 1; |
749 | lowc = d + 1; | 750 | lowc = d + 1; |
750 | } | 751 | } |
751 | fetch(ixnew, d + 1, upd, f2, ' '); | 752 | fetch(ixnew, d + 1, upd, f2, ' '); |
752 | 753 | ||
753 | context_vec_ptr = context_vec_start - 1; | 754 | context_vec_ptr = context_vec_start - 1; |
754 | } | 755 | } |
755 | 756 | ||
756 | 757 | ||
757 | static void print_header(const char *file1, const char *file2) | 758 | static void print_header(const char *file1, const char *file2) |
758 | { | 759 | { |
759 | if (label[0] != NULL) | 760 | if (label[0] != NULL) |
760 | printf("%s %s\n", "---", | 761 | printf("%s %s\n", "---", label[0]); |
761 | label[0]); | 762 | else |
762 | else | 763 | printf("%s %s\t%s", "---", file1, ctime(&stb1.st_mtime)); |
763 | printf("%s %s\t%s", "---", | 764 | if (label[1] != NULL) |
764 | file1, ctime(&stb1.st_mtime)); | 765 | printf("%s %s\n", "+++", label[1]); |
765 | if (label[1] != NULL) | 766 | else |
766 | printf("%s %s\n", "+++", | 767 | printf("%s %s\t%s", "+++", file2, ctime(&stb2.st_mtime)); |
767 | label[1]); | ||
768 | else | ||
769 | printf("%s %s\t%s", "+++", | ||
770 | file2, ctime(&stb2.st_mtime)); | ||
771 | } | 768 | } |
772 | 769 | ||
773 | 770 | ||
@@ -779,77 +776,82 @@ static void print_header(const char *file1, const char *file2) | |||
779 | * lines appended (beginning at b). If c is greater than d then there are | 776 | * lines appended (beginning at b). If c is greater than d then there are |
780 | * lines missing from the to file. | 777 | * lines missing from the to file. |
781 | */ | 778 | */ |
782 | static void change(char *file1, FILE *f1, char *file2, FILE *f2, int a, int b, int c, int d) | 779 | static void change(char *file1, FILE * f1, char *file2, FILE * f2, int a, |
780 | int b, int c, int d) | ||
783 | { | 781 | { |
784 | static size_t max_context = 64; | 782 | static size_t max_context = 64; |
785 | 783 | ||
786 | if (a > b && c > d) return; | 784 | if (a > b && c > d) |
787 | if (cmd_flags & FLAG_q) return; | 785 | return; |
788 | 786 | if (cmd_flags & FLAG_q) | |
789 | /* | 787 | return; |
790 | * Allocate change records as needed. | 788 | |
791 | */ | 789 | /* |
792 | if (context_vec_ptr == context_vec_end - 1) { | 790 | * Allocate change records as needed. |
793 | ptrdiff_t offset = context_vec_ptr - context_vec_start; | 791 | */ |
794 | max_context <<= 1; | 792 | if (context_vec_ptr == context_vec_end - 1) { |
795 | context_vec_start = xrealloc(context_vec_start, | 793 | ptrdiff_t offset = context_vec_ptr - context_vec_start; |
796 | max_context * sizeof(struct context_vec)); | 794 | |
797 | context_vec_end = context_vec_start + max_context; | 795 | max_context <<= 1; |
798 | context_vec_ptr = context_vec_start + offset; | 796 | context_vec_start = xrealloc(context_vec_start, |
799 | } | 797 | max_context * |
800 | if (anychange == 0) { | 798 | sizeof(struct context_vec)); |
801 | /* | 799 | context_vec_end = context_vec_start + max_context; |
802 | * Print the context/unidiff header first time through. | 800 | context_vec_ptr = context_vec_start + offset; |
803 | */ | 801 | } |
804 | print_header(file1, file2); | 802 | if (anychange == 0) { |
805 | anychange = 1; | 803 | /* |
806 | } else if (a > context_vec_ptr->b + (2 * context) + 1 && | 804 | * Print the context/unidiff header first time through. |
807 | c > context_vec_ptr->d + (2 * context) + 1) { | 805 | */ |
808 | /* | 806 | print_header(file1, file2); |
809 | * If this change is more than 'context' lines from the | 807 | anychange = 1; |
810 | * previous change, dump the record and reset it. | 808 | } else if (a > context_vec_ptr->b + (2 * context) + 1 && |
811 | */ | 809 | c > context_vec_ptr->d + (2 * context) + 1) { |
812 | dump_unified_vec(f1, f2); | 810 | /* |
813 | } | 811 | * If this change is more than 'context' lines from the |
814 | context_vec_ptr++; | 812 | * previous change, dump the record and reset it. |
815 | context_vec_ptr->a = a; | 813 | */ |
816 | context_vec_ptr->b = b; | 814 | dump_unified_vec(f1, f2); |
817 | context_vec_ptr->c = c; | 815 | } |
818 | context_vec_ptr->d = d; | 816 | context_vec_ptr++; |
819 | return; | 817 | context_vec_ptr->a = a; |
820 | 818 | context_vec_ptr->b = b; | |
819 | context_vec_ptr->c = c; | ||
820 | context_vec_ptr->d = d; | ||
821 | return; | ||
822 | |||
821 | } | 823 | } |
822 | 824 | ||
823 | 825 | ||
824 | static void output(char *file1, FILE *f1, char *file2, FILE *f2) | 826 | static void output(char *file1, FILE * f1, char *file2, FILE * f2) |
825 | { | 827 | { |
826 | 828 | ||
827 | /* Note that j0 and j1 can't be used as they are defined in math.h. | 829 | /* Note that j0 and j1 can't be used as they are defined in math.h. |
828 | * This also allows the rather amusing variable 'j00'... */ | 830 | * This also allows the rather amusing variable 'j00'... */ |
829 | int m, i0, i1, j00, j01; | 831 | int m, i0, i1, j00, j01; |
830 | 832 | ||
831 | rewind(f1); | 833 | rewind(f1); |
832 | rewind(f2); | 834 | rewind(f2); |
833 | m = len[0]; | 835 | m = len[0]; |
834 | J[0] = 0; | 836 | J[0] = 0; |
835 | J[m + 1] = len[1] + 1; | 837 | J[m + 1] = len[1] + 1; |
836 | for (i0 = 1; i0 <= m; i0 = i1 + 1) { | 838 | for (i0 = 1; i0 <= m; i0 = i1 + 1) { |
837 | while (i0 <= m && J[i0] == J[i0 - 1] + 1) | 839 | while (i0 <= m && J[i0] == J[i0 - 1] + 1) |
838 | i0++; | 840 | i0++; |
839 | j00 = J[i0 - 1] + 1; | 841 | j00 = J[i0 - 1] + 1; |
840 | i1 = i0 - 1; | 842 | i1 = i0 - 1; |
841 | while (i1 < m && J[i1 + 1] == 0) | 843 | while (i1 < m && J[i1 + 1] == 0) |
842 | i1++; | 844 | i1++; |
843 | j01 = J[i1 + 1] - 1; | 845 | j01 = J[i1 + 1] - 1; |
844 | J[i1] = j01; | 846 | J[i1] = j01; |
845 | change(file1, f1, file2, f2, i0, i1, j00, j01); | 847 | change(file1, f1, file2, f2, i0, i1, j00, j01); |
846 | } | 848 | } |
847 | if (m == 0) { | 849 | if (m == 0) { |
848 | change(file1, f1, file2, f2, 1, 0, 1, len[1]); | 850 | change(file1, f1, file2, f2, 1, 0, 1, len[1]); |
849 | } | 851 | } |
850 | if (anychange != 0) { | 852 | if (anychange != 0) { |
851 | dump_unified_vec(f1, f2); | 853 | dump_unified_vec(f1, f2); |
852 | } | 854 | } |
853 | } | 855 | } |
854 | 856 | ||
855 | /* | 857 | /* |
@@ -917,108 +919,109 @@ static void output(char *file1, FILE *f1, char *file2, FILE *f2) | |||
917 | 919 | ||
918 | static int diffreg(char *ofile1, char *ofile2, int flags) | 920 | static int diffreg(char *ofile1, char *ofile2, int flags) |
919 | { | 921 | { |
920 | char *file1 = ofile1; | 922 | char *file1 = ofile1; |
921 | char *file2 = ofile2; | 923 | char *file2 = ofile2; |
922 | FILE *f1 = NULL; | 924 | FILE *f1 = NULL; |
923 | FILE *f2 = NULL; | 925 | FILE *f2 = NULL; |
924 | int rval = D_SAME; | 926 | int rval = D_SAME; |
925 | int i; | 927 | int i; |
926 | 928 | ||
927 | anychange = 0; | 929 | anychange = 0; |
928 | context_vec_ptr = context_vec_start - 1; | 930 | context_vec_ptr = context_vec_start - 1; |
929 | 931 | ||
930 | if (S_ISDIR(stb1.st_mode) != S_ISDIR(stb2.st_mode)) | 932 | if (S_ISDIR(stb1.st_mode) != S_ISDIR(stb2.st_mode)) |
931 | return (S_ISDIR(stb1.st_mode) ? D_MISMATCH1 : D_MISMATCH2); | 933 | return (S_ISDIR(stb1.st_mode) ? D_MISMATCH1 : D_MISMATCH2); |
932 | if (strcmp(file1, "-") == 0 && strcmp(file2, "-") == 0) | 934 | if (strcmp(file1, "-") == 0 && strcmp(file2, "-") == 0) |
933 | goto closem; | 935 | goto closem; |
934 | 936 | ||
935 | if (flags & D_EMPTY1) | 937 | if (flags & D_EMPTY1) |
936 | f1 = bb_xfopen(bb_dev_null, "r"); | 938 | f1 = bb_xfopen(bb_dev_null, "r"); |
937 | else { | 939 | else { |
938 | if (strcmp(file1, "-") == 0) | 940 | if (strcmp(file1, "-") == 0) |
939 | f1 = stdin; | 941 | f1 = stdin; |
940 | else | 942 | else |
941 | f1 = bb_xfopen(file1, "r"); | 943 | f1 = bb_xfopen(file1, "r"); |
942 | } | 944 | } |
943 | 945 | ||
944 | if (flags & D_EMPTY2) | 946 | if (flags & D_EMPTY2) |
945 | f2 = bb_xfopen(bb_dev_null, "r"); | 947 | f2 = bb_xfopen(bb_dev_null, "r"); |
946 | else { | 948 | else { |
947 | if (strcmp(file2, "-") == 0) | 949 | if (strcmp(file2, "-") == 0) |
948 | f2 = stdin; | 950 | f2 = stdin; |
949 | else | 951 | else |
950 | f2 = bb_xfopen(file2, "r"); | 952 | f2 = bb_xfopen(file2, "r"); |
951 | } | 953 | } |
952 | 954 | ||
953 | if ((i=files_differ(f1, f2, flags)) == 0) | 955 | if ((i = files_differ(f1, f2, flags)) == 0) |
954 | goto closem; | 956 | goto closem; |
955 | else if (i != 1) {/* 1 == ok */ | 957 | else if (i != 1) { /* 1 == ok */ |
956 | /* error */ | 958 | /* error */ |
957 | status |= 2; | 959 | status |= 2; |
958 | goto closem; | 960 | goto closem; |
959 | } | 961 | } |
960 | 962 | ||
961 | if (!asciifile(f1) || !asciifile(f2)) { | 963 | if (!asciifile(f1) || !asciifile(f2)) { |
962 | rval = D_BINARY; | 964 | rval = D_BINARY; |
963 | status |= 1; | 965 | status |= 1; |
964 | goto closem; | 966 | goto closem; |
965 | } | 967 | } |
966 | 968 | ||
967 | prepare(0, f1, stb1.st_size); | 969 | prepare(0, f1, stb1.st_size); |
968 | prepare(1, f2, stb2.st_size); | 970 | prepare(1, f2, stb2.st_size); |
969 | prune(); | 971 | prune(); |
970 | sort(sfile[0], slen[0]); | 972 | sort(sfile[0], slen[0]); |
971 | sort(sfile[1], slen[1]); | 973 | sort(sfile[1], slen[1]); |
972 | 974 | ||
973 | member = (int *)file[1]; | 975 | member = (int *) file[1]; |
974 | equiv(sfile[0], slen[0], sfile[1], slen[1], member); | 976 | equiv(sfile[0], slen[0], sfile[1], slen[1], member); |
975 | member = xrealloc(member, (slen[1] + 2) * sizeof(int)); | 977 | member = xrealloc(member, (slen[1] + 2) * sizeof(int)); |
976 | 978 | ||
977 | class = (int *)file[0]; | 979 | class = (int *) file[0]; |
978 | unsort(sfile[0], slen[0], class); | 980 | unsort(sfile[0], slen[0], class); |
979 | class = xrealloc(class, (slen[0] + 2) * sizeof(int)); | 981 | class = xrealloc(class, (slen[0] + 2) * sizeof(int)); |
980 | 982 | ||
981 | klist = xmalloc((slen[0] + 2) * sizeof(int)); | 983 | klist = xmalloc((slen[0] + 2) * sizeof(int)); |
982 | clen = 0; | 984 | clen = 0; |
983 | clistlen = 100; | 985 | clistlen = 100; |
984 | clist = xmalloc(clistlen * sizeof(struct cand)); | 986 | clist = xmalloc(clistlen * sizeof(struct cand)); |
985 | i = stone(class, slen[0], member, klist); | 987 | i = stone(class, slen[0], member, klist); |
986 | free(member); | 988 | free(member); |
987 | free(class); | 989 | free(class); |
988 | 990 | ||
989 | J = xrealloc(J, (len[0] + 2) * sizeof(int)); | 991 | J = xrealloc(J, (len[0] + 2) * sizeof(int)); |
990 | unravel(klist[i]); | 992 | unravel(klist[i]); |
991 | free(clist); | 993 | free(clist); |
992 | free(klist); | 994 | free(klist); |
993 | 995 | ||
994 | ixold = xrealloc(ixold, (len[0] + 2) * sizeof(long)); | 996 | ixold = xrealloc(ixold, (len[0] + 2) * sizeof(long)); |
995 | ixnew = xrealloc(ixnew, (len[1] + 2) * sizeof(long)); | 997 | ixnew = xrealloc(ixnew, (len[1] + 2) * sizeof(long)); |
996 | check(f1, f2); | 998 | check(f1, f2); |
997 | output(file1, f1, file2, f2); | 999 | output(file1, f1, file2, f2); |
998 | 1000 | ||
999 | closem: | 1001 | closem: |
1000 | if (anychange) { | 1002 | if (anychange) { |
1001 | status |= 1; | 1003 | status |= 1; |
1002 | if (rval == D_SAME) | 1004 | if (rval == D_SAME) |
1003 | rval = D_DIFFER; | 1005 | rval = D_DIFFER; |
1004 | } | 1006 | } |
1005 | if (f1 != NULL) | 1007 | if (f1 != NULL) |
1006 | fclose(f1); | 1008 | fclose(f1); |
1007 | if (f2 != NULL) | 1009 | if (f2 != NULL) |
1008 | fclose(f2); | 1010 | fclose(f2); |
1009 | if (file1 != ofile1) | 1011 | if (file1 != ofile1) |
1010 | free(file1); | 1012 | free(file1); |
1011 | if (file2 != ofile2) | 1013 | if (file2 != ofile2) |
1012 | free(file2); | 1014 | free(file2); |
1013 | return (rval); | 1015 | return (rval); |
1014 | } | 1016 | } |
1015 | 1017 | ||
1016 | #if ENABLE_FEATURE_DIFF_DIR | 1018 | #if ENABLE_FEATURE_DIFF_DIR |
1017 | static void do_diff (char *dir1, char *path1, char *dir2, char *path2) { | 1019 | static void do_diff(char *dir1, char *path1, char *dir2, char *path2) |
1018 | 1020 | { | |
1021 | |||
1019 | int flags = D_HEADER; | 1022 | int flags = D_HEADER; |
1020 | int val; | 1023 | int val; |
1021 | 1024 | ||
1022 | char *fullpath1 = bb_xasprintf("%s/%s", dir1, path1); | 1025 | char *fullpath1 = bb_xasprintf("%s/%s", dir1, path1); |
1023 | char *fullpath2 = bb_xasprintf("%s/%s", dir2, path2); | 1026 | char *fullpath2 = bb_xasprintf("%s/%s", dir2, path2); |
1024 | 1027 | ||
@@ -1036,7 +1039,7 @@ static void do_diff (char *dir1, char *path1, char *dir2, char *path2) { | |||
1036 | 1039 | ||
1037 | if (stb1.st_mode == 0) | 1040 | if (stb1.st_mode == 0) |
1038 | stb1.st_mode = stb2.st_mode; | 1041 | stb1.st_mode = stb2.st_mode; |
1039 | 1042 | ||
1040 | if (S_ISDIR(stb1.st_mode) && S_ISDIR(stb2.st_mode)) { | 1043 | if (S_ISDIR(stb1.st_mode) && S_ISDIR(stb2.st_mode)) { |
1041 | printf("Common subdirectories: %s and %s\n", fullpath1, fullpath2); | 1044 | printf("Common subdirectories: %s and %s\n", fullpath1, fullpath2); |
1042 | return; | 1045 | return; |
@@ -1048,33 +1051,37 @@ static void do_diff (char *dir1, char *path1, char *dir2, char *path2) { | |||
1048 | val = D_SKIPPED2; | 1051 | val = D_SKIPPED2; |
1049 | else | 1052 | else |
1050 | val = diffreg(fullpath1, fullpath2, flags); | 1053 | val = diffreg(fullpath1, fullpath2, flags); |
1051 | 1054 | ||
1052 | print_status(val, fullpath1, fullpath2, NULL); | 1055 | print_status(val, fullpath1, fullpath2, NULL); |
1053 | } | 1056 | } |
1054 | #endif | 1057 | #endif |
1055 | 1058 | ||
1056 | #if ENABLE_FEATURE_DIFF_DIR | 1059 | #if ENABLE_FEATURE_DIFF_DIR |
1057 | static int dir_strcmp(const void *p1, const void *p2) { | 1060 | static int dir_strcmp(const void *p1, const void *p2) |
1058 | return strcmp(*(char * const *)p1, *(char * const *)p2); | 1061 | { |
1062 | return strcmp(*(char *const *) p1, *(char *const *) p2); | ||
1059 | } | 1063 | } |
1060 | 1064 | ||
1061 | /* This function adds a filename to dl, the directory listing. */ | 1065 | /* This function adds a filename to dl, the directory listing. */ |
1062 | 1066 | ||
1063 | static int add_to_dirlist (const char *filename, | 1067 | static int add_to_dirlist(const char *filename, |
1064 | struct stat ATTRIBUTE_UNUSED *sb, void *userdata) { | 1068 | struct stat ATTRIBUTE_UNUSED * sb, void *userdata) |
1069 | { | ||
1065 | dl_count++; | 1070 | dl_count++; |
1066 | dl = xrealloc(dl, dl_count * sizeof(char *)); | 1071 | dl = xrealloc(dl, dl_count * sizeof(char *)); |
1067 | dl[dl_count - 1] = bb_xstrdup(filename); | 1072 | dl[dl_count - 1] = bb_xstrdup(filename); |
1068 | if (cmd_flags & FLAG_r) { | 1073 | if (cmd_flags & FLAG_r) { |
1069 | int *pp = (int *) userdata; | 1074 | int *pp = (int *) userdata; |
1070 | int path_len = *pp + 1; | 1075 | int path_len = *pp + 1; |
1076 | |||
1071 | dl[dl_count - 1] = &(dl[dl_count - 1])[path_len]; | 1077 | dl[dl_count - 1] = &(dl[dl_count - 1])[path_len]; |
1072 | } | 1078 | } |
1073 | return TRUE; | 1079 | return TRUE; |
1074 | } | 1080 | } |
1075 | 1081 | ||
1076 | /* This returns a sorted directory listing. */ | 1082 | /* This returns a sorted directory listing. */ |
1077 | static char **get_dir(char *path) { | 1083 | static char **get_dir(char *path) |
1084 | { | ||
1078 | 1085 | ||
1079 | int i; | 1086 | int i; |
1080 | char **retval; | 1087 | char **retval; |
@@ -1094,7 +1101,8 @@ static char **get_dir(char *path) { | |||
1094 | 1101 | ||
1095 | /* Now fill dl with a listing. */ | 1102 | /* Now fill dl with a listing. */ |
1096 | if (cmd_flags & FLAG_r) | 1103 | if (cmd_flags & FLAG_r) |
1097 | recursive_action(path, TRUE, TRUE, FALSE, add_to_dirlist, NULL, userdata); | 1104 | recursive_action(path, TRUE, TRUE, FALSE, add_to_dirlist, NULL, |
1105 | userdata); | ||
1098 | else { | 1106 | else { |
1099 | DIR *dp; | 1107 | DIR *dp; |
1100 | struct dirent *ep; | 1108 | struct dirent *ep; |
@@ -1119,29 +1127,30 @@ static char **get_dir(char *path) { | |||
1119 | return retval; | 1127 | return retval; |
1120 | } | 1128 | } |
1121 | 1129 | ||
1122 | static void diffdir (char *p1, char *p2) { | 1130 | static void diffdir(char *p1, char *p2) |
1123 | 1131 | { | |
1132 | |||
1124 | char **dirlist1, **dirlist2; | 1133 | char **dirlist1, **dirlist2; |
1125 | char *dp1, *dp2; | 1134 | char *dp1, *dp2; |
1126 | int dirlist1_count, dirlist2_count; | 1135 | int dirlist1_count, dirlist2_count; |
1127 | int pos; | 1136 | int pos; |
1128 | 1137 | ||
1129 | /* Check for trailing slashes. */ | 1138 | /* Check for trailing slashes. */ |
1130 | 1139 | ||
1131 | if (p1[strlen(p1) - 1] == '/') | 1140 | if (p1[strlen(p1) - 1] == '/') |
1132 | p1[strlen(p1) - 1] = '\0'; | 1141 | p1[strlen(p1) - 1] = '\0'; |
1133 | if (p2[strlen(p2) - 1] == '/') | 1142 | if (p2[strlen(p2) - 1] == '/') |
1134 | p2[strlen(p2) - 1] = '\0'; | 1143 | p2[strlen(p2) - 1] = '\0'; |
1135 | 1144 | ||
1136 | /* Get directory listings for p1 and p2. */ | 1145 | /* Get directory listings for p1 and p2. */ |
1137 | 1146 | ||
1138 | dirlist1 = get_dir(p1); | 1147 | dirlist1 = get_dir(p1); |
1139 | dirlist1_count = dl_count; | 1148 | dirlist1_count = dl_count; |
1140 | dirlist1[dirlist1_count] = NULL; | 1149 | dirlist1[dirlist1_count] = NULL; |
1141 | dirlist2 = get_dir(p2); | 1150 | dirlist2 = get_dir(p2); |
1142 | dirlist2_count = dl_count; | 1151 | dirlist2_count = dl_count; |
1143 | dirlist2[dirlist2_count] = NULL; | 1152 | dirlist2[dirlist2_count] = NULL; |
1144 | 1153 | ||
1145 | /* If -S was set, find the starting point. */ | 1154 | /* If -S was set, find the starting point. */ |
1146 | if (start) { | 1155 | if (start) { |
1147 | while (*dirlist1 != NULL && strcmp(*dirlist1, start) < 0) | 1156 | while (*dirlist1 != NULL && strcmp(*dirlist1, start) < 0) |
@@ -1151,11 +1160,11 @@ static void diffdir (char *p1, char *p2) { | |||
1151 | if ((*dirlist1 == NULL) || (*dirlist2 == NULL)) | 1160 | if ((*dirlist1 == NULL) || (*dirlist2 == NULL)) |
1152 | bb_error_msg("Invalid argument to -S"); | 1161 | bb_error_msg("Invalid argument to -S"); |
1153 | } | 1162 | } |
1154 | 1163 | ||
1155 | /* Now that both dirlist1 and dirlist2 contain sorted directory | 1164 | /* Now that both dirlist1 and dirlist2 contain sorted directory |
1156 | * listings, we can start to go through dirlist1. If both listings | 1165 | * listings, we can start to go through dirlist1. If both listings |
1157 | * contain the same file, then do a normal diff. Otherwise, behaviour | 1166 | * contain the same file, then do a normal diff. Otherwise, behaviour |
1158 | * is determined by whether the -N flag is set. */ | 1167 | * is determined by whether the -N flag is set. */ |
1159 | while (*dirlist1 != NULL || *dirlist2 != NULL) { | 1168 | while (*dirlist1 != NULL || *dirlist2 != NULL) { |
1160 | dp1 = *dirlist1; | 1169 | dp1 = *dirlist1; |
1161 | dp2 = *dirlist2; | 1170 | dp2 = *dirlist2; |
@@ -1164,15 +1173,13 @@ static void diffdir (char *p1, char *p2) { | |||
1164 | do_diff(p1, dp1, p2, dp2); | 1173 | do_diff(p1, dp1, p2, dp2); |
1165 | dirlist1++; | 1174 | dirlist1++; |
1166 | dirlist2++; | 1175 | dirlist2++; |
1167 | } | 1176 | } else if (pos < 0) { |
1168 | else if (pos < 0) { | ||
1169 | if (cmd_flags & FLAG_N) | 1177 | if (cmd_flags & FLAG_N) |
1170 | do_diff(p1, dp1, p2, NULL); | 1178 | do_diff(p1, dp1, p2, NULL); |
1171 | else | 1179 | else |
1172 | print_only(p1, strlen(p1) + 1, dp1); | 1180 | print_only(p1, strlen(p1) + 1, dp1); |
1173 | dirlist1++; | 1181 | dirlist1++; |
1174 | } | 1182 | } else { |
1175 | else { | ||
1176 | if (cmd_flags & FLAG_N) | 1183 | if (cmd_flags & FLAG_N) |
1177 | do_diff(p1, NULL, p2, dp2); | 1184 | do_diff(p1, NULL, p2, dp2); |
1178 | else | 1185 | else |
@@ -1185,14 +1192,18 @@ static void diffdir (char *p1, char *p2) { | |||
1185 | 1192 | ||
1186 | 1193 | ||
1187 | 1194 | ||
1188 | int diff_main(int argc, char **argv) { | 1195 | int diff_main(int argc, char **argv) |
1196 | { | ||
1189 | char *ep; | 1197 | char *ep; |
1190 | int gotstdin = 0; | 1198 | int gotstdin = 0; |
1191 | 1199 | ||
1192 | char *U_opt; | 1200 | char *U_opt; |
1193 | llist_t *L_arg = NULL; | 1201 | llist_t *L_arg = NULL; |
1202 | |||
1194 | bb_opt_complementally = "L::"; | 1203 | bb_opt_complementally = "L::"; |
1195 | cmd_flags = bb_getopt_ulflags(argc, argv, "abdiL:NqrsS:tTU:wu", &L_arg, &start, &U_opt); | 1204 | cmd_flags = |
1205 | bb_getopt_ulflags(argc, argv, "abdiL:NqrsS:tTU:wu", &L_arg, &start, | ||
1206 | &U_opt); | ||
1196 | 1207 | ||
1197 | if (cmd_flags & FLAG_L) { | 1208 | if (cmd_flags & FLAG_L) { |
1198 | while (L_arg) { | 1209 | while (L_arg) { |
@@ -1209,13 +1220,14 @@ int diff_main(int argc, char **argv) { | |||
1209 | /* If both label[0] and label[1] were set, they need to be swapped. */ | 1220 | /* If both label[0] and label[1] were set, they need to be swapped. */ |
1210 | if (label[0] && label[1]) { | 1221 | if (label[0] && label[1]) { |
1211 | char *tmp; | 1222 | char *tmp; |
1223 | |||
1212 | tmp = label[1]; | 1224 | tmp = label[1]; |
1213 | label[1] = label[0]; | 1225 | label[1] = label[0]; |
1214 | label[0] = tmp; | 1226 | label[0] = tmp; |
1215 | } | 1227 | } |
1216 | } | 1228 | } |
1217 | 1229 | ||
1218 | context = 3; /* This is the default number of lines of context. */ | 1230 | context = 3; /* This is the default number of lines of context. */ |
1219 | if (cmd_flags & FLAG_U) { | 1231 | if (cmd_flags & FLAG_U) { |
1220 | context = strtol(U_opt, &ep, 10); | 1232 | context = strtol(U_opt, &ep, 10); |
1221 | if (context == 0) { | 1233 | if (context == 0) { |
@@ -1224,36 +1236,35 @@ int diff_main(int argc, char **argv) { | |||
1224 | } | 1236 | } |
1225 | } | 1237 | } |
1226 | argc -= optind; | 1238 | argc -= optind; |
1227 | argv += optind; | 1239 | argv += optind; |
1228 | 1240 | ||
1229 | /* | 1241 | /* |
1230 | * Do sanity checks, fill in stb1 and stb2 and call the appropriate | 1242 | * Do sanity checks, fill in stb1 and stb2 and call the appropriate |
1231 | * driver routine. Both drivers use the contents of stb1 and stb2. | 1243 | * driver routine. Both drivers use the contents of stb1 and stb2. |
1232 | */ | 1244 | */ |
1233 | if (argc < 2) { | 1245 | if (argc < 2) { |
1234 | bb_error_msg("Missing filename"); | 1246 | bb_error_msg("Missing filename"); |
1235 | bb_show_usage(); | 1247 | bb_show_usage(); |
1236 | } | 1248 | } |
1237 | if (strcmp(argv[0], "-") == 0) { | 1249 | if (strcmp(argv[0], "-") == 0) { |
1238 | fstat(STDIN_FILENO, &stb1); | 1250 | fstat(STDIN_FILENO, &stb1); |
1239 | gotstdin = 1; | 1251 | gotstdin = 1; |
1240 | } else | 1252 | } else |
1241 | xstat(argv[0], &stb1); | 1253 | xstat(argv[0], &stb1); |
1242 | if (strcmp(argv[1], "-") == 0) { | 1254 | if (strcmp(argv[1], "-") == 0) { |
1243 | fstat(STDIN_FILENO, &stb2); | 1255 | fstat(STDIN_FILENO, &stb2); |
1244 | gotstdin = 1; | 1256 | gotstdin = 1; |
1245 | } else | 1257 | } else |
1246 | xstat(argv[1], &stb2); | 1258 | xstat(argv[1], &stb2); |
1247 | if (gotstdin && (S_ISDIR(stb1.st_mode) || S_ISDIR(stb2.st_mode))) | 1259 | if (gotstdin && (S_ISDIR(stb1.st_mode) || S_ISDIR(stb2.st_mode))) |
1248 | bb_error_msg_and_die("Can't compare - to a directory"); | 1260 | bb_error_msg_and_die("Can't compare - to a directory"); |
1249 | if (S_ISDIR(stb1.st_mode) && S_ISDIR(stb2.st_mode)) { | 1261 | if (S_ISDIR(stb1.st_mode) && S_ISDIR(stb2.st_mode)) { |
1250 | #if ENABLE_FEATURE_DIFF_DIR | 1262 | #if ENABLE_FEATURE_DIFF_DIR |
1251 | diffdir(argv[0], argv[1]); | 1263 | diffdir(argv[0], argv[1]); |
1252 | #else | 1264 | #else |
1253 | bb_error_msg_and_die("Directory comparison not supported"); | 1265 | bb_error_msg_and_die("Directory comparison not supported"); |
1254 | #endif | 1266 | #endif |
1255 | } | 1267 | } else { |
1256 | else { | ||
1257 | if (S_ISDIR(stb1.st_mode)) { | 1268 | if (S_ISDIR(stb1.st_mode)) { |
1258 | argv[0] = concat_path_file(argv[0], argv[1]); | 1269 | argv[0] = concat_path_file(argv[0], argv[1]); |
1259 | xstat(argv[0], &stb1); | 1270 | xstat(argv[0], &stb1); |
@@ -1266,4 +1277,3 @@ int diff_main(int argc, char **argv) { | |||
1266 | } | 1277 | } |
1267 | exit(status); | 1278 | exit(status); |
1268 | } | 1279 | } |
1269 | |||