diff options
author | Adam <you@example.com> | 2020-05-17 05:51:50 +0200 |
---|---|---|
committer | Adam <you@example.com> | 2020-05-17 05:51:50 +0200 |
commit | e611b132f9b8abe35b362e5870b74bce94a1e58e (patch) | |
tree | a5781d2ec0e085eeca33cf350cf878f2efea6fe5 /private/crt32/exec/execvpe.c | |
download | NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar.gz NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar.bz2 NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar.lz NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar.xz NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.tar.zst NT4.0-e611b132f9b8abe35b362e5870b74bce94a1e58e.zip |
Diffstat (limited to 'private/crt32/exec/execvpe.c')
-rw-r--r-- | private/crt32/exec/execvpe.c | 171 |
1 files changed, 171 insertions, 0 deletions
diff --git a/private/crt32/exec/execvpe.c b/private/crt32/exec/execvpe.c new file mode 100644 index 000000000..650f34b2a --- /dev/null +++ b/private/crt32/exec/execvpe.c @@ -0,0 +1,171 @@ +/*** +*execvpe.c - execute a file with given environ; search along PATH +* +* Copyright (c) 1985-1992, Microsoft Corporation. All rights reserved. +* +*Purpose: +* defines _execvpe() - execute a file with given environ +* +*Revision History: +* 10-17-83 RN written +* 10-29-85 TC added execvpe capability +* 11-19-86 SKS handle both kinds of slashes +* 12-01-86 JMB added Kanji file name support under conditional KANJI +* switches, corrected header info +* removed bogus check for env = b after call to strncpy(). +* 12-11-87 JCR Added "_LOAD_DS" to declaration +* 09-05-88 SKS Treat EACCES the same as ENOENT -- keep trying +* 10-18-88 GJF Removed copy of PATH string to local array, changed +* bbuf to be a malloc-ed buffer. Removed bogus limits +* on the size of that PATH string. +* 10-26-88 GJF Don't search PATH when relative pathname is given (per +* Stevesa). Also, if the name built from PATH component +* and filename is a UNC name, allow any error. +* 11-20-89 GJF Fixed copyright. Added const attribute to types of +* filename, argvector and envptr. Also, added "#include +* <jstring.h>" under KANJI switch (same as 5-17-89 change +* to CRT version). +* 03-08-90 GJF Replaced _LOAD_DS with _CALLTYPE1, added #include +* <cruntime.h> and removed #include <register.h>. Also, +* cleaned up the formatting a bit. +* 07-24-90 SBM Removed redundant includes, replaced <assertm.h> by +* <assert.h> +* 09-27-90 GJF New-style function declarator. +* 01-17-91 GJF ANSI naming. +* 11-30-92 KRS Port _MBCS code from 16-bit tree. +* +*******************************************************************************/ + +#include <cruntime.h> +#include <errno.h> +#include <assert.h> +#include <stdlib.h> +#include <string.h> +#include <process.h> +#include <mbstring.h> + +#define SLASH "\\" +#define SLASHCHAR '\\' +#define XSLASHCHAR '/' +#define DELIMITER ";" + +#ifdef _MBCS +/* note, the macro below assumes p is to pointer to a single-byte character + * or the 1st byte of a double-byte character, in a string. + */ +#define ISPSLASH(p) ( ((p) == _mbschr((p), SLASHCHAR)) || ((p) == \ +_mbschr((p), XSLASHCHAR)) ) +#else +#define ISSLASH(c) ( ((c) == SLASHCHAR) || ((c) == XSLASHCHAR) ) +#endif + + +/*** +*int _execvpe(filename, argvector, envvector) - execute a file +* +*Purpose: +* Executes a file with given arguments and environment. +* try to execute the file. start with the name itself (directory '.'), +* and if that doesn't work start prepending pathnames from the +* environment until one works or we run out. if the file is a pathname, +* don't go to the environment to get alternate paths. If a needed text +* file is busy, wait a little while and try again before despairing +* completely +* +*Entry: +* char *filename - file to execute +* char **argvector - vector of arguments +* char **envvector - vector of environment variables +* +*Exit: +* destroys the calling process (hopefully) +* if fails, returns -1 +* +*Exceptions: +* +*******************************************************************************/ + +int _CALLTYPE1 _execvpe ( + REG3 const char *filename, + const char * const *argvector, + const char * const *envptr + ) +{ + REG1 char *env; + char *bbuf = NULL; + REG2 char *buf; + char *pfin; + + assert(filename != NULL); + assert(*filename != '\0'); + assert(argvector != NULL); + assert(*argvector != NULL); + assert(**argvector != '\0'); + + _execve(filename,argvector,envptr); + + /* In a SBCS build, _mbschr will map to strchr. */ + + if ( (errno != ENOENT) + || (_mbschr(filename, SLASHCHAR) != NULL) + || (_mbschr(filename, XSLASHCHAR) != NULL) + || *filename && *(filename+1) == ':' + || !(env=getenv("PATH")) ) + goto reterror; + + /* allocate a buffer to hold alternate pathnames for the executable + */ + if ( (buf = bbuf = (char *)malloc(_MAX_PATH)) == NULL ) goto reterror; + + do { + /* copy a component into bbuf[], taking care not to overflow it + */ + /* UNDONE: make sure ';' isn't 2nd byte of DBCS char */ + while ( (*env) && (*env != ';') && (buf < bbuf+_MAX_PATH-2) ) + *buf++ = *env++; + + *buf = '\0'; + pfin = --buf; + buf = bbuf; + +#ifdef _MBCS + if (*pfin == SLASHCHAR) { + if (pfin != _mbsrchr(buf,SLASHCHAR)) + /* *pfin is the second byte of a double-byte + * character + */ + strcat( buf, SLASH ); + } + else if (*pfin != XSLASHCHAR) + strcat(buf, SLASH); +#else + if (*pfin != SLASHCHAR && *pfin != XSLASHCHAR) + strcat(buf, SLASH); +#endif + + /* check that the final path will be of legal size. if so, + * build it. otherwise, return to the caller (return value + * and errno rename set from initial call to _execve()). + */ + if ( (strlen(buf) + strlen(filename)) < _MAX_PATH ) + strcat(buf, filename); + else + break; + + _execve(buf, argvector, envptr); + + if ( (errno != ENOENT) +#ifdef _MBCS + && (!ISPSLASH(buf) || !ISPSLASH(buf+1)) ) +#else + && (!ISSLASH(*buf) || !ISSLASH(*(buf+1))) ) +#endif + break; + } while ( *env && env++ ); + +reterror: + if (bbuf != NULL) + free(bbuf); + + return(-1); +} |