4 This file is part of polypaudio.
6 polypaudio 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 of the License,
9 or (at your option) any later version.
11 polypaudio 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 polypaudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
30 #include <sys/types.h>
35 #include <polypcore/module.h>
36 #include <polypcore/util.h>
37 #include <polypcore/modargs.h>
38 #include <polypcore/log.h>
39 #include <polypcore/core-subscribe.h>
40 #include <polypcore/xmalloc.h>
41 #include <polypcore/sink-input.h>
43 #include "module-match-symdef.h"
45 PA_MODULE_AUTHOR("Lennart Poettering")
46 PA_MODULE_DESCRIPTION("Sink input matching module")
47 PA_MODULE_USAGE("table=<filename>")
48 PA_MODULE_VERSION(PACKAGE_VERSION
)
50 #define WHITESPACE "\n\r \t"
52 #ifndef DEFAULT_CONFIG_DIR
53 #define DEFAULT_CONFIG_DIR "/etc/polypaudio"
56 #define DEFAULT_MATCH_TABLE_FILE DEFAULT_CONFIG_DIR"/match.table"
57 #define DEFAULT_MATCH_TABLE_FILE_USER ".polypaudio/match.table"
59 static const char* const valid_modargs
[] = {
72 pa_subscription
*subscription
;
75 static int load_rules(struct userdata
*u
, const char *filename
) {
79 struct rule
*end
= NULL
;
83 fopen(fn
= pa_xstrdup(filename
), "r") :
84 pa_open_config_file(DEFAULT_MATCH_TABLE_FILE
, DEFAULT_MATCH_TABLE_FILE_USER
, NULL
, &fn
);
87 pa_log(__FILE__
": failed to open file '%s': %s", fn
, strerror(errno
));
99 if (!fgets(ln
, sizeof(ln
), f
))
106 if (ln
[0] == '#' || !*ln
)
109 d
= ln
+strcspn(ln
, WHITESPACE
);
110 v
= d
+strspn(d
, WHITESPACE
);
114 pa_log(__FILE__
": [%s:%u] failed to parse line - too few words", filename
, n
);
119 if (pa_atou(v
, &k
) < 0) {
120 pa_log(__FILE__
": [%s:%u] failed to parse volume", filename
, n
);
124 volume
= (pa_volume_t
) k
;
127 if (regcomp(®ex
, ln
, REG_EXTENDED
|REG_NOSUB
) != 0) {
128 pa_log(__FILE__
": [%s:%u] invalid regular expression", filename
, n
);
132 rule
= pa_xmalloc(sizeof(struct rule
));
134 rule
->volume
= volume
;
158 static void callback(pa_core
*c
, pa_subscription_event_type_t t
, uint32_t idx
, void *userdata
) {
159 struct userdata
*u
= userdata
;
164 if (t
!= (PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_NEW
))
167 if (!(si
= pa_idxset_get_by_index(c
->sink_inputs
, idx
)))
173 for (r
= u
->rules
; r
; r
= r
->next
) {
174 if (!regexec(&r
->regex
, si
->name
, 0, NULL
, 0)) {
176 pa_log_debug(__FILE__
": changing volume of sink input '%s' to 0x%03x", si
->name
, r
->volume
);
177 pa_cvolume_set(&cv
, r
->volume
, si
->sample_spec
.channels
);
178 pa_sink_input_set_volume(si
, &cv
);
183 int pa__init(pa_core
*c
, pa_module
*m
) {
184 pa_modargs
*ma
= NULL
;
188 if (!(ma
= pa_modargs_new(m
->argument
, valid_modargs
))) {
189 pa_log(__FILE__
": Failed to parse module arguments");
193 u
= pa_xmalloc(sizeof(struct userdata
));
195 u
->subscription
= NULL
;
198 if (load_rules(u
, pa_modargs_get_value(ma
, "table", NULL
)) < 0)
201 u
->subscription
= pa_subscription_new(c
, PA_SUBSCRIPTION_MASK_SINK_INPUT
, callback
, u
);
214 void pa__done(pa_core
*c
, pa_module
*m
) {
219 if (!(u
= m
->userdata
))
223 pa_subscription_free(u
->subscription
);
225 for (r
= u
->rules
; r
; r
= n
) {