blob: 82241cd81d5823880058d800d941e2a1190cf9c2 [file] [log] [blame]
Denis Vlasenko5cb46482006-10-12 06:12:11 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Utility routines.
4 *
5 * Copyright (C) 2006 Gabriel Somlo <somlo at cmu.edu>
6 *
7 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
8 */
9
10#include "libbb.h"
11
12/* check if path points to an executable file;
13 * return 1 if found;
14 * return 0 otherwise;
15 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000016int FAST_FUNC execable_file(const char *name)
Denis Vlasenko5cb46482006-10-12 06:12:11 +000017{
18 struct stat s;
19 return (!access(name, X_OK) && !stat(name, &s) && S_ISREG(s.st_mode));
20}
21
Denis Vlasenkof592aa32008-06-05 13:33:59 +000022/* search (*PATHp) for an executable file;
Denis Vlasenko5cb46482006-10-12 06:12:11 +000023 * return allocated string containing full path if found;
Denis Vlasenkof592aa32008-06-05 13:33:59 +000024 * PATHp points to the component after the one where it was found
25 * (or NULL),
26 * you may call find_execable again with this PATHp to continue
27 * (if it's not NULL).
28 * return NULL otherwise; (PATHp is undefined)
29 * in all cases (*PATHp) contents will be trashed (s/:/NUL/).
Denis Vlasenko5cb46482006-10-12 06:12:11 +000030 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000031char* FAST_FUNC find_execable(const char *filename, char **PATHp)
Denis Vlasenko5cb46482006-10-12 06:12:11 +000032{
Denis Vlasenkof592aa32008-06-05 13:33:59 +000033 char *p, *n;
Denis Vlasenko5cb46482006-10-12 06:12:11 +000034
Denis Vlasenkof592aa32008-06-05 13:33:59 +000035 p = *PATHp;
Denis Vlasenko5cb46482006-10-12 06:12:11 +000036 while (p) {
37 n = strchr(p, ':');
38 if (n)
39 *n++ = '\0';
40 if (*p != '\0') { /* it's not a PATH="foo::bar" situation */
41 p = concat_path_file(p, filename);
42 if (execable_file(p)) {
Denis Vlasenkof592aa32008-06-05 13:33:59 +000043 *PATHp = n;
Denis Vlasenko5cb46482006-10-12 06:12:11 +000044 return p;
45 }
46 free(p);
47 }
48 p = n;
Denis Vlasenkof592aa32008-06-05 13:33:59 +000049 } /* on loop exit p == NULL */
50 return p;
Denis Vlasenko5cb46482006-10-12 06:12:11 +000051}
52
53/* search $PATH for an executable file;
54 * return 1 if found;
55 * return 0 otherwise;
56 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000057int FAST_FUNC exists_execable(const char *filename)
Denis Vlasenko5cb46482006-10-12 06:12:11 +000058{
Denis Vlasenkof592aa32008-06-05 13:33:59 +000059 char *path = xstrdup(getenv("PATH"));
60 char *tmp = path;
61 char *ret = find_execable(filename, &tmp);
62 free(path);
Denis Vlasenko5cb46482006-10-12 06:12:11 +000063 if (ret) {
64 free(ret);
65 return 1;
66 }
67 return 0;
68}
Denis Vlasenko1d76f432007-02-06 01:20:12 +000069
Denis Vlasenko80d14be2007-04-10 23:03:30 +000070#if ENABLE_FEATURE_PREFER_APPLETS
Denis Vlasenko1d76f432007-02-06 01:20:12 +000071/* just like the real execvp, but try to launch an applet named 'file' first
72 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000073int FAST_FUNC bb_execvp(const char *file, char *const argv[])
Denis Vlasenko1d76f432007-02-06 01:20:12 +000074{
Denis Vlasenko1aa7e472007-11-28 06:49:03 +000075 return execvp(find_applet_by_name(file) >= 0 ? bb_busybox_exec_path : file,
Denis Vlasenko1d76f432007-02-06 01:20:12 +000076 argv);
77}
78#endif
Pascal Bellard21e8e8d2010-07-04 00:57:03 +020079
80int FAST_FUNC BB_EXECVP_or_die(char **argv)
81{
82 BB_EXECVP(argv[0], argv);
83 /* SUSv3-mandated exit codes */
84 xfunc_error_retval = (errno == ENOENT) ? 127 : 126;
85 bb_perror_msg_and_die("can't execute '%s'", argv[0]);
86}