xref: /aosp_15_r20/external/toybox/toys/pending/diff.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* diff.c - compare files line by line
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2014 Sandeep Sharma <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2014 Ashwini Kumar <[email protected]>
5*cf5a6c84SAndroid Build Coastguard Worker  *
6*cf5a6c84SAndroid Build Coastguard Worker  * See https://pubs.opengroup.org/onlinepubs/9699919799/utilities/diff.html
7*cf5a6c84SAndroid Build Coastguard Worker  * and https://www.cs.dartmouth.edu/~doug/diff.pdf
8*cf5a6c84SAndroid Build Coastguard Worker  *
9*cf5a6c84SAndroid Build Coastguard Worker  * Deviations from posix: always does -u
10*cf5a6c84SAndroid Build Coastguard Worker 
11*cf5a6c84SAndroid Build Coastguard Worker USE_DIFF(NEWTOY(diff, "<2>2(unchanged-line-format):;(old-line-format):;(no-dereference);(new-line-format):;(color)(strip-trailing-cr)B(ignore-blank-lines)d(minimal)b(ignore-space-change)ut(expand-tabs)w(ignore-all-space)i(ignore-case)T(initial-tab)s(report-identical-files)q(brief)a(text)S(starting-file):F(show-function-line):;L(label)*N(new-file)r(recursive)U(unified)#<0=3", TOYFLAG_USR|TOYFLAG_BIN|TOYFLAG_ARGFAIL(2)))
12*cf5a6c84SAndroid Build Coastguard Worker 
13*cf5a6c84SAndroid Build Coastguard Worker config DIFF
14*cf5a6c84SAndroid Build Coastguard Worker   bool "diff"
15*cf5a6c84SAndroid Build Coastguard Worker   default n
16*cf5a6c84SAndroid Build Coastguard Worker   help
17*cf5a6c84SAndroid Build Coastguard Worker     usage: diff [-abBdiNqrTstw] [-L LABEL] [-S FILE] [-U LINES] [-F REGEX ] FILE1 FILE2
18*cf5a6c84SAndroid Build Coastguard Worker 
19*cf5a6c84SAndroid Build Coastguard Worker     -a	Treat all files as text
20*cf5a6c84SAndroid Build Coastguard Worker     -b	Ignore changes in the amount of whitespace
21*cf5a6c84SAndroid Build Coastguard Worker     -B	Ignore changes whose lines are all blank
22*cf5a6c84SAndroid Build Coastguard Worker     -d	Try hard to find a smaller set of changes
23*cf5a6c84SAndroid Build Coastguard Worker     -F 	Show the most recent line matching the regex
24*cf5a6c84SAndroid Build Coastguard Worker     -i	Ignore case differences
25*cf5a6c84SAndroid Build Coastguard Worker     -L	Use LABEL instead of the filename in the unified header
26*cf5a6c84SAndroid Build Coastguard Worker     -N	Treat absent files as empty
27*cf5a6c84SAndroid Build Coastguard Worker     -q	Output only whether files differ
28*cf5a6c84SAndroid Build Coastguard Worker     -r	Recurse
29*cf5a6c84SAndroid Build Coastguard Worker     -S	Start with FILE when comparing directories
30*cf5a6c84SAndroid Build Coastguard Worker     -s	Report when two files are the same
31*cf5a6c84SAndroid Build Coastguard Worker     -T	Make tabs line up by prefixing a tab when necessary
32*cf5a6c84SAndroid Build Coastguard Worker     -t	Expand tabs to spaces in output
33*cf5a6c84SAndroid Build Coastguard Worker     -u	Unified diff
34*cf5a6c84SAndroid Build Coastguard Worker     -U	Output LINES lines of context
35*cf5a6c84SAndroid Build Coastguard Worker     -w	Ignore all whitespace
36*cf5a6c84SAndroid Build Coastguard Worker 
37*cf5a6c84SAndroid Build Coastguard Worker     --color     Color output   --strip-trailing-cr   Strip '\r' from input lines
38*cf5a6c84SAndroid Build Coastguard Worker     --no-dereference Don't follow symbolic links
39*cf5a6c84SAndroid Build Coastguard Worker     --TYPE-line-format=FORMAT  Display TYPE (unchanged/old/new) lines using FORMAT
40*cf5a6c84SAndroid Build Coastguard Worker       FORMAT uses printf integer escapes (ala %-2.4x) followed by LETTER: FELMNn
41*cf5a6c84SAndroid Build Coastguard Worker     Supported format specifiers are:
42*cf5a6c84SAndroid Build Coastguard Worker     * %l, the contents of the line, without the trailing newline
43*cf5a6c84SAndroid Build Coastguard Worker     * %L, the contents of the line, including the trailing newline
44*cf5a6c84SAndroid Build Coastguard Worker     * %%, the character '%'
45*cf5a6c84SAndroid Build Coastguard Worker */
46*cf5a6c84SAndroid Build Coastguard Worker 
47*cf5a6c84SAndroid Build Coastguard Worker #define FOR_diff
48*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
49*cf5a6c84SAndroid Build Coastguard Worker 
50*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
51*cf5a6c84SAndroid Build Coastguard Worker   long U;
52*cf5a6c84SAndroid Build Coastguard Worker   struct arg_list *L;
53*cf5a6c84SAndroid Build Coastguard Worker   char *F, *S, *new_line_format, *old_line_format, *unchanged_line_format;
54*cf5a6c84SAndroid Build Coastguard Worker 
55*cf5a6c84SAndroid Build Coastguard Worker   int dir_num, size, is_binary, is_symlink, differ, change, len[2], *offset[2];
56*cf5a6c84SAndroid Build Coastguard Worker   struct stat st[2];
57*cf5a6c84SAndroid Build Coastguard Worker   struct {
58*cf5a6c84SAndroid Build Coastguard Worker     char **list;
59*cf5a6c84SAndroid Build Coastguard Worker     int nr_elm;
60*cf5a6c84SAndroid Build Coastguard Worker   } dir[2];
61*cf5a6c84SAndroid Build Coastguard Worker   struct {
62*cf5a6c84SAndroid Build Coastguard Worker     FILE *fp;
63*cf5a6c84SAndroid Build Coastguard Worker     int len;
64*cf5a6c84SAndroid Build Coastguard Worker   } file[2];
65*cf5a6c84SAndroid Build Coastguard Worker   struct {
66*cf5a6c84SAndroid Build Coastguard Worker     char *name;
67*cf5a6c84SAndroid Build Coastguard Worker     int len;
68*cf5a6c84SAndroid Build Coastguard Worker   } link[2];
69*cf5a6c84SAndroid Build Coastguard Worker )
70*cf5a6c84SAndroid Build Coastguard Worker 
71*cf5a6c84SAndroid Build Coastguard Worker #define IS_STDIN(s)     (*(s)=='-' && !(s)[1])
72*cf5a6c84SAndroid Build Coastguard Worker 
73*cf5a6c84SAndroid Build Coastguard Worker struct v_vector {
74*cf5a6c84SAndroid Build Coastguard Worker   unsigned serial:31;
75*cf5a6c84SAndroid Build Coastguard Worker   unsigned last:1;
76*cf5a6c84SAndroid Build Coastguard Worker   union {
77*cf5a6c84SAndroid Build Coastguard Worker     unsigned hash;
78*cf5a6c84SAndroid Build Coastguard Worker     unsigned p;
79*cf5a6c84SAndroid Build Coastguard Worker   };
80*cf5a6c84SAndroid Build Coastguard Worker };
81*cf5a6c84SAndroid Build Coastguard Worker 
82*cf5a6c84SAndroid Build Coastguard Worker struct diff {
83*cf5a6c84SAndroid Build Coastguard Worker   long a, b, c, d, prev, suff;
84*cf5a6c84SAndroid Build Coastguard Worker };
85*cf5a6c84SAndroid Build Coastguard Worker 
86*cf5a6c84SAndroid Build Coastguard Worker struct candidate {
87*cf5a6c84SAndroid Build Coastguard Worker   struct candidate *next, *prev;
88*cf5a6c84SAndroid Build Coastguard Worker   int a, b;
89*cf5a6c84SAndroid Build Coastguard Worker };
90*cf5a6c84SAndroid Build Coastguard Worker 
91*cf5a6c84SAndroid Build Coastguard Worker enum {
92*cf5a6c84SAndroid Build Coastguard Worker   empty = 1 << 9,
93*cf5a6c84SAndroid Build Coastguard Worker   eol = 1 << 10,
94*cf5a6c84SAndroid Build Coastguard Worker   eof = 1 << 11,
95*cf5a6c84SAndroid Build Coastguard Worker   space = 1 << 12
96*cf5a6c84SAndroid Build Coastguard Worker };
97*cf5a6c84SAndroid Build Coastguard Worker 
xlstat(char * path,struct stat * st)98*cf5a6c84SAndroid Build Coastguard Worker void xlstat(char *path, struct stat *st)
99*cf5a6c84SAndroid Build Coastguard Worker {
100*cf5a6c84SAndroid Build Coastguard Worker   if(lstat(path, st)) perror_exit("Can't lstat %s", path);
101*cf5a6c84SAndroid Build Coastguard Worker }
102*cf5a6c84SAndroid Build Coastguard Worker 
comp(void * a,void * b)103*cf5a6c84SAndroid Build Coastguard Worker static int comp(void *a, void *b)
104*cf5a6c84SAndroid Build Coastguard Worker {
105*cf5a6c84SAndroid Build Coastguard Worker   int i = ((struct v_vector *)a)->hash - ((struct v_vector *)b)->hash;
106*cf5a6c84SAndroid Build Coastguard Worker 
107*cf5a6c84SAndroid Build Coastguard Worker   return i ? : ((struct v_vector *)a)->serial - ((struct v_vector *)b)->serial;
108*cf5a6c84SAndroid Build Coastguard Worker }
109*cf5a6c84SAndroid Build Coastguard Worker 
search(struct candidate ** K,int r,int k,int j)110*cf5a6c84SAndroid Build Coastguard Worker static int search(struct candidate **K, int r, int k, int j)
111*cf5a6c84SAndroid Build Coastguard Worker {
112*cf5a6c84SAndroid Build Coastguard Worker   int low = r, upper = k, mid;
113*cf5a6c84SAndroid Build Coastguard Worker 
114*cf5a6c84SAndroid Build Coastguard Worker   while (low<=(mid = (low+upper)/2)) {
115*cf5a6c84SAndroid Build Coastguard Worker     if (K[mid]->b < j && K[mid + 1]->b > j) return mid;
116*cf5a6c84SAndroid Build Coastguard Worker     if (K[mid]->b < j) low = mid + 1;
117*cf5a6c84SAndroid Build Coastguard Worker     else if (K[mid]->b > j) upper = mid - 1;
118*cf5a6c84SAndroid Build Coastguard Worker     else return -1;
119*cf5a6c84SAndroid Build Coastguard Worker   }
120*cf5a6c84SAndroid Build Coastguard Worker   return -1;
121*cf5a6c84SAndroid Build Coastguard Worker }
122*cf5a6c84SAndroid Build Coastguard Worker 
new_candidate(int i,int j,struct candidate * prev)123*cf5a6c84SAndroid Build Coastguard Worker static struct candidate *new_candidate(int i, int j, struct candidate *prev)
124*cf5a6c84SAndroid Build Coastguard Worker {
125*cf5a6c84SAndroid Build Coastguard Worker   struct candidate *c = xzalloc(sizeof(struct candidate));
126*cf5a6c84SAndroid Build Coastguard Worker 
127*cf5a6c84SAndroid Build Coastguard Worker   c->a = i;
128*cf5a6c84SAndroid Build Coastguard Worker   c->b = j;
129*cf5a6c84SAndroid Build Coastguard Worker   c->prev = prev;
130*cf5a6c84SAndroid Build Coastguard Worker   return c;
131*cf5a6c84SAndroid Build Coastguard Worker }
132*cf5a6c84SAndroid Build Coastguard Worker 
133*cf5a6c84SAndroid Build Coastguard Worker /* 1. Search K[r: k] for an element K[s] such that K[s]-> b < j and K[s + 1]->b > j
134*cf5a6c84SAndroid Build Coastguard Worker  * 2. if found do
135*cf5a6c84SAndroid Build Coastguard Worker  *  2.a. If K[s + 1]->b > j do K[r] = c; r = s+1 and c = candidate(i, j, K[s]) //we have a candidate
136*cf5a6c84SAndroid Build Coastguard Worker  *  2.b. if s = k (fence reached move it further) do K[k + 2] = K[k + 1], k++
137*cf5a6c84SAndroid Build Coastguard Worker  * 3. if E[p].last true break i.e we have reached at the end of an equiv class
138*cf5a6c84SAndroid Build Coastguard Worker  *    else p = p + 1 //keep traversing the equiv class.
139*cf5a6c84SAndroid Build Coastguard Worker  * 4. K[r] = c //Save the sucessfully filled k-candidate.
140*cf5a6c84SAndroid Build Coastguard Worker  */
do_merge(struct candidate ** K,int * k,int i,struct v_vector * E,int p)141*cf5a6c84SAndroid Build Coastguard Worker static void do_merge(struct candidate **K, int *k, int i,
142*cf5a6c84SAndroid Build Coastguard Worker     struct v_vector *E, int p)
143*cf5a6c84SAndroid Build Coastguard Worker {
144*cf5a6c84SAndroid Build Coastguard Worker   int r = 0, s, j;
145*cf5a6c84SAndroid Build Coastguard Worker   struct candidate *pr = 0, *c = K[0];
146*cf5a6c84SAndroid Build Coastguard Worker 
147*cf5a6c84SAndroid Build Coastguard Worker   for (;;) {
148*cf5a6c84SAndroid Build Coastguard Worker     j = E[p].serial;
149*cf5a6c84SAndroid Build Coastguard Worker     s = search(K, r, *k, j);
150*cf5a6c84SAndroid Build Coastguard Worker     if (s>=0 && K[s]->b<j && K[s+1]->b>j) {
151*cf5a6c84SAndroid Build Coastguard Worker       if (K[s+1]->b>j) {
152*cf5a6c84SAndroid Build Coastguard Worker         pr = K[s];
153*cf5a6c84SAndroid Build Coastguard Worker         if (r && K[r]) c->next = K[r];
154*cf5a6c84SAndroid Build Coastguard Worker         K[r] = c;
155*cf5a6c84SAndroid Build Coastguard Worker         r = s+1;
156*cf5a6c84SAndroid Build Coastguard Worker         c = new_candidate(i , j, pr);
157*cf5a6c84SAndroid Build Coastguard Worker       }
158*cf5a6c84SAndroid Build Coastguard Worker       if (s == *k) {
159*cf5a6c84SAndroid Build Coastguard Worker         ++*k;
160*cf5a6c84SAndroid Build Coastguard Worker         K[*k+1] = K[*k];
161*cf5a6c84SAndroid Build Coastguard Worker         break;
162*cf5a6c84SAndroid Build Coastguard Worker       }
163*cf5a6c84SAndroid Build Coastguard Worker     }
164*cf5a6c84SAndroid Build Coastguard Worker     if (E[p].last) break;
165*cf5a6c84SAndroid Build Coastguard Worker     else p++;
166*cf5a6c84SAndroid Build Coastguard Worker   }
167*cf5a6c84SAndroid Build Coastguard Worker   K[r] = c;
168*cf5a6c84SAndroid Build Coastguard Worker }
169*cf5a6c84SAndroid Build Coastguard Worker 
read_tok(FILE * fp,off_t * off,int tok)170*cf5a6c84SAndroid Build Coastguard Worker static int read_tok(FILE *fp, off_t *off, int tok)
171*cf5a6c84SAndroid Build Coastguard Worker {
172*cf5a6c84SAndroid Build Coastguard Worker   int t = 0, is_space;
173*cf5a6c84SAndroid Build Coastguard Worker 
174*cf5a6c84SAndroid Build Coastguard Worker   tok |= empty;
175*cf5a6c84SAndroid Build Coastguard Worker   while (!(tok & eol)) {
176*cf5a6c84SAndroid Build Coastguard Worker     t = fgetc(fp);
177*cf5a6c84SAndroid Build Coastguard Worker 
178*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(strip_trailing_cr) && t == '\r') {
179*cf5a6c84SAndroid Build Coastguard Worker       int t2 = fgetc(fp);
180*cf5a6c84SAndroid Build Coastguard Worker       if (t2 == '\n') {
181*cf5a6c84SAndroid Build Coastguard Worker         t = t2;
182*cf5a6c84SAndroid Build Coastguard Worker         if (off) (*off)++;
183*cf5a6c84SAndroid Build Coastguard Worker       } else {
184*cf5a6c84SAndroid Build Coastguard Worker         ungetc(t2, fp);
185*cf5a6c84SAndroid Build Coastguard Worker       }
186*cf5a6c84SAndroid Build Coastguard Worker     }
187*cf5a6c84SAndroid Build Coastguard Worker 
188*cf5a6c84SAndroid Build Coastguard Worker     if (off && t != EOF) *off += 1;
189*cf5a6c84SAndroid Build Coastguard Worker     is_space = isspace(t) || (t == EOF);
190*cf5a6c84SAndroid Build Coastguard Worker     tok |= (t & (eof + eol)); //set tok eof+eol when t is eof
191*cf5a6c84SAndroid Build Coastguard Worker 
192*cf5a6c84SAndroid Build Coastguard Worker     if (t == '\n') tok |= eol;
193*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(i)) if (t >= 'A' && t <= 'Z') t = tolower(t);
194*cf5a6c84SAndroid Build Coastguard Worker 
195*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(w) && is_space) continue;
196*cf5a6c84SAndroid Build Coastguard Worker 
197*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(b)) {
198*cf5a6c84SAndroid Build Coastguard Worker       if (tok & space) {
199*cf5a6c84SAndroid Build Coastguard Worker         if (is_space) continue;
200*cf5a6c84SAndroid Build Coastguard Worker         tok &= ~space;
201*cf5a6c84SAndroid Build Coastguard Worker       } else if (is_space) t = space + ' ';
202*cf5a6c84SAndroid Build Coastguard Worker     }
203*cf5a6c84SAndroid Build Coastguard Worker     tok &= ~(empty + 0xff);  //remove empty and char too.
204*cf5a6c84SAndroid Build Coastguard Worker     tok |= t; //add most recent char
205*cf5a6c84SAndroid Build Coastguard Worker     break;
206*cf5a6c84SAndroid Build Coastguard Worker   }
207*cf5a6c84SAndroid Build Coastguard Worker 
208*cf5a6c84SAndroid Build Coastguard Worker   return tok;
209*cf5a6c84SAndroid Build Coastguard Worker }
210*cf5a6c84SAndroid Build Coastguard Worker 
bcomp(void * a,void * b)211*cf5a6c84SAndroid Build Coastguard Worker int bcomp(void *a, void *b)
212*cf5a6c84SAndroid Build Coastguard Worker {
213*cf5a6c84SAndroid Build Coastguard Worker   struct v_vector *l = (struct v_vector *)a, *r = (struct v_vector *)b;
214*cf5a6c84SAndroid Build Coastguard Worker 
215*cf5a6c84SAndroid Build Coastguard Worker   return (l->hash-r->hash) ? : r[-1].last ? 0 : -1;
216*cf5a6c84SAndroid Build Coastguard Worker }
217*cf5a6c84SAndroid Build Coastguard Worker 
218*cf5a6c84SAndroid Build Coastguard Worker /*  file[0] corresponds file 1 and file[1] correspond file 2.
219*cf5a6c84SAndroid Build Coastguard Worker  * 1. calc hashes for both the files and store them in vector(v[0], v[1])
220*cf5a6c84SAndroid Build Coastguard Worker  * 2. sort file[1] with hash as primary and serial as sec. key
221*cf5a6c84SAndroid Build Coastguard Worker  * 3. Form the equivalance class of file[1] stored in e vector. It lists all the equivalence
222*cf5a6c84SAndroid Build Coastguard Worker  *    classes of lines in file[1], with e.last = true on the last element of each class.
223*cf5a6c84SAndroid Build Coastguard Worker  *    The elements are ordered by serial within classes.
224*cf5a6c84SAndroid Build Coastguard Worker  * 4. Form the p vector stored in  p_vector. p_vector[i], if non-zero, now points in e vector
225*cf5a6c84SAndroid Build Coastguard Worker  *    to the beginning of the equiv class of lines in file[1] equivalent to line
226*cf5a6c84SAndroid Build Coastguard Worker  *    i in file[0].
227*cf5a6c84SAndroid Build Coastguard Worker  * 5. Form the k-candidates as discribed in do_merge.
228*cf5a6c84SAndroid Build Coastguard Worker  * 6. Create a vector J[i] = j, such that i'th line in file[0] is j'th line of
229*cf5a6c84SAndroid Build Coastguard Worker  *    file[1], i.e J comprises LCS
230*cf5a6c84SAndroid Build Coastguard Worker  */
create_j_vector()231*cf5a6c84SAndroid Build Coastguard Worker static int *create_j_vector()
232*cf5a6c84SAndroid Build Coastguard Worker {
233*cf5a6c84SAndroid Build Coastguard Worker   int tok, i, j, size = 100, k;
234*cf5a6c84SAndroid Build Coastguard Worker   off_t off;
235*cf5a6c84SAndroid Build Coastguard Worker   long hash;
236*cf5a6c84SAndroid Build Coastguard Worker   int *p_vector, *J;
237*cf5a6c84SAndroid Build Coastguard Worker   struct v_vector *v[2], *e;
238*cf5a6c84SAndroid Build Coastguard Worker   struct candidate **kcand, *pr;
239*cf5a6c84SAndroid Build Coastguard Worker 
240*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i < 2; i++) {
241*cf5a6c84SAndroid Build Coastguard Worker     tok = off = 0;
242*cf5a6c84SAndroid Build Coastguard Worker     hash = 5831;
243*cf5a6c84SAndroid Build Coastguard Worker     v[i] = xzalloc(size * sizeof(struct v_vector));
244*cf5a6c84SAndroid Build Coastguard Worker     TT.offset[i] = xzalloc(size * sizeof(int));
245*cf5a6c84SAndroid Build Coastguard Worker     TT.file[i].len = 0;
246*cf5a6c84SAndroid Build Coastguard Worker     if (fseek(TT.file[i].fp, 0, SEEK_SET)) perror_exit("fseek failed");
247*cf5a6c84SAndroid Build Coastguard Worker 
248*cf5a6c84SAndroid Build Coastguard Worker     while (1) {
249*cf5a6c84SAndroid Build Coastguard Worker       tok  = read_tok(TT.file[i].fp, &off, tok);
250*cf5a6c84SAndroid Build Coastguard Worker       if (!(tok & empty)) {
251*cf5a6c84SAndroid Build Coastguard Worker         hash = ((hash << 5) + hash) + (tok & 0xff);
252*cf5a6c84SAndroid Build Coastguard Worker         continue;
253*cf5a6c84SAndroid Build Coastguard Worker       }
254*cf5a6c84SAndroid Build Coastguard Worker 
255*cf5a6c84SAndroid Build Coastguard Worker       if (size == ++TT.file[i].len) {
256*cf5a6c84SAndroid Build Coastguard Worker         size = size * 11 / 10;
257*cf5a6c84SAndroid Build Coastguard Worker         v[i] = xrealloc(v[i], size*sizeof(struct v_vector));
258*cf5a6c84SAndroid Build Coastguard Worker         TT.offset[i] = xrealloc(TT.offset[i], size*sizeof(int));
259*cf5a6c84SAndroid Build Coastguard Worker       }
260*cf5a6c84SAndroid Build Coastguard Worker 
261*cf5a6c84SAndroid Build Coastguard Worker       v[i][TT.file[i].len].hash = hash & INT_MAX;
262*cf5a6c84SAndroid Build Coastguard Worker       TT.offset[i][TT.file[i].len] = off;
263*cf5a6c84SAndroid Build Coastguard Worker       if ((tok & eof)) {
264*cf5a6c84SAndroid Build Coastguard Worker         TT.offset[i][TT.file[i].len] = ++off;
265*cf5a6c84SAndroid Build Coastguard Worker         break;
266*cf5a6c84SAndroid Build Coastguard Worker       }
267*cf5a6c84SAndroid Build Coastguard Worker       hash = 5831;  //next line
268*cf5a6c84SAndroid Build Coastguard Worker       tok = 0;
269*cf5a6c84SAndroid Build Coastguard Worker     }
270*cf5a6c84SAndroid Build Coastguard Worker     if (TT.offset[i][TT.file[i].len]-TT.offset[i][TT.file[i].len-1] == 1)
271*cf5a6c84SAndroid Build Coastguard Worker       TT.file[i].len--;
272*cf5a6c84SAndroid Build Coastguard Worker   }
273*cf5a6c84SAndroid Build Coastguard Worker 
274*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i<=TT.file[1].len; i++) v[1][i].serial = i;
275*cf5a6c84SAndroid Build Coastguard Worker   qsort(v[1]+1, TT.file[1].len, sizeof(struct v_vector), (void *)comp);
276*cf5a6c84SAndroid Build Coastguard Worker 
277*cf5a6c84SAndroid Build Coastguard Worker   e = v[1];
278*cf5a6c84SAndroid Build Coastguard Worker   e[0].serial = 0;
279*cf5a6c84SAndroid Build Coastguard Worker   e[0].last = 1;
280*cf5a6c84SAndroid Build Coastguard Worker   for (i = 1; i<=TT.file[1].len; i++)
281*cf5a6c84SAndroid Build Coastguard Worker     e[i].last = i==TT.file[1].len || v[1][i].hash!=v[1][i+1].hash;
282*cf5a6c84SAndroid Build Coastguard Worker 
283*cf5a6c84SAndroid Build Coastguard Worker   p_vector = xzalloc((TT.file[0].len+2)*sizeof(int));
284*cf5a6c84SAndroid Build Coastguard Worker   for (i = 1; i<=TT.file[0].len; i++) {
285*cf5a6c84SAndroid Build Coastguard Worker     void *r = bsearch(&v[0][i], e+1, TT.file[1].len, sizeof(*e), (void *)bcomp);
286*cf5a6c84SAndroid Build Coastguard Worker     if (r) p_vector[i] = (struct v_vector *)r - e;
287*cf5a6c84SAndroid Build Coastguard Worker   }
288*cf5a6c84SAndroid Build Coastguard Worker 
289*cf5a6c84SAndroid Build Coastguard Worker   for (i = 1; i<=TT.file[0].len; i++) e[i].p = p_vector[i];
290*cf5a6c84SAndroid Build Coastguard Worker   free(p_vector);
291*cf5a6c84SAndroid Build Coastguard Worker 
292*cf5a6c84SAndroid Build Coastguard Worker   size = 100;
293*cf5a6c84SAndroid Build Coastguard Worker   kcand = xzalloc(size * sizeof(struct candidate *));
294*cf5a6c84SAndroid Build Coastguard Worker 
295*cf5a6c84SAndroid Build Coastguard Worker   kcand[0] = new_candidate(0 , 0, 0);
296*cf5a6c84SAndroid Build Coastguard Worker   kcand[1] = new_candidate(TT.file[0].len+1, TT.file[1].len+1, 0); //the fence
297*cf5a6c84SAndroid Build Coastguard Worker 
298*cf5a6c84SAndroid Build Coastguard Worker   k = 0;  //last successfully filled k candidate.
299*cf5a6c84SAndroid Build Coastguard Worker   for (i = 1; i<=TT.file[0].len; i++) {
300*cf5a6c84SAndroid Build Coastguard Worker     if (!e[i].p) continue;
301*cf5a6c84SAndroid Build Coastguard Worker     if ((size - 2) == k) {
302*cf5a6c84SAndroid Build Coastguard Worker       size = size * 11 / 10;
303*cf5a6c84SAndroid Build Coastguard Worker       kcand = xrealloc(kcand, (size*sizeof(struct candidate *)));
304*cf5a6c84SAndroid Build Coastguard Worker     }
305*cf5a6c84SAndroid Build Coastguard Worker     do_merge(kcand, &k, i, e, e[i].p);
306*cf5a6c84SAndroid Build Coastguard Worker   }
307*cf5a6c84SAndroid Build Coastguard Worker   free(v[0]); //no need for v_vector now.
308*cf5a6c84SAndroid Build Coastguard Worker   free(v[1]);
309*cf5a6c84SAndroid Build Coastguard Worker 
310*cf5a6c84SAndroid Build Coastguard Worker   J = xzalloc((TT.file[0].len+2)*sizeof(int));
311*cf5a6c84SAndroid Build Coastguard Worker 
312*cf5a6c84SAndroid Build Coastguard Worker   for (pr = kcand[k]; pr; pr = pr->prev) J[pr->a] = pr->b;
313*cf5a6c84SAndroid Build Coastguard Worker   J[TT.file[0].len+1] = TT.file[1].len+1; //mark boundary
314*cf5a6c84SAndroid Build Coastguard Worker 
315*cf5a6c84SAndroid Build Coastguard Worker   for (i = k+1; i>=0; i--) llist_traverse(kcand[i], free);
316*cf5a6c84SAndroid Build Coastguard Worker   free(kcand);
317*cf5a6c84SAndroid Build Coastguard Worker 
318*cf5a6c84SAndroid Build Coastguard Worker   for (i = 1; i<=TT.file[0].len; i++) { // jackpot?
319*cf5a6c84SAndroid Build Coastguard Worker     if (!J[i]) continue;
320*cf5a6c84SAndroid Build Coastguard Worker 
321*cf5a6c84SAndroid Build Coastguard Worker     if (fseek(TT.file[0].fp, TT.offset[0][i-1], SEEK_SET)
322*cf5a6c84SAndroid Build Coastguard Worker      || fseek(TT.file[1].fp, TT.offset[1][J[i]-1], SEEK_SET))
323*cf5a6c84SAndroid Build Coastguard Worker        perror_exit("fseek");
324*cf5a6c84SAndroid Build Coastguard Worker 
325*cf5a6c84SAndroid Build Coastguard Worker     for (j = J[i]; i<=TT.file[0].len && J[i]==j; i++, j++) {
326*cf5a6c84SAndroid Build Coastguard Worker       int tok0 = 0, tok1 = 0;
327*cf5a6c84SAndroid Build Coastguard Worker 
328*cf5a6c84SAndroid Build Coastguard Worker       do {
329*cf5a6c84SAndroid Build Coastguard Worker         tok0 = read_tok(TT.file[0].fp, NULL, tok0);
330*cf5a6c84SAndroid Build Coastguard Worker         tok1 = read_tok(TT.file[1].fp, NULL, tok1);
331*cf5a6c84SAndroid Build Coastguard Worker         if (((tok0 ^ tok1) & empty) || ((tok0 & 0xff) != (tok1 & 0xff)))
332*cf5a6c84SAndroid Build Coastguard Worker           J[i] = 0;
333*cf5a6c84SAndroid Build Coastguard Worker       } while (!(tok0 & tok1 & empty));
334*cf5a6c84SAndroid Build Coastguard Worker     }
335*cf5a6c84SAndroid Build Coastguard Worker   }
336*cf5a6c84SAndroid Build Coastguard Worker   return J;
337*cf5a6c84SAndroid Build Coastguard Worker }
338*cf5a6c84SAndroid Build Coastguard Worker 
diff(char ** files)339*cf5a6c84SAndroid Build Coastguard Worker static int *diff(char **files)
340*cf5a6c84SAndroid Build Coastguard Worker {
341*cf5a6c84SAndroid Build Coastguard Worker   size_t i ,j;
342*cf5a6c84SAndroid Build Coastguard Worker   int s, t;
343*cf5a6c84SAndroid Build Coastguard Worker   char *bufi, *bufj;
344*cf5a6c84SAndroid Build Coastguard Worker 
345*cf5a6c84SAndroid Build Coastguard Worker   TT.is_binary = 0; //loop calls to diff
346*cf5a6c84SAndroid Build Coastguard Worker   TT.differ = 0;
347*cf5a6c84SAndroid Build Coastguard Worker 
348*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i < 2; i++) {
349*cf5a6c84SAndroid Build Coastguard Worker     if ((j = !strcmp(files[i], "-")) || S_ISFIFO(TT.st[i].st_mode)) {
350*cf5a6c84SAndroid Build Coastguard Worker       char *tmp_name;
351*cf5a6c84SAndroid Build Coastguard Worker       int srcfd = j ? 0 : open(files[i], O_RDONLY),
352*cf5a6c84SAndroid Build Coastguard Worker         tmpfd = xtempfile("fifo", &tmp_name);
353*cf5a6c84SAndroid Build Coastguard Worker 
354*cf5a6c84SAndroid Build Coastguard Worker       unlink(tmp_name);
355*cf5a6c84SAndroid Build Coastguard Worker       free(tmp_name);
356*cf5a6c84SAndroid Build Coastguard Worker 
357*cf5a6c84SAndroid Build Coastguard Worker       xsendfile(srcfd, tmpfd);
358*cf5a6c84SAndroid Build Coastguard Worker       if (!j) close(srcfd);
359*cf5a6c84SAndroid Build Coastguard Worker       TT.file[i].fp = fdopen(tmpfd, "r");
360*cf5a6c84SAndroid Build Coastguard Worker     } else TT.file[i].fp = fopen(files[i], "r");
361*cf5a6c84SAndroid Build Coastguard Worker 
362*cf5a6c84SAndroid Build Coastguard Worker     if (!TT.file[i].fp) {
363*cf5a6c84SAndroid Build Coastguard Worker       perror_msg("%s", files[i]);
364*cf5a6c84SAndroid Build Coastguard Worker       TT.differ = 2;
365*cf5a6c84SAndroid Build Coastguard Worker       return 0; //return SAME
366*cf5a6c84SAndroid Build Coastguard Worker     }
367*cf5a6c84SAndroid Build Coastguard Worker   }
368*cf5a6c84SAndroid Build Coastguard Worker 
369*cf5a6c84SAndroid Build Coastguard Worker   s = sizeof(toybuf)/2;
370*cf5a6c84SAndroid Build Coastguard Worker   bufi = toybuf;
371*cf5a6c84SAndroid Build Coastguard Worker   bufj = toybuf+s;
372*cf5a6c84SAndroid Build Coastguard Worker 
373*cf5a6c84SAndroid Build Coastguard Worker   if (fseek(TT.file[0].fp, 0, SEEK_SET) || fseek(TT.file[1].fp, 0, SEEK_SET))
374*cf5a6c84SAndroid Build Coastguard Worker     perror_exit("fseek");
375*cf5a6c84SAndroid Build Coastguard Worker 
376*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(a)) return create_j_vector();
377*cf5a6c84SAndroid Build Coastguard Worker 
378*cf5a6c84SAndroid Build Coastguard Worker   while (1) {
379*cf5a6c84SAndroid Build Coastguard Worker     i = fread(bufi, 1, s, TT.file[0].fp);
380*cf5a6c84SAndroid Build Coastguard Worker     j = fread(bufj, 1, s, TT.file[1].fp);
381*cf5a6c84SAndroid Build Coastguard Worker 
382*cf5a6c84SAndroid Build Coastguard Worker     if (i != j) TT.differ = 1;
383*cf5a6c84SAndroid Build Coastguard Worker 
384*cf5a6c84SAndroid Build Coastguard Worker     for (t = 0; t < i && !TT.is_binary; t++) if (!bufi[t]) TT.is_binary = 1;
385*cf5a6c84SAndroid Build Coastguard Worker     for (t = 0; t < j && !TT.is_binary; t++) if (!bufj[t]) TT.is_binary = 1;
386*cf5a6c84SAndroid Build Coastguard Worker 
387*cf5a6c84SAndroid Build Coastguard Worker     i = minof(i, j);
388*cf5a6c84SAndroid Build Coastguard Worker     for (t = 0; t < i; t++) if (bufi[t] != bufj[t]) TT.differ = 1;
389*cf5a6c84SAndroid Build Coastguard Worker 
390*cf5a6c84SAndroid Build Coastguard Worker     if (!i || !j) break;
391*cf5a6c84SAndroid Build Coastguard Worker   }
392*cf5a6c84SAndroid Build Coastguard Worker   if (TT.is_binary || !TT.differ) return 0;
393*cf5a6c84SAndroid Build Coastguard Worker 
394*cf5a6c84SAndroid Build Coastguard Worker   return create_j_vector();
395*cf5a6c84SAndroid Build Coastguard Worker }
396*cf5a6c84SAndroid Build Coastguard Worker 
print_line_matching_regex(int a,regex_t * reg,int * off_set,FILE * fp)397*cf5a6c84SAndroid Build Coastguard Worker static void print_line_matching_regex(int a, regex_t *reg, int *off_set, FILE *fp) {
398*cf5a6c84SAndroid Build Coastguard Worker   int i = 0, j = 0, line_buf_size = 100, cc = 0;
399*cf5a6c84SAndroid Build Coastguard Worker   char* line = xzalloc(line_buf_size * sizeof(char));
400*cf5a6c84SAndroid Build Coastguard Worker   for (i = a; a > 0; --i) {
401*cf5a6c84SAndroid Build Coastguard Worker     int line_len = 0;
402*cf5a6c84SAndroid Build Coastguard Worker     if (fseek(fp, off_set[i - 1], SEEK_SET)) perror_exit("fseek failed");
403*cf5a6c84SAndroid Build Coastguard Worker     for (j = 0; j < (off_set[i] - off_set[i - 1]); j++) {
404*cf5a6c84SAndroid Build Coastguard Worker       cc = fgetc(fp);
405*cf5a6c84SAndroid Build Coastguard Worker       if (cc == EOF || cc == '\n') {
406*cf5a6c84SAndroid Build Coastguard Worker         break;
407*cf5a6c84SAndroid Build Coastguard Worker       }
408*cf5a6c84SAndroid Build Coastguard Worker       ++line_len;
409*cf5a6c84SAndroid Build Coastguard Worker       if (line_len >= line_buf_size) {
410*cf5a6c84SAndroid Build Coastguard Worker         line_buf_size = line_buf_size * 11 / 10;
411*cf5a6c84SAndroid Build Coastguard Worker         line = xrealloc(line, line_buf_size*sizeof(char));
412*cf5a6c84SAndroid Build Coastguard Worker       }
413*cf5a6c84SAndroid Build Coastguard Worker       line[j] = cc;
414*cf5a6c84SAndroid Build Coastguard Worker     }
415*cf5a6c84SAndroid Build Coastguard Worker     line[line_len] = '\0';
416*cf5a6c84SAndroid Build Coastguard Worker     if (!regexec0(reg, line, line_len, 0, NULL, 0)) {
417*cf5a6c84SAndroid Build Coastguard Worker       printf(" %s", line);
418*cf5a6c84SAndroid Build Coastguard Worker       break;
419*cf5a6c84SAndroid Build Coastguard Worker     }
420*cf5a6c84SAndroid Build Coastguard Worker   }
421*cf5a6c84SAndroid Build Coastguard Worker   free(line);
422*cf5a6c84SAndroid Build Coastguard Worker }
423*cf5a6c84SAndroid Build Coastguard Worker 
print_diff(int a,int b,char c,int * off_set,FILE * fp)424*cf5a6c84SAndroid Build Coastguard Worker static void print_diff(int a, int b, char c, int *off_set, FILE *fp)
425*cf5a6c84SAndroid Build Coastguard Worker {
426*cf5a6c84SAndroid Build Coastguard Worker   int i, j, cc, cl;
427*cf5a6c84SAndroid Build Coastguard Worker   char *reset = 0, *fmt = 0;
428*cf5a6c84SAndroid Build Coastguard Worker 
429*cf5a6c84SAndroid Build Coastguard Worker   if (!TT.new_line_format && c!=' ' && FLAG(color)) {
430*cf5a6c84SAndroid Build Coastguard Worker     printf("\e[%dm", 31+(c=='+'));
431*cf5a6c84SAndroid Build Coastguard Worker     reset = "\e[0m";
432*cf5a6c84SAndroid Build Coastguard Worker   }
433*cf5a6c84SAndroid Build Coastguard Worker 
434*cf5a6c84SAndroid Build Coastguard Worker   for (i = a; i <= b; i++) {
435*cf5a6c84SAndroid Build Coastguard Worker     if (fseek(fp, off_set[i - 1], SEEK_SET)) perror_exit("fseek failed");
436*cf5a6c84SAndroid Build Coastguard Worker     if (TT.new_line_format) {
437*cf5a6c84SAndroid Build Coastguard Worker       if (c == '+') fmt = TT.new_line_format;
438*cf5a6c84SAndroid Build Coastguard Worker       else if (c == '-') fmt = TT.old_line_format;
439*cf5a6c84SAndroid Build Coastguard Worker       else fmt = TT.unchanged_line_format;
440*cf5a6c84SAndroid Build Coastguard Worker       while (*fmt) {
441*cf5a6c84SAndroid Build Coastguard Worker         if (*fmt == '%') {
442*cf5a6c84SAndroid Build Coastguard Worker           fmt++;
443*cf5a6c84SAndroid Build Coastguard Worker           char f = *fmt++;
444*cf5a6c84SAndroid Build Coastguard Worker           if (f == '%') putchar('%');
445*cf5a6c84SAndroid Build Coastguard Worker           else if (f == 'l' || f == 'L') {
446*cf5a6c84SAndroid Build Coastguard Worker             for (j = 0; j <  (off_set[i] - off_set[i - 1]); j++) {
447*cf5a6c84SAndroid Build Coastguard Worker               cc = fgetc(fp);
448*cf5a6c84SAndroid Build Coastguard Worker               if (cc == EOF) break;
449*cf5a6c84SAndroid Build Coastguard Worker               if (cc != '\n' || f == 'L') putchar(cc);
450*cf5a6c84SAndroid Build Coastguard Worker             }
451*cf5a6c84SAndroid Build Coastguard Worker           } else error_exit("Unrecognized format specifier %%%c", f);
452*cf5a6c84SAndroid Build Coastguard Worker         } else putchar(*fmt++);
453*cf5a6c84SAndroid Build Coastguard Worker       }
454*cf5a6c84SAndroid Build Coastguard Worker       continue;
455*cf5a6c84SAndroid Build Coastguard Worker     }
456*cf5a6c84SAndroid Build Coastguard Worker     putchar(c);
457*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(T)) putchar('\t');
458*cf5a6c84SAndroid Build Coastguard Worker     for (j = 0, cl = 0; j <  (off_set[i] - off_set[i - 1]); j++) {
459*cf5a6c84SAndroid Build Coastguard Worker       cc = fgetc(fp);
460*cf5a6c84SAndroid Build Coastguard Worker       if (cc == EOF) {
461*cf5a6c84SAndroid Build Coastguard Worker         printf("%s\n\\ No newline at end of file\n", reset ? : "");
462*cf5a6c84SAndroid Build Coastguard Worker         return;
463*cf5a6c84SAndroid Build Coastguard Worker       }
464*cf5a6c84SAndroid Build Coastguard Worker       if ((cc == '\t') && FLAG(t)) do putchar(' '); while (++cl & 7);
465*cf5a6c84SAndroid Build Coastguard Worker       else {
466*cf5a6c84SAndroid Build Coastguard Worker         putchar(cc); //xputc has calls to fflush, it hurts performance badly.
467*cf5a6c84SAndroid Build Coastguard Worker         cl++;
468*cf5a6c84SAndroid Build Coastguard Worker       }
469*cf5a6c84SAndroid Build Coastguard Worker     }
470*cf5a6c84SAndroid Build Coastguard Worker   }
471*cf5a6c84SAndroid Build Coastguard Worker   if (reset) xputsn(reset);
472*cf5a6c84SAndroid Build Coastguard Worker }
473*cf5a6c84SAndroid Build Coastguard Worker 
concat_file_path(char * path,char * default_path)474*cf5a6c84SAndroid Build Coastguard Worker static char *concat_file_path(char *path, char *default_path)
475*cf5a6c84SAndroid Build Coastguard Worker {
476*cf5a6c84SAndroid Build Coastguard Worker   char *final_path;
477*cf5a6c84SAndroid Build Coastguard Worker 
478*cf5a6c84SAndroid Build Coastguard Worker   if ('/' == path[strlen(path) - 1]) {
479*cf5a6c84SAndroid Build Coastguard Worker     while (*default_path == '/') ++default_path;
480*cf5a6c84SAndroid Build Coastguard Worker     final_path = xmprintf("%s%s", path, default_path);
481*cf5a6c84SAndroid Build Coastguard Worker   }
482*cf5a6c84SAndroid Build Coastguard Worker   else if (*default_path != '/')
483*cf5a6c84SAndroid Build Coastguard Worker     final_path = xmprintf("%s/%s", path, default_path);
484*cf5a6c84SAndroid Build Coastguard Worker   else final_path = xmprintf("%s%s", path, default_path);
485*cf5a6c84SAndroid Build Coastguard Worker   return final_path;
486*cf5a6c84SAndroid Build Coastguard Worker }
487*cf5a6c84SAndroid Build Coastguard Worker 
skip(struct dirtree * node)488*cf5a6c84SAndroid Build Coastguard Worker static int skip(struct dirtree *node)
489*cf5a6c84SAndroid Build Coastguard Worker {
490*cf5a6c84SAndroid Build Coastguard Worker   int len = strlen(toys.optargs[TT.dir_num]), ret = 0;
491*cf5a6c84SAndroid Build Coastguard Worker   char *tmp = NULL, *ptr, *f_path = dirtree_path(node, NULL);
492*cf5a6c84SAndroid Build Coastguard Worker   struct stat st;
493*cf5a6c84SAndroid Build Coastguard Worker 
494*cf5a6c84SAndroid Build Coastguard Worker   ptr = f_path;
495*cf5a6c84SAndroid Build Coastguard Worker   ptr += len;
496*cf5a6c84SAndroid Build Coastguard Worker   if (ptr[0]) {
497*cf5a6c84SAndroid Build Coastguard Worker     tmp = concat_file_path(toys.optargs[1 - TT.dir_num], ptr);
498*cf5a6c84SAndroid Build Coastguard Worker     if (tmp && !stat(tmp, &st)) ret = 0; //it is there on other side
499*cf5a6c84SAndroid Build Coastguard Worker     else ret = 1; //not present on other side.
500*cf5a6c84SAndroid Build Coastguard Worker   }
501*cf5a6c84SAndroid Build Coastguard Worker   free(f_path);
502*cf5a6c84SAndroid Build Coastguard Worker   if (tmp) free(tmp);
503*cf5a6c84SAndroid Build Coastguard Worker   return ret; //add otherwise
504*cf5a6c84SAndroid Build Coastguard Worker }
505*cf5a6c84SAndroid Build Coastguard Worker 
add_to_list(struct dirtree * node)506*cf5a6c84SAndroid Build Coastguard Worker static void add_to_list(struct dirtree *node)
507*cf5a6c84SAndroid Build Coastguard Worker {
508*cf5a6c84SAndroid Build Coastguard Worker   char *full_path;
509*cf5a6c84SAndroid Build Coastguard Worker 
510*cf5a6c84SAndroid Build Coastguard Worker   TT.dir[TT.dir_num].list = xrealloc(TT.dir[TT.dir_num].list,
511*cf5a6c84SAndroid Build Coastguard Worker       (TT.size + 1)*sizeof(char*));
512*cf5a6c84SAndroid Build Coastguard Worker   TT.size++;
513*cf5a6c84SAndroid Build Coastguard Worker   full_path = dirtree_path(node, NULL);
514*cf5a6c84SAndroid Build Coastguard Worker   TT.dir[TT.dir_num].list[TT.size - 1] = full_path;
515*cf5a6c84SAndroid Build Coastguard Worker }
516*cf5a6c84SAndroid Build Coastguard Worker 
list_dir(struct dirtree * node)517*cf5a6c84SAndroid Build Coastguard Worker static int list_dir(struct dirtree *node)
518*cf5a6c84SAndroid Build Coastguard Worker {
519*cf5a6c84SAndroid Build Coastguard Worker   int ret = 0;
520*cf5a6c84SAndroid Build Coastguard Worker 
521*cf5a6c84SAndroid Build Coastguard Worker   if (!dirtree_notdotdot(node)) return 0;
522*cf5a6c84SAndroid Build Coastguard Worker 
523*cf5a6c84SAndroid Build Coastguard Worker   if (S_ISDIR(node->st.st_mode) && !node->parent) { //add root dirs.
524*cf5a6c84SAndroid Build Coastguard Worker     add_to_list(node);
525*cf5a6c84SAndroid Build Coastguard Worker     return (DIRTREE_RECURSE|((FLAG(no_dereference)) ? 0 : DIRTREE_SYMFOLLOW));
526*cf5a6c84SAndroid Build Coastguard Worker   }
527*cf5a6c84SAndroid Build Coastguard Worker 
528*cf5a6c84SAndroid Build Coastguard Worker   if (S_ISDIR(node->st.st_mode) && FLAG(r)) {
529*cf5a6c84SAndroid Build Coastguard Worker     if (!FLAG(N)) ret = skip(node);
530*cf5a6c84SAndroid Build Coastguard Worker     if (!ret) return DIRTREE_RECURSE|((FLAG(no_dereference)) ? 0 : DIRTREE_SYMFOLLOW);
531*cf5a6c84SAndroid Build Coastguard Worker     else {
532*cf5a6c84SAndroid Build Coastguard Worker       add_to_list(node); //only at one side.
533*cf5a6c84SAndroid Build Coastguard Worker       return 0;
534*cf5a6c84SAndroid Build Coastguard Worker     }
535*cf5a6c84SAndroid Build Coastguard Worker   } else {
536*cf5a6c84SAndroid Build Coastguard Worker     add_to_list(node);
537*cf5a6c84SAndroid Build Coastguard Worker     return S_ISDIR(node->st.st_mode) ? 0 : (DIRTREE_RECURSE|((FLAG(no_dereference)) ? 0 : DIRTREE_SYMFOLLOW));
538*cf5a6c84SAndroid Build Coastguard Worker   }
539*cf5a6c84SAndroid Build Coastguard Worker }
540*cf5a6c84SAndroid Build Coastguard Worker 
cmp(void * p1,void * p2)541*cf5a6c84SAndroid Build Coastguard Worker static int cmp(void *p1, void *p2)
542*cf5a6c84SAndroid Build Coastguard Worker {
543*cf5a6c84SAndroid Build Coastguard Worker    return strcmp(*(char **)p1, *(char **)p2);
544*cf5a6c84SAndroid Build Coastguard Worker }
545*cf5a6c84SAndroid Build Coastguard Worker 
546*cf5a6c84SAndroid Build Coastguard Worker // quote and escape filenames that have awkward characters
quote_filename(char * filename)547*cf5a6c84SAndroid Build Coastguard Worker char *quote_filename(char *filename)
548*cf5a6c84SAndroid Build Coastguard Worker {
549*cf5a6c84SAndroid Build Coastguard Worker   char *to = "abfnrtv\"\\", *from = "\a\b\f\n\r\t\v\"\\", *s, *t=0, *u;
550*cf5a6c84SAndroid Build Coastguard Worker   int len, quote = 0;
551*cf5a6c84SAndroid Build Coastguard Worker 
552*cf5a6c84SAndroid Build Coastguard Worker   for (;;) {
553*cf5a6c84SAndroid Build Coastguard Worker     // measure escapes on first pass, write on second
554*cf5a6c84SAndroid Build Coastguard Worker     len = 0;
555*cf5a6c84SAndroid Build Coastguard Worker     for (s = filename; *s; s++) {
556*cf5a6c84SAndroid Build Coastguard Worker       if ((u = strchr(from, *s))) {
557*cf5a6c84SAndroid Build Coastguard Worker         if (t) t[len] = '\\', t[len+1] = to[u-from];
558*cf5a6c84SAndroid Build Coastguard Worker         len += 2;
559*cf5a6c84SAndroid Build Coastguard Worker       } else if (*s<0x20 || *s>=0x80)
560*cf5a6c84SAndroid Build Coastguard Worker         len += snprintf(t+len, 5*!!t, "\\%.3o", *s);
561*cf5a6c84SAndroid Build Coastguard Worker       else {
562*cf5a6c84SAndroid Build Coastguard Worker         if (t) t[len] = *s;
563*cf5a6c84SAndroid Build Coastguard Worker         len++;
564*cf5a6c84SAndroid Build Coastguard Worker       }
565*cf5a6c84SAndroid Build Coastguard Worker     }
566*cf5a6c84SAndroid Build Coastguard Worker     if (t) {
567*cf5a6c84SAndroid Build Coastguard Worker       if (quote) t[len++] = '"';
568*cf5a6c84SAndroid Build Coastguard Worker       t[len] = 0;
569*cf5a6c84SAndroid Build Coastguard Worker 
570*cf5a6c84SAndroid Build Coastguard Worker       return t-quote;
571*cf5a6c84SAndroid Build Coastguard Worker     }
572*cf5a6c84SAndroid Build Coastguard Worker 
573*cf5a6c84SAndroid Build Coastguard Worker     // construct the new string
574*cf5a6c84SAndroid Build Coastguard Worker     quote = strlen(filename)!=len || strchr(filename, ' ');
575*cf5a6c84SAndroid Build Coastguard Worker     t = xmalloc(len+1+2*quote);
576*cf5a6c84SAndroid Build Coastguard Worker     if (quote) *t++ = '"';
577*cf5a6c84SAndroid Build Coastguard Worker   }
578*cf5a6c84SAndroid Build Coastguard Worker }
579*cf5a6c84SAndroid Build Coastguard Worker 
show_label(char * prefix,char * filename,struct stat * sb)580*cf5a6c84SAndroid Build Coastguard Worker static void show_label(char *prefix, char *filename, struct stat *sb)
581*cf5a6c84SAndroid Build Coastguard Worker {
582*cf5a6c84SAndroid Build Coastguard Worker   char date[36];
583*cf5a6c84SAndroid Build Coastguard Worker   char *quoted_file;
584*cf5a6c84SAndroid Build Coastguard Worker 
585*cf5a6c84SAndroid Build Coastguard Worker   quoted_file = quote_filename(filename);
586*cf5a6c84SAndroid Build Coastguard Worker   printf("%s %s\t%s\n", prefix, quoted_file,
587*cf5a6c84SAndroid Build Coastguard Worker     format_iso_time(date, sizeof(date), &sb->st_mtim));
588*cf5a6c84SAndroid Build Coastguard Worker   free(quoted_file);
589*cf5a6c84SAndroid Build Coastguard Worker }
590*cf5a6c84SAndroid Build Coastguard Worker 
do_symlink_diff(char ** files)591*cf5a6c84SAndroid Build Coastguard Worker static void do_symlink_diff(char **files)
592*cf5a6c84SAndroid Build Coastguard Worker {
593*cf5a6c84SAndroid Build Coastguard Worker   size_t i;
594*cf5a6c84SAndroid Build Coastguard Worker   int s = sizeof(toybuf)/2;
595*cf5a6c84SAndroid Build Coastguard Worker 
596*cf5a6c84SAndroid Build Coastguard Worker   TT.is_symlink = 1;
597*cf5a6c84SAndroid Build Coastguard Worker   TT.differ = 0;
598*cf5a6c84SAndroid Build Coastguard Worker   TT.link[0].name = TT.link[1].name = NULL;
599*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i < 2; i++) {
600*cf5a6c84SAndroid Build Coastguard Worker     TT.link[i].name = xreadlink(files[i]);
601*cf5a6c84SAndroid Build Coastguard Worker     if (TT.link[i].name == 0) {
602*cf5a6c84SAndroid Build Coastguard Worker       perror_msg("readlink failed");
603*cf5a6c84SAndroid Build Coastguard Worker       TT.differ = 2;
604*cf5a6c84SAndroid Build Coastguard Worker       free(TT.link[0].name);
605*cf5a6c84SAndroid Build Coastguard Worker       return;
606*cf5a6c84SAndroid Build Coastguard Worker     }
607*cf5a6c84SAndroid Build Coastguard Worker     TT.link[i].len = strlen(TT.link[i].name);
608*cf5a6c84SAndroid Build Coastguard Worker   }
609*cf5a6c84SAndroid Build Coastguard Worker 
610*cf5a6c84SAndroid Build Coastguard Worker   if (TT.link[0].len != TT.link[1].len) TT.differ = 1;
611*cf5a6c84SAndroid Build Coastguard Worker   else if (smemcmp(TT.link[0].name, TT.link[1].name, TT.link[0].len))
612*cf5a6c84SAndroid Build Coastguard Worker     TT.differ = 1;
613*cf5a6c84SAndroid Build Coastguard Worker   free(TT.link[0].name);
614*cf5a6c84SAndroid Build Coastguard Worker   free(TT.link[1].name);
615*cf5a6c84SAndroid Build Coastguard Worker   return;
616*cf5a6c84SAndroid Build Coastguard Worker }
617*cf5a6c84SAndroid Build Coastguard Worker 
do_diff(char ** files)618*cf5a6c84SAndroid Build Coastguard Worker static void do_diff(char **files)
619*cf5a6c84SAndroid Build Coastguard Worker {
620*cf5a6c84SAndroid Build Coastguard Worker   long i = 1, size = 1, x = 0, change = 0, ignore_white,
621*cf5a6c84SAndroid Build Coastguard Worker    start1, end1, start2, end2;
622*cf5a6c84SAndroid Build Coastguard Worker   struct diff *d;
623*cf5a6c84SAndroid Build Coastguard Worker   struct arg_list *llist = TT.L;
624*cf5a6c84SAndroid Build Coastguard Worker   int *J;
625*cf5a6c84SAndroid Build Coastguard Worker   regex_t reg;
626*cf5a6c84SAndroid Build Coastguard Worker 
627*cf5a6c84SAndroid Build Coastguard Worker   TT.offset[0] = TT.offset[1] = NULL;
628*cf5a6c84SAndroid Build Coastguard Worker   J = diff(files);
629*cf5a6c84SAndroid Build Coastguard Worker 
630*cf5a6c84SAndroid Build Coastguard Worker   if (!J) return; //No need to compare, have to status only
631*cf5a6c84SAndroid Build Coastguard Worker 
632*cf5a6c84SAndroid Build Coastguard Worker   if (TT.F) {
633*cf5a6c84SAndroid Build Coastguard Worker     xregcomp(&reg, TT.F, 0);
634*cf5a6c84SAndroid Build Coastguard Worker   }
635*cf5a6c84SAndroid Build Coastguard Worker 
636*cf5a6c84SAndroid Build Coastguard Worker   d = xzalloc(size *sizeof(struct diff));
637*cf5a6c84SAndroid Build Coastguard Worker   do {
638*cf5a6c84SAndroid Build Coastguard Worker     ignore_white = 0;
639*cf5a6c84SAndroid Build Coastguard Worker     for (d[x].a = i; d[x].a<=TT.file[0].len; d[x].a++) {
640*cf5a6c84SAndroid Build Coastguard Worker       if (J[d[x].a] != (J[d[x].a - 1] + 1)) break;
641*cf5a6c84SAndroid Build Coastguard Worker       else continue;
642*cf5a6c84SAndroid Build Coastguard Worker     }
643*cf5a6c84SAndroid Build Coastguard Worker     d[x].c = (J[d[x].a - 1] + 1);
644*cf5a6c84SAndroid Build Coastguard Worker 
645*cf5a6c84SAndroid Build Coastguard Worker     for (d[x].b = (d[x].a - 1); d[x].b<=TT.file[0].len; d[x].b++) {
646*cf5a6c84SAndroid Build Coastguard Worker       if (J[d[x].b + 1]) break;
647*cf5a6c84SAndroid Build Coastguard Worker       else continue;
648*cf5a6c84SAndroid Build Coastguard Worker     }
649*cf5a6c84SAndroid Build Coastguard Worker     d[x].d = (J[d[x].b + 1] - 1);
650*cf5a6c84SAndroid Build Coastguard Worker 
651*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(B)) {
652*cf5a6c84SAndroid Build Coastguard Worker       if (d[x].a <= d[x].b) {
653*cf5a6c84SAndroid Build Coastguard Worker         if ((TT.offset[0][d[x].b] - TT.offset[0][d[x].a - 1])
654*cf5a6c84SAndroid Build Coastguard Worker             == (d[x].b - d[x].a + 1))
655*cf5a6c84SAndroid Build Coastguard Worker           ignore_white = 1;
656*cf5a6c84SAndroid Build Coastguard Worker       } else if (d[x].c <= d[x].d){
657*cf5a6c84SAndroid Build Coastguard Worker         if ((TT.offset[1][d[x].d] - TT.offset[1][d[x].c - 1])
658*cf5a6c84SAndroid Build Coastguard Worker             == (d[x].d - d[x].c + 1))
659*cf5a6c84SAndroid Build Coastguard Worker           ignore_white = 1;
660*cf5a6c84SAndroid Build Coastguard Worker       }
661*cf5a6c84SAndroid Build Coastguard Worker     }
662*cf5a6c84SAndroid Build Coastguard Worker 
663*cf5a6c84SAndroid Build Coastguard Worker     //is we have diff ?   TODO: lolcat?
664*cf5a6c84SAndroid Build Coastguard Worker     if ((d[x].a <= d[x].b || d[x].c <= d[x].d) && !ignore_white) change = 1;
665*cf5a6c84SAndroid Build Coastguard Worker 
666*cf5a6c84SAndroid Build Coastguard Worker     if (!ignore_white) d = xrealloc(d, (x + 2) *sizeof(struct diff));
667*cf5a6c84SAndroid Build Coastguard Worker     i = d[x].b + 1;
668*cf5a6c84SAndroid Build Coastguard Worker     if (i>TT.file[0].len) break;
669*cf5a6c84SAndroid Build Coastguard Worker     J[d[x].b] = d[x].d;
670*cf5a6c84SAndroid Build Coastguard Worker     if (!ignore_white) x++;
671*cf5a6c84SAndroid Build Coastguard Worker   } while (i<=TT.file[0].len);
672*cf5a6c84SAndroid Build Coastguard Worker 
673*cf5a6c84SAndroid Build Coastguard Worker   i = x+1;
674*cf5a6c84SAndroid Build Coastguard Worker   TT.differ = change; //update status, may change bcoz of -w etc.
675*cf5a6c84SAndroid Build Coastguard Worker 
676*cf5a6c84SAndroid Build Coastguard Worker   if (!FLAG(q) && change) {
677*cf5a6c84SAndroid Build Coastguard Worker     if (!TT.new_line_format) {
678*cf5a6c84SAndroid Build Coastguard Worker       if (FLAG(color)) printf("\e[1m");
679*cf5a6c84SAndroid Build Coastguard Worker       if (FLAG(L)) printf("--- %s\n", llist->arg);
680*cf5a6c84SAndroid Build Coastguard Worker       else show_label("---", files[0], &(TT).st[0]);
681*cf5a6c84SAndroid Build Coastguard Worker       if (!FLAG(L) || !llist->next) show_label("+++", files[1], &(TT).st[1]);
682*cf5a6c84SAndroid Build Coastguard Worker       else {
683*cf5a6c84SAndroid Build Coastguard Worker         while (llist->next) llist = llist->next;
684*cf5a6c84SAndroid Build Coastguard Worker         printf("+++ %s\n", llist->arg);
685*cf5a6c84SAndroid Build Coastguard Worker       }
686*cf5a6c84SAndroid Build Coastguard Worker       if (FLAG(color)) printf("\e[0m");
687*cf5a6c84SAndroid Build Coastguard Worker     }
688*cf5a6c84SAndroid Build Coastguard Worker 
689*cf5a6c84SAndroid Build Coastguard Worker     struct diff *t, *ptr1 = d, *ptr2 = d;
690*cf5a6c84SAndroid Build Coastguard Worker     while (i) {
691*cf5a6c84SAndroid Build Coastguard Worker       long a,b;
692*cf5a6c84SAndroid Build Coastguard Worker 
693*cf5a6c84SAndroid Build Coastguard Worker       // trim context to file len.
694*cf5a6c84SAndroid Build Coastguard Worker       if (TT.new_line_format || TT.U>TT.file[0].len) TT.U = TT.file[0].len;
695*cf5a6c84SAndroid Build Coastguard Worker       if (ptr1->b < ptr1->a && ptr1->d < ptr1->c) {
696*cf5a6c84SAndroid Build Coastguard Worker         i--;
697*cf5a6c84SAndroid Build Coastguard Worker         continue;
698*cf5a6c84SAndroid Build Coastguard Worker       }
699*cf5a6c84SAndroid Build Coastguard Worker       //Handle the context stuff
700*cf5a6c84SAndroid Build Coastguard Worker       a =  ptr1->a;
701*cf5a6c84SAndroid Build Coastguard Worker       b = minof(TT.file[0].len, ptr1->b);
702*cf5a6c84SAndroid Build Coastguard Worker       if (i == x + 1) ptr1->suff = maxof(1, a-TT.U);
703*cf5a6c84SAndroid Build Coastguard Worker       else if (ptr1[-1].prev >= ptr1->a-TT.U) ptr1->suff = ptr1[-1].prev+1;
704*cf5a6c84SAndroid Build Coastguard Worker       else ptr1->suff =  ptr1->a-TT.U;
705*cf5a6c84SAndroid Build Coastguard Worker calc_ct:
706*cf5a6c84SAndroid Build Coastguard Worker       if (i > 1) {
707*cf5a6c84SAndroid Build Coastguard Worker         if ((ptr2->b + TT.U) >= (ptr2  + 1)->a) {
708*cf5a6c84SAndroid Build Coastguard Worker           ptr2++;
709*cf5a6c84SAndroid Build Coastguard Worker           i--;
710*cf5a6c84SAndroid Build Coastguard Worker           goto calc_ct;
711*cf5a6c84SAndroid Build Coastguard Worker         } else ptr2->prev = ptr2->b + TT.U;
712*cf5a6c84SAndroid Build Coastguard Worker       } else ptr2->prev = ptr2->b;
713*cf5a6c84SAndroid Build Coastguard Worker       start1 = (ptr2->prev - ptr1->suff + 1);
714*cf5a6c84SAndroid Build Coastguard Worker       end1 = (start1 == 1) ? -1 : start1;
715*cf5a6c84SAndroid Build Coastguard Worker       start2 = maxof(1, ptr1->c - (ptr1->a - ptr1->suff));
716*cf5a6c84SAndroid Build Coastguard Worker       end2 = ptr2->prev - ptr2->b + ptr2->d;
717*cf5a6c84SAndroid Build Coastguard Worker 
718*cf5a6c84SAndroid Build Coastguard Worker       if (!TT.new_line_format) {
719*cf5a6c84SAndroid Build Coastguard Worker         if (FLAG(color)) printf("\e[36m");
720*cf5a6c84SAndroid Build Coastguard Worker         printf("@@ -%ld", start1 ? ptr1->suff: (ptr1->suff -1));
721*cf5a6c84SAndroid Build Coastguard Worker         if (end1 != -1) printf(",%ld ", ptr2->prev-ptr1->suff + 1);
722*cf5a6c84SAndroid Build Coastguard Worker         else putchar(' ');
723*cf5a6c84SAndroid Build Coastguard Worker 
724*cf5a6c84SAndroid Build Coastguard Worker         printf("+%ld", (end2 - start2 + 1) ? start2: (start2 -1));
725*cf5a6c84SAndroid Build Coastguard Worker         if ((end2 - start2 +1) != 1) printf(",%ld ", (end2 - start2 +1));
726*cf5a6c84SAndroid Build Coastguard Worker         else putchar(' ');
727*cf5a6c84SAndroid Build Coastguard Worker         printf("@@");
728*cf5a6c84SAndroid Build Coastguard Worker         if (FLAG(color)) printf("\e[0m");
729*cf5a6c84SAndroid Build Coastguard Worker         if (TT.F) {
730*cf5a6c84SAndroid Build Coastguard Worker           print_line_matching_regex(ptr1->suff-1, &reg, TT.offset[0], TT.file[0].fp);
731*cf5a6c84SAndroid Build Coastguard Worker         }
732*cf5a6c84SAndroid Build Coastguard Worker         putchar('\n');
733*cf5a6c84SAndroid Build Coastguard Worker       }
734*cf5a6c84SAndroid Build Coastguard Worker 
735*cf5a6c84SAndroid Build Coastguard Worker       for (t = ptr1; t <= ptr2; t++) {
736*cf5a6c84SAndroid Build Coastguard Worker         if (t==ptr1) print_diff(t->suff, t->a-1, ' ', TT.offset[0], TT.file[0].fp);
737*cf5a6c84SAndroid Build Coastguard Worker         print_diff(t->a, t->b, '-', TT.offset[0], TT.file[0].fp);
738*cf5a6c84SAndroid Build Coastguard Worker         print_diff(t->c, t->d, '+', TT.offset[1], TT.file[1].fp);
739*cf5a6c84SAndroid Build Coastguard Worker         if (t == ptr2)
740*cf5a6c84SAndroid Build Coastguard Worker           print_diff(t->b+1, (t)->prev, ' ', TT.offset[0], TT.file[0].fp);
741*cf5a6c84SAndroid Build Coastguard Worker         else print_diff(t->b+1, (t+1)->a-1, ' ', TT.offset[0], TT.file[0].fp);
742*cf5a6c84SAndroid Build Coastguard Worker       }
743*cf5a6c84SAndroid Build Coastguard Worker       ptr2++;
744*cf5a6c84SAndroid Build Coastguard Worker       ptr1 = ptr2;
745*cf5a6c84SAndroid Build Coastguard Worker       i--;
746*cf5a6c84SAndroid Build Coastguard Worker     } //end of while
747*cf5a6c84SAndroid Build Coastguard Worker   } //End of !FLAG_q
748*cf5a6c84SAndroid Build Coastguard Worker   free(d);
749*cf5a6c84SAndroid Build Coastguard Worker   free(J);
750*cf5a6c84SAndroid Build Coastguard Worker   free(TT.offset[0]);
751*cf5a6c84SAndroid Build Coastguard Worker   free(TT.offset[1]);
752*cf5a6c84SAndroid Build Coastguard Worker }
753*cf5a6c84SAndroid Build Coastguard Worker 
show_status(char ** files)754*cf5a6c84SAndroid Build Coastguard Worker static void show_status(char **files)
755*cf5a6c84SAndroid Build Coastguard Worker {
756*cf5a6c84SAndroid Build Coastguard Worker   if (TT.differ==2) return; // TODO: needed?
757*cf5a6c84SAndroid Build Coastguard Worker   if (TT.differ ? FLAG(q) || TT.is_binary || TT.is_symlink : FLAG(s))
758*cf5a6c84SAndroid Build Coastguard Worker     printf("%s %s and %s %s\n", TT.is_symlink ? "Symbolic links" : "Files",
759*cf5a6c84SAndroid Build Coastguard Worker       files[0], files[1], TT.differ ? "differ" : "are identical");
760*cf5a6c84SAndroid Build Coastguard Worker }
761*cf5a6c84SAndroid Build Coastguard Worker 
create_empty_entry(int l,int r,int j)762*cf5a6c84SAndroid Build Coastguard Worker static void create_empty_entry(int l , int r, int j)
763*cf5a6c84SAndroid Build Coastguard Worker {
764*cf5a6c84SAndroid Build Coastguard Worker   struct stat st[2];
765*cf5a6c84SAndroid Build Coastguard Worker   char *f[2], *path[2];
766*cf5a6c84SAndroid Build Coastguard Worker   int i;
767*cf5a6c84SAndroid Build Coastguard Worker 
768*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i < 2; i++) {
769*cf5a6c84SAndroid Build Coastguard Worker     if (j) {
770*cf5a6c84SAndroid Build Coastguard Worker       if (!FLAG(N) || i!=(j>0)) continue;
771*cf5a6c84SAndroid Build Coastguard Worker       path[!i] = concat_file_path(TT.dir[!i].list[0],
772*cf5a6c84SAndroid Build Coastguard Worker         TT.dir[i].list[i ? r : l]+TT.len[i]);
773*cf5a6c84SAndroid Build Coastguard Worker       f[!i] = "/dev/null";
774*cf5a6c84SAndroid Build Coastguard Worker     }
775*cf5a6c84SAndroid Build Coastguard Worker     path[i] = f[i] = TT.dir[i].list[i ? r : l];
776*cf5a6c84SAndroid Build Coastguard Worker     (FLAG(no_dereference) ? lstat : stat)(f[i], st+i);
777*cf5a6c84SAndroid Build Coastguard Worker     if (j) st[!i] = st[i];
778*cf5a6c84SAndroid Build Coastguard Worker   }
779*cf5a6c84SAndroid Build Coastguard Worker 
780*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i<2; i++) {
781*cf5a6c84SAndroid Build Coastguard Worker     if (!S_ISREG(st[i].st_mode) && !S_ISDIR(st[i].st_mode) && !S_ISLNK(st[i].st_mode)) {
782*cf5a6c84SAndroid Build Coastguard Worker       printf("File %s is not a regular file, symbolic link, or directory and was skipped\n",
783*cf5a6c84SAndroid Build Coastguard Worker         path[i]);
784*cf5a6c84SAndroid Build Coastguard Worker       break;
785*cf5a6c84SAndroid Build Coastguard Worker     }
786*cf5a6c84SAndroid Build Coastguard Worker   }
787*cf5a6c84SAndroid Build Coastguard Worker 
788*cf5a6c84SAndroid Build Coastguard Worker   if (i != 2);
789*cf5a6c84SAndroid Build Coastguard Worker   else if ((st[0].st_mode & S_IFMT) != (st[1].st_mode & S_IFMT)) {
790*cf5a6c84SAndroid Build Coastguard Worker     i = S_ISREG(st[0].st_mode) + 2 * S_ISLNK(st[0].st_mode);
791*cf5a6c84SAndroid Build Coastguard Worker     int k = S_ISREG(st[1].st_mode) + 2 * S_ISLNK(st[1].st_mode);
792*cf5a6c84SAndroid Build Coastguard Worker     char *fidir[] = {"directory", "regular file", "symbolic link"};
793*cf5a6c84SAndroid Build Coastguard Worker     printf("File %s is a %s while file %s is a %s\n",
794*cf5a6c84SAndroid Build Coastguard Worker       path[0], fidir[i], path[1], fidir[k]);
795*cf5a6c84SAndroid Build Coastguard Worker   } else if (S_ISDIR(st[0].st_mode))
796*cf5a6c84SAndroid Build Coastguard Worker     printf("Common subdirectories: %s and %s\n", path[0], path[1]);
797*cf5a6c84SAndroid Build Coastguard Worker   else if (S_ISLNK(st[0].st_mode)) {
798*cf5a6c84SAndroid Build Coastguard Worker     do_symlink_diff(f);
799*cf5a6c84SAndroid Build Coastguard Worker     show_status(path);
800*cf5a6c84SAndroid Build Coastguard Worker   } else {
801*cf5a6c84SAndroid Build Coastguard Worker     do_diff(f);
802*cf5a6c84SAndroid Build Coastguard Worker     show_status(path);
803*cf5a6c84SAndroid Build Coastguard Worker     if (TT.file[0].fp) fclose(TT.file[0].fp);
804*cf5a6c84SAndroid Build Coastguard Worker     if (TT.file[1].fp) fclose(TT.file[1].fp);
805*cf5a6c84SAndroid Build Coastguard Worker   }
806*cf5a6c84SAndroid Build Coastguard Worker 
807*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(N) && j) free(path[j<=0]);
808*cf5a6c84SAndroid Build Coastguard Worker }
809*cf5a6c84SAndroid Build Coastguard Worker 
diff_dir(int * start)810*cf5a6c84SAndroid Build Coastguard Worker static void diff_dir(int *start)
811*cf5a6c84SAndroid Build Coastguard Worker {
812*cf5a6c84SAndroid Build Coastguard Worker   int l, r, j = 0;
813*cf5a6c84SAndroid Build Coastguard Worker 
814*cf5a6c84SAndroid Build Coastguard Worker   l = start[0]; //left side file start
815*cf5a6c84SAndroid Build Coastguard Worker   r = start[1]; //right side file start
816*cf5a6c84SAndroid Build Coastguard Worker   while (l < TT.dir[0].nr_elm && r < TT.dir[1].nr_elm) {
817*cf5a6c84SAndroid Build Coastguard Worker     if ((j = strcmp (TT.dir[0].list[l]+TT.len[0],
818*cf5a6c84SAndroid Build Coastguard Worker             (TT.dir[1].list[r]+TT.len[1]))) && !FLAG(N)) {
819*cf5a6c84SAndroid Build Coastguard Worker       if (j > 0) {
820*cf5a6c84SAndroid Build Coastguard Worker         printf("Only in %s: %s\n", TT.dir[1].list[0], TT.dir[1].list[r]+TT.len[1]);
821*cf5a6c84SAndroid Build Coastguard Worker         free(TT.dir[1].list[r++]);
822*cf5a6c84SAndroid Build Coastguard Worker       } else {
823*cf5a6c84SAndroid Build Coastguard Worker         printf ("Only in %s: %s\n", TT.dir[0].list[0], TT.dir[0].list[l]+TT.len[0]);
824*cf5a6c84SAndroid Build Coastguard Worker         free(TT.dir[0].list[l++]);
825*cf5a6c84SAndroid Build Coastguard Worker       }
826*cf5a6c84SAndroid Build Coastguard Worker       TT.differ = 1;
827*cf5a6c84SAndroid Build Coastguard Worker     } else {
828*cf5a6c84SAndroid Build Coastguard Worker       create_empty_entry(l, r, j); //create non empty dirs/files if -N.
829*cf5a6c84SAndroid Build Coastguard Worker       if (j>=0) free(TT.dir[1].list[r++]);
830*cf5a6c84SAndroid Build Coastguard Worker       if (j<=0) free(TT.dir[0].list[l++]);
831*cf5a6c84SAndroid Build Coastguard Worker     }
832*cf5a6c84SAndroid Build Coastguard Worker   }
833*cf5a6c84SAndroid Build Coastguard Worker 
834*cf5a6c84SAndroid Build Coastguard Worker   if (l == TT.dir[0].nr_elm) {
835*cf5a6c84SAndroid Build Coastguard Worker     while (r<TT.dir[1].nr_elm) {
836*cf5a6c84SAndroid Build Coastguard Worker       if (!FLAG(N)) {
837*cf5a6c84SAndroid Build Coastguard Worker         printf ("Only in %s: %s\n", TT.dir[1].list[0], TT.dir[1].list[r]+TT.len[1]);
838*cf5a6c84SAndroid Build Coastguard Worker         TT.differ = 1;
839*cf5a6c84SAndroid Build Coastguard Worker       } else create_empty_entry(l, r, 1);
840*cf5a6c84SAndroid Build Coastguard Worker       free(TT.dir[1].list[r++]);
841*cf5a6c84SAndroid Build Coastguard Worker     }
842*cf5a6c84SAndroid Build Coastguard Worker   } else if (r == TT.dir[1].nr_elm) {
843*cf5a6c84SAndroid Build Coastguard Worker     while (l<TT.dir[0].nr_elm) {
844*cf5a6c84SAndroid Build Coastguard Worker       if (!FLAG(N)) {
845*cf5a6c84SAndroid Build Coastguard Worker         printf ("Only in %s: %s\n", TT.dir[0].list[0], TT.dir[0].list[l]+TT.len[0]);
846*cf5a6c84SAndroid Build Coastguard Worker         TT.differ = 1;
847*cf5a6c84SAndroid Build Coastguard Worker       } else create_empty_entry(l, r, -1);
848*cf5a6c84SAndroid Build Coastguard Worker       free(TT.dir[0].list[l++]);
849*cf5a6c84SAndroid Build Coastguard Worker     }
850*cf5a6c84SAndroid Build Coastguard Worker   }
851*cf5a6c84SAndroid Build Coastguard Worker   free(TT.dir[0].list[0]); //we are done, free root nodes too
852*cf5a6c84SAndroid Build Coastguard Worker   free(TT.dir[0].list);
853*cf5a6c84SAndroid Build Coastguard Worker   free(TT.dir[1].list[0]);
854*cf5a6c84SAndroid Build Coastguard Worker   free(TT.dir[1].list);
855*cf5a6c84SAndroid Build Coastguard Worker }
856*cf5a6c84SAndroid Build Coastguard Worker 
diff_main(void)857*cf5a6c84SAndroid Build Coastguard Worker void diff_main(void)
858*cf5a6c84SAndroid Build Coastguard Worker {
859*cf5a6c84SAndroid Build Coastguard Worker   int i, j = 0, k = 1, start[2] = {1, 1};
860*cf5a6c84SAndroid Build Coastguard Worker   char **files = toys.optargs;
861*cf5a6c84SAndroid Build Coastguard Worker 
862*cf5a6c84SAndroid Build Coastguard Worker   toys.exitval = 2;
863*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(color) && !isatty(1)) toys.optflags ^= FLAG_color;
864*cf5a6c84SAndroid Build Coastguard Worker 
865*cf5a6c84SAndroid Build Coastguard Worker   for (j = 0; j < 2; j++) {
866*cf5a6c84SAndroid Build Coastguard Worker     if (IS_STDIN(files[j])) fstat(0, &TT.st[j]);
867*cf5a6c84SAndroid Build Coastguard Worker     else if (FLAG(no_dereference)) xlstat(files[j], &TT.st[j]);
868*cf5a6c84SAndroid Build Coastguard Worker     else xstat(files[j], &TT.st[j]);
869*cf5a6c84SAndroid Build Coastguard Worker   }
870*cf5a6c84SAndroid Build Coastguard Worker 
871*cf5a6c84SAndroid Build Coastguard Worker   if (S_ISDIR(TT.st[0].st_mode) != S_ISDIR(TT.st[1].st_mode))
872*cf5a6c84SAndroid Build Coastguard Worker     error_exit("can't compare directory to non-directory");
873*cf5a6c84SAndroid Build Coastguard Worker 
874*cf5a6c84SAndroid Build Coastguard Worker   if (TT.unchanged_line_format || TT.old_line_format || TT.new_line_format) {
875*cf5a6c84SAndroid Build Coastguard Worker     if (S_ISDIR(TT.st[0].st_mode) && S_ISDIR(TT.st[1].st_mode))
876*cf5a6c84SAndroid Build Coastguard Worker       error_exit("can't use line format with directories");
877*cf5a6c84SAndroid Build Coastguard Worker     if (!TT.unchanged_line_format) TT.unchanged_line_format = "%l\n";
878*cf5a6c84SAndroid Build Coastguard Worker     if (!TT.old_line_format) TT.old_line_format = "%l\n";
879*cf5a6c84SAndroid Build Coastguard Worker     if (!TT.new_line_format) TT.new_line_format = "%l\n";
880*cf5a6c84SAndroid Build Coastguard Worker   }
881*cf5a6c84SAndroid Build Coastguard Worker 
882*cf5a6c84SAndroid Build Coastguard Worker   if (same_file(TT.st, TT.st+1)) {
883*cf5a6c84SAndroid Build Coastguard Worker     toys.exitval = 0;
884*cf5a6c84SAndroid Build Coastguard Worker     return show_status(files);
885*cf5a6c84SAndroid Build Coastguard Worker   }
886*cf5a6c84SAndroid Build Coastguard Worker 
887*cf5a6c84SAndroid Build Coastguard Worker   if (S_ISDIR(TT.st[0].st_mode) && S_ISDIR(TT.st[1].st_mode)) {
888*cf5a6c84SAndroid Build Coastguard Worker     for (j = 0; j < 2; j++) {
889*cf5a6c84SAndroid Build Coastguard Worker       memset(TT.dir+j, 0, sizeof(*TT.dir));
890*cf5a6c84SAndroid Build Coastguard Worker       dirtree_flagread(files[j], (FLAG(no_dereference)) ? 0 : DIRTREE_SYMFOLLOW, list_dir);
891*cf5a6c84SAndroid Build Coastguard Worker       TT.dir[j].nr_elm = TT.size; //size updated in list_dir
892*cf5a6c84SAndroid Build Coastguard Worker       qsort(&TT.dir[j].list[1], TT.size-1, sizeof(char *), (void *)cmp);
893*cf5a6c84SAndroid Build Coastguard Worker 
894*cf5a6c84SAndroid Build Coastguard Worker       TT.len[j] = strlen(TT.dir[j].list[0]); //calc root node len
895*cf5a6c84SAndroid Build Coastguard Worker       TT.len[j] += TT.dir[j].list[0][TT.len[j]-1] != '/';
896*cf5a6c84SAndroid Build Coastguard Worker 
897*cf5a6c84SAndroid Build Coastguard Worker       if (FLAG(S)) {
898*cf5a6c84SAndroid Build Coastguard Worker         while (k<TT.size && strcmp(TT.dir[j].list[k]+TT.len[j], TT.S)<0) {
899*cf5a6c84SAndroid Build Coastguard Worker           start[j]++;
900*cf5a6c84SAndroid Build Coastguard Worker           k++;
901*cf5a6c84SAndroid Build Coastguard Worker         }
902*cf5a6c84SAndroid Build Coastguard Worker       }
903*cf5a6c84SAndroid Build Coastguard Worker       TT.dir_num++;
904*cf5a6c84SAndroid Build Coastguard Worker       TT.size = 0;
905*cf5a6c84SAndroid Build Coastguard Worker       k = 1;
906*cf5a6c84SAndroid Build Coastguard Worker     }
907*cf5a6c84SAndroid Build Coastguard Worker     diff_dir(start);
908*cf5a6c84SAndroid Build Coastguard Worker   } else {
909*cf5a6c84SAndroid Build Coastguard Worker     if (S_ISDIR(TT.st[0].st_mode) || S_ISDIR(TT.st[1].st_mode)) {
910*cf5a6c84SAndroid Build Coastguard Worker       int d = S_ISDIR(TT.st[0].st_mode);
911*cf5a6c84SAndroid Build Coastguard Worker       char *slash = strrchr(files[d], '/');
912*cf5a6c84SAndroid Build Coastguard Worker 
913*cf5a6c84SAndroid Build Coastguard Worker       files[!d] = concat_file_path(files[!d], slash ? slash+1 : files[d]);
914*cf5a6c84SAndroid Build Coastguard Worker       if ((FLAG(no_dereference) ? lstat : stat)(files[!d], &TT.st[!d]))
915*cf5a6c84SAndroid Build Coastguard Worker         perror_exit("%s", files[!d]);
916*cf5a6c84SAndroid Build Coastguard Worker     }
917*cf5a6c84SAndroid Build Coastguard Worker     if ((S_ISLNK(TT.st[0].st_mode)) != S_ISLNK(TT.st[1].st_mode)) {
918*cf5a6c84SAndroid Build Coastguard Worker       i = !strcmp(files[0], "-") ? 0 : S_ISREG(TT.st[0].st_mode) + 2 * S_ISLNK(TT.st[0].st_mode);
919*cf5a6c84SAndroid Build Coastguard Worker       int k = !strcmp(files[0], "-") ? 0 : S_ISREG(TT.st[1].st_mode) + 2 * S_ISLNK(TT.st[1].st_mode);
920*cf5a6c84SAndroid Build Coastguard Worker       char *fidir[] = {"fifo", "regular file", "symbolic link"};
921*cf5a6c84SAndroid Build Coastguard Worker       printf("File %s is a %s while file %s is a %s\n",
922*cf5a6c84SAndroid Build Coastguard Worker         files[0], fidir[i], files[1], fidir[k]);
923*cf5a6c84SAndroid Build Coastguard Worker       TT.differ = 1;
924*cf5a6c84SAndroid Build Coastguard Worker     } else {
925*cf5a6c84SAndroid Build Coastguard Worker       if (S_ISLNK(TT.st[0].st_mode)) do_symlink_diff(files);
926*cf5a6c84SAndroid Build Coastguard Worker       else do_diff(files);
927*cf5a6c84SAndroid Build Coastguard Worker       show_status(files);
928*cf5a6c84SAndroid Build Coastguard Worker     }
929*cf5a6c84SAndroid Build Coastguard Worker     if (TT.file[0].fp) fclose(TT.file[0].fp);
930*cf5a6c84SAndroid Build Coastguard Worker     if (TT.file[1].fp) fclose(TT.file[1].fp);
931*cf5a6c84SAndroid Build Coastguard Worker   }
932*cf5a6c84SAndroid Build Coastguard Worker   toys.exitval = TT.differ; //exit status will be the status
933*cf5a6c84SAndroid Build Coastguard Worker }
934