153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2005-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 <string.h>
2753a5a1b3Sopenharmony_ci#include <stdlib.h>
2853a5a1b3Sopenharmony_ci
2953a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h>
3053a5a1b3Sopenharmony_ci
3153a5a1b3Sopenharmony_ci#include <pulsecore/module.h>
3253a5a1b3Sopenharmony_ci#include <pulsecore/log.h>
3353a5a1b3Sopenharmony_ci#include <pulsecore/namereg.h>
3453a5a1b3Sopenharmony_ci#include <pulsecore/sink.h>
3553a5a1b3Sopenharmony_ci#include <pulsecore/modargs.h>
3653a5a1b3Sopenharmony_ci#include <pulsecore/macro.h>
3753a5a1b3Sopenharmony_ci
3853a5a1b3Sopenharmony_ciPA_MODULE_AUTHOR("Lennart Poettering");
3953a5a1b3Sopenharmony_ciPA_MODULE_DESCRIPTION("LIRC volume control");
4053a5a1b3Sopenharmony_ciPA_MODULE_VERSION(PACKAGE_VERSION);
4153a5a1b3Sopenharmony_ciPA_MODULE_LOAD_ONCE(true);
4253a5a1b3Sopenharmony_ciPA_MODULE_USAGE("config=<config file> sink=<sink name> appname=<lirc application name> volume_limit=<volume limit> volume_step=<volume change step>");
4353a5a1b3Sopenharmony_ci
4453a5a1b3Sopenharmony_ci/* LIRC would provide it's own definition of PACKAGE_VERSION, include it after
4553a5a1b3Sopenharmony_ci * pulseaudio module definition block to prevent module loader version mismatch.
4653a5a1b3Sopenharmony_ci */
4753a5a1b3Sopenharmony_ci#include <lirc/lirc_client.h>
4853a5a1b3Sopenharmony_ci
4953a5a1b3Sopenharmony_cistatic const char* const valid_modargs[] = {
5053a5a1b3Sopenharmony_ci    "config",
5153a5a1b3Sopenharmony_ci    "sink",
5253a5a1b3Sopenharmony_ci    "appname",
5353a5a1b3Sopenharmony_ci    "volume_limit",
5453a5a1b3Sopenharmony_ci    "volume_step",
5553a5a1b3Sopenharmony_ci    NULL,
5653a5a1b3Sopenharmony_ci};
5753a5a1b3Sopenharmony_ci
5853a5a1b3Sopenharmony_cistruct userdata {
5953a5a1b3Sopenharmony_ci    int lirc_fd;
6053a5a1b3Sopenharmony_ci    pa_io_event *io;
6153a5a1b3Sopenharmony_ci    struct lirc_config *config;
6253a5a1b3Sopenharmony_ci    char *sink_name;
6353a5a1b3Sopenharmony_ci    pa_module *module;
6453a5a1b3Sopenharmony_ci    float mute_toggle_save;
6553a5a1b3Sopenharmony_ci    pa_volume_t volume_limit;
6653a5a1b3Sopenharmony_ci    pa_volume_t volume_step;
6753a5a1b3Sopenharmony_ci};
6853a5a1b3Sopenharmony_ci
6953a5a1b3Sopenharmony_cistatic void io_callback(pa_mainloop_api *io, pa_io_event *e, int fd, pa_io_event_flags_t events, void*userdata) {
7053a5a1b3Sopenharmony_ci    struct userdata *u = userdata;
7153a5a1b3Sopenharmony_ci    char *name = NULL, *code = NULL;
7253a5a1b3Sopenharmony_ci
7353a5a1b3Sopenharmony_ci    pa_assert(io);
7453a5a1b3Sopenharmony_ci    pa_assert(u);
7553a5a1b3Sopenharmony_ci
7653a5a1b3Sopenharmony_ci    if (events & (PA_IO_EVENT_HANGUP|PA_IO_EVENT_ERROR)) {
7753a5a1b3Sopenharmony_ci        pa_log("Lost connection to LIRC daemon.");
7853a5a1b3Sopenharmony_ci        goto fail;
7953a5a1b3Sopenharmony_ci    }
8053a5a1b3Sopenharmony_ci
8153a5a1b3Sopenharmony_ci    if (events & PA_IO_EVENT_INPUT) {
8253a5a1b3Sopenharmony_ci        char *c;
8353a5a1b3Sopenharmony_ci
8453a5a1b3Sopenharmony_ci        if (lirc_nextcode(&code) != 0 || !code) {
8553a5a1b3Sopenharmony_ci            pa_log("lirc_nextcode() failed.");
8653a5a1b3Sopenharmony_ci            goto fail;
8753a5a1b3Sopenharmony_ci        }
8853a5a1b3Sopenharmony_ci
8953a5a1b3Sopenharmony_ci        c = pa_xstrdup(code);
9053a5a1b3Sopenharmony_ci        c[strcspn(c, "\n\r")] = 0;
9153a5a1b3Sopenharmony_ci        pa_log_debug("Raw IR code '%s'", c);
9253a5a1b3Sopenharmony_ci        pa_xfree(c);
9353a5a1b3Sopenharmony_ci
9453a5a1b3Sopenharmony_ci        while (lirc_code2char(u->config, code, &name) == 0 && name) {
9553a5a1b3Sopenharmony_ci            enum {
9653a5a1b3Sopenharmony_ci                INVALID,
9753a5a1b3Sopenharmony_ci                UP,
9853a5a1b3Sopenharmony_ci                DOWN,
9953a5a1b3Sopenharmony_ci                MUTE,
10053a5a1b3Sopenharmony_ci                RESET,
10153a5a1b3Sopenharmony_ci                MUTE_TOGGLE
10253a5a1b3Sopenharmony_ci            } volchange = INVALID;
10353a5a1b3Sopenharmony_ci
10453a5a1b3Sopenharmony_ci            pa_log_info("Translated IR code '%s'", name);
10553a5a1b3Sopenharmony_ci
10653a5a1b3Sopenharmony_ci            if (strcasecmp(name, "volume-up") == 0)
10753a5a1b3Sopenharmony_ci                volchange = UP;
10853a5a1b3Sopenharmony_ci            else if (strcasecmp(name, "volume-down") == 0)
10953a5a1b3Sopenharmony_ci                volchange = DOWN;
11053a5a1b3Sopenharmony_ci            else if (strcasecmp(name, "mute") == 0)
11153a5a1b3Sopenharmony_ci                volchange = MUTE;
11253a5a1b3Sopenharmony_ci            else if (strcasecmp(name, "mute-toggle") == 0)
11353a5a1b3Sopenharmony_ci                volchange = MUTE_TOGGLE;
11453a5a1b3Sopenharmony_ci            else if (strcasecmp(name, "reset") == 0)
11553a5a1b3Sopenharmony_ci                volchange = RESET;
11653a5a1b3Sopenharmony_ci
11753a5a1b3Sopenharmony_ci            if (volchange == INVALID)
11853a5a1b3Sopenharmony_ci                pa_log_warn("Received unknown IR code '%s'", name);
11953a5a1b3Sopenharmony_ci            else {
12053a5a1b3Sopenharmony_ci                pa_sink *s;
12153a5a1b3Sopenharmony_ci
12253a5a1b3Sopenharmony_ci                if (!(s = pa_namereg_get(u->module->core, u->sink_name, PA_NAMEREG_SINK)))
12353a5a1b3Sopenharmony_ci                    pa_log("Failed to get sink '%s'", u->sink_name);
12453a5a1b3Sopenharmony_ci                else {
12553a5a1b3Sopenharmony_ci                    pa_cvolume cv = *pa_sink_get_volume(s, false);
12653a5a1b3Sopenharmony_ci
12753a5a1b3Sopenharmony_ci                    switch (volchange) {
12853a5a1b3Sopenharmony_ci                        case UP:
12953a5a1b3Sopenharmony_ci                            pa_cvolume_inc_clamp(&cv, u->volume_step, u->volume_limit);
13053a5a1b3Sopenharmony_ci                            pa_sink_set_volume(s, &cv, true, true);
13153a5a1b3Sopenharmony_ci                            break;
13253a5a1b3Sopenharmony_ci
13353a5a1b3Sopenharmony_ci                        case DOWN:
13453a5a1b3Sopenharmony_ci                            pa_cvolume_dec(&cv, u->volume_step);
13553a5a1b3Sopenharmony_ci                            pa_sink_set_volume(s, &cv, true, true);
13653a5a1b3Sopenharmony_ci                            break;
13753a5a1b3Sopenharmony_ci
13853a5a1b3Sopenharmony_ci                        case MUTE:
13953a5a1b3Sopenharmony_ci                            pa_sink_set_mute(s, true, true);
14053a5a1b3Sopenharmony_ci                            break;
14153a5a1b3Sopenharmony_ci
14253a5a1b3Sopenharmony_ci                        case RESET:
14353a5a1b3Sopenharmony_ci                            pa_sink_set_mute(s, false, true);
14453a5a1b3Sopenharmony_ci                            break;
14553a5a1b3Sopenharmony_ci
14653a5a1b3Sopenharmony_ci                        case MUTE_TOGGLE:
14753a5a1b3Sopenharmony_ci                            pa_sink_set_mute(s, !pa_sink_get_mute(s, false), true);
14853a5a1b3Sopenharmony_ci                            break;
14953a5a1b3Sopenharmony_ci
15053a5a1b3Sopenharmony_ci                        case INVALID:
15153a5a1b3Sopenharmony_ci                            pa_assert_not_reached();
15253a5a1b3Sopenharmony_ci                    }
15353a5a1b3Sopenharmony_ci                }
15453a5a1b3Sopenharmony_ci            }
15553a5a1b3Sopenharmony_ci        }
15653a5a1b3Sopenharmony_ci    }
15753a5a1b3Sopenharmony_ci
15853a5a1b3Sopenharmony_ci    pa_xfree(code);
15953a5a1b3Sopenharmony_ci
16053a5a1b3Sopenharmony_ci    return;
16153a5a1b3Sopenharmony_ci
16253a5a1b3Sopenharmony_cifail:
16353a5a1b3Sopenharmony_ci    u->module->core->mainloop->io_free(u->io);
16453a5a1b3Sopenharmony_ci    u->io = NULL;
16553a5a1b3Sopenharmony_ci
16653a5a1b3Sopenharmony_ci    pa_module_unload_request(u->module, true);
16753a5a1b3Sopenharmony_ci
16853a5a1b3Sopenharmony_ci    pa_xfree(code);
16953a5a1b3Sopenharmony_ci}
17053a5a1b3Sopenharmony_ci
17153a5a1b3Sopenharmony_ciint pa__init(pa_module*m) {
17253a5a1b3Sopenharmony_ci    pa_modargs *ma = NULL;
17353a5a1b3Sopenharmony_ci    struct userdata *u;
17453a5a1b3Sopenharmony_ci    pa_volume_t volume_limit = PA_CLAMP_VOLUME(PA_VOLUME_NORM*3/2);
17553a5a1b3Sopenharmony_ci    pa_volume_t volume_step = PA_VOLUME_NORM/20;
17653a5a1b3Sopenharmony_ci
17753a5a1b3Sopenharmony_ci    pa_assert(m);
17853a5a1b3Sopenharmony_ci
17953a5a1b3Sopenharmony_ci    if (!(ma = pa_modargs_new(m->argument, valid_modargs))) {
18053a5a1b3Sopenharmony_ci        pa_log("Failed to parse module arguments");
18153a5a1b3Sopenharmony_ci        goto fail;
18253a5a1b3Sopenharmony_ci    }
18353a5a1b3Sopenharmony_ci
18453a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_u32(ma, "volume_limit", &volume_limit) < 0) {
18553a5a1b3Sopenharmony_ci        pa_log("Failed to parse volume limit");
18653a5a1b3Sopenharmony_ci        goto fail;
18753a5a1b3Sopenharmony_ci    }
18853a5a1b3Sopenharmony_ci
18953a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_u32(ma, "volume_step", &volume_step) < 0) {
19053a5a1b3Sopenharmony_ci        pa_log("Failed to parse volume step");
19153a5a1b3Sopenharmony_ci        goto fail;
19253a5a1b3Sopenharmony_ci    }
19353a5a1b3Sopenharmony_ci
19453a5a1b3Sopenharmony_ci    m->userdata = u = pa_xnew(struct userdata, 1);
19553a5a1b3Sopenharmony_ci    u->module = m;
19653a5a1b3Sopenharmony_ci    u->io = NULL;
19753a5a1b3Sopenharmony_ci    u->config = NULL;
19853a5a1b3Sopenharmony_ci    u->sink_name = pa_xstrdup(pa_modargs_get_value(ma, "sink", NULL));
19953a5a1b3Sopenharmony_ci    u->lirc_fd = -1;
20053a5a1b3Sopenharmony_ci    u->mute_toggle_save = 0;
20153a5a1b3Sopenharmony_ci    u->volume_limit = PA_CLAMP_VOLUME(volume_limit);
20253a5a1b3Sopenharmony_ci    u->volume_step = PA_CLAMP_VOLUME(volume_step);
20353a5a1b3Sopenharmony_ci
20453a5a1b3Sopenharmony_ci    if ((u->lirc_fd = lirc_init((char*) pa_modargs_get_value(ma, "appname", "pulseaudio"), 1)) < 0) {
20553a5a1b3Sopenharmony_ci        pa_log("lirc_init() failed.");
20653a5a1b3Sopenharmony_ci        goto fail;
20753a5a1b3Sopenharmony_ci    }
20853a5a1b3Sopenharmony_ci
20953a5a1b3Sopenharmony_ci    if (lirc_readconfig((char*) pa_modargs_get_value(ma, "config", NULL), &u->config, NULL) < 0) {
21053a5a1b3Sopenharmony_ci        pa_log("lirc_readconfig() failed.");
21153a5a1b3Sopenharmony_ci        goto fail;
21253a5a1b3Sopenharmony_ci    }
21353a5a1b3Sopenharmony_ci
21453a5a1b3Sopenharmony_ci    u->io = m->core->mainloop->io_new(m->core->mainloop, u->lirc_fd, PA_IO_EVENT_INPUT|PA_IO_EVENT_HANGUP, io_callback, u);
21553a5a1b3Sopenharmony_ci
21653a5a1b3Sopenharmony_ci    pa_modargs_free(ma);
21753a5a1b3Sopenharmony_ci
21853a5a1b3Sopenharmony_ci    return 0;
21953a5a1b3Sopenharmony_ci
22053a5a1b3Sopenharmony_cifail:
22153a5a1b3Sopenharmony_ci
22253a5a1b3Sopenharmony_ci    if (ma)
22353a5a1b3Sopenharmony_ci        pa_modargs_free(ma);
22453a5a1b3Sopenharmony_ci
22553a5a1b3Sopenharmony_ci    pa__done(m);
22653a5a1b3Sopenharmony_ci    return -1;
22753a5a1b3Sopenharmony_ci}
22853a5a1b3Sopenharmony_ci
22953a5a1b3Sopenharmony_civoid pa__done(pa_module*m) {
23053a5a1b3Sopenharmony_ci    struct userdata *u;
23153a5a1b3Sopenharmony_ci    pa_assert(m);
23253a5a1b3Sopenharmony_ci
23353a5a1b3Sopenharmony_ci    if (!(u = m->userdata))
23453a5a1b3Sopenharmony_ci        return;
23553a5a1b3Sopenharmony_ci
23653a5a1b3Sopenharmony_ci    if (u->io)
23753a5a1b3Sopenharmony_ci        m->core->mainloop->io_free(u->io);
23853a5a1b3Sopenharmony_ci
23953a5a1b3Sopenharmony_ci    if (u->config)
24053a5a1b3Sopenharmony_ci        lirc_freeconfig(u->config);
24153a5a1b3Sopenharmony_ci
24253a5a1b3Sopenharmony_ci    if (u->lirc_fd >= 0)
24353a5a1b3Sopenharmony_ci        lirc_deinit();
24453a5a1b3Sopenharmony_ci
24553a5a1b3Sopenharmony_ci    pa_xfree(u->sink_name);
24653a5a1b3Sopenharmony_ci    pa_xfree(u);
24753a5a1b3Sopenharmony_ci}
248