123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101 |
- /* vi: set sw=4 ts=4: */
- /*
- * Utility routines.
- *
- * Copyright (C) 2006 Gabriel Somlo <somlo at cmu.edu>
- *
- * Licensed under GPLv2 or later, see file LICENSE in this source tree.
- */
- #include "libbb.h"
- /* check if path points to an executable file;
- * return 1 if found;
- * return 0 otherwise;
- */
- int FAST_FUNC file_is_executable(const char *name)
- {
- struct stat s;
- return (!access(name, X_OK) && !stat(name, &s) && S_ISREG(s.st_mode));
- }
- /* search (*PATHp) for an executable file;
- * return allocated string containing full path if found;
- * PATHp points to the component after the one where it was found
- * (or NULL),
- * you may call find_executable again with this PATHp to continue
- * (if it's not NULL).
- * return NULL otherwise; (PATHp is undefined)
- * in all cases (*PATHp) contents will be trashed (s/:/NUL/).
- */
- char* FAST_FUNC find_executable(const char *filename, char **PATHp)
- {
- /* About empty components in $PATH:
- * http://pubs.opengroup.org/onlinepubs/009695399/basedefs/xbd_chap08.html
- * 8.3 Other Environment Variables - PATH
- * A zero-length prefix is a legacy feature that indicates the current
- * working directory. It appears as two adjacent colons ( "::" ), as an
- * initial colon preceding the rest of the list, or as a trailing colon
- * following the rest of the list.
- */
- char *p, *n;
- p = *PATHp;
- while (p) {
- n = strchr(p, ':');
- if (n)
- *n++ = '\0';
- p = concat_path_file(
- p[0] ? p : ".", /* handle "::" case */
- filename
- );
- if (file_is_executable(p)) {
- *PATHp = n;
- return p;
- }
- free(p);
- p = n;
- } /* on loop exit p == NULL */
- return p;
- }
- /* search $PATH for an executable file;
- * return 1 if found;
- * return 0 otherwise;
- */
- int FAST_FUNC executable_exists(const char *filename)
- {
- char *path = xstrdup(getenv("PATH"));
- char *tmp = path;
- char *ret = find_executable(filename, &tmp);
- free(path);
- free(ret);
- return ret != NULL;
- }
- #if ENABLE_FEATURE_PREFER_APPLETS
- /* just like the real execvp, but try to launch an applet named 'file' first */
- int FAST_FUNC BB_EXECVP(const char *file, char *const argv[])
- {
- if (find_applet_by_name(file) >= 0)
- execvp(bb_busybox_exec_path, argv);
- return execvp(file, argv);
- }
- #endif
- void FAST_FUNC BB_EXECVP_or_die(char **argv)
- {
- BB_EXECVP(argv[0], argv);
- /* SUSv3-mandated exit codes */
- xfunc_error_retval = (errno == ENOENT) ? 127 : 126;
- bb_perror_msg_and_die("can't execute '%s'", argv[0]);
- }
- /* Typical idiom for applets which exec *optional* PROG [ARGS] */
- void FAST_FUNC exec_prog_or_SHELL(char **argv)
- {
- if (argv[0]) {
- BB_EXECVP_or_die(argv);
- }
- run_shell(getenv("SHELL"), /*login:*/ 1, NULL);
- }
|