[SV 56834] Support local PATH search with posix_spawnp

When using exec we install the child's environment before invoking
execlp(), so commands are found on the child's PATH.  posix_spawnp
searches on the parent's PATH, which we don't want.

Import gnulib's findprog-in module and use it to search the child's
PATH, then use posix_spawn() to run it.

Also, posix_spawn() does not fall back to trying sh on ENOEXEC, as
execlp() does, so implement that as well.

* bootstrap.conf: Add the findprog-in gnulib module
* src/job.c: Include findprog.h if we're using posix_spawn.
(start_job_command): Remove the handling of child->cmd_name,
(child_execute_job): and add it here.  Look up the command to be
run in the child's path and invoke it if found.  If it fails with
ENOEXEC then retry it as an argument to the default shell.
* tests/scripts/misc/general4: Test makefile PATH assignments.
* tests/scripts/features/targetvars: Ditto, for target variables.
This commit is contained in:
Paul Smith 2019-09-06 22:24:46 -04:00
parent ebe1d37104
commit 60905a8afb
4 changed files with 132 additions and 8 deletions

View File

@ -46,6 +46,7 @@ gnulib_files=doc/make-stds.texi
gnulib_modules="\ gnulib_modules="\
alloca alloca
fdl fdl
findprog-in
getloadavg getloadavg
host-cpu-c-abi host-cpu-c-abi
strerror strerror

View File

@ -17,6 +17,7 @@ this program. If not, see <http://www.gnu.org/licenses/>. */
#include "makeint.h" #include "makeint.h"
#include <assert.h> #include <assert.h>
#include <string.h>
#include "job.h" #include "job.h"
#include "debug.h" #include "debug.h"
@ -25,8 +26,6 @@ this program. If not, see <http://www.gnu.org/licenses/>. */
#include "variable.h" #include "variable.h"
#include "os.h" #include "os.h"
#include <string.h>
/* Default shell to use. */ /* Default shell to use. */
#ifdef WINDOWS32 #ifdef WINDOWS32
# ifdef HAVE_STRINGS_H # ifdef HAVE_STRINGS_H
@ -139,6 +138,7 @@ extern int wait3 ();
#ifdef USE_POSIX_SPAWN #ifdef USE_POSIX_SPAWN
# include <spawn.h> # include <spawn.h>
# include "findprog.h"
#endif #endif
#if !defined (wait) && !defined (POSIX) #if !defined (wait) && !defined (POSIX)
@ -1466,8 +1466,6 @@ start_job_command (struct child *child)
environ = parent_environ; /* Restore value child may have clobbered. */ environ = parent_environ; /* Restore value child may have clobbered. */
jobserver_post_child (flags & COMMANDS_RECURSE); jobserver_post_child (flags & COMMANDS_RECURSE);
free (child->cmd_name);
child->cmd_name = child->pid > 0 ? xstrdup(argv[0]) : NULL;
#endif /* !VMS */ #endif /* !VMS */
} }
@ -2271,9 +2269,10 @@ child_execute_job (struct childbase *child, int good_stdin, char **argv)
pid_t pid; pid_t pid;
int r; int r;
#if defined(USE_POSIX_SPAWN) #if defined(USE_POSIX_SPAWN)
short flags = 0; char *cmd;
posix_spawnattr_t attr; posix_spawnattr_t attr;
posix_spawn_file_actions_t fa; posix_spawn_file_actions_t fa;
short flags = 0;
#endif #endif
/* Divert child output if we want to capture it. */ /* Divert child output if we want to capture it. */
@ -2312,7 +2311,7 @@ child_execute_job (struct childbase *child, int good_stdin, char **argv)
/* Run the command. */ /* Run the command. */
exec_command (argv, child->environment); exec_command (argv, child->environment);
#else /* use posix_spawn() */ #else /* USE_POSIX_SPAWN */
if ((r = posix_spawnattr_init (&attr)) != 0) if ((r = posix_spawnattr_init (&attr)) != 0)
goto done; goto done;
@ -2359,10 +2358,67 @@ child_execute_job (struct childbase *child, int good_stdin, char **argv)
if ((r = posix_spawnattr_setflags (&attr, flags)) != 0) if ((r = posix_spawnattr_setflags (&attr, flags)) != 0)
goto cleanup; goto cleanup;
/* Look up the program on the child's PATH, if needed. */
{
const char *p = NULL;
char **pp;
for (pp = child->environment; *pp != NULL; ++pp)
if ((*pp)[0] == 'P' && (*pp)[1] == 'A' && (*pp)[2] == 'T'
&& (*pp)[3] == 'H' &&(*pp)[4] == '=')
{
p = (*pp) + 5;
break;
}
cmd = (char *)find_in_given_path (argv[0], p);
}
if (!cmd)
{
r = ENOENT;
goto cleanup;
}
/* Start the program. */ /* Start the program. */
while ((r = posix_spawnp (&pid, argv[0], &fa, &attr, argv, child->environment)) == EINTR) while ((r = posix_spawn (&pid, cmd, &fa, &attr, argv,
child->environment)) == EINTR)
; ;
/* posix_spawn() doesn't provide sh fallback like exec() does; implement
it here. POSIX doesn't specify the path to sh so use the default. */
if (r == ENOEXEC)
{
char **nargv;
char **pp;
size_t l = 0;
for (pp = argv; *pp != NULL; ++pp)
++l;
nargv = xmalloc (sizeof (char *) * (l + 2));
nargv[0] = (char *)default_shell;
nargv[1] = cmd;
memcpy (&nargv[2], &argv[1], sizeof (char *) * l);
while ((r = posix_spawn (&pid, nargv[0], &fa, &attr, nargv,
child->environment)) == EINTR)
;
free (nargv);
}
if (r == 0)
{
/* Spawn succeeded but may fail later: remember the command. */
free (child->cmd_name);
if (cmd != argv[0])
child->cmd_name = cmd;
else
child->cmd_name = xstrdup(cmd);
}
cleanup: cleanup:
posix_spawn_file_actions_destroy (&fa); posix_spawn_file_actions_destroy (&fa);
posix_spawnattr_destroy (&attr); posix_spawnattr_destroy (&attr);
@ -2371,7 +2427,7 @@ child_execute_job (struct childbase *child, int good_stdin, char **argv)
if (r != 0) if (r != 0)
pid = -1; pid = -1;
#endif /* have posix_spawn() */ #endif /* USE_POSIX_SPAWN */
if (pid < 0) if (pid < 0)
OSS (error, NILF, "%s: %s", argv[0], strerror (r)); OSS (error, NILF, "%s: %s", argv[0], strerror (r));

