Home | History | Annotate | Line # | Download | only in apropos
apropos.c revision 1.5
      1  1.5    tls /*      $NetBSD: apropos.c,v 1.5 1995/09/04 20:46:20 tls Exp $      */
      2  1.3  glass 
      3  1.1    cgd /*
      4  1.2  glass  * Copyright (c) 1987, 1993, 1994
      5  1.1    cgd  *	The Regents of the University of California.  All rights reserved.
      6  1.1    cgd  *
      7  1.1    cgd  * Redistribution and use in source and binary forms, with or without
      8  1.1    cgd  * modification, are permitted provided that the following conditions
      9  1.1    cgd  * are met:
     10  1.1    cgd  * 1. Redistributions of source code must retain the above copyright
     11  1.1    cgd  *    notice, this list of conditions and the following disclaimer.
     12  1.1    cgd  * 2. Redistributions in binary form must reproduce the above copyright
     13  1.1    cgd  *    notice, this list of conditions and the following disclaimer in the
     14  1.1    cgd  *    documentation and/or other materials provided with the distribution.
     15  1.1    cgd  * 3. All advertising materials mentioning features or use of this software
     16  1.1    cgd  *    must display the following acknowledgement:
     17  1.1    cgd  *	This product includes software developed by the University of
     18  1.1    cgd  *	California, Berkeley and its contributors.
     19  1.1    cgd  * 4. Neither the name of the University nor the names of its contributors
     20  1.1    cgd  *    may be used to endorse or promote products derived from this software
     21  1.1    cgd  *    without specific prior written permission.
     22  1.1    cgd  *
     23  1.1    cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     24  1.1    cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     25  1.1    cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     26  1.1    cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     27  1.1    cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     28  1.1    cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     29  1.1    cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     30  1.1    cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     31  1.1    cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     32  1.1    cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     33  1.1    cgd  * SUCH DAMAGE.
     34  1.1    cgd  */
     35  1.1    cgd 
     36  1.1    cgd #ifndef lint
     37  1.1    cgd static char copyright[] =
     38  1.2  glass "@(#) Copyright (c) 1987, 1993, 1994\n\
     39  1.1    cgd 	The Regents of the University of California.  All rights reserved.\n";
     40  1.1    cgd #endif /* not lint */
     41  1.1    cgd 
     42  1.1    cgd #ifndef lint
     43  1.3  glass #if 0
     44  1.5    tls static char sccsid[] = "@(#)apropos.c	8.8 (Berkeley) 5/4/95";
     45  1.3  glass #else
     46  1.5    tls static char rcsid[] = "$NetBSD: apropos.c,v 1.5 1995/09/04 20:46:20 tls Exp $";
     47  1.3  glass #endif
     48  1.1    cgd #endif /* not lint */
     49  1.1    cgd 
     50  1.1    cgd #include <sys/param.h>
     51  1.1    cgd #include <sys/queue.h>
     52  1.1    cgd 
     53  1.1    cgd #include <ctype.h>
     54  1.1    cgd #include <err.h>
     55  1.2  glass #include <limits.h>
     56  1.1    cgd #include <stdio.h>
     57  1.1    cgd #include <stdlib.h>
     58  1.1    cgd #include <string.h>
     59  1.5    tls #include <unistd.h>
     60  1.1    cgd 
     61  1.1    cgd #include "../man/config.h"
     62  1.1    cgd #include "../man/pathnames.h"
     63  1.1    cgd 
     64  1.2  glass static int *found, foundman;
     65  1.1    cgd 
     66  1.2  glass void apropos __P((char **, char *, int));
     67  1.2  glass void lowstr __P((char *, char *));
     68  1.2  glass int match __P((char *, char *));
     69  1.2  glass void usage __P((void));
     70  1.1    cgd 
     71  1.1    cgd int
     72  1.1    cgd main(argc, argv)
     73  1.1    cgd 	int argc;
     74  1.1    cgd 	char *argv[];
     75  1.1    cgd {
     76  1.1    cgd 	ENTRY *ep;
     77  1.1    cgd 	TAG *tp;
     78  1.1    cgd 	int ch, rv;
     79  1.1    cgd 	char *conffile, **p, *p_augment, *p_path;
     80  1.1    cgd 
     81  1.1    cgd 	conffile = NULL;
     82  1.1    cgd 	p_augment = p_path = NULL;
     83  1.1    cgd 	while ((ch = getopt(argc, argv, "C:M:m:P:")) != EOF)
     84  1.1    cgd 		switch (ch) {
     85  1.1    cgd 		case 'C':
     86  1.1    cgd 			conffile = optarg;
     87  1.1    cgd 			break;
     88  1.1    cgd 		case 'M':
     89  1.1    cgd 		case 'P':		/* backward compatible */
     90  1.1    cgd 			p_path = optarg;
     91  1.1    cgd 			break;
     92  1.1    cgd 		case 'm':
     93  1.1    cgd 			p_augment = optarg;
     94  1.1    cgd 			break;
     95  1.1    cgd 		case '?':
     96  1.1    cgd 		default:
     97  1.1    cgd 			usage();
     98  1.1    cgd 		}
     99  1.1    cgd 	argv += optind;
    100  1.1    cgd 	argc -= optind;
    101  1.1    cgd 
    102  1.1    cgd 	if (argc < 1)
    103  1.1    cgd 		usage();
    104  1.1    cgd 
    105  1.1    cgd 	if ((found = malloc((u_int)argc * sizeof(int))) == NULL)
    106  1.1    cgd 		err(1, NULL);
    107  1.1    cgd 	memset(found, 0, argc * sizeof(int));
    108  1.1    cgd 
    109  1.1    cgd 	for (p = argv; *p; ++p)			/* convert to lower-case */
    110  1.1    cgd 		lowstr(*p, *p);
    111  1.1    cgd 
    112  1.1    cgd 	if (p_augment)
    113  1.1    cgd 		apropos(argv, p_augment, 1);
    114  1.1    cgd 	if (p_path || (p_path = getenv("MANPATH")))
    115  1.1    cgd 		apropos(argv, p_path, 1);
    116  1.1    cgd 	else {
    117  1.1    cgd 		config(conffile);
    118  1.1    cgd 		ep = (tp = getlist("_whatdb")) == NULL ?
    119  1.1    cgd 		    NULL : tp->list.tqh_first;
    120  1.1    cgd 		for (; ep != NULL; ep = ep->q.tqe_next)
    121  1.1    cgd 			apropos(argv, ep->s, 0);
    122  1.1    cgd 	}
    123  1.1    cgd 
    124  1.2  glass 	if (!foundman)
    125  1.2  glass 		errx(1, "no %s file found", _PATH_WHATIS);
    126  1.2  glass 
    127  1.1    cgd 	rv = 1;
    128  1.1    cgd 	for (p = argv; *p; ++p)
    129  1.1    cgd 		if (found[p - argv])
    130  1.1    cgd 			rv = 0;
    131  1.1    cgd 		else
    132  1.1    cgd 			(void)printf("%s: nothing appropriate\n", *p);
    133  1.1    cgd 	exit(rv);
    134  1.1    cgd }
    135  1.1    cgd 
    136  1.2  glass void
    137  1.1    cgd apropos(argv, path, buildpath)
    138  1.1    cgd 	char **argv, *path;
    139  1.1    cgd 	int buildpath;
    140  1.1    cgd {
    141  1.2  glass 	char *end, *name, **p;
    142  1.2  glass 	char buf[LINE_MAX + 1], wbuf[LINE_MAX + 1];
    143  1.1    cgd 
    144  1.1    cgd 	for (name = path; name; name = end) {	/* through name list */
    145  1.2  glass 		if (end = strchr(name, ':'))
    146  1.1    cgd 			*end++ = '\0';
    147  1.1    cgd 
    148  1.1    cgd 		if (buildpath) {
    149  1.1    cgd 			char hold[MAXPATHLEN + 1];
    150  1.1    cgd 
    151  1.1    cgd 			(void)sprintf(hold, "%s/%s", name, _PATH_WHATIS);
    152  1.1    cgd 			name = hold;
    153  1.1    cgd 		}
    154  1.1    cgd 
    155  1.1    cgd 		if (!freopen(name, "r", stdin))
    156  1.1    cgd 			continue;
    157  1.1    cgd 
    158  1.1    cgd 		foundman = 1;
    159  1.1    cgd 
    160  1.1    cgd 		/* for each file found */
    161  1.1    cgd 		while (fgets(buf, sizeof(buf), stdin)) {
    162  1.2  glass 			if (!strchr(buf, '\n')) {
    163  1.2  glass 				warnx("%s: line too long", name);
    164  1.1    cgd 				continue;
    165  1.1    cgd 			}
    166  1.1    cgd 			lowstr(buf, wbuf);
    167  1.1    cgd 			for (p = argv; *p; ++p)
    168  1.1    cgd 				if (match(wbuf, *p)) {
    169  1.1    cgd 					(void)printf("%s", buf);
    170  1.1    cgd 					found[p - argv] = 1;
    171  1.1    cgd 
    172  1.1    cgd 					/* only print line once */
    173  1.1    cgd 					while (*++p)
    174  1.1    cgd 						if (match(wbuf, *p))
    175  1.1    cgd 							found[p - argv] = 1;
    176  1.1    cgd 					break;
    177  1.1    cgd 				}
    178  1.1    cgd 		}
    179  1.1    cgd 	}
    180  1.1    cgd }
    181  1.1    cgd 
    182  1.1    cgd /*
    183  1.1    cgd  * match --
    184  1.1    cgd  *	match anywhere the string appears
    185  1.1    cgd  */
    186  1.2  glass int
    187  1.1    cgd match(bp, str)
    188  1.2  glass 	char *bp, *str;
    189  1.1    cgd {
    190  1.2  glass 	int len;
    191  1.2  glass 	char test;
    192  1.1    cgd 
    193  1.1    cgd 	if (!*bp)
    194  1.2  glass 		return (0);
    195  1.1    cgd 	/* backward compatible: everything matches empty string */
    196  1.1    cgd 	if (!*str)
    197  1.2  glass 		return (1);
    198  1.1    cgd 	for (test = *str++, len = strlen(str); *bp;)
    199  1.1    cgd 		if (test == *bp++ && !strncmp(bp, str, len))
    200  1.2  glass 			return (1);
    201  1.2  glass 	return (0);
    202  1.1    cgd }
    203  1.1    cgd 
    204  1.1    cgd /*
    205  1.1    cgd  * lowstr --
    206  1.1    cgd  *	convert a string to lower case
    207  1.1    cgd  */
    208  1.2  glass void
    209  1.1    cgd lowstr(from, to)
    210  1.2  glass 	char *from, *to;
    211  1.1    cgd {
    212  1.2  glass 	char ch;
    213  1.1    cgd 
    214  1.1    cgd 	while ((ch = *from++) && ch != '\n')
    215  1.1    cgd 		*to++ = isupper(ch) ? tolower(ch) : ch;
    216  1.1    cgd 	*to = '\0';
    217  1.1    cgd }
    218  1.1    cgd 
    219  1.1    cgd /*
    220  1.1    cgd  * usage --
    221  1.1    cgd  *	print usage message and die
    222  1.1    cgd  */
    223  1.2  glass void
    224  1.1    cgd usage()
    225  1.1    cgd {
    226  1.2  glass 
    227  1.1    cgd 	(void)fprintf(stderr,
    228  1.1    cgd 	    "usage: apropos [-C file] [-M path] [-m path] keyword ...\n");
    229  1.1    cgd 	exit(1);
    230  1.1    cgd }
    231