summaryrefslogtreecommitdiff
path: root/src/libposixplz/execvep_internal.c
blob: 17858f0da1b84cc12bf08462678f2f2d8c4c8624 (plain)
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
/* ISC license. */

#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <skalibs/bytestr.h>
#include <skalibs/posixplz.h>

void execvep_internal (char const *file, char const *const *argv, char const *const *envp, char const *path)
{
  if (!path) errno = EINVAL ;
  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 ;
  }
}