xref: /aosp_15_r20/external/toybox/toys/posix/comm.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* comm.c - select or reject lines common to two files
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2012 Ilya Kuzmich <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  *
5*cf5a6c84SAndroid Build Coastguard Worker  * See http://opengroup.org/onlinepubs/9699919799/utilities/comm.html
6*cf5a6c84SAndroid Build Coastguard Worker 
7*cf5a6c84SAndroid Build Coastguard Worker // <# and ># take single digit, so 321 define flags
8*cf5a6c84SAndroid Build Coastguard Worker USE_COMM(NEWTOY(comm, "<2>2321", TOYFLAG_USR|TOYFLAG_BIN))
9*cf5a6c84SAndroid Build Coastguard Worker 
10*cf5a6c84SAndroid Build Coastguard Worker config COMM
11*cf5a6c84SAndroid Build Coastguard Worker   bool "comm"
12*cf5a6c84SAndroid Build Coastguard Worker   default y
13*cf5a6c84SAndroid Build Coastguard Worker   help
14*cf5a6c84SAndroid Build Coastguard Worker     usage: comm [-123] FILE1 FILE2
15*cf5a6c84SAndroid Build Coastguard Worker 
16*cf5a6c84SAndroid Build Coastguard Worker     Read FILE1 and FILE2, which should be ordered, and produce three text
17*cf5a6c84SAndroid Build Coastguard Worker     columns as output: lines only in FILE1; lines only in FILE2; and lines
18*cf5a6c84SAndroid Build Coastguard Worker     in both files. Filename "-" is a synonym for stdin.
19*cf5a6c84SAndroid Build Coastguard Worker 
20*cf5a6c84SAndroid Build Coastguard Worker     -1	Suppress the output column of lines unique to FILE1
21*cf5a6c84SAndroid Build Coastguard Worker     -2	Suppress the output column of lines unique to FILE2
22*cf5a6c84SAndroid Build Coastguard Worker     -3	Suppress the output column of lines duplicated in FILE1 and FILE2
23*cf5a6c84SAndroid Build Coastguard Worker */
24*cf5a6c84SAndroid Build Coastguard Worker 
25*cf5a6c84SAndroid Build Coastguard Worker #define FOR_comm
26*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
27*cf5a6c84SAndroid Build Coastguard Worker 
writeline(const char * line,int col)28*cf5a6c84SAndroid Build Coastguard Worker static void writeline(const char *line, int col)
29*cf5a6c84SAndroid Build Coastguard Worker {
30*cf5a6c84SAndroid Build Coastguard Worker   if (!col && FLAG(1)) return;
31*cf5a6c84SAndroid Build Coastguard Worker   else if (col == 1) {
32*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(2)) return;
33*cf5a6c84SAndroid Build Coastguard Worker     if (!FLAG(1)) putchar('\t');
34*cf5a6c84SAndroid Build Coastguard Worker   } else if (col == 2) {
35*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(3)) return;
36*cf5a6c84SAndroid Build Coastguard Worker     if (!FLAG(1)) putchar('\t');
37*cf5a6c84SAndroid Build Coastguard Worker     if (!FLAG(2)) putchar('\t');
38*cf5a6c84SAndroid Build Coastguard Worker   }
39*cf5a6c84SAndroid Build Coastguard Worker   puts(line);
40*cf5a6c84SAndroid Build Coastguard Worker }
41*cf5a6c84SAndroid Build Coastguard Worker 
comm_main(void)42*cf5a6c84SAndroid Build Coastguard Worker void comm_main(void)
43*cf5a6c84SAndroid Build Coastguard Worker {
44*cf5a6c84SAndroid Build Coastguard Worker   FILE *file[2];
45*cf5a6c84SAndroid Build Coastguard Worker   char *line[2];
46*cf5a6c84SAndroid Build Coastguard Worker   int i = 0;
47*cf5a6c84SAndroid Build Coastguard Worker 
48*cf5a6c84SAndroid Build Coastguard Worker   for (i = 0; i<2; i++) {
49*cf5a6c84SAndroid Build Coastguard Worker     file[i] = strcmp(toys.optargs[i], "-")?xfopen(toys.optargs[i], "r"):stdin;
50*cf5a6c84SAndroid Build Coastguard Worker     line[i] = xgetline(file[i]);
51*cf5a6c84SAndroid Build Coastguard Worker   }
52*cf5a6c84SAndroid Build Coastguard Worker 
53*cf5a6c84SAndroid Build Coastguard Worker   if (toys.optflags == 7) return;
54*cf5a6c84SAndroid Build Coastguard Worker 
55*cf5a6c84SAndroid Build Coastguard Worker   while (line[0] && line[1]) {
56*cf5a6c84SAndroid Build Coastguard Worker     int order = strcmp(line[0], line[1]);
57*cf5a6c84SAndroid Build Coastguard Worker 
58*cf5a6c84SAndroid Build Coastguard Worker     if (!order) {
59*cf5a6c84SAndroid Build Coastguard Worker       writeline(line[0], 2);
60*cf5a6c84SAndroid Build Coastguard Worker       for (i = 0; i < 2; i++) {
61*cf5a6c84SAndroid Build Coastguard Worker         free(line[i]);
62*cf5a6c84SAndroid Build Coastguard Worker         line[i] = xgetline(file[i]);
63*cf5a6c84SAndroid Build Coastguard Worker       }
64*cf5a6c84SAndroid Build Coastguard Worker     } else {
65*cf5a6c84SAndroid Build Coastguard Worker       i = order>0;
66*cf5a6c84SAndroid Build Coastguard Worker       writeline(line[i], i);
67*cf5a6c84SAndroid Build Coastguard Worker       free(line[i]);
68*cf5a6c84SAndroid Build Coastguard Worker       line[i] = xgetline(file[i]);
69*cf5a6c84SAndroid Build Coastguard Worker     }
70*cf5a6c84SAndroid Build Coastguard Worker   }
71*cf5a6c84SAndroid Build Coastguard Worker 
72*cf5a6c84SAndroid Build Coastguard Worker   // Print rest of the longer file.
73*cf5a6c84SAndroid Build Coastguard Worker   for (i = line[0] ? 0 : 1; line[i];) {
74*cf5a6c84SAndroid Build Coastguard Worker     writeline(line[i], i);
75*cf5a6c84SAndroid Build Coastguard Worker     free(line[i]);
76*cf5a6c84SAndroid Build Coastguard Worker     line[i] = xgetline(file[i]);
77*cf5a6c84SAndroid Build Coastguard Worker   }
78*cf5a6c84SAndroid Build Coastguard Worker 
79*cf5a6c84SAndroid Build Coastguard Worker   if (CFG_TOYBOX_FREE) fclose(file[0]), fclose(file[1]);
80*cf5a6c84SAndroid Build Coastguard Worker }
81