}
static void route_sink_input(struct userdata *u, pa_sink_input *si) {
+ const char *filter_device;
const char *role;
uint32_t role_index, device_index;
pa_sink *sink;
if (!si->sink)
return;
+ /* If module-filter-apply has loaded a filter for the stream, let's not
+ * break the filtering. The pa_streq() check is needed, because
+ * module-filter-apply doesn't unset the property when the stream moves
+ * away from the filter device for whatever reason (this is ugly, but
+ * easier to do this way than appropriately unsetting the property). */
+ filter_device = pa_proplist_gets(si->proplist, "module-filter-apply.filter_device");
+ if (filter_device && pa_streq(filter_device, si->sink->name))
+ return;
+
/* It might happen that a stream and a sink are set up at the
same time, in which case we want to make sure we don't
interfere with that */
}
static void route_source_output(struct userdata *u, pa_source_output *so) {
+ const char *filter_device;
const char *role;
uint32_t role_index, device_index;
pa_source *source;
if (!so->source)
return;
+ /* If module-filter-apply has loaded a filter for the stream, let's not
+ * break the filtering. The pa_streq() check is needed, because
+ * module-filter-apply doesn't unset the property when the stream moves
+ * away from the filter device for whatever reason (this is ugly, but
+ * easier to do this way than appropriately unsetting the property). */
+ filter_device = pa_proplist_gets(so->proplist, "module-filter-apply.filter_device");
+ if (filter_device && pa_streq(filter_device, so->source->name))
+ return;
+
/* It might happen that a stream and a source are set up at the
same time, in which case we want to make sure we don't
interfere with that */
}
static int do_move(pa_object *obj, pa_object *parent, bool restore, bool is_input) {
- if (is_input)
+ if (is_input) {
+ if (!restore) {
+ char *old_value;
+
+ if (pa_proplist_contains(PA_SINK_INPUT(obj)->proplist, "module-filter-apply.filter_device")) {
+ old_value = pa_xstrdup(pa_proplist_gets(PA_SINK_INPUT(obj)->proplist, "module-filter-apply.filter_device"));
+ if (!old_value)
+ old_value = pa_xstrdup("(data)");
+ } else
+ old_value = pa_xstrdup("(unset)");
+
+ if (!pa_streq(PA_SINK(parent)->name, old_value)) {
+ pa_proplist *pl;
+
+ pl = pa_proplist_new();
+ pa_proplist_sets(pl, "module-filter-apply.filter_device", PA_SINK(parent)->name);
+ pa_sink_input_update_proplist(PA_SINK_INPUT(obj), PA_UPDATE_REPLACE, pl);
+ pa_proplist_free(pl);
+ pa_log_debug("Sink input %u: proplist[module-filter-apply.filter_device]: %s -> %s",
+ PA_SINK_INPUT(obj)->index, old_value, PA_SINK(parent)->name);
+ }
+
+ pa_xfree(old_value);
+ }
+
return pa_sink_input_move_to(PA_SINK_INPUT(obj), PA_SINK(parent), restore);
- else
+ } else {
+ if (!restore) {
+ char *old_value;
+
+ if (pa_proplist_contains(PA_SOURCE_OUTPUT(obj)->proplist, "module-filter-apply.filter_device")) {
+ old_value = pa_xstrdup(pa_proplist_gets(PA_SOURCE_OUTPUT(obj)->proplist, "module-filter-apply.filter_device"));
+ if (!old_value)
+ old_value = pa_xstrdup("(data)");
+ } else
+ old_value = pa_xstrdup("(unset)");
+
+ if (!pa_streq(PA_SOURCE(parent)->name, old_value)) {
+ pa_proplist *pl;
+
+ pl = pa_proplist_new();
+ pa_proplist_sets(pl, "module-filter-apply.filter_device", PA_SOURCE(parent)->name);
+ pa_source_output_update_proplist(PA_SOURCE_OUTPUT(obj), PA_UPDATE_REPLACE, pl);
+ pa_proplist_free(pl);
+ pa_log_debug("Source output %u: proplist[module-filter-apply.filter_device]: %s -> %s",
+ PA_SOURCE_OUTPUT(obj)->index, old_value, PA_SOURCE(parent)->name);
+ }
+
+ pa_xfree(old_value);
+ }
+
return pa_source_output_move_to(PA_SOURCE_OUTPUT(obj), PA_SOURCE(parent), restore);
+ }
}
static void move_object_for_filter(pa_object *o, struct filter* filter, bool restore, bool is_sink_input) {