153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2004-2006 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 published
853a5a1b3Sopenharmony_ci  by the Free Software Foundation; either version 2.1 of the License,
953a5a1b3Sopenharmony_ci  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  General Public License for more details.
1553a5a1b3Sopenharmony_ci
1653a5a1b3Sopenharmony_ci  You should have received a copy of the GNU Lesser General Public License
1753a5a1b3Sopenharmony_ci  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 <stdio.h>
2553a5a1b3Sopenharmony_ci#include <unistd.h>
2653a5a1b3Sopenharmony_ci#include <fcntl.h>
2753a5a1b3Sopenharmony_ci#include <errno.h>
2853a5a1b3Sopenharmony_ci
2953a5a1b3Sopenharmony_ci#include <pulsecore/module.h>
3053a5a1b3Sopenharmony_ci#include <pulsecore/iochannel.h>
3153a5a1b3Sopenharmony_ci#include <pulsecore/cli.h>
3253a5a1b3Sopenharmony_ci#include <pulsecore/sioman.h>
3353a5a1b3Sopenharmony_ci#include <pulsecore/log.h>
3453a5a1b3Sopenharmony_ci#include <pulsecore/modargs.h>
3553a5a1b3Sopenharmony_ci#include <pulsecore/macro.h>
3653a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h>
3753a5a1b3Sopenharmony_ci
3853a5a1b3Sopenharmony_ciPA_MODULE_AUTHOR("Lennart Poettering");
3953a5a1b3Sopenharmony_ciPA_MODULE_DESCRIPTION("Command line interface");
4053a5a1b3Sopenharmony_ciPA_MODULE_VERSION(PACKAGE_VERSION);
4153a5a1b3Sopenharmony_ciPA_MODULE_LOAD_ONCE(true);
4253a5a1b3Sopenharmony_ciPA_MODULE_USAGE("exit_on_eof=<exit daemon after EOF?>");
4353a5a1b3Sopenharmony_ci
4453a5a1b3Sopenharmony_cistatic const char* const valid_modargs[] = {
4553a5a1b3Sopenharmony_ci    "exit_on_eof",
4653a5a1b3Sopenharmony_ci    NULL
4753a5a1b3Sopenharmony_ci};
4853a5a1b3Sopenharmony_ci
4953a5a1b3Sopenharmony_cistatic void eof_and_unload_cb(pa_cli*c, void *userdata) {
5053a5a1b3Sopenharmony_ci    pa_module *m = userdata;
5153a5a1b3Sopenharmony_ci
5253a5a1b3Sopenharmony_ci    pa_assert(c);
5353a5a1b3Sopenharmony_ci    pa_assert(m);
5453a5a1b3Sopenharmony_ci
5553a5a1b3Sopenharmony_ci    pa_module_unload_request(m, true);
5653a5a1b3Sopenharmony_ci}
5753a5a1b3Sopenharmony_ci
5853a5a1b3Sopenharmony_cistatic void eof_and_exit_cb(pa_cli*c, void *userdata) {
5953a5a1b3Sopenharmony_ci    pa_module *m = userdata;
6053a5a1b3Sopenharmony_ci
6153a5a1b3Sopenharmony_ci    pa_assert(c);
6253a5a1b3Sopenharmony_ci    pa_assert(m);
6353a5a1b3Sopenharmony_ci
6453a5a1b3Sopenharmony_ci    pa_core_exit(m->core, false, 0);
6553a5a1b3Sopenharmony_ci}
6653a5a1b3Sopenharmony_ci
6753a5a1b3Sopenharmony_ciint pa__init(pa_module*m) {
6853a5a1b3Sopenharmony_ci    pa_iochannel *io;
6953a5a1b3Sopenharmony_ci    pa_modargs *ma;
7053a5a1b3Sopenharmony_ci    bool exit_on_eof = false;
7153a5a1b3Sopenharmony_ci#ifndef OS_IS_WIN32
7253a5a1b3Sopenharmony_ci    int fd;
7353a5a1b3Sopenharmony_ci#endif
7453a5a1b3Sopenharmony_ci
7553a5a1b3Sopenharmony_ci    pa_assert(m);
7653a5a1b3Sopenharmony_ci
7753a5a1b3Sopenharmony_ci    if (m->core->running_as_daemon) {
7853a5a1b3Sopenharmony_ci        pa_log_info("Running as daemon, refusing to load this module.");
7953a5a1b3Sopenharmony_ci        return 0;
8053a5a1b3Sopenharmony_ci    }
8153a5a1b3Sopenharmony_ci
8253a5a1b3Sopenharmony_ci    if (!(ma = pa_modargs_new(m->argument, valid_modargs))) {
8353a5a1b3Sopenharmony_ci        pa_log("failed to parse module arguments.");
8453a5a1b3Sopenharmony_ci        goto fail;
8553a5a1b3Sopenharmony_ci    }
8653a5a1b3Sopenharmony_ci
8753a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "exit_on_eof", &exit_on_eof) < 0) {
8853a5a1b3Sopenharmony_ci        pa_log("exit_on_eof= expects boolean argument.");
8953a5a1b3Sopenharmony_ci        goto fail;
9053a5a1b3Sopenharmony_ci    }
9153a5a1b3Sopenharmony_ci
9253a5a1b3Sopenharmony_ci    if (pa_stdio_acquire() < 0) {
9353a5a1b3Sopenharmony_ci        pa_log("STDIN/STDOUT already in use.");
9453a5a1b3Sopenharmony_ci        goto fail;
9553a5a1b3Sopenharmony_ci    }
9653a5a1b3Sopenharmony_ci
9753a5a1b3Sopenharmony_ci    /* We try to open the controlling tty anew here. This has the
9853a5a1b3Sopenharmony_ci     * benefit of giving us a new fd that doesn't share the O_NDELAY
9953a5a1b3Sopenharmony_ci     * flag with fds 0, 1, or 2. Since pa_iochannel_xxx needs O_NDELAY
10053a5a1b3Sopenharmony_ci     * on its fd using those fds directly could set O_NDELAY which
10153a5a1b3Sopenharmony_ci     * fprintf() doesn't really like, resulting in truncated output
10253a5a1b3Sopenharmony_ci     * of log messages, particularly because if stdout and stderr are
10353a5a1b3Sopenharmony_ci     * dup'ed they share the same O_NDELAY, too. */
10453a5a1b3Sopenharmony_ci
10553a5a1b3Sopenharmony_ci#ifndef OS_IS_WIN32
10653a5a1b3Sopenharmony_ci    if ((fd = pa_open_cloexec("/dev/tty", O_RDWR|O_NONBLOCK, 0)) >= 0) {
10753a5a1b3Sopenharmony_ci        io = pa_iochannel_new(m->core->mainloop, fd, fd);
10853a5a1b3Sopenharmony_ci        pa_log_debug("Managed to open /dev/tty.");
10953a5a1b3Sopenharmony_ci    }
11053a5a1b3Sopenharmony_ci    else
11153a5a1b3Sopenharmony_ci#endif
11253a5a1b3Sopenharmony_ci    {
11353a5a1b3Sopenharmony_ci        io = pa_iochannel_new(m->core->mainloop, STDIN_FILENO, STDOUT_FILENO);
11453a5a1b3Sopenharmony_ci        pa_iochannel_set_noclose(io, true);
11553a5a1b3Sopenharmony_ci        pa_log_debug("Failed to open /dev/tty, using stdin/stdout fds instead.");
11653a5a1b3Sopenharmony_ci    }
11753a5a1b3Sopenharmony_ci
11853a5a1b3Sopenharmony_ci    m->userdata = pa_cli_new(m->core, io, m);
11953a5a1b3Sopenharmony_ci    pa_cli_set_eof_callback(m->userdata, exit_on_eof ? eof_and_exit_cb : eof_and_unload_cb, m);
12053a5a1b3Sopenharmony_ci
12153a5a1b3Sopenharmony_ci    pa_modargs_free(ma);
12253a5a1b3Sopenharmony_ci
12353a5a1b3Sopenharmony_ci    return 0;
12453a5a1b3Sopenharmony_ci
12553a5a1b3Sopenharmony_cifail:
12653a5a1b3Sopenharmony_ci
12753a5a1b3Sopenharmony_ci    if (ma)
12853a5a1b3Sopenharmony_ci        pa_modargs_free(ma);
12953a5a1b3Sopenharmony_ci
13053a5a1b3Sopenharmony_ci    return -1;
13153a5a1b3Sopenharmony_ci}
13253a5a1b3Sopenharmony_ci
13353a5a1b3Sopenharmony_civoid pa__done(pa_module*m) {
13453a5a1b3Sopenharmony_ci    pa_assert(m);
13553a5a1b3Sopenharmony_ci
13653a5a1b3Sopenharmony_ci    if (m->userdata) {
13753a5a1b3Sopenharmony_ci        pa_cli_free(m->userdata);
13853a5a1b3Sopenharmony_ci        pa_stdio_release();
13953a5a1b3Sopenharmony_ci    }
14053a5a1b3Sopenharmony_ci}
141