xref: /dflybsd-src/gnu/usr.bin/rcs/ident/ident.c (revision 86d7f5d305c6adaa56ff4582ece9859d73106103)
1*86d7f5d3SJohn Marino /* Identify RCS keyword strings in files.  */
2*86d7f5d3SJohn Marino 
3*86d7f5d3SJohn Marino /* Copyright 1982, 1988, 1989 Walter Tichy
4*86d7f5d3SJohn Marino    Copyright 1990, 1991, 1992, 1993, 1994, 1995 Paul Eggert
5*86d7f5d3SJohn Marino    Distributed under license by the Free Software Foundation, Inc.
6*86d7f5d3SJohn Marino 
7*86d7f5d3SJohn Marino This file is part of RCS.
8*86d7f5d3SJohn Marino 
9*86d7f5d3SJohn Marino RCS is free software; you can redistribute it and/or modify
10*86d7f5d3SJohn Marino it under the terms of the GNU General Public License as published by
11*86d7f5d3SJohn Marino the Free Software Foundation; either version 2, or (at your option)
12*86d7f5d3SJohn Marino any later version.
13*86d7f5d3SJohn Marino 
14*86d7f5d3SJohn Marino RCS is distributed in the hope that it will be useful,
15*86d7f5d3SJohn Marino but WITHOUT ANY WARRANTY; without even the implied warranty of
16*86d7f5d3SJohn Marino MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17*86d7f5d3SJohn Marino GNU General Public License for more details.
18*86d7f5d3SJohn Marino 
19*86d7f5d3SJohn Marino You should have received a copy of the GNU General Public License
20*86d7f5d3SJohn Marino along with RCS; see the file COPYING.
21*86d7f5d3SJohn Marino If not, write to the Free Software Foundation,
22*86d7f5d3SJohn Marino 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23*86d7f5d3SJohn Marino 
24*86d7f5d3SJohn Marino Report problems and direct all questions to:
25*86d7f5d3SJohn Marino 
26*86d7f5d3SJohn Marino     rcs-bugs@cs.purdue.edu
27*86d7f5d3SJohn Marino 
28*86d7f5d3SJohn Marino */
29*86d7f5d3SJohn Marino 
30*86d7f5d3SJohn Marino /*
31*86d7f5d3SJohn Marino  * $FreeBSD: src/gnu/usr.bin/rcs/ident/ident.c,v 1.7.2.1 2001/12/10 20:49:37 peter Exp $
32*86d7f5d3SJohn Marino  * $DragonFly: src/gnu/usr.bin/rcs/ident/ident.c,v 1.2 2003/06/17 04:25:47 dillon Exp $
33*86d7f5d3SJohn Marino  *
34*86d7f5d3SJohn Marino  * Revision 5.9  1995/06/16 06:19:24  eggert
35*86d7f5d3SJohn Marino  * Update FSF address.
36*86d7f5d3SJohn Marino  *
37*86d7f5d3SJohn Marino  * Revision 5.8  1995/06/01 16:23:43  eggert
38*86d7f5d3SJohn Marino  * (exiterr, reportError): New functions, needed for DOS and OS/2 ports.
39*86d7f5d3SJohn Marino  * (scanfile): Use them.
40*86d7f5d3SJohn Marino  *
41*86d7f5d3SJohn Marino  * Revision 5.7  1994/03/20 04:52:58  eggert
42*86d7f5d3SJohn Marino  * Remove `exiting' from identExit.
43*86d7f5d3SJohn Marino  *
44*86d7f5d3SJohn Marino  * Revision 5.6  1993/11/09 17:40:15  eggert
45*86d7f5d3SJohn Marino  * Add -V.
46*86d7f5d3SJohn Marino  *
47*86d7f5d3SJohn Marino  * Revision 5.5  1993/11/03 17:42:27  eggert
48*86d7f5d3SJohn Marino  * Test for char == EOF, not char < 0.
49*86d7f5d3SJohn Marino  *
50*86d7f5d3SJohn Marino  * Revision 5.4  1992/01/24  18:44:19  eggert
51*86d7f5d3SJohn Marino  * lint -> RCS_lint
52*86d7f5d3SJohn Marino  *
53*86d7f5d3SJohn Marino  * Revision 5.3  1991/09/10  22:15:46  eggert
54*86d7f5d3SJohn Marino  * Open files with FOPEN_R, not FOPEN_R_WORK,
55*86d7f5d3SJohn Marino  * because they might be executables, not working files.
56*86d7f5d3SJohn Marino  *
57*86d7f5d3SJohn Marino  * Revision 5.2  1991/08/19  03:13:55  eggert
58*86d7f5d3SJohn Marino  * Report read errors immediately.
59*86d7f5d3SJohn Marino  *
60*86d7f5d3SJohn Marino  * Revision 5.1  1991/02/25  07:12:37  eggert
61*86d7f5d3SJohn Marino  * Don't report empty keywords.  Check for I/O errors.
62*86d7f5d3SJohn Marino  *
63*86d7f5d3SJohn Marino  * Revision 5.0  1990/08/22  08:12:37  eggert
64*86d7f5d3SJohn Marino  * Don't limit output to known keywords.
65*86d7f5d3SJohn Marino  * Remove arbitrary limits and lint.  Ansify and Posixate.
66*86d7f5d3SJohn Marino  *
67*86d7f5d3SJohn Marino  * Revision 4.5  89/05/01  15:11:54  narten
68*86d7f5d3SJohn Marino  * changed copyright header to reflect current distribution rules
69*86d7f5d3SJohn Marino  *
70*86d7f5d3SJohn Marino  * Revision 4.4  87/10/23  17:09:57  narten
71*86d7f5d3SJohn Marino  * added exit(0) so exit return code would be non random
72*86d7f5d3SJohn Marino  *
73*86d7f5d3SJohn Marino  * Revision 4.3  87/10/18  10:23:55  narten
74*86d7f5d3SJohn Marino  * Updating version numbers. Changes relative to 1.1 are actually relative
75*86d7f5d3SJohn Marino  * to 4.1
76*86d7f5d3SJohn Marino  *
77*86d7f5d3SJohn Marino  * Revision 1.3  87/07/09  09:20:52  trinkle
78*86d7f5d3SJohn Marino  * Added check to make sure there is at least one arg before comparing argv[1]
79*86d7f5d3SJohn Marino  * with "-q".  This necessary on machines that don't allow dereferncing null
80*86d7f5d3SJohn Marino  * pointers (i.e. Suns).
81*86d7f5d3SJohn Marino  *
82*86d7f5d3SJohn Marino  * Revision 1.2  87/03/27  14:21:47  jenkins
83*86d7f5d3SJohn Marino  * Port to suns
84*86d7f5d3SJohn Marino  *
85*86d7f5d3SJohn Marino  * Revision 4.1  83/05/10  16:31:02  wft
86*86d7f5d3SJohn Marino  * Added option -q and input from reading stdin.
87*86d7f5d3SJohn Marino  * Marker matching is now done with trymatch() (independent of keywords).
88*86d7f5d3SJohn Marino  *
89*86d7f5d3SJohn Marino  * Revision 3.4  83/02/18  17:37:49  wft
90*86d7f5d3SJohn Marino  * removed printing of new line after last file.
91*86d7f5d3SJohn Marino  *
92*86d7f5d3SJohn Marino  * Revision 3.3  82/12/04  12:48:55  wft
93*86d7f5d3SJohn Marino  * Added LOCKER.
94*86d7f5d3SJohn Marino  *
95*86d7f5d3SJohn Marino  * Revision 3.2  82/11/28  18:24:17  wft
96*86d7f5d3SJohn Marino  * removed Suffix; added ungetc to avoid skipping over trailing KDELIM.
97*86d7f5d3SJohn Marino  *
98*86d7f5d3SJohn Marino  * Revision 3.1  82/10/13  15:58:51  wft
99*86d7f5d3SJohn Marino  * fixed type of variables receiving from getc() (char-->int).
100*86d7f5d3SJohn Marino */
101*86d7f5d3SJohn Marino 
102*86d7f5d3SJohn Marino #include  "rcsbase.h"
103*86d7f5d3SJohn Marino 
104*86d7f5d3SJohn Marino static int match P((FILE*));
105*86d7f5d3SJohn Marino static int scanfile P((FILE*,char const*,int));
106*86d7f5d3SJohn Marino static void reportError P((char const*));
107*86d7f5d3SJohn Marino 
108*86d7f5d3SJohn Marino mainProg(identId, "ident", "$DragonFly: src/gnu/usr.bin/rcs/ident/ident.c,v 1.2 2003/06/17 04:25:47 dillon Exp $")
109*86d7f5d3SJohn Marino /*  Ident searches the named files for all occurrences
110*86d7f5d3SJohn Marino  *  of the pattern $@: text $ where @ is a keyword.
111*86d7f5d3SJohn Marino  */
112*86d7f5d3SJohn Marino 
113*86d7f5d3SJohn Marino {
114*86d7f5d3SJohn Marino    FILE *fp;
115*86d7f5d3SJohn Marino    int quiet = 0;
116*86d7f5d3SJohn Marino    int status = EXIT_SUCCESS;
117*86d7f5d3SJohn Marino    char const *a;
118*86d7f5d3SJohn Marino 
119*86d7f5d3SJohn Marino    while ((a = *++argv)  &&  *a=='-')
120*86d7f5d3SJohn Marino 	while (*++a)
121*86d7f5d3SJohn Marino 	    switch (*a) {
122*86d7f5d3SJohn Marino 		case 'q':
123*86d7f5d3SJohn Marino 		    quiet = 1;
124*86d7f5d3SJohn Marino 		    break;
125*86d7f5d3SJohn Marino 
126*86d7f5d3SJohn Marino 		case 'V':
127*86d7f5d3SJohn Marino 		    VOID printf("RCS version %s\n", RCS_version_string);
128*86d7f5d3SJohn Marino 		    quiet = -1;
129*86d7f5d3SJohn Marino 		    break;
130*86d7f5d3SJohn Marino 
131*86d7f5d3SJohn Marino 		default:
132*86d7f5d3SJohn Marino 		    VOID fprintf(stderr,
133*86d7f5d3SJohn Marino 			"ident: usage: ident -{qV} [file...]\n"
134*86d7f5d3SJohn Marino 		    );
135*86d7f5d3SJohn Marino 		    exitmain(EXIT_FAILURE);
136*86d7f5d3SJohn Marino 		    break;
137*86d7f5d3SJohn Marino 	    }
138*86d7f5d3SJohn Marino 
139*86d7f5d3SJohn Marino    if (0 <= quiet) {
140*86d7f5d3SJohn Marino        if (!a)
141*86d7f5d3SJohn Marino 	    VOID scanfile(stdin, (char*)0, quiet);
142*86d7f5d3SJohn Marino        else
143*86d7f5d3SJohn Marino 	    do {
144*86d7f5d3SJohn Marino 		if (!(fp = fopen(a, FOPEN_RB))) {
145*86d7f5d3SJohn Marino 		    reportError(a);
146*86d7f5d3SJohn Marino 		    status = EXIT_FAILURE;
147*86d7f5d3SJohn Marino 		} else if (
148*86d7f5d3SJohn Marino 		    scanfile(fp, a, quiet) != 0
149*86d7f5d3SJohn Marino 		    || (argv[1]  &&  putchar('\n') == EOF)
150*86d7f5d3SJohn Marino 		)
151*86d7f5d3SJohn Marino 		    break;
152*86d7f5d3SJohn Marino 	    } while ((a = *++argv));
153*86d7f5d3SJohn Marino    }
154*86d7f5d3SJohn Marino 
155*86d7f5d3SJohn Marino    if (ferror(stdout) || fclose(stdout)!=0) {
156*86d7f5d3SJohn Marino       reportError("standard output");
157*86d7f5d3SJohn Marino       status = EXIT_FAILURE;
158*86d7f5d3SJohn Marino    }
159*86d7f5d3SJohn Marino    exitmain(status);
160*86d7f5d3SJohn Marino }
161*86d7f5d3SJohn Marino 
162*86d7f5d3SJohn Marino #if RCS_lint
163*86d7f5d3SJohn Marino #	define exiterr identExit
164*86d7f5d3SJohn Marino #endif
165*86d7f5d3SJohn Marino 	void
exiterr()166*86d7f5d3SJohn Marino exiterr()
167*86d7f5d3SJohn Marino {
168*86d7f5d3SJohn Marino 	_exit(EXIT_FAILURE);
169*86d7f5d3SJohn Marino }
170*86d7f5d3SJohn Marino 
171*86d7f5d3SJohn Marino 	static void
reportError(s)172*86d7f5d3SJohn Marino reportError(s)
173*86d7f5d3SJohn Marino 	char const *s;
174*86d7f5d3SJohn Marino {
175*86d7f5d3SJohn Marino 	int e = errno;
176*86d7f5d3SJohn Marino 	VOID fprintf(stderr, "%s error: ", cmdid);
177*86d7f5d3SJohn Marino 	errno = e;
178*86d7f5d3SJohn Marino 	perror(s);
179*86d7f5d3SJohn Marino }
180*86d7f5d3SJohn Marino 
181*86d7f5d3SJohn Marino 
182*86d7f5d3SJohn Marino 	static int
scanfile(file,name,quiet)183*86d7f5d3SJohn Marino scanfile(file, name, quiet)
184*86d7f5d3SJohn Marino 	register FILE *file;
185*86d7f5d3SJohn Marino 	char const *name;
186*86d7f5d3SJohn Marino 	int quiet;
187*86d7f5d3SJohn Marino /* Function: scan an open file with descriptor file for keywords.
188*86d7f5d3SJohn Marino  * Return -1 if there's a write error; exit immediately on a read error.
189*86d7f5d3SJohn Marino  */
190*86d7f5d3SJohn Marino {
191*86d7f5d3SJohn Marino    register int c;
192*86d7f5d3SJohn Marino 
193*86d7f5d3SJohn Marino    if (name) {
194*86d7f5d3SJohn Marino       VOID printf("%s:\n", name);
195*86d7f5d3SJohn Marino       if (ferror(stdout))
196*86d7f5d3SJohn Marino 	 return -1;
197*86d7f5d3SJohn Marino    } else
198*86d7f5d3SJohn Marino       name = "standard input";
199*86d7f5d3SJohn Marino    c = 0;
200*86d7f5d3SJohn Marino    while (c != EOF  ||  ! (feof(file)|ferror(file))) {
201*86d7f5d3SJohn Marino       if (c == KDELIM) {
202*86d7f5d3SJohn Marino 	 if ((c = match(file)))
203*86d7f5d3SJohn Marino 	    continue;
204*86d7f5d3SJohn Marino 	 if (ferror(stdout))
205*86d7f5d3SJohn Marino 	    return -1;
206*86d7f5d3SJohn Marino 	 quiet = true;
207*86d7f5d3SJohn Marino       }
208*86d7f5d3SJohn Marino       c = getc(file);
209*86d7f5d3SJohn Marino    }
210*86d7f5d3SJohn Marino    if (ferror(file) || fclose(file) != 0) {
211*86d7f5d3SJohn Marino       reportError(name);
212*86d7f5d3SJohn Marino       /*
213*86d7f5d3SJohn Marino       * The following is equivalent to exit(EXIT_FAILURE), but we invoke
214*86d7f5d3SJohn Marino       * exiterr to keep lint happy.  The DOS and OS/2 ports need exiterr.
215*86d7f5d3SJohn Marino       */
216*86d7f5d3SJohn Marino       VOID fflush(stderr);
217*86d7f5d3SJohn Marino       VOID fflush(stdout);
218*86d7f5d3SJohn Marino       exiterr();
219*86d7f5d3SJohn Marino    }
220*86d7f5d3SJohn Marino    if (!quiet)
221*86d7f5d3SJohn Marino       VOID fprintf(stderr, "%s warning: no id keywords in %s\n", cmdid, name);
222*86d7f5d3SJohn Marino    return 0;
223*86d7f5d3SJohn Marino }
224*86d7f5d3SJohn Marino 
225*86d7f5d3SJohn Marino 
226*86d7f5d3SJohn Marino 
227*86d7f5d3SJohn Marino 	static int
match(fp)228*86d7f5d3SJohn Marino match(fp)   /* group substring between two KDELIM's; then do pattern match */
229*86d7f5d3SJohn Marino    register FILE *fp;
230*86d7f5d3SJohn Marino {
231*86d7f5d3SJohn Marino    char line[BUFSIZ];
232*86d7f5d3SJohn Marino    register int c;
233*86d7f5d3SJohn Marino    register char * tp;
234*86d7f5d3SJohn Marino 
235*86d7f5d3SJohn Marino    tp = line;
236*86d7f5d3SJohn Marino    while ((c = getc(fp)) != VDELIM) {
237*86d7f5d3SJohn Marino       if (c == EOF  &&  feof(fp) | ferror(fp))
238*86d7f5d3SJohn Marino 	 return c;
239*86d7f5d3SJohn Marino       switch (ctab[c]) {
240*86d7f5d3SJohn Marino 	 case LETTER: case Letter: case DIGIT:
241*86d7f5d3SJohn Marino 	    *tp++ = c;
242*86d7f5d3SJohn Marino 	    if (tp < line+sizeof(line)-4)
243*86d7f5d3SJohn Marino 	       break;
244*86d7f5d3SJohn Marino 	    /* fall into */
245*86d7f5d3SJohn Marino 	 default:
246*86d7f5d3SJohn Marino 	    return c ? c : '\n'/* anything but 0 or KDELIM or EOF */;
247*86d7f5d3SJohn Marino       }
248*86d7f5d3SJohn Marino    }
249*86d7f5d3SJohn Marino    if (tp == line)
250*86d7f5d3SJohn Marino       return c;
251*86d7f5d3SJohn Marino    *tp++ = c;
252*86d7f5d3SJohn Marino    if ((c = getc(fp)) != ' ')
253*86d7f5d3SJohn Marino       return c ? c : '\n';
254*86d7f5d3SJohn Marino    *tp++ = c;
255*86d7f5d3SJohn Marino    while( (c = getc(fp)) != KDELIM ) {
256*86d7f5d3SJohn Marino       if (c == EOF  &&  feof(fp) | ferror(fp))
257*86d7f5d3SJohn Marino 	    return c;
258*86d7f5d3SJohn Marino       switch (ctab[c]) {
259*86d7f5d3SJohn Marino 	 default:
260*86d7f5d3SJohn Marino 	    *tp++ = c;
261*86d7f5d3SJohn Marino 	    if (tp < line+sizeof(line)-2)
262*86d7f5d3SJohn Marino 	       break;
263*86d7f5d3SJohn Marino 	    /* fall into */
264*86d7f5d3SJohn Marino 	 case NEWLN: case UNKN:
265*86d7f5d3SJohn Marino 	    return c ? c : '\n';
266*86d7f5d3SJohn Marino       }
267*86d7f5d3SJohn Marino    }
268*86d7f5d3SJohn Marino    if (tp[-1] != ' ')
269*86d7f5d3SJohn Marino       return c;
270*86d7f5d3SJohn Marino    *tp++ = c;     /*append trailing KDELIM*/
271*86d7f5d3SJohn Marino    *tp   = '\0';
272*86d7f5d3SJohn Marino    VOID printf("     %c%s\n", KDELIM, line);
273*86d7f5d3SJohn Marino    return 0;
274*86d7f5d3SJohn Marino }
275