1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
/* ISC license. */
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <skalibs/bytestr.h>
#include <skalibs/posixplz.h>
void execvep (char const *file, char const *const *argv, char const *const *envp, char const *path)
{
if (!path) errno = EINVAL ;
else if (file[str_chr(file, '/')])
execve(file, (char *const *)argv, (char *const *)envp) ; /* execve prototype sucks */
else
{
size_t pathlen = strlen(path) + 1 ;
size_t filelen = strlen(file) ;
int savederrno = 0 ;
while (pathlen)
{
size_t split = byte_chr(path, pathlen - 1, ':') ;
if (split)
{
char tmp[split + 2 + filelen] ;
memcpy(tmp, path, split) ;
tmp[split] = '/' ;
memcpy(tmp + split + 1, file, filelen + 1) ;
execve(tmp, (char *const *)argv, (char *const *)envp) ;
if (errno != ENOENT)
{
savederrno = errno ;
if ((errno != EACCES) && (errno != EPERM) && (errno != EISDIR)) break ;
}
}
path += split+1 ; pathlen -= split+1 ;
}
if (savederrno) errno = savederrno ;
}
}
|