From cc70d84c4a6b6e1c0224bbdfd5bd306252d9fb4f Mon Sep 17 00:00:00 2001 From: Brian Feldman Date: Tue, 29 Aug 2000 23:30:52 +0000 Subject: [PATCH] Use a C version of which(1). Submitted by: Dan Papasian Reviewed by: jhb --- usr.bin/which/Makefile | 9 +-- usr.bin/which/which.c | 145 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 146 insertions(+), 8 deletions(-) create mode 100644 usr.bin/which/which.c diff --git a/usr.bin/which/Makefile b/usr.bin/which/Makefile index 94646408bc84..fdc121bb9820 100644 --- a/usr.bin/which/Makefile +++ b/usr.bin/which/Makefile @@ -1,13 +1,6 @@ # @(#)Makefile 8.1 (Berkeley) 6/6/93 # $FreeBSD$ -MAINTAINER= wosch - - -MAN1= which.1 - -beforeinstall: - ${INSTALL} -c -o ${BINOWN} -g ${BINGRP} -m ${BINMODE} \ - ${.CURDIR}/which.pl ${DESTDIR}${BINDIR}/which +PROG= which .include diff --git a/usr.bin/which/which.c b/usr.bin/which/which.c new file mode 100644 index 000000000000..c0435fbbb950 --- /dev/null +++ b/usr.bin/which/which.c @@ -0,0 +1,145 @@ +/** + * Copyright (c) 2000 Dan Papasian. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. The name of the author may not be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT + * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF + * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + * $FreeBSD$ + */ + +#include +#include + +#include +#include +#include +#include +#include + +static void usage(void); +static int print_matches(char *, char *); + +int silent; +int allpaths; + +int +main(int argc, char **argv) +{ + char *p, *path; + ssize_t pathlen; + int opt, status; + + status = EXIT_SUCCESS; + + /* If called without args, die silently to conform */ + if (argc < 2) + exit(EXIT_FAILURE); + + while ((opt = getopt(argc, argv, "as")) != -1) { + switch (opt) { + case 'a': + allpaths = 1; + break; + case 's': + silent = 1; + break; + default: + usage(); + break; + } + } + + argv += optind; + argc -= optind; + + if ((p = getenv("PATH")) == NULL) + exit(EXIT_FAILURE); + pathlen = strlen(p) + 1; + path = malloc(pathlen); + if (path == NULL) + err(EXIT_FAILURE, NULL); + + if (argc == 0) + status = EXIT_FAILURE; + + while (argc > 0) { + memcpy(path, p, pathlen); + + if (strlen(*argv) > FILENAME_MAX || + print_matches(path, *argv) == -1) + status = EXIT_FAILURE; + + argv++; + argc--; + } + + exit(status); +} + +static void +usage(void) +{ + + errx(EXIT_FAILURE, "usage: which [-as] program ..."); +} + +static int +is_there(char *candidate) +{ + struct stat fin; + + /* XXX work around access(2) false positives for superuser */ + if (access(candidate, X_OK) == 0 && + stat(candidate, &fin) == 0 && + S_ISREG(fin.st_mode) && + (getuid() != 0 || + (fin.st_mode & (S_IXUSR | S_IXGRP | S_IXOTH)) != 0)) { + if (!silent) + printf("%s\n", candidate); + return (1); + } + return (0); +} + +static int +print_matches(char *path, char *filename) +{ + char candidate[PATH_MAX]; + char *d; + int found; + + if (*filename == '/') + return (is_there(filename) ? 0 : -1); + found = 0; + while ((d = strsep(&path, ":")) != NULL) { + if (snprintf(candidate, sizeof(candidate), "%s/%s", d, + filename) >= sizeof(candidate)) + continue; + if (is_there(candidate)) { + found = 1; + if (!allpaths) + break; + } + } + return (found ? 0 : -1); +} +