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