xref: /freebsd/lib/libc/stdlib/system.c (revision dc36d6f9bb1753f3808552f3afd30eda9a7b206a)
1*8a16b7a1SPedro F. Giffuni /*-
2*8a16b7a1SPedro F. Giffuni  * SPDX-License-Identifier: BSD-3-Clause
3*8a16b7a1SPedro F. Giffuni  *
458f0484fSRodney W. Grimes  * Copyright (c) 1988, 1993
558f0484fSRodney W. Grimes  *	The Regents of the University of California.  All rights reserved.
658f0484fSRodney W. Grimes  *
758f0484fSRodney W. Grimes  * Redistribution and use in source and binary forms, with or without
858f0484fSRodney W. Grimes  * modification, are permitted provided that the following conditions
958f0484fSRodney W. Grimes  * are met:
1058f0484fSRodney W. Grimes  * 1. Redistributions of source code must retain the above copyright
1158f0484fSRodney W. Grimes  *    notice, this list of conditions and the following disclaimer.
1258f0484fSRodney W. Grimes  * 2. Redistributions in binary form must reproduce the above copyright
1358f0484fSRodney W. Grimes  *    notice, this list of conditions and the following disclaimer in the
1458f0484fSRodney W. Grimes  *    documentation and/or other materials provided with the distribution.
15580b4d18SEd Maste  * 3. Neither the name of the University nor the names of its contributors
1658f0484fSRodney W. Grimes  *    may be used to endorse or promote products derived from this software
1758f0484fSRodney W. Grimes  *    without specific prior written permission.
1858f0484fSRodney W. Grimes  *
1958f0484fSRodney W. Grimes  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2058f0484fSRodney W. Grimes  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2158f0484fSRodney W. Grimes  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2258f0484fSRodney W. Grimes  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2358f0484fSRodney W. Grimes  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2458f0484fSRodney W. Grimes  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2558f0484fSRodney W. Grimes  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2658f0484fSRodney W. Grimes  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2758f0484fSRodney W. Grimes  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2858f0484fSRodney W. Grimes  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2958f0484fSRodney W. Grimes  * SUCH DAMAGE.
3058f0484fSRodney W. Grimes  */
3158f0484fSRodney W. Grimes 
32d201fe46SDaniel Eischen #include "namespace.h"
3358f0484fSRodney W. Grimes #include <sys/types.h>
3458f0484fSRodney W. Grimes #include <sys/wait.h>
3558f0484fSRodney W. Grimes #include <signal.h>
3658f0484fSRodney W. Grimes #include <stdlib.h>
3758f0484fSRodney W. Grimes #include <stddef.h>
387d1ffcb7SPeter Wemm #include <string.h>
3958f0484fSRodney W. Grimes #include <unistd.h>
4058f0484fSRodney W. Grimes #include <paths.h>
41b487e9d3SJames Raynard #include <errno.h>
42d201fe46SDaniel Eischen #include "un-namespace.h"
43d201fe46SDaniel Eischen #include "libc_private.h"
4458f0484fSRodney W. Grimes 
458495e8b1SKonstantin Belousov #pragma weak system
4692927338SJason Evans int
system(const char * command)478495e8b1SKonstantin Belousov system(const char *command)
488495e8b1SKonstantin Belousov {
498495e8b1SKonstantin Belousov 
508495e8b1SKonstantin Belousov 	return (((int (*)(const char *))
518495e8b1SKonstantin Belousov 	    __libc_interposing[INTERPOS_system])(command));
528495e8b1SKonstantin Belousov }
538495e8b1SKonstantin Belousov 
548495e8b1SKonstantin Belousov int
__libc_system(const char * command)558495e8b1SKonstantin Belousov __libc_system(const char *command)
5658f0484fSRodney W. Grimes {
579c2ccf27SAlfred Perlstein 	pid_t pid, savedpid;
58b487e9d3SJames Raynard 	int pstat;
59b487e9d3SJames Raynard 	struct sigaction ign, intact, quitact;
60b487e9d3SJames Raynard 	sigset_t newsigblock, oldsigblock;
6158f0484fSRodney W. Grimes 
6258f0484fSRodney W. Grimes 	if (!command)		/* just checking... */
6358f0484fSRodney W. Grimes 		return(1);
6458f0484fSRodney W. Grimes 
65b487e9d3SJames Raynard 	(void)sigemptyset(&newsigblock);
661638d6b6SJames Raynard 	(void)sigaddset(&newsigblock, SIGCHLD);
67865ca149SJilles Tjoelker 	(void)sigaddset(&newsigblock, SIGINT);
68865ca149SJilles Tjoelker 	(void)sigaddset(&newsigblock, SIGQUIT);
69bd6060a1SKonstantin Belousov 	(void)__libc_sigprocmask(SIG_BLOCK, &newsigblock, &oldsigblock);
707d1ffcb7SPeter Wemm 	switch(pid = vfork()) {
71bd6060a1SKonstantin Belousov 	/*
72bd6060a1SKonstantin Belousov 	 * In the child, use unwrapped syscalls.  libthr is in
73bd6060a1SKonstantin Belousov 	 * undefined state after vfork().
74bd6060a1SKonstantin Belousov 	 */
7558f0484fSRodney W. Grimes 	case -1:			/* error */
76bd6060a1SKonstantin Belousov 		(void)__libc_sigprocmask(SIG_SETMASK, &oldsigblock, NULL);
777d1ffcb7SPeter Wemm 		return (-1);
7858f0484fSRodney W. Grimes 	case 0:				/* child */
79b487e9d3SJames Raynard 		/*
80b487e9d3SJames Raynard 		 * Restore original signal dispositions and exec the command.
81b487e9d3SJames Raynard 		 */
82bd6060a1SKonstantin Belousov 		(void)__sys_sigprocmask(SIG_SETMASK, &oldsigblock, NULL);
8358f0484fSRodney W. Grimes 		execl(_PATH_BSHELL, "sh", "-c", command, (char *)NULL);
8458f0484fSRodney W. Grimes 		_exit(127);
857d1ffcb7SPeter Wemm 	}
867d1ffcb7SPeter Wemm 	/*
877d1ffcb7SPeter Wemm 	 * If we are running means that the child has either completed
887d1ffcb7SPeter Wemm 	 * its execve, or has failed.
897d1ffcb7SPeter Wemm 	 * Block SIGINT/QUIT because sh -c handles it and wait for
907d1ffcb7SPeter Wemm 	 * it to clean up.
917d1ffcb7SPeter Wemm 	 */
927d1ffcb7SPeter Wemm 	memset(&ign, 0, sizeof(ign));
937d1ffcb7SPeter Wemm 	ign.sa_handler = SIG_IGN;
947d1ffcb7SPeter Wemm 	(void)sigemptyset(&ign.sa_mask);
95bd6060a1SKonstantin Belousov 	(void)__libc_sigaction(SIGINT, &ign, &intact);
96bd6060a1SKonstantin Belousov 	(void)__libc_sigaction(SIGQUIT, &ign, &quitact);
979c2ccf27SAlfred Perlstein 	savedpid = pid;
98b487e9d3SJames Raynard 	do {
999c2ccf27SAlfred Perlstein 		pid = _wait4(savedpid, &pstat, 0, (struct rusage *)0);
100b487e9d3SJames Raynard 	} while (pid == -1 && errno == EINTR);
101bd6060a1SKonstantin Belousov 	(void)__libc_sigaction(SIGINT, &intact, NULL);
102bd6060a1SKonstantin Belousov 	(void)__libc_sigaction(SIGQUIT,  &quitact, NULL);
103bd6060a1SKonstantin Belousov 	(void)__libc_sigprocmask(SIG_SETMASK, &oldsigblock, NULL);
104b487e9d3SJames Raynard 	return (pid == -1 ? -1 : pstat);
10558f0484fSRodney W. Grimes }
10692927338SJason Evans 
1078495e8b1SKonstantin Belousov __weak_reference(__libc_system, __system);
1088495e8b1SKonstantin Belousov __weak_reference(__libc_system, _system);
109