View File

@ -255,6 +255,32 @@ a: ; @echo $(A)
!, !,
'', "hello; world\n"); '', "hello; world\n");
# TEST #21: SV-56834 Ensure setting PATH in a target var works properly
mkdir('sd', 0775);
open(my $fh, '>', 'sd/foobar');
print $fh "exit 0";
close($fh);
chmod 0755, 'sd/foobar';
run_make_test(q!
all: PATH := sd
all: ; foobar
!,
'', "foobar\n");
# Don't use the general PATH if not found on the target path
$extraENV{PATH} = "$ENV{PATH}:sd";
run_make_test(q!
all: PATH := ..
all: ; foobar
!,
'', "foobar\n#MAKE#: foobar: $ERR_no_such_file\n#MAKE#: *** [#MAKEFILE#;3: all] Error 127", 512);
unlink('sd/foobar');
rmdir ('sd');
# TEST #19: Test define/endef variables as target-specific vars # TEST #19: Test define/endef variables as target-specific vars
# run_make_test(' # run_make_test('

View File

@ -79,4 +79,45 @@ all: ; \@echo hi
", ",
'', "hi\n"); '', "hi\n");
# SV-56834 Ensure setting PATH in the makefile works properly
mkdir('sd', 0775);
open(my $fh, '>', 'sd/foobar');
print $fh "exit 0\n";
close($fh);
chmod 0755, 'sd/foobar';
run_make_test(q!
PATH := sd
all: ; foobar
!,
'', "foobar\n");
# Don't use the general PATH if not found on the target path
$extraENV{PATH} = "$ENV{PATH}:sd";
run_make_test(q!
PATH := ..
all: ; foobar
!,
'', "foobar\n#MAKE#: foobar: $ERR_no_such_file\n#MAKE#: *** [#MAKEFILE#;3: all] Error 127", 512);
unlink('sd/foobar');
rmdir('sd');
# Ensure that local programs are not found if "." is not on the PATH
open(my $fh, '>', 'foobar');
print $fh "exit 0\n";
close($fh);
chmod 0755, 'foobar';
run_make_test(q!
PATH := ..
all: ; foobar
!,
'', "foobar\n#MAKE#: foobar: $ERR_no_such_file\n#MAKE#: *** [#MAKEFILE#;3: all] Error 127", 512);
unlink('foobar');
1; 1;