123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966 |
- /* Guts of POSIX spawn interface. Generic POSIX.1 version.
- Copyright (C) 2000-2006, 2008-2022 Free Software Foundation, Inc.
- This file is part of the GNU C Library.
- This file is free software: you can redistribute it and/or modify
- it under the terms of the GNU Lesser General Public License as
- published by the Free Software Foundation; either version 2.1 of the
- License, or (at your option) any later version.
- This file is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Lesser General Public License for more details.
- You should have received a copy of the GNU Lesser General Public License
- along with this program. If not, see <https://www.gnu.org/licenses/>. */
- #include <config.h>
- /* Specification. */
- #include <spawn.h>
- #include "spawn_int.h"
- #include <alloca.h>
- #include <errno.h>
- #include <fcntl.h>
- #ifndef O_LARGEFILE
- # define O_LARGEFILE 0
- #endif
- #if _LIBC || HAVE_PATHS_H
- # include <paths.h>
- #else
- # define _PATH_BSHELL BOURNE_SHELL
- #endif
- #include <signal.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #if _LIBC
- # include <not-cancel.h>
- #else
- # define close_not_cancel close
- # define open_not_cancel open
- #endif
- #if _LIBC
- # include <local-setxid.h>
- #else
- # if !HAVE_SETEUID
- # define seteuid(id) setresuid (-1, id, -1)
- # endif
- # if !HAVE_SETEGID
- # define setegid(id) setresgid (-1, id, -1)
- # endif
- # define local_seteuid(id) seteuid (id)
- # define local_setegid(id) setegid (id)
- #endif
- #if _LIBC
- # define alloca __alloca
- # define execve __execve
- # define dup2 __dup2
- # define fork __fork
- # define getgid __getgid
- # define getuid __getuid
- # define sched_setparam __sched_setparam
- # define sched_setscheduler __sched_setscheduler
- # define setpgid __setpgid
- # define sigaction __sigaction
- # define sigismember __sigismember
- # define sigprocmask __sigprocmask
- # define strchrnul __strchrnul
- # define vfork __vfork
- #endif
- /* The Unix standard contains a long explanation of the way to signal
- an error after the fork() was successful. Since no new wait status
- was wanted there is no way to signal an error using one of the
- available methods. The committee chose to signal an error by a
- normal program exit with the exit code 127. */
- #define SPAWN_ERROR 127
- #if defined _WIN32 && ! defined __CYGWIN__
- /* Native Windows API. */
- /* Get declarations of the native Windows API functions. */
- # define WIN32_LEAN_AND_MEAN
- # include <windows.h>
- # include <stdbool.h>
- # include <stdio.h>
- # include "filename.h"
- # include "concat-filename.h"
- # include "findprog.h"
- # include "malloca.h"
- # include "windows-spawn.h"
- /* Don't assume that UNICODE is not defined. */
- # undef CreateFile
- # define CreateFile CreateFileA
- # undef STARTUPINFO
- # define STARTUPINFO STARTUPINFOA
- # undef CreateProcess
- # define CreateProcess CreateProcessA
- /* Grows inh_handles->count so that it becomes > newfd.
- Returns 0 upon success. In case of failure, -1 is returned, with errno set.
- */
- static int
- grow_inheritable_handles (struct inheritable_handles *inh_handles, int newfd)
- {
- if (inh_handles->allocated <= newfd)
- {
- size_t new_allocated = 2 * inh_handles->allocated + 1;
- if (new_allocated <= newfd)
- new_allocated = newfd + 1;
- HANDLE *new_handles_array =
- (HANDLE *)
- realloc (inh_handles->handles, new_allocated * sizeof (HANDLE));
- if (new_handles_array == NULL)
- {
- errno = ENOMEM;
- return -1;
- }
- unsigned char *new_flags_array =
- (unsigned char *)
- realloc (inh_handles->flags, new_allocated * sizeof (unsigned char));
- if (new_flags_array == NULL)
- {
- free (new_handles_array);
- errno = ENOMEM;
- return -1;
- }
- inh_handles->allocated = new_allocated;
- inh_handles->handles = new_handles_array;
- inh_handles->flags = new_flags_array;
- }
- HANDLE *handles = inh_handles->handles;
- for (; inh_handles->count <= newfd; inh_handles->count++)
- handles[inh_handles->count] = INVALID_HANDLE_VALUE;
- return 0;
- }
- /* Reduces inh_handles->count to the minimum needed. */
- static void
- shrink_inheritable_handles (struct inheritable_handles *inh_handles)
- {
- HANDLE *handles = inh_handles->handles;
- while (inh_handles->count > 3
- && handles[inh_handles->count - 1] == INVALID_HANDLE_VALUE)
- inh_handles->count--;
- }
- /* Closes all handles in inh_handles. */
- static void
- close_inheritable_handles (struct inheritable_handles *inh_handles)
- {
- HANDLE *handles = inh_handles->handles;
- size_t handles_count = inh_handles->count;
- unsigned int fd;
- for (fd = 0; fd < handles_count; fd++)
- {
- HANDLE handle = handles[fd];
- if (handle != INVALID_HANDLE_VALUE)
- CloseHandle (handle);
- }
- }
- /* Tests whether a memory region, starting at P and N bytes long, contains only
- zeroes. */
- static bool
- memiszero (const void *p, size_t n)
- {
- const char *cp = p;
- for (; n > 0; cp++, n--)
- if (*cp != 0)
- return 0;
- return 1;
- }
- /* Tests whether *S contains no signals. */
- static bool
- sigisempty (const sigset_t *s)
- {
- return memiszero (s, sizeof (sigset_t));
- }
- /* Opens a HANDLE to a file.
- Upon failure, returns INVALID_HANDLE_VALUE with errno set. */
- static HANDLE
- open_handle (const char *name, int flags, mode_t mode)
- {
- /* To ease portability. Like in open.c. */
- if (strcmp (name, "/dev/null") == 0)
- name = "NUL";
- /* POSIX <https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap04.html#tag_04_13>
- specifies: "More than two leading <slash> characters shall be treated as
- a single <slash> character." */
- if (ISSLASH (name[0]) && ISSLASH (name[1]) && ISSLASH (name[2]))
- {
- name += 2;
- while (ISSLASH (name[1]))
- name++;
- }
- size_t len = strlen (name);
- size_t drive_prefix_len = (HAS_DEVICE (name) ? 2 : 0);
- /* Remove trailing slashes (except the very first one, at position
- drive_prefix_len), but remember their presence. */
- size_t rlen;
- bool check_dir = false;
- rlen = len;
- while (rlen > drive_prefix_len && ISSLASH (name[rlen-1]))
- {
- check_dir = true;
- if (rlen == drive_prefix_len + 1)
- break;
- rlen--;
- }
- /* Handle '' and 'C:'. */
- if (!check_dir && rlen == drive_prefix_len)
- {
- errno = ENOENT;
- return INVALID_HANDLE_VALUE;
- }
- /* Handle '\\'. */
- if (rlen == 1 && ISSLASH (name[0]) && len >= 2)
- {
- errno = ENOENT;
- return INVALID_HANDLE_VALUE;
- }
- const char *rname;
- char *malloca_rname;
- if (rlen == len)
- {
- rname = name;
- malloca_rname = NULL;
- }
- else
- {
- malloca_rname = malloca (rlen + 1);
- if (malloca_rname == NULL)
- {
- errno = ENOMEM;
- return INVALID_HANDLE_VALUE;
- }
- memcpy (malloca_rname, name, rlen);
- malloca_rname[rlen] = '\0';
- rname = malloca_rname;
- }
- /* For the meaning of the flags, see
- <https://docs.microsoft.com/en-us/cpp/c-runtime-library/reference/open-wopen> */
- /* Open a handle to the file.
- CreateFile
- <https://docs.microsoft.com/en-us/windows/desktop/api/fileapi/nf-fileapi-createfilea>
- <https://docs.microsoft.com/en-us/windows/desktop/FileIO/creating-and-opening-files> */
- HANDLE handle =
- CreateFile (rname,
- ((flags & (O_WRONLY | O_RDWR)) != 0
- ? GENERIC_READ | GENERIC_WRITE
- : GENERIC_READ),
- FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
- NULL,
- ((flags & O_CREAT) != 0
- ? ((flags & O_EXCL) != 0
- ? CREATE_NEW
- : ((flags & O_TRUNC) != 0 ? CREATE_ALWAYS : OPEN_ALWAYS))
- : ((flags & O_TRUNC) != 0
- ? TRUNCATE_EXISTING
- : OPEN_EXISTING)),
- /* FILE_FLAG_BACKUP_SEMANTICS is useful for opening directories,
- which is out-of-scope here. */
- /* FILE_FLAG_POSIX_SEMANTICS (treat file names that differ only
- in case as different) makes sense only when applied to *all*
- filesystem operations. */
- /* FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_POSIX_SEMANTICS */
- FILE_ATTRIBUTE_NORMAL
- | ((flags & O_TEMPORARY) != 0 ? FILE_FLAG_DELETE_ON_CLOSE : 0)
- | ((flags & O_SEQUENTIAL ) != 0 ? FILE_FLAG_SEQUENTIAL_SCAN : 0)
- | ((flags & O_RANDOM) != 0 ? FILE_FLAG_RANDOM_ACCESS : 0),
- NULL);
- if (handle == INVALID_HANDLE_VALUE)
- switch (GetLastError ())
- {
- /* Some of these errors probably cannot happen with the specific flags
- that we pass to CreateFile. But who knows... */
- case ERROR_FILE_NOT_FOUND: /* The last component of rname does not exist. */
- case ERROR_PATH_NOT_FOUND: /* Some directory component in rname does not exist. */
- case ERROR_BAD_PATHNAME: /* rname is such as '\\server'. */
- case ERROR_BAD_NETPATH: /* rname is such as '\\nonexistentserver\share'. */
- case ERROR_BAD_NET_NAME: /* rname is such as '\\server\nonexistentshare'. */
- case ERROR_INVALID_NAME: /* rname contains wildcards, misplaced colon, etc. */
- case ERROR_DIRECTORY:
- errno = ENOENT;
- break;
- case ERROR_ACCESS_DENIED: /* rname is such as 'C:\System Volume Information\foo'. */
- case ERROR_SHARING_VIOLATION: /* rname is such as 'C:\pagefile.sys'. */
- /* XXX map to EACCES or EPERM? */
- errno = EACCES;
- break;
- case ERROR_OUTOFMEMORY:
- errno = ENOMEM;
- break;
- case ERROR_WRITE_PROTECT:
- errno = EROFS;
- break;
- case ERROR_WRITE_FAULT:
- case ERROR_READ_FAULT:
- case ERROR_GEN_FAILURE:
- errno = EIO;
- break;
- case ERROR_BUFFER_OVERFLOW:
- case ERROR_FILENAME_EXCED_RANGE:
- errno = ENAMETOOLONG;
- break;
- case ERROR_DELETE_PENDING: /* XXX map to EACCES or EPERM? */
- errno = EPERM;
- break;
- default:
- errno = EINVAL;
- break;
- }
- if (malloca_rname != NULL)
- {
- int saved_errno = errno;
- freea (malloca_rname);
- errno = saved_errno;
- }
- return handle;
- }
- /* Executes an 'open' action.
- Returns 0 upon success. In case of failure, -1 is returned, with errno set.
- */
- static int
- do_open (struct inheritable_handles *inh_handles, int newfd,
- const char *filename, const char *directory,
- int flags, mode_t mode, HANDLE curr_process)
- {
- if (!(newfd >= 0 && newfd < _getmaxstdio ()))
- {
- errno = EBADF;
- return -1;
- }
- if (grow_inheritable_handles (inh_handles, newfd) < 0)
- return -1;
- if (inh_handles->handles[newfd] != INVALID_HANDLE_VALUE
- && !CloseHandle (inh_handles->handles[newfd]))
- {
- errno = EIO;
- return -1;
- }
- if (filename == NULL)
- {
- errno = EINVAL;
- return -1;
- }
- char *filename_to_free = NULL;
- if (directory != NULL && IS_RELATIVE_FILE_NAME (filename))
- {
- char *real_filename = concatenated_filename (directory, filename, NULL);
- if (real_filename == NULL)
- {
- errno = ENOMEM;
- return -1;
- }
- filename = real_filename;
- filename_to_free = real_filename;
- }
- HANDLE handle = open_handle (filename, flags, mode);
- if (handle == INVALID_HANDLE_VALUE)
- {
- free (filename_to_free);
- return -1;
- }
- free (filename_to_free);
- /* Duplicate the handle, so that it becomes inheritable. */
- if (!DuplicateHandle (curr_process, handle,
- curr_process, &inh_handles->handles[newfd],
- 0, TRUE,
- DUPLICATE_CLOSE_SOURCE | DUPLICATE_SAME_ACCESS))
- {
- errno = EBADF; /* arbitrary */
- return -1;
- }
- inh_handles->flags[newfd] = ((flags & O_APPEND) != 0 ? 32 : 0);
- return 0;
- }
- /* Executes a 'dup2' action.
- Returns 0 upon success. In case of failure, -1 is returned, with errno set.
- */
- static int
- do_dup2 (struct inheritable_handles *inh_handles, int oldfd, int newfd,
- HANDLE curr_process)
- {
- if (!(oldfd >= 0 && oldfd < inh_handles->count
- && inh_handles->handles[oldfd] != INVALID_HANDLE_VALUE))
- {
- errno = EBADF;
- return -1;
- }
- if (!(newfd >= 0 && newfd < _getmaxstdio ()))
- {
- errno = EBADF;
- return -1;
- }
- if (newfd != oldfd)
- {
- if (grow_inheritable_handles (inh_handles, newfd) < 0)
- return -1;
- if (inh_handles->handles[newfd] != INVALID_HANDLE_VALUE
- && !CloseHandle (inh_handles->handles[newfd]))
- {
- errno = EIO;
- return -1;
- }
- /* Duplicate the handle, so that it a forthcoming do_close action on oldfd
- has no effect on newfd. */
- if (!DuplicateHandle (curr_process, inh_handles->handles[oldfd],
- curr_process, &inh_handles->handles[newfd],
- 0, TRUE, DUPLICATE_SAME_ACCESS))
- {
- errno = EBADF; /* arbitrary */
- return -1;
- }
- inh_handles->flags[newfd] = 0;
- }
- return 0;
- }
- /* Executes a 'close' action.
- Returns 0 upon success. In case of failure, -1 is returned, with errno set.
- */
- static int
- do_close (struct inheritable_handles *inh_handles, int fd)
- {
- if (!(fd >= 0 && fd < inh_handles->count
- && inh_handles->handles[fd] != INVALID_HANDLE_VALUE))
- {
- errno = EBADF;
- return -1;
- }
- if (!CloseHandle (inh_handles->handles[fd]))
- {
- errno = EIO;
- return -1;
- }
- inh_handles->handles[fd] = INVALID_HANDLE_VALUE;
- return 0;
- }
- int
- __spawni (pid_t *pid, const char *prog_filename,
- const posix_spawn_file_actions_t *file_actions,
- const posix_spawnattr_t *attrp, const char *const prog_argv[],
- const char *const envp[], int use_path)
- {
- /* Validate the arguments. */
- if (prog_filename == NULL
- || (attrp != NULL
- && ((attrp->_flags & ~POSIX_SPAWN_SETPGROUP) != 0
- || attrp->_pgrp != 0
- || ! sigisempty (&attrp->_sd)
- || ! sigisempty (&attrp->_ss)
- || attrp->_sp.sched_priority != 0
- || attrp->_policy != 0)))
- return EINVAL;
- /* Process group handling:
- Native Windows does not have the concept of process group, but it has the
- concept of a console attached to a process.
- So, we interpret the three cases as follows:
- - Flag POSIX_SPAWN_SETPGROUP not set: Means, the child process is in the
- same process group as the parent process. We interpret this as a
- request to reuse the same console.
- - Flag POSIX_SPAWN_SETPGROUP set with attrp->_pgrp == 0: Means the child
- process starts a process group of its own. See
- <https://pubs.opengroup.org/onlinepubs/9699919799/functions/posix_spawnattr_getpgroup.html>
- <https://pubs.opengroup.org/onlinepubs/9699919799/functions/setpgrp.html>
- We interpret this as a request to detach from the current console.
- - Flag POSIX_SPAWN_SETPGROUP set with attrp->_pgrp != 0: Means the child
- process joins another, existing process group. See
- <https://pubs.opengroup.org/onlinepubs/9699919799/functions/posix_spawnattr_getpgroup.html>
- <https://pubs.opengroup.org/onlinepubs/9699919799/functions/setpgid.html>
- We don't support this case; it produces error EINVAL above. */
- /* <https://docs.microsoft.com/en-us/windows/win32/procthread/process-creation-flags> */
- DWORD process_creation_flags =
- (attrp != NULL && (attrp->_flags & POSIX_SPAWN_SETPGROUP) != 0 ? DETACHED_PROCESS : 0);
- char *argv_mem_to_free;
- const char **argv = prepare_spawn (prog_argv, &argv_mem_to_free);
- if (argv == NULL)
- return errno; /* errno is set here */
- argv++;
- /* Compose the command. */
- char *command = compose_command (argv);
- if (command == NULL)
- {
- free (argv_mem_to_free);
- return ENOMEM;
- }
- /* Copy *ENVP into a contiguous block of memory. */
- char *envblock;
- if (envp == NULL)
- envblock = NULL;
- else
- {
- envblock = compose_envblock (envp);
- if (envblock == NULL)
- {
- free (command);
- free (argv_mem_to_free);
- return ENOMEM;
- }
- }
- /* Set up the array of handles to inherit.
- Duplicate each handle, so that a spawn_do_close action (below) has no
- effect on the file descriptors of the current process. Alternatively,
- we could store, for each handle, a bit that tells whether it is shared
- with the current process. But this is simpler. */
- struct inheritable_handles inh_handles;
- if (init_inheritable_handles (&inh_handles, true) < 0)
- goto failed_1;
- /* Directory in which to execute the new process. */
- const char *directory = NULL;
- /* Execute the file_actions, modifying the inh_handles instead of the
- file descriptors of the current process. */
- if (file_actions != NULL)
- {
- HANDLE curr_process = GetCurrentProcess ();
- int cnt;
- for (cnt = 0; cnt < file_actions->_used; ++cnt)
- {
- struct __spawn_action *action = &file_actions->_actions[cnt];
- switch (action->tag)
- {
- case spawn_do_close:
- {
- int fd = action->action.close_action.fd;
- if (do_close (&inh_handles, fd) < 0)
- goto failed_2;
- }
- break;
- case spawn_do_open:
- {
- int newfd = action->action.open_action.fd;
- const char *filename = action->action.open_action.path;
- int flags = action->action.open_action.oflag;
- mode_t mode = action->action.open_action.mode;
- if (do_open (&inh_handles, newfd, filename, directory,
- flags, mode, curr_process)
- < 0)
- goto failed_2;
- }
- break;
- case spawn_do_dup2:
- {
- int oldfd = action->action.dup2_action.fd;
- int newfd = action->action.dup2_action.newfd;
- if (do_dup2 (&inh_handles, oldfd, newfd, curr_process) < 0)
- goto failed_2;
- }
- break;
- case spawn_do_chdir:
- {
- char *newdir = action->action.chdir_action.path;
- if (directory != NULL && IS_RELATIVE_FILE_NAME (newdir))
- {
- newdir = concatenated_filename (directory, newdir, NULL);
- if (newdir == NULL)
- {
- errno = ENOMEM;
- goto failed_2;
- }
- }
- directory = newdir;
- }
- break;
- case spawn_do_fchdir:
- /* Not supported in this implementation. */
- errno = EINVAL;
- goto failed_2;
- }
- }
- }
- /* Reduce inh_handles.count to the minimum needed. */
- shrink_inheritable_handles (&inh_handles);
- /* CreateProcess
- <https://docs.microsoft.com/en-us/windows/win32/api/processthreadsapi/nf-processthreadsapi-createprocessa> */
- /* STARTUPINFO
- <https://docs.microsoft.com/en-us/windows/win32/api/processthreadsapi/ns-processthreadsapi-startupinfoa> */
- STARTUPINFO sinfo;
- sinfo.cb = sizeof (STARTUPINFO);
- sinfo.lpReserved = NULL;
- sinfo.lpDesktop = NULL;
- sinfo.lpTitle = NULL;
- if (compose_handles_block (&inh_handles, &sinfo) < 0)
- goto failed_2;
- /* Perform the PATH search now, considering the final DIRECTORY. */
- char *resolved_prog_filename_to_free = NULL;
- {
- const char *resolved_prog_filename =
- find_in_given_path (prog_filename, use_path ? getenv ("PATH") : "",
- directory, false);
- if (resolved_prog_filename == NULL)
- goto failed_3;
- if (resolved_prog_filename != prog_filename)
- resolved_prog_filename_to_free = (char *) resolved_prog_filename;
- prog_filename = resolved_prog_filename;
- }
- PROCESS_INFORMATION pinfo;
- if (!CreateProcess (prog_filename, command, NULL, NULL, TRUE,
- process_creation_flags, envblock, directory, &sinfo,
- &pinfo))
- {
- DWORD error = GetLastError ();
- free (resolved_prog_filename_to_free);
- free (sinfo.lpReserved2);
- close_inheritable_handles (&inh_handles);
- free_inheritable_handles (&inh_handles);
- free (envblock);
- free (command);
- free (argv_mem_to_free);
- return convert_CreateProcess_error (error);
- }
- if (pinfo.hThread)
- CloseHandle (pinfo.hThread);
- free (resolved_prog_filename_to_free);
- free (sinfo.lpReserved2);
- close_inheritable_handles (&inh_handles);
- free_inheritable_handles (&inh_handles);
- free (envblock);
- free (command);
- free (argv_mem_to_free);
- if (pid != NULL)
- *pid = (intptr_t) pinfo.hProcess;
- return 0;
- failed_3:
- {
- int saved_errno = errno;
- free (sinfo.lpReserved2);
- close_inheritable_handles (&inh_handles);
- free_inheritable_handles (&inh_handles);
- free (envblock);
- free (command);
- free (argv_mem_to_free);
- return saved_errno;
- }
- failed_2:
- {
- int saved_errno = errno;
- close_inheritable_handles (&inh_handles);
- free_inheritable_handles (&inh_handles);
- free (envblock);
- free (command);
- free (argv_mem_to_free);
- return saved_errno;
- }
- failed_1:
- free (envblock);
- free (command);
- free (argv_mem_to_free);
- return errno;
- }
- #else
- /* Spawn a new process executing PATH with the attributes describes in *ATTRP.
- Before running the process perform the actions described in FILE-ACTIONS. */
- int
- __spawni (pid_t *pid, const char *file,
- const posix_spawn_file_actions_t *file_actions,
- const posix_spawnattr_t *attrp, const char *const argv[],
- const char *const envp[], int use_path)
- {
- pid_t new_pid;
- char *path, *p, *name;
- size_t len;
- size_t pathlen;
- /* Do this once. */
- short int flags = attrp == NULL ? 0 : attrp->_flags;
- /* Avoid gcc warning
- "variable 'flags' might be clobbered by 'longjmp' or 'vfork'" */
- (void) &flags;
- /* Generate the new process. */
- #if HAVE_VFORK
- if ((flags & POSIX_SPAWN_USEVFORK) != 0
- /* If no major work is done, allow using vfork. Note that we
- might perform the path searching. But this would be done by
- a call to execvp(), too, and such a call must be OK according
- to POSIX. */
- || ((flags & (POSIX_SPAWN_SETSIGMASK | POSIX_SPAWN_SETSIGDEF
- | POSIX_SPAWN_SETSCHEDPARAM | POSIX_SPAWN_SETSCHEDULER
- | POSIX_SPAWN_SETPGROUP | POSIX_SPAWN_RESETIDS)) == 0
- && file_actions == NULL))
- new_pid = vfork ();
- else
- #endif
- new_pid = fork ();
- if (new_pid != 0)
- {
- if (new_pid < 0)
- return errno;
- /* The call was successful. Store the PID if necessary. */
- if (pid != NULL)
- *pid = new_pid;
- return 0;
- }
- /* Set signal mask. */
- if ((flags & POSIX_SPAWN_SETSIGMASK) != 0
- && sigprocmask (SIG_SETMASK, &attrp->_ss, NULL) != 0)
- _exit (SPAWN_ERROR);
- /* Set signal default action. */
- if ((flags & POSIX_SPAWN_SETSIGDEF) != 0)
- {
- /* We have to iterate over all signals. This could possibly be
- done better but it requires system specific solutions since
- the sigset_t data type can be very different on different
- architectures. */
- int sig;
- struct sigaction sa;
- memset (&sa, '\0', sizeof (sa));
- sa.sa_handler = SIG_DFL;
- for (sig = 1; sig <= NSIG; ++sig)
- if (sigismember (&attrp->_sd, sig) != 0
- && sigaction (sig, &sa, NULL) != 0)
- _exit (SPAWN_ERROR);
- }
- #if (_LIBC ? defined _POSIX_PRIORITY_SCHEDULING : HAVE_SCHED_SETPARAM && HAVE_SCHED_SETSCHEDULER)
- /* Set the scheduling algorithm and parameters. */
- if ((flags & (POSIX_SPAWN_SETSCHEDPARAM | POSIX_SPAWN_SETSCHEDULER))
- == POSIX_SPAWN_SETSCHEDPARAM)
- {
- if (sched_setparam (0, &attrp->_sp) == -1)
- _exit (SPAWN_ERROR);
- }
- else if ((flags & POSIX_SPAWN_SETSCHEDULER) != 0)
- {
- if (sched_setscheduler (0, attrp->_policy,
- (flags & POSIX_SPAWN_SETSCHEDPARAM) != 0
- ? &attrp->_sp : NULL) == -1)
- _exit (SPAWN_ERROR);
- }
- #endif
- /* Set the process group ID. */
- if ((flags & POSIX_SPAWN_SETPGROUP) != 0
- && setpgid (0, attrp->_pgrp) != 0)
- _exit (SPAWN_ERROR);
- /* Set the effective user and group IDs. */
- if ((flags & POSIX_SPAWN_RESETIDS) != 0
- && (local_seteuid (getuid ()) != 0
- || local_setegid (getgid ()) != 0))
- _exit (SPAWN_ERROR);
- /* Execute the file actions. */
- if (file_actions != NULL)
- {
- int cnt;
- for (cnt = 0; cnt < file_actions->_used; ++cnt)
- {
- struct __spawn_action *action = &file_actions->_actions[cnt];
- switch (action->tag)
- {
- case spawn_do_close:
- if (close_not_cancel (action->action.close_action.fd) != 0)
- /* Signal the error. */
- _exit (SPAWN_ERROR);
- break;
- case spawn_do_open:
- {
- int new_fd = open_not_cancel (action->action.open_action.path,
- action->action.open_action.oflag
- | O_LARGEFILE,
- action->action.open_action.mode);
- if (new_fd == -1)
- /* The 'open' call failed. */
- _exit (SPAWN_ERROR);
- /* Make sure the desired file descriptor is used. */
- if (new_fd != action->action.open_action.fd)
- {
- if (dup2 (new_fd, action->action.open_action.fd)
- != action->action.open_action.fd)
- /* The 'dup2' call failed. */
- _exit (SPAWN_ERROR);
- if (close_not_cancel (new_fd) != 0)
- /* The 'close' call failed. */
- _exit (SPAWN_ERROR);
- }
- }
- break;
- case spawn_do_dup2:
- if (dup2 (action->action.dup2_action.fd,
- action->action.dup2_action.newfd)
- != action->action.dup2_action.newfd)
- /* The 'dup2' call failed. */
- _exit (SPAWN_ERROR);
- break;
- case spawn_do_chdir:
- if (chdir (action->action.chdir_action.path) < 0)
- /* The 'chdir' call failed. */
- _exit (SPAWN_ERROR);
- break;
- case spawn_do_fchdir:
- if (fchdir (action->action.fchdir_action.fd) < 0)
- /* The 'fchdir' call failed. */
- _exit (SPAWN_ERROR);
- break;
- }
- }
- }
- if (! use_path || strchr (file, '/') != NULL)
- {
- /* The FILE parameter is actually a path. */
- execve (file, (char * const *) argv, (char * const *) envp);
- /* Oh, oh. 'execve' returns. This is bad. */
- _exit (SPAWN_ERROR);
- }
- /* We have to search for FILE on the path. */
- path = getenv ("PATH");
- if (path == NULL)
- {
- #if HAVE_CONFSTR
- /* There is no 'PATH' in the environment.
- The default search path is the current directory
- followed by the path 'confstr' returns for '_CS_PATH'. */
- len = confstr (_CS_PATH, (char *) NULL, 0);
- path = (char *) alloca (1 + len);
- path[0] = ':';
- (void) confstr (_CS_PATH, path + 1, len);
- #else
- /* Pretend that the PATH contains only the current directory. */
- path = "";
- #endif
- }
- len = strlen (file) + 1;
- pathlen = strlen (path);
- name = alloca (pathlen + len + 1);
- /* Copy the file name at the top. */
- name = (char *) memcpy (name + pathlen + 1, file, len);
- /* And add the slash. */
- *--name = '/';
- p = path;
- do
- {
- char *startp;
- path = p;
- p = strchrnul (path, ':');
- if (p == path)
- /* Two adjacent colons, or a colon at the beginning or the end
- of 'PATH' means to search the current directory. */
- startp = name + 1;
- else
- startp = (char *) memcpy (name - (p - path), path, p - path);
- /* Try to execute this name. If it works, execv will not return. */
- execve (startp, (char * const *) argv, (char * const *) envp);
- switch (errno)
- {
- case EACCES:
- case ENOENT:
- case ESTALE:
- case ENOTDIR:
- /* Those errors indicate the file is missing or not executable
- by us, in which case we want to just try the next path
- directory. */
- break;
- default:
- /* Some other error means we found an executable file, but
- something went wrong executing it; return the error to our
- caller. */
- _exit (SPAWN_ERROR);
- }
- }
- while (*p++ != '\0');
- /* Return with an error. */
- _exit (SPAWN_ERROR);
- }
- #endif
|