execvp.c 949 B

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849
  1. #include <stdlib.h>
  2. #include <string.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <limits.h>
  6. extern char **__environ;
  7. int __execvpe(const char *file, char *const argv[], char *const envp[])
  8. {
  9. const char *p, *z, *path = getenv("PATH");
  10. size_t l, k;
  11. errno = ENOENT;
  12. if (!*file) return -1;
  13. if (strchr(file, '/'))
  14. return execve(file, argv, envp);
  15. if (!path) path = "/usr/local/bin:/bin:/usr/bin";
  16. k = strnlen(file, NAME_MAX+1);
  17. if (k > NAME_MAX) {
  18. errno = ENAMETOOLONG;
  19. return -1;
  20. }
  21. l = strnlen(path, PATH_MAX-1)+1;
  22. for(p=path; ; p=z) {
  23. char b[l+k+1];
  24. z = strchr(p, ':');
  25. if (!z) z = p+strlen(p);
  26. if (z-p >= l) {
  27. if (!*z++) break;
  28. continue;
  29. }
  30. memcpy(b, p, z-p);
  31. b[z-p] = '/';
  32. memcpy(b+(z-p)+(z>p), file, k+1);
  33. execve(b, argv, envp);
  34. if (errno != ENOENT) return -1;
  35. if (!*z++) break;
  36. }
  37. return -1;
  38. }
  39. int execvp(const char *file, char *const argv[])
  40. {
  41. return __execvpe(file, argv, __environ);
  42. }