2 This file is part of PulseAudio.
4 Copyright 2005-2006 Lennart Poettering
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as published
8 by the Free Software Foundation; either version 2.1 of the License,
9 or (at your option) any later version.
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public License
17 along with PulseAudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
31 #include <lirc/lirc_client.h>
33 #include <pulse/xmalloc.h>
35 #include <pulsecore/module.h>
36 #include <pulsecore/log.h>
37 #include <pulsecore/namereg.h>
38 #include <pulsecore/sink.h>
39 #include <pulsecore/modargs.h>
40 #include <pulsecore/macro.h>
42 #include "module-lirc-symdef.h"
44 PA_MODULE_AUTHOR("Lennart Poettering");
45 PA_MODULE_DESCRIPTION("LIRC volume control");
46 PA_MODULE_VERSION(PACKAGE_VERSION
);
47 PA_MODULE_LOAD_ONCE(TRUE
);
48 PA_MODULE_USAGE("config=<config file> sink=<sink name> appname=<lirc application name>");
50 static const char* const valid_modargs
[] = {
60 struct lirc_config
*config
;
63 float mute_toggle_save
;
66 static void io_callback(pa_mainloop_api
*io
, pa_io_event
*e
, int fd
, pa_io_event_flags_t events
, void*userdata
) {
67 struct userdata
*u
= userdata
;
68 char *name
= NULL
, *code
= NULL
;
73 if (events
& (PA_IO_EVENT_HANGUP
|PA_IO_EVENT_ERROR
)) {
74 pa_log("Lost connection to LIRC daemon.");
78 if (events
& PA_IO_EVENT_INPUT
) {
81 if (lirc_nextcode(&code
) != 0 || !code
) {
82 pa_log("lirc_nextcode() failed.");
87 c
[strcspn(c
, "\n\r")] = 0;
88 pa_log_debug("Raw IR code '%s'", c
);
91 while (lirc_code2char(u
->config
, code
, &name
) == 0 && name
) {
99 } volchange
= INVALID
;
101 pa_log_info("Translated IR code '%s'", name
);
103 if (strcasecmp(name
, "volume-up") == 0)
105 else if (strcasecmp(name
, "volume-down") == 0)
107 else if (strcasecmp(name
, "mute") == 0)
109 else if (strcasecmp(name
, "mute-toggle") == 0)
110 volchange
= MUTE_TOGGLE
;
111 else if (strcasecmp(name
, "reset") == 0)
114 if (volchange
== INVALID
)
115 pa_log_warn("Received unknown IR code '%s'", name
);
119 if (!(s
= pa_namereg_get(u
->module
->core
, u
->sink_name
, PA_NAMEREG_SINK
)))
120 pa_log("Failed to get sink '%s'", u
->sink_name
);
123 pa_cvolume cv
= *pa_sink_get_volume(s
, FALSE
, FALSE
);
125 #define DELTA (PA_VOLUME_NORM/20)
129 for (i
= 0; i
< cv
.channels
; i
++) {
130 if (cv
.values
[i
] < PA_VOLUME_MAX
- DELTA
)
131 cv
.values
[i
] += DELTA
;
133 cv
.values
[i
] = PA_VOLUME_MAX
;
136 pa_sink_set_volume(s
, &cv
, TRUE
, TRUE
, TRUE
, TRUE
);
140 for (i
= 0; i
< cv
.channels
; i
++) {
141 if (cv
.values
[i
] > DELTA
)
142 cv
.values
[i
] -= DELTA
;
144 cv
.values
[i
] = PA_VOLUME_MUTED
;
147 pa_sink_set_volume(s
, &cv
, TRUE
, TRUE
, TRUE
, TRUE
);
151 pa_sink_set_mute(s
, TRUE
, TRUE
);
155 pa_sink_set_mute(s
, FALSE
, TRUE
);
160 pa_sink_set_mute(s
, !pa_sink_get_mute(s
, FALSE
), TRUE
);
164 pa_assert_not_reached();
176 u
->module
->core
->mainloop
->io_free(u
->io
);
179 pa_module_unload_request(u
->module
, TRUE
);
184 int pa__init(pa_module
*m
) {
185 pa_modargs
*ma
= NULL
;
190 if (!(ma
= pa_modargs_new(m
->argument
, valid_modargs
))) {
191 pa_log("Failed to parse module arguments");
195 m
->userdata
= u
= pa_xnew(struct userdata
, 1);
199 u
->sink_name
= pa_xstrdup(pa_modargs_get_value(ma
, "sink", NULL
));
201 u
->mute_toggle_save
= 0;
203 if ((u
->lirc_fd
= lirc_init((char*) pa_modargs_get_value(ma
, "appname", "pulseaudio"), 1)) < 0) {
204 pa_log("lirc_init() failed.");
208 if (lirc_readconfig((char*) pa_modargs_get_value(ma
, "config", NULL
), &u
->config
, NULL
) < 0) {
209 pa_log("lirc_readconfig() failed.");
213 u
->io
= m
->core
->mainloop
->io_new(m
->core
->mainloop
, u
->lirc_fd
, PA_IO_EVENT_INPUT
|PA_IO_EVENT_HANGUP
, io_callback
, u
);
228 void pa__done(pa_module
*m
) {
232 if (!(u
= m
->userdata
))
236 m
->core
->mainloop
->io_free(u
->io
);
239 lirc_freeconfig(u
->config
);
244 pa_xfree(u
->sink_name
);