From c389d9118152e45303173c221a33a8083e769884 Mon Sep 17 00:00:00 2001 From: Pavel Roskin Date: Mon, 5 Jun 2000 23:41:27 +0000 Subject: "which" rewritten to use stat(). Fixes to improve its compatability with traditional implementations --- Changelog | 2 ++ findutils/which.c | 63 +++++++++++++++++++++++++++++++++++++++---------------- which.c | 63 +++++++++++++++++++++++++++++++++++++++---------------- 3 files changed, 92 insertions(+), 36 deletions(-) diff --git a/Changelog b/Changelog index 004bca4..57b936e 100644 --- a/Changelog +++ b/Changelog @@ -50,6 +50,8 @@ * Fixed all fatalError() calls lacking a "\n", thanks to Pavel Roskin. * Fixed a segfault in yes when no args were given -- Pavel Roskin. * Simplified freeramdisk and added argument checking -- Pavel Roskin. + * "which" rewritten to use stat(). Fixes to improve its compatability + with traditional implementations -- Pavel Roskin. * More doc updates diff --git a/findutils/which.c b/findutils/which.c index 46c2204..7a11bf7 100644 --- a/findutils/which.c +++ b/findutils/which.c @@ -23,15 +23,18 @@ #include "internal.h" #include -#include +#include +#include extern int which_main(int argc, char **argv) { - char *path_list, *test, *tmp; - struct dirent *next; + char *path_list, *test, *tmp, *path_parsed; + char buf[PATH_MAX]; + struct stat filestat; + int count = 0; - if (**(argv + 1) == '-') { + if (argc <= 1 || **(argv + 1) == '-') { usage("which [COMMAND ...]\n" #ifndef BB_FEATURE_TRIVIAL_HELP "\nLocates a COMMAND.\n" @@ -44,21 +47,45 @@ extern int which_main(int argc, char **argv) if (!path_list) path_list = "/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin"; - while(argc-- > 0 && *(argv++) != '\0' && strlen(*argv)) { - for( test=path_list; (tmp=strchr(test, ':')) && (tmp+1)!=NULL; test=++tmp) { - DIR *dir; - *tmp='\0'; - //printf("Checking directory '%s'\n", test); - dir = opendir(test); - if (!dir) - continue; - while ((next = readdir(dir)) != NULL) { - //printf("Checking file '%s'\n", next->d_name); - if ((strcmp(next->d_name, *argv) == 0)) { - printf("%s/%s\n", test, next->d_name); - exit(TRUE); - } + path_parsed = malloc (strlen(path_list) + 1); + strcpy (path_parsed, path_list); + + /* Replace colons with zeros in path_parsed and count them */ + count = 1; + test = path_parsed; + while (1) { + tmp = strchr(test, ':'); + if (tmp == NULL) + break; + *tmp = 0; + test = tmp + 1; + count++; + } + + + while(argc-- > 0) { + int i; + int found = FALSE; + test = path_parsed; + argv++; + for (i = 0; i < count; i++) { + strcpy (buf, test); + strcat (buf, "/"); + strcat (buf, *argv); + if (stat (buf, &filestat) == 0 + && filestat.st_mode & S_IXUSR) + { + found = TRUE; + break; } + test += (strlen(test) + 1); + } + if (found == TRUE) + printf ("%s\n", buf); + else + { + printf ("which: no %s in (%s)\n", *argv, path_list); + exit (FALSE); } } exit(TRUE); diff --git a/which.c b/which.c index 46c2204..7a11bf7 100644 --- a/which.c +++ b/which.c @@ -23,15 +23,18 @@ #include "internal.h" #include -#include +#include +#include extern int which_main(int argc, char **argv) { - char *path_list, *test, *tmp; - struct dirent *next; + char *path_list, *test, *tmp, *path_parsed; + char buf[PATH_MAX]; + struct stat filestat; + int count = 0; - if (**(argv + 1) == '-') { + if (argc <= 1 || **(argv + 1) == '-') { usage("which [COMMAND ...]\n" #ifndef BB_FEATURE_TRIVIAL_HELP "\nLocates a COMMAND.\n" @@ -44,21 +47,45 @@ extern int which_main(int argc, char **argv) if (!path_list) path_list = "/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin"; - while(argc-- > 0 && *(argv++) != '\0' && strlen(*argv)) { - for( test=path_list; (tmp=strchr(test, ':')) && (tmp+1)!=NULL; test=++tmp) { - DIR *dir; - *tmp='\0'; - //printf("Checking directory '%s'\n", test); - dir = opendir(test); - if (!dir) - continue; - while ((next = readdir(dir)) != NULL) { - //printf("Checking file '%s'\n", next->d_name); - if ((strcmp(next->d_name, *argv) == 0)) { - printf("%s/%s\n", test, next->d_name); - exit(TRUE); - } + path_parsed = malloc (strlen(path_list) + 1); + strcpy (path_parsed, path_list); + + /* Replace colons with zeros in path_parsed and count them */ + count = 1; + test = path_parsed; + while (1) { + tmp = strchr(test, ':'); + if (tmp == NULL) + break; + *tmp = 0; + test = tmp + 1; + count++; + } + + + while(argc-- > 0) { + int i; + int found = FALSE; + test = path_parsed; + argv++; + for (i = 0; i < count; i++) { + strcpy (buf, test); + strcat (buf, "/"); + strcat (buf, *argv); + if (stat (buf, &filestat) == 0 + && filestat.st_mode & S_IXUSR) + { + found = TRUE; + break; } + test += (strlen(test) + 1); + } + if (found == TRUE) + printf ("%s\n", buf); + else + { + printf ("which: no %s in (%s)\n", *argv, path_list); + exit (FALSE); } } exit(TRUE); -- cgit v1.1