Add searchpath().

This commit is contained in:
rsc 2005-01-23 22:33:59 +00:00
parent 28125cbd4c
commit 4a62371140
3 changed files with 64 additions and 0 deletions

View file

@ -390,6 +390,7 @@ extern int postnote(int, int, char *);
extern double p9pow10(int);
/* extern int putenv(char*, char*); <stdlib.h. */
/* extern void qsort(void*, long, long, int (*)(void*, void*)); <stdlib.h> */
extern char* searchpath(char*);
/* extern int p9setjmp(p9jmp_buf); */
#define p9setjmp(b) sigsetjmp((void*)(b), 1)
/*

View file

@ -131,6 +131,7 @@ LIB9OFILES=\
read9pmsg.$O\
readn.$O\
rfork.$O\
searchpath.$O\
seek.$O\
sendfd.$O\
sleep.$O\

62
src/lib9/searchpath.c Normal file
View file

@ -0,0 +1,62 @@
#include <u.h>
#include <libc.h>
/*
* Search $PATH for an executable with the given name.
* Like in rc, mid-name slashes do not disable search.
* Should probably handle escaped colons,
* but I don't know what the syntax is.
*/
char*
searchpath(char *name)
{
char *path, *p, *next;
char *s, *ss;
int ns, l;
s = nil;
ns = 0;
if((name[0] == '.' && name[1] == '/')
|| (name[0] == '.' && name[1] == '.' && name[2] == '/')
|| (name[0] == '/')){
if(access(name, AEXEC) >= 0)
return strdup(name);
return nil;
}
path = getenv("PATH");
for(p=path; p && *p; p=next){
if((next = strchr(p, ':')) != nil)
*next++ = 0;
if(*p == 0){
if(access(name, AEXEC) >= 0){
free(s);
free(path);
return strdup(name);
}
}else{
l = strlen(p)+1+strlen(name)+1;
if(l > ns){
ss = realloc(s, l);
if(ss == nil){
free(s);
free(path);
return nil;
}
s = ss;
ns = l;
}
strcpy(s, p);
strcat(s, "/");
strcat(s, name);
if(access(s, AEXEC) >= 0){
free(path);
return s;
}
}
}
free(s);
free(path);
return nil;
}