summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorTanu Kaskinen <tanuk@iki.fi>2016-12-08 01:59:04 +0200
committerTanu Kaskinen <tanuk@iki.fi>2016-12-20 01:38:17 +0200
commitd404d8d1abc7c92d62e3edf27dd36fac882852ca (patch)
tree9d3df2a7aba20368c298cce2976008e94eb54305 /src
parentaf45c0e3cd5808aac712836e58a8e65189aa60a1 (diff)
sink, source: remove some assumptions about stream "attached" state
Streams are detached when they are removed or moved away from a device, or when a filter device that they're connected to is removed or moved. If these cases overlap, a crash will happen due to "double-detaching". This can happen if a filter sink is removed, and a stream connected to that filter sink removes itself when its sink goes away. Here are the steps in more detail: When a filter sink is unloaded, first it will unlink its own sink input. This will cause the filter sink's input to be detached. The filter sink propagates the detachment to all inputs connected to it using pa_sink_detach_within_thread(). After the filter sink is done unlinking its own sink input, it will unlink the sink. This will cause at least module-combine-sink to remove its sink input if it had one connected to the removed filter sink. When the combine sink removes its sink input, that input will get detached again, and a crash follows. We can relax the assertions a bit, and skip the detach() call if the sink input is already detached. I think a better fix would be to unlink the sink before the sink input when unloading a filter sink - that way we could avoid the double-detaching - but that would be a much more complicated change. I decided to go with this simple fix for now. BugLink: https://bugs.freedesktop.org/show_bug.cgi?id=98617
Diffstat (limited to 'src')
-rw-r--r--src/pulsecore/sink.c27
-rw-r--r--src/pulsecore/source.c18
2 files changed, 25 insertions, 20 deletions
diff --git a/src/pulsecore/sink.c b/src/pulsecore/sink.c
index 475e03bd0..4031bcd04 100644
--- a/src/pulsecore/sink.c
+++ b/src/pulsecore/sink.c
@@ -2536,11 +2536,12 @@ int pa_sink_process_msg(pa_msgobject *o, int code, void *userdata, int64_t offse
* sink input handling a few lines down at
* PA_SINK_MESSAGE_START_MOVE, too. */
- pa_assert(i->thread_info.attached);
- i->thread_info.attached = false;
+ if (i->thread_info.attached) {
+ i->thread_info.attached = false;
- if (i->detach)
- i->detach(i);
+ if (i->detach)
+ i->detach(i);
+ }
pa_sink_input_set_state_within_thread(i, i->state);
@@ -2635,11 +2636,12 @@ int pa_sink_process_msg(pa_msgobject *o, int code, void *userdata, int64_t offse
}
}
- pa_assert(i->thread_info.attached);
- i->thread_info.attached = false;
+ if (i->thread_info.attached) {
+ i->thread_info.attached = false;
- if (i->detach)
- i->detach(i);
+ if (i->detach)
+ i->detach(i);
+ }
/* Let's remove the sink input ...*/
pa_hashmap_remove_and_free(s->thread_info.inputs, PA_UINT32_TO_PTR(i->index));
@@ -2932,11 +2934,12 @@ void pa_sink_detach_within_thread(pa_sink *s) {
pa_assert(PA_SINK_IS_LINKED(s->thread_info.state));
PA_HASHMAP_FOREACH(i, s->thread_info.inputs, state) {
- pa_assert(i->thread_info.attached);
- i->thread_info.attached = false;
+ if (i->thread_info.attached) {
+ i->thread_info.attached = false;
- if (i->detach)
- i->detach(i);
+ if (i->detach)
+ i->detach(i);
+ }
}
if (s->monitor_source)
diff --git a/src/pulsecore/source.c b/src/pulsecore/source.c
index 3c4a3cfdc..9bc7520ee 100644
--- a/src/pulsecore/source.c
+++ b/src/pulsecore/source.c
@@ -2062,11 +2062,12 @@ int pa_source_process_msg(pa_msgobject *object, int code, void *userdata, int64_
pa_source_output_set_state_within_thread(o, o->state);
- pa_assert(o->thread_info.attached);
- o->thread_info.attached = false;
+ if (o->thread_info.attached) {
+ o->thread_info.attached = false;
- if (o->detach)
- o->detach(o);
+ if (o->detach)
+ o->detach(o);
+ }
if (o->thread_info.direct_on_input) {
pa_hashmap_remove(o->thread_info.direct_on_input->thread_info.direct_outputs, PA_UINT32_TO_PTR(o->index));
@@ -2289,11 +2290,12 @@ void pa_source_detach_within_thread(pa_source *s) {
pa_assert(PA_SOURCE_IS_LINKED(s->thread_info.state));
PA_HASHMAP_FOREACH(o, s->thread_info.outputs, state) {
- pa_assert(o->thread_info.attached);
- o->thread_info.attached = false;
+ if (o->thread_info.attached) {
+ o->thread_info.attached = false;
- if (o->detach)
- o->detach(o);
+ if (o->detach)
+ o->detach(o);
+ }
}
}