summaryrefslogtreecommitdiff
path: root/usr/src/lib/libcmd/common/comm.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr/src/lib/libcmd/common/comm.c')
-rw-r--r--usr/src/lib/libcmd/common/comm.c201
1 files changed, 201 insertions, 0 deletions
diff --git a/usr/src/lib/libcmd/common/comm.c b/usr/src/lib/libcmd/common/comm.c
new file mode 100644
index 0000000000..3e28867e7e
--- /dev/null
+++ b/usr/src/lib/libcmd/common/comm.c
@@ -0,0 +1,201 @@
+/***********************************************************************
+* *
+* This software is part of the ast package *
+* Copyright (c) 1992-2007 AT&T Knowledge Ventures *
+* and is licensed under the *
+* Common Public License, Version 1.0 *
+* by AT&T Knowledge Ventures *
+* *
+* A copy of the License is available at *
+* http://www.opensource.org/licenses/cpl1.0.txt *
+* (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
+* *
+* Information and Software Systems Research *
+* AT&T Research *
+* Florham Park NJ *
+* *
+* Glenn Fowler <gsf@research.att.com> *
+* David Korn <dgk@research.att.com> *
+* *
+***********************************************************************/
+#pragma prototyped
+/*
+ * David Korn
+ * AT&T Bell Laboratories
+ *
+ * comm
+ */
+
+static const char usage[] =
+"[-?\n@(#)$Id: comm (AT&T Research) 1999-04-28 $\n]"
+USAGE_LICENSE
+"[+NAME?comm - select or reject lines common to two files]"
+"[+DESCRIPTION?\bcomm\b reads two files \afile1\a and \afile2\a "
+ "which should be ordered in the collating sequence of the "
+ "current locale, and produces three text columns as output:]{"
+ "[+1?Lines only in \afile1\a.]"
+ "[+2?Lines only in \afile2\a.]"
+ "[+3?Lines in both files.]"
+ "}"
+"[+?If lines in either file are not ordered according to the collating "
+ "sequence of the current locale, the results are not specified.]"
+"[+?If either \afile1\a or \afile2\a is \b-\b, \bcomm\b "
+ "uses standard input starting at the current location.]"
+
+"[1?Suppress the output column of lines unique to \afile1\a.]"
+"[2?Suppress the output column of lines unique to \afile2\a.]"
+"[3?Suppress the output column of lines duplicate in \afile1\a and \afile2\a.]"
+"\n"
+"\nfile1 file2\n"
+"\n"
+"[+EXIT STATUS?]{"
+ "[+0?Both files processed successfully.]"
+ "[+>0?An error occurred.]"
+"}"
+"[+SEE ALSO?\bcmp\b(1), \bdiff\b(1)]"
+;
+
+
+#include <cmd.h>
+
+#define C_FILE1 1
+#define C_FILE2 2
+#define C_COMMON 4
+#define C_ALL (C_FILE1|C_FILE2|C_COMMON)
+
+static int comm(Sfio_t *in1, Sfio_t *in2, register Sfio_t *out,register int mode)
+{
+ register char *cp1, *cp2;
+ register int n1, n2, n, comp;
+ if(cp1 = sfgetr(in1,'\n',0))
+ n1 = sfvalue(in1);
+ if(cp2 = sfgetr(in2,'\n',0))
+ n2 = sfvalue(in2);
+ while(cp1 && cp2)
+ {
+ n=(n1<n2?n1:n2);
+ if((comp=memcmp(cp1,cp2,n-1))==0 && (comp=n1-n2)==0)
+ {
+ if(mode&C_COMMON)
+ {
+ if(mode!=C_COMMON)
+ {
+ sfputc(out,'\t');
+ if(mode==C_ALL)
+ sfputc(out,'\t');
+ }
+ if(sfwrite(out,cp1,n) < 0)
+ return(-1);
+ }
+ if(cp1 = sfgetr(in1,'\n',0))
+ n1 = sfvalue(in1);
+ if(cp2 = sfgetr(in2,'\n',0))
+ n2 = sfvalue(in2);
+ }
+ else if(comp > 0)
+ {
+ if(mode&C_FILE2)
+ {
+ if(mode&C_FILE1)
+ sfputc(out,'\t');
+ if(sfwrite(out,cp2,n2) < 0)
+ return(-1);
+ }
+ if(cp2 = sfgetr(in2,'\n',0))
+ n2 = sfvalue(in2);
+ }
+ else
+ {
+ if((mode&C_FILE1) && sfwrite(out,cp1,n1) < 0)
+ return(-1);
+ if(cp1 = sfgetr(in1,'\n',0))
+ n1 = sfvalue(in1);
+ }
+ }
+ n = 0;
+ if(cp2)
+ {
+ cp1 = cp2;
+ in1 = in2;
+ n1 = n2;
+ if(mode&C_FILE1)
+ n = 1;
+ mode &= C_FILE2;
+ }
+ else
+ mode &= C_FILE1;
+ if(!mode || !cp1)
+ {
+ if(cp1 && in1==sfstdin)
+ sfseek(in1,(Sfoff_t)0,SEEK_END);
+ return(0);
+ }
+ /* process the remaining stream */
+ while(1)
+ {
+ if(n)
+ sfputc(out,'\t');
+ if(sfwrite(out,cp1,n1) < 0)
+ return(-1);
+ if(!(cp1 = sfgetr(in1,'\n',0)))
+ return(0);
+ n1 = sfvalue(in1);
+ }
+ /* NOT REACHED */
+}
+
+int
+b_comm(int argc, char *argv[], void* context)
+{
+ register int n;
+ register int mode = C_FILE1|C_FILE2|C_COMMON;
+ register char *cp;
+ Sfio_t *f1, *f2;
+
+ cmdinit(argc, argv, context, ERROR_CATALOG, 0);
+ while (n = optget(argv, usage)) switch (n)
+ {
+ case '1':
+ mode &= ~C_FILE1;
+ break;
+ case '2':
+ mode &= ~C_FILE2;
+ break;
+ case '3':
+ mode &= ~C_COMMON;
+ break;
+ case ':':
+ error(2, "%s",opt_info.arg);
+ break;
+ case '?':
+ error(ERROR_usage(2), "%s",opt_info.arg);
+ break;
+ }
+ argv += opt_info.index;
+ argc -= opt_info.index;
+ if(error_info.errors || argc!=2)
+ error(ERROR_usage(2),"%s",optusage(NiL));
+ cp = *argv++;
+ if(streq(cp,"-"))
+ f1 = sfstdin;
+ else if(!(f1 = sfopen(NiL, cp,"r")))
+ error(ERROR_system(1),"%s: cannot open",cp);
+ cp = *argv;
+ if(streq(cp,"-"))
+ f2 = sfstdin;
+ else if(!(f2 = sfopen(NiL, cp,"r")))
+ error(ERROR_system(1),"%s: cannot open",cp);
+ if(mode)
+ {
+ if(comm(f1,f2,sfstdout,mode) < 0)
+ error(ERROR_system(1)," write error");
+ }
+ else if(f1==sfstdin || f2==sfstdin)
+ sfseek(sfstdin,(Sfoff_t)0,SEEK_END);
+ if(f1!=sfstdin)
+ sfclose(f1);
+ if(f2!=sfstdin)
+ sfclose(f2);
+ return(error_info.errors);
+}
+