153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2007 Lennart Poettering
553a5a1b3Sopenharmony_ci
653a5a1b3Sopenharmony_ci  PulseAudio is free software; you can redistribute it and/or modify
753a5a1b3Sopenharmony_ci  it under the terms of the GNU Lesser General Public License as
853a5a1b3Sopenharmony_ci  published by the Free Software Foundation; either version 2.1 of the
953a5a1b3Sopenharmony_ci  License, or (at your option) any later version.
1053a5a1b3Sopenharmony_ci
1153a5a1b3Sopenharmony_ci  PulseAudio is distributed in the hope that it will be useful, but
1253a5a1b3Sopenharmony_ci  WITHOUT ANY WARRANTY; without even the implied warranty of
1353a5a1b3Sopenharmony_ci  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1453a5a1b3Sopenharmony_ci  Lesser General Public License for more details.
1553a5a1b3Sopenharmony_ci
1653a5a1b3Sopenharmony_ci  You should have received a copy of the GNU Lesser General Public
1753a5a1b3Sopenharmony_ci  License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
1853a5a1b3Sopenharmony_ci***/
1953a5a1b3Sopenharmony_ci
2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H
2153a5a1b3Sopenharmony_ci#include <config.h>
2253a5a1b3Sopenharmony_ci#endif
2353a5a1b3Sopenharmony_ci
2453a5a1b3Sopenharmony_ci#include <errno.h>
2553a5a1b3Sopenharmony_ci#include <stdio.h>
2653a5a1b3Sopenharmony_ci#include <sys/types.h>
2753a5a1b3Sopenharmony_ci#include <dirent.h>
2853a5a1b3Sopenharmony_ci#include <unistd.h>
2953a5a1b3Sopenharmony_ci#include <fcntl.h>
3053a5a1b3Sopenharmony_ci#include <signal.h>
3153a5a1b3Sopenharmony_ci
3253a5a1b3Sopenharmony_ci#ifdef HAVE_SYS_PRCTL_H
3353a5a1b3Sopenharmony_ci#include <sys/prctl.h>
3453a5a1b3Sopenharmony_ci#endif
3553a5a1b3Sopenharmony_ci#ifdef HAVE_SYS_RESOURCE_H
3653a5a1b3Sopenharmony_ci#include <sys/resource.h>
3753a5a1b3Sopenharmony_ci#endif
3853a5a1b3Sopenharmony_ci
3953a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h>
4053a5a1b3Sopenharmony_ci#include <pulsecore/core-error.h>
4153a5a1b3Sopenharmony_ci#include <pulsecore/pipe.h>
4253a5a1b3Sopenharmony_ci
4353a5a1b3Sopenharmony_ci#include "start-child.h"
4453a5a1b3Sopenharmony_ci
4553a5a1b3Sopenharmony_ciint pa_start_child_for_read(const char *name, const char *argv1, pid_t *pid) {
4653a5a1b3Sopenharmony_ci#ifdef HAVE_FORK
4753a5a1b3Sopenharmony_ci    pid_t child;
4853a5a1b3Sopenharmony_ci    int pipe_fds[2] = { -1, -1 };
4953a5a1b3Sopenharmony_ci
5053a5a1b3Sopenharmony_ci    if (pipe(pipe_fds) < 0) {
5153a5a1b3Sopenharmony_ci        pa_log("pipe() failed: %s", pa_cstrerror(errno));
5253a5a1b3Sopenharmony_ci        goto fail;
5353a5a1b3Sopenharmony_ci    }
5453a5a1b3Sopenharmony_ci
5553a5a1b3Sopenharmony_ci    if ((child = fork()) == (pid_t) -1) {
5653a5a1b3Sopenharmony_ci        pa_log("fork() failed: %s", pa_cstrerror(errno));
5753a5a1b3Sopenharmony_ci        goto fail;
5853a5a1b3Sopenharmony_ci
5953a5a1b3Sopenharmony_ci    } else if (child != 0) {
6053a5a1b3Sopenharmony_ci
6153a5a1b3Sopenharmony_ci        /* Parent */
6253a5a1b3Sopenharmony_ci        pa_assert_se(pa_close(pipe_fds[1]) == 0);
6353a5a1b3Sopenharmony_ci
6453a5a1b3Sopenharmony_ci        if (pid)
6553a5a1b3Sopenharmony_ci            *pid = child;
6653a5a1b3Sopenharmony_ci
6753a5a1b3Sopenharmony_ci        return pipe_fds[0];
6853a5a1b3Sopenharmony_ci    } else {
6953a5a1b3Sopenharmony_ci        /* child */
7053a5a1b3Sopenharmony_ci
7153a5a1b3Sopenharmony_ci        pa_reset_personality();
7253a5a1b3Sopenharmony_ci
7353a5a1b3Sopenharmony_ci        pa_assert_se(pa_close(pipe_fds[0]) == 0);
7453a5a1b3Sopenharmony_ci        pa_assert_se(dup2(pipe_fds[1], STDOUT_FILENO) == STDOUT_FILENO);
7553a5a1b3Sopenharmony_ci
7653a5a1b3Sopenharmony_ci        if (pipe_fds[1] != STDOUT_FILENO)
7753a5a1b3Sopenharmony_ci            pa_assert_se(pa_close(pipe_fds[1]) == 0);
7853a5a1b3Sopenharmony_ci
7953a5a1b3Sopenharmony_ci        pa_close(STDIN_FILENO);
8053a5a1b3Sopenharmony_ci        pa_assert_se(open("/dev/null", O_RDONLY) == STDIN_FILENO);
8153a5a1b3Sopenharmony_ci
8253a5a1b3Sopenharmony_ci        pa_close(STDERR_FILENO);
8353a5a1b3Sopenharmony_ci        pa_assert_se(open("/dev/null", O_WRONLY) == STDERR_FILENO);
8453a5a1b3Sopenharmony_ci
8553a5a1b3Sopenharmony_ci        pa_close_all(-1);
8653a5a1b3Sopenharmony_ci        pa_reset_sigs(-1);
8753a5a1b3Sopenharmony_ci        pa_unblock_sigs(-1);
8853a5a1b3Sopenharmony_ci        pa_reset_priority();
8953a5a1b3Sopenharmony_ci        pa_unset_env_recorded();
9053a5a1b3Sopenharmony_ci
9153a5a1b3Sopenharmony_ci        /* Make sure our children are not influenced by the
9253a5a1b3Sopenharmony_ci         * LD_BIND_NOW we set for ourselves. */
9353a5a1b3Sopenharmony_ci        pa_unset_env("LD_BIND_NOW");
9453a5a1b3Sopenharmony_ci
9553a5a1b3Sopenharmony_ci#ifdef PR_SET_PDEATHSIG
9653a5a1b3Sopenharmony_ci        /* On Linux we can use PR_SET_PDEATHSIG to have the helper
9753a5a1b3Sopenharmony_ci        process killed when the daemon dies abnormally. On non-Linux
9853a5a1b3Sopenharmony_ci        machines the client will die as soon as it writes data to
9953a5a1b3Sopenharmony_ci        stdout again (SIGPIPE) */
10053a5a1b3Sopenharmony_ci
10153a5a1b3Sopenharmony_ci        prctl(PR_SET_PDEATHSIG, SIGTERM, 0, 0, 0);
10253a5a1b3Sopenharmony_ci#endif
10353a5a1b3Sopenharmony_ci
10453a5a1b3Sopenharmony_ci        execl(name, name, argv1, NULL);
10553a5a1b3Sopenharmony_ci        _exit(1);
10653a5a1b3Sopenharmony_ci    }
10753a5a1b3Sopenharmony_ci
10853a5a1b3Sopenharmony_cifail:
10953a5a1b3Sopenharmony_ci    pa_close_pipe(pipe_fds);
11053a5a1b3Sopenharmony_ci#endif /* HAVE_FORK */
11153a5a1b3Sopenharmony_ci
11253a5a1b3Sopenharmony_ci    return -1;
11353a5a1b3Sopenharmony_ci}
114