mirror of
				https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
				synced 2025-11-03 09:01:50 -05:00 
			
		
		
		
	core: make fixed latency dynamically changeable
This of course makes the name 'fixed' a bit of a misnomer. However the definitions are now like this: fixed latency: the latency may change during runtime, but is solely controlled by the backend, the client has no influence. dynamic latency: the latency may change during runtime, influenced by the requests of the clients. i.e. fixed vs. dynamic is from the perspective of the client.
This commit is contained in:
		
							parent
							
								
									4eb59fb90e
								
							
						
					
					
						commit
						350a2bc846
					
				
					 10 changed files with 212 additions and 52 deletions
				
			
		| 
						 | 
					@ -881,7 +881,7 @@ static int sink_process_msg(pa_msgobject *o, int code, void *data, int64_t offse
 | 
				
			||||||
                *((pa_usec_t*) data) = wi > ri ? wi - ri : 0;
 | 
					                *((pa_usec_t*) data) = wi > ri ? wi - ri : 0;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            *((pa_usec_t*) data) += u->sink->fixed_latency;
 | 
					            *((pa_usec_t*) data) += u->sink->thread_info.fixed_latency;
 | 
				
			||||||
            return 0;
 | 
					            return 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
| 
						 | 
					@ -943,7 +943,7 @@ static int source_process_msg(pa_msgobject *o, int code, void *data, int64_t off
 | 
				
			||||||
            wi = pa_smoother_get(u->read_smoother, pa_rtclock_now());
 | 
					            wi = pa_smoother_get(u->read_smoother, pa_rtclock_now());
 | 
				
			||||||
            ri = pa_bytes_to_usec(u->read_index, &u->sample_spec);
 | 
					            ri = pa_bytes_to_usec(u->read_index, &u->sample_spec);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            *((pa_usec_t*) data) = (wi > ri ? wi - ri : 0) + u->source->fixed_latency;
 | 
					            *((pa_usec_t*) data) = (wi > ri ? wi - ri : 0) + u->source->thread_info.fixed_latency;
 | 
				
			||||||
            return 0;
 | 
					            return 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -296,7 +296,7 @@ char *pa_sink_list_to_string(pa_core *c) {
 | 
				
			||||||
            pa_strbuf_printf(
 | 
					            pa_strbuf_printf(
 | 
				
			||||||
                    s,
 | 
					                    s,
 | 
				
			||||||
                    "\tfixed latency: %0.2f ms\n",
 | 
					                    "\tfixed latency: %0.2f ms\n",
 | 
				
			||||||
                    (double) pa_sink_get_requested_latency(sink) / PA_USEC_PER_MSEC);
 | 
					                    (double) pa_sink_get_fixed_latency(sink) / PA_USEC_PER_MSEC);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (sink->card)
 | 
					        if (sink->card)
 | 
				
			||||||
            pa_strbuf_printf(s, "\tcard: %u <%s>\n", sink->card->index, sink->card->name);
 | 
					            pa_strbuf_printf(s, "\tcard: %u <%s>\n", sink->card->index, sink->card->name);
 | 
				
			||||||
| 
						 | 
					@ -415,7 +415,7 @@ char *pa_source_list_to_string(pa_core *c) {
 | 
				
			||||||
            pa_strbuf_printf(
 | 
					            pa_strbuf_printf(
 | 
				
			||||||
                    s,
 | 
					                    s,
 | 
				
			||||||
                    "\tfixed latency: %0.2f ms\n",
 | 
					                    "\tfixed latency: %0.2f ms\n",
 | 
				
			||||||
                    (double) pa_source_get_requested_latency(source) / PA_USEC_PER_MSEC);
 | 
					                    (double) pa_source_get_fixed_latency(source) / PA_USEC_PER_MSEC);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (source->monitor_of)
 | 
					        if (source->monitor_of)
 | 
				
			||||||
            pa_strbuf_printf(s, "\tmonitor_of: %u\n", source->monitor_of->index);
 | 
					            pa_strbuf_printf(s, "\tmonitor_of: %u\n", source->monitor_of->index);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -114,6 +114,7 @@ static void reset_callbacks(pa_sink_input *i) {
 | 
				
			||||||
    i->update_max_request = NULL;
 | 
					    i->update_max_request = NULL;
 | 
				
			||||||
    i->update_sink_requested_latency = NULL;
 | 
					    i->update_sink_requested_latency = NULL;
 | 
				
			||||||
    i->update_sink_latency_range = NULL;
 | 
					    i->update_sink_latency_range = NULL;
 | 
				
			||||||
 | 
					    i->update_sink_fixed_latency = NULL;
 | 
				
			||||||
    i->attach = NULL;
 | 
					    i->attach = NULL;
 | 
				
			||||||
    i->detach = NULL;
 | 
					    i->detach = NULL;
 | 
				
			||||||
    i->suspend = NULL;
 | 
					    i->suspend = NULL;
 | 
				
			||||||
| 
						 | 
					@ -851,13 +852,13 @@ pa_usec_t pa_sink_input_set_requested_latency_within_thread(pa_sink_input *i, pa
 | 
				
			||||||
    pa_sink_input_assert_io_context(i);
 | 
					    pa_sink_input_assert_io_context(i);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(i->sink->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
					    if (!(i->sink->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
				
			||||||
        usec = i->sink->fixed_latency;
 | 
					        usec = i->sink->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (usec != (pa_usec_t) -1)
 | 
					    if (usec != (pa_usec_t) -1)
 | 
				
			||||||
        usec = PA_CLAMP(usec, i->sink->thread_info.min_latency, i->sink->thread_info.max_latency);
 | 
					        usec = PA_CLAMP(usec, i->sink->thread_info.min_latency, i->sink->thread_info.max_latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    i->thread_info.requested_sink_latency = usec;
 | 
					    i->thread_info.requested_sink_latency = usec;
 | 
				
			||||||
    pa_sink_invalidate_requested_latency(i->sink);
 | 
					    pa_sink_invalidate_requested_latency(i->sink, TRUE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return usec;
 | 
					    return usec;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -877,7 +878,7 @@ pa_usec_t pa_sink_input_set_requested_latency(pa_sink_input *i, pa_usec_t usec)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (i->sink) {
 | 
					    if (i->sink) {
 | 
				
			||||||
        if (!(i->sink->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
					        if (!(i->sink->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
				
			||||||
            usec = i->sink->fixed_latency;
 | 
					            usec = pa_sink_get_fixed_latency(i->sink);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (usec != (pa_usec_t) -1) {
 | 
					        if (usec != (pa_usec_t) -1) {
 | 
				
			||||||
            pa_usec_t min_latency, max_latency;
 | 
					            pa_usec_t min_latency, max_latency;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -138,6 +138,10 @@ struct pa_sink_input {
 | 
				
			||||||
     * from IO context. */
 | 
					     * from IO context. */
 | 
				
			||||||
    void (*update_sink_latency_range) (pa_sink_input *i); /* may be NULL */
 | 
					    void (*update_sink_latency_range) (pa_sink_input *i); /* may be NULL */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Called whenver the fixed latency of the sink changes, if there
 | 
				
			||||||
 | 
					     * is one. Called from IO context. */
 | 
				
			||||||
 | 
					    void (*update_sink_fixed_latency) (pa_sink_input *i); /* may be NULL */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* If non-NULL this function is called when the input is first
 | 
					    /* If non-NULL this function is called when the input is first
 | 
				
			||||||
     * connected to a sink or when the rtpoll/asyncmsgq fields
 | 
					     * connected to a sink or when the rtpoll/asyncmsgq fields
 | 
				
			||||||
     * change. You usually don't need to implement this function
 | 
					     * change. You usually don't need to implement this function
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -256,8 +256,6 @@ pa_sink* pa_sink_new(
 | 
				
			||||||
    s->muted = data->muted;
 | 
					    s->muted = data->muted;
 | 
				
			||||||
    s->refresh_volume = s->refresh_muted = FALSE;
 | 
					    s->refresh_volume = s->refresh_muted = FALSE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    s->fixed_latency = flags & PA_SINK_DYNAMIC_LATENCY ? 0 : DEFAULT_FIXED_LATENCY;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    reset_callbacks(s);
 | 
					    reset_callbacks(s);
 | 
				
			||||||
    s->userdata = NULL;
 | 
					    s->userdata = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -307,6 +305,7 @@ pa_sink* pa_sink_new(
 | 
				
			||||||
    s->thread_info.requested_latency = 0;
 | 
					    s->thread_info.requested_latency = 0;
 | 
				
			||||||
    s->thread_info.min_latency = ABSOLUTE_MIN_LATENCY;
 | 
					    s->thread_info.min_latency = ABSOLUTE_MIN_LATENCY;
 | 
				
			||||||
    s->thread_info.max_latency = ABSOLUTE_MAX_LATENCY;
 | 
					    s->thread_info.max_latency = ABSOLUTE_MAX_LATENCY;
 | 
				
			||||||
 | 
					    s->thread_info.fixed_latency = flags & PA_SINK_DYNAMIC_LATENCY ? 0 : DEFAULT_FIXED_LATENCY;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert_se(pa_idxset_put(core->sinks, s, &s->index) >= 0);
 | 
					    pa_assert_se(pa_idxset_put(core->sinks, s, &s->index) >= 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -349,6 +348,7 @@ pa_sink* pa_sink_new(
 | 
				
			||||||
    s->monitor_source->monitor_of = s;
 | 
					    s->monitor_source->monitor_of = s;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_source_set_latency_range(s->monitor_source, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
					    pa_source_set_latency_range(s->monitor_source, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
				
			||||||
 | 
					    pa_source_set_fixed_latency(s->monitor_source, s->thread_info.fixed_latency);
 | 
				
			||||||
    pa_source_set_max_rewind(s->monitor_source, s->thread_info.max_rewind);
 | 
					    pa_source_set_max_rewind(s->monitor_source, s->thread_info.max_rewind);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return s;
 | 
					    return s;
 | 
				
			||||||
| 
						 | 
					@ -438,11 +438,11 @@ void pa_sink_put(pa_sink* s) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert((s->flags & PA_SINK_HW_VOLUME_CTRL) || (s->base_volume == PA_VOLUME_NORM && s->flags & PA_SINK_DECIBEL_VOLUME));
 | 
					    pa_assert((s->flags & PA_SINK_HW_VOLUME_CTRL) || (s->base_volume == PA_VOLUME_NORM && s->flags & PA_SINK_DECIBEL_VOLUME));
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SINK_DECIBEL_VOLUME) || s->n_volume_steps == PA_VOLUME_NORM+1);
 | 
					    pa_assert(!(s->flags & PA_SINK_DECIBEL_VOLUME) || s->n_volume_steps == PA_VOLUME_NORM+1);
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SINK_DYNAMIC_LATENCY) == (s->fixed_latency != 0));
 | 
					    pa_assert(!(s->flags & PA_SINK_DYNAMIC_LATENCY) == (s->thread_info.fixed_latency != 0));
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SINK_LATENCY) == !(s->monitor_source->flags & PA_SOURCE_LATENCY));
 | 
					    pa_assert(!(s->flags & PA_SINK_LATENCY) == !(s->monitor_source->flags & PA_SOURCE_LATENCY));
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SINK_DYNAMIC_LATENCY) == !(s->monitor_source->flags & PA_SOURCE_DYNAMIC_LATENCY));
 | 
					    pa_assert(!(s->flags & PA_SINK_DYNAMIC_LATENCY) == !(s->monitor_source->flags & PA_SOURCE_DYNAMIC_LATENCY));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert(s->monitor_source->fixed_latency == s->fixed_latency);
 | 
					    pa_assert(s->monitor_source->thread_info.fixed_latency == s->thread_info.fixed_latency);
 | 
				
			||||||
    pa_assert(s->monitor_source->thread_info.min_latency == s->thread_info.min_latency);
 | 
					    pa_assert(s->monitor_source->thread_info.min_latency == s->thread_info.min_latency);
 | 
				
			||||||
    pa_assert(s->monitor_source->thread_info.max_latency == s->thread_info.max_latency);
 | 
					    pa_assert(s->monitor_source->thread_info.max_latency == s->thread_info.max_latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1748,7 +1748,7 @@ int pa_sink_process_msg(pa_msgobject *o, int code, void *userdata, int64_t offse
 | 
				
			||||||
            if (pa_hashmap_remove(s->thread_info.inputs, PA_UINT32_TO_PTR(i->index)))
 | 
					            if (pa_hashmap_remove(s->thread_info.inputs, PA_UINT32_TO_PTR(i->index)))
 | 
				
			||||||
                pa_sink_input_unref(i);
 | 
					                pa_sink_input_unref(i);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            pa_sink_invalidate_requested_latency(s);
 | 
					            pa_sink_invalidate_requested_latency(s, TRUE);
 | 
				
			||||||
            pa_sink_request_rewind(s, (size_t) -1);
 | 
					            pa_sink_request_rewind(s, (size_t) -1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            /* In flat volume mode we need to update the volume as
 | 
					            /* In flat volume mode we need to update the volume as
 | 
				
			||||||
| 
						 | 
					@ -1794,7 +1794,7 @@ int pa_sink_process_msg(pa_msgobject *o, int code, void *userdata, int64_t offse
 | 
				
			||||||
            if (pa_hashmap_remove(s->thread_info.inputs, PA_UINT32_TO_PTR(i->index)))
 | 
					            if (pa_hashmap_remove(s->thread_info.inputs, PA_UINT32_TO_PTR(i->index)))
 | 
				
			||||||
                pa_sink_input_unref(i);
 | 
					                pa_sink_input_unref(i);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            pa_sink_invalidate_requested_latency(s);
 | 
					            pa_sink_invalidate_requested_latency(s, TRUE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            pa_log_debug("Requesting rewind due to started move");
 | 
					            pa_log_debug("Requesting rewind due to started move");
 | 
				
			||||||
            pa_sink_request_rewind(s, (size_t) -1);
 | 
					            pa_sink_request_rewind(s, (size_t) -1);
 | 
				
			||||||
| 
						 | 
					@ -1946,6 +1946,16 @@ int pa_sink_process_msg(pa_msgobject *o, int code, void *userdata, int64_t offse
 | 
				
			||||||
            return 0;
 | 
					            return 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case PA_SINK_MESSAGE_GET_FIXED_LATENCY:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            *((pa_usec_t*) userdata) = s->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					            return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case PA_SINK_MESSAGE_SET_FIXED_LATENCY:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            pa_sink_set_fixed_latency_within_thread(s, (pa_usec_t) offset);
 | 
				
			||||||
 | 
					            return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        case PA_SINK_MESSAGE_GET_MAX_REWIND:
 | 
					        case PA_SINK_MESSAGE_GET_MAX_REWIND:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            *((size_t*) userdata) = s->thread_info.max_rewind;
 | 
					            *((size_t*) userdata) = s->thread_info.max_rewind;
 | 
				
			||||||
| 
						 | 
					@ -2082,13 +2092,12 @@ pa_usec_t pa_sink_get_requested_latency_within_thread(pa_sink *s) {
 | 
				
			||||||
    pa_sink_assert_io_context(s);
 | 
					    pa_sink_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(s->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
					    if (!(s->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
				
			||||||
        return PA_CLAMP(s->fixed_latency, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
					        return PA_CLAMP(s->thread_info.fixed_latency, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (s->thread_info.requested_latency_valid)
 | 
					    if (s->thread_info.requested_latency_valid)
 | 
				
			||||||
        return s->thread_info.requested_latency;
 | 
					        return s->thread_info.requested_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    while ((i = pa_hashmap_iterate(s->thread_info.inputs, &state, NULL)))
 | 
					    PA_HASHMAP_FOREACH(i, s->thread_info.inputs, state)
 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (i->thread_info.requested_sink_latency != (pa_usec_t) -1 &&
 | 
					        if (i->thread_info.requested_sink_latency != (pa_usec_t) -1 &&
 | 
				
			||||||
            (result == (pa_usec_t) -1 || result > i->thread_info.requested_sink_latency))
 | 
					            (result == (pa_usec_t) -1 || result > i->thread_info.requested_sink_latency))
 | 
				
			||||||
            result = i->thread_info.requested_sink_latency;
 | 
					            result = i->thread_info.requested_sink_latency;
 | 
				
			||||||
| 
						 | 
					@ -2190,17 +2199,17 @@ void pa_sink_set_max_request(pa_sink *s, size_t max_request) {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from IO thread */
 | 
					/* Called from IO thread */
 | 
				
			||||||
void pa_sink_invalidate_requested_latency(pa_sink *s) {
 | 
					void pa_sink_invalidate_requested_latency(pa_sink *s, pa_bool_t dynamic) {
 | 
				
			||||||
    pa_sink_input *i;
 | 
					    pa_sink_input *i;
 | 
				
			||||||
    void *state = NULL;
 | 
					    void *state = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_sink_assert_ref(s);
 | 
					    pa_sink_assert_ref(s);
 | 
				
			||||||
    pa_sink_assert_io_context(s);
 | 
					    pa_sink_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(s->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
					    if ((s->flags & PA_SINK_DYNAMIC_LATENCY))
 | 
				
			||||||
        return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        s->thread_info.requested_latency_valid = FALSE;
 | 
					        s->thread_info.requested_latency_valid = FALSE;
 | 
				
			||||||
 | 
					    else if (dynamic)
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (PA_SINK_IS_LINKED(s->thread_info.state)) {
 | 
					    if (PA_SINK_IS_LINKED(s->thread_info.state)) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2295,16 +2304,20 @@ void pa_sink_set_latency_range_within_thread(pa_sink *s, pa_usec_t min_latency,
 | 
				
			||||||
                i->update_sink_latency_range(i);
 | 
					                i->update_sink_latency_range(i);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_sink_invalidate_requested_latency(s);
 | 
					    pa_sink_invalidate_requested_latency(s, FALSE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_source_set_latency_range_within_thread(s->monitor_source, min_latency, max_latency);
 | 
					    pa_source_set_latency_range_within_thread(s->monitor_source, min_latency, max_latency);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from main thread, before the sink is put */
 | 
					/* Called from main thread */
 | 
				
			||||||
void pa_sink_set_fixed_latency(pa_sink *s, pa_usec_t latency) {
 | 
					void pa_sink_set_fixed_latency(pa_sink *s, pa_usec_t latency) {
 | 
				
			||||||
    pa_sink_assert_ref(s);
 | 
					    pa_sink_assert_ref(s);
 | 
				
			||||||
    pa_assert_ctl_context();
 | 
					    pa_assert_ctl_context();
 | 
				
			||||||
    pa_assert(pa_sink_get_state(s) == PA_SINK_INIT);
 | 
					
 | 
				
			||||||
 | 
					    if (s->flags & PA_SINK_DYNAMIC_LATENCY) {
 | 
				
			||||||
 | 
					        pa_assert(latency == 0);
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (latency < ABSOLUTE_MIN_LATENCY)
 | 
					    if (latency < ABSOLUTE_MIN_LATENCY)
 | 
				
			||||||
        latency = ABSOLUTE_MIN_LATENCY;
 | 
					        latency = ABSOLUTE_MIN_LATENCY;
 | 
				
			||||||
| 
						 | 
					@ -2312,10 +2325,64 @@ void pa_sink_set_fixed_latency(pa_sink *s, pa_usec_t latency) {
 | 
				
			||||||
    if (latency > ABSOLUTE_MAX_LATENCY)
 | 
					    if (latency > ABSOLUTE_MAX_LATENCY)
 | 
				
			||||||
        latency = ABSOLUTE_MAX_LATENCY;
 | 
					        latency = ABSOLUTE_MAX_LATENCY;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    s->fixed_latency = latency;
 | 
					    if (PA_SINK_IS_LINKED(s->state))
 | 
				
			||||||
 | 
					        pa_assert_se(pa_asyncmsgq_send(s->asyncmsgq, PA_MSGOBJECT(s), PA_SINK_MESSAGE_SET_FIXED_LATENCY, NULL, (int64_t) latency, NULL) == 0);
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        s->thread_info.fixed_latency = latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_source_set_fixed_latency(s->monitor_source, latency);
 | 
					    pa_source_set_fixed_latency(s->monitor_source, latency);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Called from main thread */
 | 
				
			||||||
 | 
					pa_usec_t pa_sink_get_fixed_latency(pa_sink *s) {
 | 
				
			||||||
 | 
					    pa_usec_t latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_sink_assert_ref(s);
 | 
				
			||||||
 | 
					    pa_assert_ctl_context();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->flags & PA_SINK_DYNAMIC_LATENCY)
 | 
				
			||||||
 | 
					        return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (PA_SINK_IS_LINKED(s->state))
 | 
				
			||||||
 | 
					        pa_assert_se(pa_asyncmsgq_send(s->asyncmsgq, PA_MSGOBJECT(s), PA_SINK_MESSAGE_GET_FIXED_LATENCY, &latency, 0, NULL) == 0);
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        latency = s->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return latency;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Called from IO thread */
 | 
				
			||||||
 | 
					void pa_sink_set_fixed_latency_within_thread(pa_sink *s, pa_usec_t latency) {
 | 
				
			||||||
 | 
					    pa_sink_assert_ref(s);
 | 
				
			||||||
 | 
					    pa_sink_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->flags & PA_SINK_DYNAMIC_LATENCY) {
 | 
				
			||||||
 | 
					        pa_assert(latency == 0);
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_assert(latency >= ABSOLUTE_MIN_LATENCY);
 | 
				
			||||||
 | 
					    pa_assert(latency <= ABSOLUTE_MAX_LATENCY);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->thread_info.fixed_latency == latency)
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    s->thread_info.fixed_latency = latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (PA_SINK_IS_LINKED(s->thread_info.state)) {
 | 
				
			||||||
 | 
					        pa_sink_input *i;
 | 
				
			||||||
 | 
					        void *state = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        PA_HASHMAP_FOREACH(i, s->thread_info.inputs, state)
 | 
				
			||||||
 | 
					            if (i->update_sink_fixed_latency)
 | 
				
			||||||
 | 
					                i->update_sink_fixed_latency(i);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_sink_invalidate_requested_latency(s, FALSE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_source_set_fixed_latency_within_thread(s->monitor_source, latency);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from main context */
 | 
					/* Called from main context */
 | 
				
			||||||
size_t pa_sink_get_max_rewind(pa_sink *s) {
 | 
					size_t pa_sink_get_max_rewind(pa_sink *s) {
 | 
				
			||||||
    size_t r;
 | 
					    size_t r;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -105,8 +105,6 @@ struct pa_sink {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_memchunk silence;
 | 
					    pa_memchunk silence;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_usec_t fixed_latency; /* for sinks with PA_SINK_DYNAMIC_LATENCY this is 0 */
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    pa_hashmap *ports;
 | 
					    pa_hashmap *ports;
 | 
				
			||||||
    pa_device_port *active_port;
 | 
					    pa_device_port *active_port;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -160,6 +158,9 @@ struct pa_sink {
 | 
				
			||||||
        pa_cvolume soft_volume;
 | 
					        pa_cvolume soft_volume;
 | 
				
			||||||
        pa_bool_t soft_muted:1;
 | 
					        pa_bool_t soft_muted:1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /* The requested latency is used for dynamic latency
 | 
				
			||||||
 | 
					         * sinks. For fixed latency sinks it is always identical to
 | 
				
			||||||
 | 
					         * the fixed_latency. See below. */
 | 
				
			||||||
        pa_bool_t requested_latency_valid:1;
 | 
					        pa_bool_t requested_latency_valid:1;
 | 
				
			||||||
        pa_usec_t requested_latency;
 | 
					        pa_usec_t requested_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -175,8 +176,15 @@ struct pa_sink {
 | 
				
			||||||
        size_t rewind_nbytes;
 | 
					        size_t rewind_nbytes;
 | 
				
			||||||
        pa_bool_t rewind_requested;
 | 
					        pa_bool_t rewind_requested;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /* Both dynamic and fixed latencies will be clamped to this
 | 
				
			||||||
 | 
					         * range. */
 | 
				
			||||||
        pa_usec_t min_latency; /* we won't go below this latency */
 | 
					        pa_usec_t min_latency; /* we won't go below this latency */
 | 
				
			||||||
        pa_usec_t max_latency; /* An upper limit for the latencies */
 | 
					        pa_usec_t max_latency; /* An upper limit for the latencies */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /* 'Fixed' simply means that the latency is exclusively
 | 
				
			||||||
 | 
					         * decided on by the sink, and the clients have no influence
 | 
				
			||||||
 | 
					         * in changing it */
 | 
				
			||||||
 | 
					        pa_usec_t fixed_latency; /* for sinks with PA_SINK_DYNAMIC_LATENCY this is 0 */
 | 
				
			||||||
    } thread_info;
 | 
					    } thread_info;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    void *userdata;
 | 
					    void *userdata;
 | 
				
			||||||
| 
						 | 
					@ -202,6 +210,8 @@ typedef enum pa_sink_message {
 | 
				
			||||||
    PA_SINK_MESSAGE_DETACH,
 | 
					    PA_SINK_MESSAGE_DETACH,
 | 
				
			||||||
    PA_SINK_MESSAGE_SET_LATENCY_RANGE,
 | 
					    PA_SINK_MESSAGE_SET_LATENCY_RANGE,
 | 
				
			||||||
    PA_SINK_MESSAGE_GET_LATENCY_RANGE,
 | 
					    PA_SINK_MESSAGE_GET_LATENCY_RANGE,
 | 
				
			||||||
 | 
					    PA_SINK_MESSAGE_SET_FIXED_LATENCY,
 | 
				
			||||||
 | 
					    PA_SINK_MESSAGE_GET_FIXED_LATENCY,
 | 
				
			||||||
    PA_SINK_MESSAGE_GET_MAX_REWIND,
 | 
					    PA_SINK_MESSAGE_GET_MAX_REWIND,
 | 
				
			||||||
    PA_SINK_MESSAGE_GET_MAX_REQUEST,
 | 
					    PA_SINK_MESSAGE_GET_MAX_REQUEST,
 | 
				
			||||||
    PA_SINK_MESSAGE_SET_MAX_REWIND,
 | 
					    PA_SINK_MESSAGE_SET_MAX_REWIND,
 | 
				
			||||||
| 
						 | 
					@ -282,6 +292,7 @@ pa_bool_t pa_device_init_intended_roles(pa_proplist *p);
 | 
				
			||||||
pa_usec_t pa_sink_get_latency(pa_sink *s);
 | 
					pa_usec_t pa_sink_get_latency(pa_sink *s);
 | 
				
			||||||
pa_usec_t pa_sink_get_requested_latency(pa_sink *s);
 | 
					pa_usec_t pa_sink_get_requested_latency(pa_sink *s);
 | 
				
			||||||
void pa_sink_get_latency_range(pa_sink *s, pa_usec_t *min_latency, pa_usec_t *max_latency);
 | 
					void pa_sink_get_latency_range(pa_sink *s, pa_usec_t *min_latency, pa_usec_t *max_latency);
 | 
				
			||||||
 | 
					pa_usec_t pa_sink_get_fixed_latency(pa_sink *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
size_t pa_sink_get_max_rewind(pa_sink *s);
 | 
					size_t pa_sink_get_max_rewind(pa_sink *s);
 | 
				
			||||||
size_t pa_sink_get_max_request(pa_sink *s);
 | 
					size_t pa_sink_get_max_request(pa_sink *s);
 | 
				
			||||||
| 
						 | 
					@ -333,12 +344,13 @@ void pa_sink_set_max_rewind_within_thread(pa_sink *s, size_t max_rewind);
 | 
				
			||||||
void pa_sink_set_max_request_within_thread(pa_sink *s, size_t max_request);
 | 
					void pa_sink_set_max_request_within_thread(pa_sink *s, size_t max_request);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_sink_set_latency_range_within_thread(pa_sink *s, pa_usec_t min_latency, pa_usec_t max_latency);
 | 
					void pa_sink_set_latency_range_within_thread(pa_sink *s, pa_usec_t min_latency, pa_usec_t max_latency);
 | 
				
			||||||
 | 
					void pa_sink_set_fixed_latency_within_thread(pa_sink *s, pa_usec_t latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*** To be called exclusively by sink input drivers, from IO context */
 | 
					/*** To be called exclusively by sink input drivers, from IO context */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_sink_request_rewind(pa_sink*s, size_t nbytes);
 | 
					void pa_sink_request_rewind(pa_sink*s, size_t nbytes);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_sink_invalidate_requested_latency(pa_sink *s);
 | 
					void pa_sink_invalidate_requested_latency(pa_sink *s, pa_bool_t dynamic);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
pa_usec_t pa_sink_get_latency_within_thread(pa_sink *s);
 | 
					pa_usec_t pa_sink_get_latency_within_thread(pa_sink *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -84,6 +84,7 @@ static void reset_callbacks(pa_source_output *o) {
 | 
				
			||||||
    o->update_max_rewind = NULL;
 | 
					    o->update_max_rewind = NULL;
 | 
				
			||||||
    o->update_source_requested_latency = NULL;
 | 
					    o->update_source_requested_latency = NULL;
 | 
				
			||||||
    o->update_source_latency_range = NULL;
 | 
					    o->update_source_latency_range = NULL;
 | 
				
			||||||
 | 
					    o->update_source_fixed_latency = NULL;
 | 
				
			||||||
    o->attach = NULL;
 | 
					    o->attach = NULL;
 | 
				
			||||||
    o->detach = NULL;
 | 
					    o->detach = NULL;
 | 
				
			||||||
    o->suspend = NULL;
 | 
					    o->suspend = NULL;
 | 
				
			||||||
| 
						 | 
					@ -561,13 +562,13 @@ pa_usec_t pa_source_output_set_requested_latency_within_thread(pa_source_output
 | 
				
			||||||
    pa_source_output_assert_io_context(o);
 | 
					    pa_source_output_assert_io_context(o);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(o->source->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
					    if (!(o->source->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
				
			||||||
        usec = o->source->fixed_latency;
 | 
					        usec = o->source->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (usec != (pa_usec_t) -1)
 | 
					    if (usec != (pa_usec_t) -1)
 | 
				
			||||||
        usec = PA_CLAMP(usec, o->source->thread_info.min_latency, o->source->thread_info.max_latency);
 | 
					        usec = PA_CLAMP(usec, o->source->thread_info.min_latency, o->source->thread_info.max_latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    o->thread_info.requested_source_latency = usec;
 | 
					    o->thread_info.requested_source_latency = usec;
 | 
				
			||||||
    pa_source_invalidate_requested_latency(o->source);
 | 
					    pa_source_invalidate_requested_latency(o->source, TRUE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return usec;
 | 
					    return usec;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -587,7 +588,7 @@ pa_usec_t pa_source_output_set_requested_latency(pa_source_output *o, pa_usec_t
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (o->source) {
 | 
					    if (o->source) {
 | 
				
			||||||
        if (!(o->source->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
					        if (!(o->source->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
				
			||||||
            usec = o->source->fixed_latency;
 | 
					            usec = pa_source_get_fixed_latency(o->source);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (usec != (pa_usec_t) -1) {
 | 
					        if (usec != (pa_usec_t) -1) {
 | 
				
			||||||
            pa_usec_t min_latency, max_latency;
 | 
					            pa_usec_t min_latency, max_latency;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -109,6 +109,10 @@ struct pa_source_output {
 | 
				
			||||||
     * from IO context. */
 | 
					     * from IO context. */
 | 
				
			||||||
    void (*update_source_latency_range) (pa_source_output *o); /* may be NULL */
 | 
					    void (*update_source_latency_range) (pa_source_output *o); /* may be NULL */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Called whenver the fixed latency of the source changes, if there
 | 
				
			||||||
 | 
					     * is one. Called from IO context. */
 | 
				
			||||||
 | 
					    void (*update_source_fixed_latency) (pa_source_output *i); /* may be NULL */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* If non-NULL this function is called when the output is first
 | 
					    /* If non-NULL this function is called when the output is first
 | 
				
			||||||
     * connected to a source. Called from IO thread context */
 | 
					     * connected to a source. Called from IO thread context */
 | 
				
			||||||
    void (*attach) (pa_source_output *o);           /* may be NULL */
 | 
					    void (*attach) (pa_source_output *o);           /* may be NULL */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -226,8 +226,6 @@ pa_source* pa_source_new(
 | 
				
			||||||
    s->muted = data->muted;
 | 
					    s->muted = data->muted;
 | 
				
			||||||
    s->refresh_volume = s->refresh_muted = FALSE;
 | 
					    s->refresh_volume = s->refresh_muted = FALSE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    s->fixed_latency = flags & PA_SOURCE_DYNAMIC_LATENCY ? 0 : DEFAULT_FIXED_LATENCY;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    reset_callbacks(s);
 | 
					    reset_callbacks(s);
 | 
				
			||||||
    s->userdata = NULL;
 | 
					    s->userdata = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -274,6 +272,7 @@ pa_source* pa_source_new(
 | 
				
			||||||
    s->thread_info.requested_latency = 0;
 | 
					    s->thread_info.requested_latency = 0;
 | 
				
			||||||
    s->thread_info.min_latency = ABSOLUTE_MIN_LATENCY;
 | 
					    s->thread_info.min_latency = ABSOLUTE_MIN_LATENCY;
 | 
				
			||||||
    s->thread_info.max_latency = ABSOLUTE_MAX_LATENCY;
 | 
					    s->thread_info.max_latency = ABSOLUTE_MAX_LATENCY;
 | 
				
			||||||
 | 
					    s->thread_info.fixed_latency = flags & PA_SOURCE_DYNAMIC_LATENCY ? 0 : DEFAULT_FIXED_LATENCY;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert_se(pa_idxset_put(core->sources, s, &s->index) >= 0);
 | 
					    pa_assert_se(pa_idxset_put(core->sources, s, &s->index) >= 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -370,7 +369,7 @@ void pa_source_put(pa_source *s) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert((s->flags & PA_SOURCE_HW_VOLUME_CTRL) || (s->base_volume == PA_VOLUME_NORM && s->flags & PA_SOURCE_DECIBEL_VOLUME));
 | 
					    pa_assert((s->flags & PA_SOURCE_HW_VOLUME_CTRL) || (s->base_volume == PA_VOLUME_NORM && s->flags & PA_SOURCE_DECIBEL_VOLUME));
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SOURCE_DECIBEL_VOLUME) || s->n_volume_steps == PA_VOLUME_NORM+1);
 | 
					    pa_assert(!(s->flags & PA_SOURCE_DECIBEL_VOLUME) || s->n_volume_steps == PA_VOLUME_NORM+1);
 | 
				
			||||||
    pa_assert(!(s->flags & PA_SOURCE_DYNAMIC_LATENCY) == (s->fixed_latency != 0));
 | 
					    pa_assert(!(s->flags & PA_SOURCE_DYNAMIC_LATENCY) == (s->thread_info.fixed_latency != 0));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert_se(source_set_state(s, PA_SOURCE_IDLE) == 0);
 | 
					    pa_assert_se(source_set_state(s, PA_SOURCE_IDLE) == 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1037,7 +1036,7 @@ int pa_source_process_msg(pa_msgobject *object, int code, void *userdata, int64_
 | 
				
			||||||
            if (pa_hashmap_remove(s->thread_info.outputs, PA_UINT32_TO_PTR(o->index)))
 | 
					            if (pa_hashmap_remove(s->thread_info.outputs, PA_UINT32_TO_PTR(o->index)))
 | 
				
			||||||
                pa_source_output_unref(o);
 | 
					                pa_source_output_unref(o);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            pa_source_invalidate_requested_latency(s);
 | 
					            pa_source_invalidate_requested_latency(s, TRUE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            return 0;
 | 
					            return 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
| 
						 | 
					@ -1117,6 +1116,16 @@ int pa_source_process_msg(pa_msgobject *object, int code, void *userdata, int64_
 | 
				
			||||||
            return 0;
 | 
					            return 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case PA_SOURCE_MESSAGE_GET_FIXED_LATENCY:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            *((pa_usec_t*) userdata) = s->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					            return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case PA_SOURCE_MESSAGE_SET_FIXED_LATENCY:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            pa_source_set_fixed_latency_within_thread(s, (pa_usec_t) offset);
 | 
				
			||||||
 | 
					            return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        case PA_SOURCE_MESSAGE_GET_MAX_REWIND:
 | 
					        case PA_SOURCE_MESSAGE_GET_MAX_REWIND:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            *((size_t*) userdata) = s->thread_info.max_rewind;
 | 
					            *((size_t*) userdata) = s->thread_info.max_rewind;
 | 
				
			||||||
| 
						 | 
					@ -1223,13 +1232,12 @@ pa_usec_t pa_source_get_requested_latency_within_thread(pa_source *s) {
 | 
				
			||||||
    pa_source_assert_io_context(s);
 | 
					    pa_source_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(s->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
					    if (!(s->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
				
			||||||
        return PA_CLAMP(s->fixed_latency, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
					        return PA_CLAMP(s->thread_info.fixed_latency, s->thread_info.min_latency, s->thread_info.max_latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (s->thread_info.requested_latency_valid)
 | 
					    if (s->thread_info.requested_latency_valid)
 | 
				
			||||||
        return s->thread_info.requested_latency;
 | 
					        return s->thread_info.requested_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    while ((o = pa_hashmap_iterate(s->thread_info.outputs, &state, NULL)))
 | 
					    PA_HASHMAP_FOREACH(o, s->thread_info.outputs, state)
 | 
				
			||||||
 | 
					 | 
				
			||||||
        if (o->thread_info.requested_source_latency != (pa_usec_t) -1 &&
 | 
					        if (o->thread_info.requested_source_latency != (pa_usec_t) -1 &&
 | 
				
			||||||
            (result == (pa_usec_t) -1 || result > o->thread_info.requested_source_latency))
 | 
					            (result == (pa_usec_t) -1 || result > o->thread_info.requested_source_latency))
 | 
				
			||||||
            result = o->thread_info.requested_source_latency;
 | 
					            result = o->thread_info.requested_source_latency;
 | 
				
			||||||
| 
						 | 
					@ -1292,17 +1300,17 @@ void pa_source_set_max_rewind(pa_source *s, size_t max_rewind) {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from IO thread */
 | 
					/* Called from IO thread */
 | 
				
			||||||
void pa_source_invalidate_requested_latency(pa_source *s) {
 | 
					void pa_source_invalidate_requested_latency(pa_source *s, pa_bool_t dynamic) {
 | 
				
			||||||
    pa_source_output *o;
 | 
					    pa_source_output *o;
 | 
				
			||||||
    void *state = NULL;
 | 
					    void *state = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_source_assert_ref(s);
 | 
					    pa_source_assert_ref(s);
 | 
				
			||||||
    pa_source_assert_io_context(s);
 | 
					    pa_source_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!(s->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
					    if ((s->flags & PA_SOURCE_DYNAMIC_LATENCY))
 | 
				
			||||||
        return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        s->thread_info.requested_latency_valid = FALSE;
 | 
					        s->thread_info.requested_latency_valid = FALSE;
 | 
				
			||||||
 | 
					    else if (dynamic)
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (PA_SOURCE_IS_LINKED(s->thread_info.state)) {
 | 
					    if (PA_SOURCE_IS_LINKED(s->thread_info.state)) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1315,7 +1323,7 @@ void pa_source_invalidate_requested_latency(pa_source *s) {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (s->monitor_of)
 | 
					    if (s->monitor_of)
 | 
				
			||||||
        pa_sink_invalidate_requested_latency(s->monitor_of);
 | 
					        pa_sink_invalidate_requested_latency(s->monitor_of, dynamic);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from main thread */
 | 
					/* Called from main thread */
 | 
				
			||||||
| 
						 | 
					@ -1375,8 +1383,6 @@ void pa_source_get_latency_range(pa_source *s, pa_usec_t *min_latency, pa_usec_t
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from IO thread, and from main thread before pa_source_put() is called */
 | 
					/* Called from IO thread, and from main thread before pa_source_put() is called */
 | 
				
			||||||
void pa_source_set_latency_range_within_thread(pa_source *s, pa_usec_t min_latency, pa_usec_t max_latency) {
 | 
					void pa_source_set_latency_range_within_thread(pa_source *s, pa_usec_t min_latency, pa_usec_t max_latency) {
 | 
				
			||||||
    void *state = NULL;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    pa_source_assert_ref(s);
 | 
					    pa_source_assert_ref(s);
 | 
				
			||||||
    pa_source_assert_io_context(s);
 | 
					    pa_source_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1390,18 +1396,23 @@ void pa_source_set_latency_range_within_thread(pa_source *s, pa_usec_t min_laten
 | 
				
			||||||
              (s->flags & PA_SOURCE_DYNAMIC_LATENCY) ||
 | 
					              (s->flags & PA_SOURCE_DYNAMIC_LATENCY) ||
 | 
				
			||||||
              s->monitor_of);
 | 
					              s->monitor_of);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->thread_info.min_latency == min_latency &&
 | 
				
			||||||
 | 
					        s->thread_info.max_latency == max_latency)
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    s->thread_info.min_latency = min_latency;
 | 
					    s->thread_info.min_latency = min_latency;
 | 
				
			||||||
    s->thread_info.max_latency = max_latency;
 | 
					    s->thread_info.max_latency = max_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (PA_SOURCE_IS_LINKED(s->thread_info.state)) {
 | 
					    if (PA_SOURCE_IS_LINKED(s->thread_info.state)) {
 | 
				
			||||||
        pa_source_output *o;
 | 
					        pa_source_output *o;
 | 
				
			||||||
 | 
					        void *state = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        while ((o = pa_hashmap_iterate(s->thread_info.outputs, &state, NULL)))
 | 
					        PA_HASHMAP_FOREACH(o, s->thread_info.outputs, state)
 | 
				
			||||||
            if (o->update_source_latency_range)
 | 
					            if (o->update_source_latency_range)
 | 
				
			||||||
                o->update_source_latency_range(o);
 | 
					                o->update_source_latency_range(o);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_source_invalidate_requested_latency(s);
 | 
					    pa_source_invalidate_requested_latency(s, FALSE);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from main thread, before the source is put */
 | 
					/* Called from main thread, before the source is put */
 | 
				
			||||||
| 
						 | 
					@ -1409,7 +1420,10 @@ void pa_source_set_fixed_latency(pa_source *s, pa_usec_t latency) {
 | 
				
			||||||
    pa_source_assert_ref(s);
 | 
					    pa_source_assert_ref(s);
 | 
				
			||||||
    pa_assert_ctl_context();
 | 
					    pa_assert_ctl_context();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_assert(pa_source_get_state(s) == PA_SOURCE_INIT);
 | 
					    if (s->flags & PA_SOURCE_DYNAMIC_LATENCY) {
 | 
				
			||||||
 | 
					        pa_assert(latency == 0);
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (latency < ABSOLUTE_MIN_LATENCY)
 | 
					    if (latency < ABSOLUTE_MIN_LATENCY)
 | 
				
			||||||
        latency = ABSOLUTE_MIN_LATENCY;
 | 
					        latency = ABSOLUTE_MIN_LATENCY;
 | 
				
			||||||
| 
						 | 
					@ -1417,7 +1431,58 @@ void pa_source_set_fixed_latency(pa_source *s, pa_usec_t latency) {
 | 
				
			||||||
    if (latency > ABSOLUTE_MAX_LATENCY)
 | 
					    if (latency > ABSOLUTE_MAX_LATENCY)
 | 
				
			||||||
        latency = ABSOLUTE_MAX_LATENCY;
 | 
					        latency = ABSOLUTE_MAX_LATENCY;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    s->fixed_latency = latency;
 | 
					    if (PA_SOURCE_IS_LINKED(s->state))
 | 
				
			||||||
 | 
					        pa_assert_se(pa_asyncmsgq_send(s->asyncmsgq, PA_MSGOBJECT(s), PA_SOURCE_MESSAGE_SET_FIXED_LATENCY, NULL, (int64_t) latency, NULL) == 0);
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        s->thread_info.fixed_latency = latency;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Called from main thread */
 | 
				
			||||||
 | 
					pa_usec_t pa_source_get_fixed_latency(pa_source *s) {
 | 
				
			||||||
 | 
					    pa_usec_t latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_source_assert_ref(s);
 | 
				
			||||||
 | 
					    pa_assert_ctl_context();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->flags & PA_SOURCE_DYNAMIC_LATENCY)
 | 
				
			||||||
 | 
					        return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (PA_SOURCE_IS_LINKED(s->state))
 | 
				
			||||||
 | 
					        pa_assert_se(pa_asyncmsgq_send(s->asyncmsgq, PA_MSGOBJECT(s), PA_SOURCE_MESSAGE_GET_FIXED_LATENCY, &latency, 0, NULL) == 0);
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        latency = s->thread_info.fixed_latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return latency;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Called from IO thread */
 | 
				
			||||||
 | 
					void pa_source_set_fixed_latency_within_thread(pa_source *s, pa_usec_t latency) {
 | 
				
			||||||
 | 
					    pa_source_assert_ref(s);
 | 
				
			||||||
 | 
					    pa_source_assert_io_context(s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->flags & PA_SOURCE_DYNAMIC_LATENCY) {
 | 
				
			||||||
 | 
					        pa_assert(latency == 0);
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_assert(latency >= ABSOLUTE_MIN_LATENCY);
 | 
				
			||||||
 | 
					    pa_assert(latency <= ABSOLUTE_MAX_LATENCY);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (s->thread_info.fixed_latency == latency)
 | 
				
			||||||
 | 
					        return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    s->thread_info.fixed_latency = latency;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (PA_SOURCE_IS_LINKED(s->thread_info.state)) {
 | 
				
			||||||
 | 
					        pa_source_output *o;
 | 
				
			||||||
 | 
					        void *state = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        PA_HASHMAP_FOREACH(o, s->thread_info.outputs, state)
 | 
				
			||||||
 | 
					            if (o->update_source_fixed_latency)
 | 
				
			||||||
 | 
					                o->update_source_fixed_latency(o);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pa_source_invalidate_requested_latency(s, FALSE);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called from main thread */
 | 
					/* Called from main thread */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -93,8 +93,6 @@ struct pa_source {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_memchunk silence;
 | 
					    pa_memchunk silence;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    pa_usec_t fixed_latency; /* for sources with PA_SOURCE_DYNAMIC_LATENCY this is 0 */
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    pa_hashmap *ports;
 | 
					    pa_hashmap *ports;
 | 
				
			||||||
    pa_device_port *active_port;
 | 
					    pa_device_port *active_port;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -153,6 +151,8 @@ struct pa_source {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        pa_usec_t min_latency; /* we won't go below this latency */
 | 
					        pa_usec_t min_latency; /* we won't go below this latency */
 | 
				
			||||||
        pa_usec_t max_latency; /* An upper limit for the latencies */
 | 
					        pa_usec_t max_latency; /* An upper limit for the latencies */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        pa_usec_t fixed_latency; /* for sources with PA_SOURCE_DYNAMIC_LATENCY this is 0 */
 | 
				
			||||||
 } thread_info;
 | 
					 } thread_info;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    void *userdata;
 | 
					    void *userdata;
 | 
				
			||||||
| 
						 | 
					@ -175,6 +175,8 @@ typedef enum pa_source_message {
 | 
				
			||||||
    PA_SOURCE_MESSAGE_DETACH,
 | 
					    PA_SOURCE_MESSAGE_DETACH,
 | 
				
			||||||
    PA_SOURCE_MESSAGE_SET_LATENCY_RANGE,
 | 
					    PA_SOURCE_MESSAGE_SET_LATENCY_RANGE,
 | 
				
			||||||
    PA_SOURCE_MESSAGE_GET_LATENCY_RANGE,
 | 
					    PA_SOURCE_MESSAGE_GET_LATENCY_RANGE,
 | 
				
			||||||
 | 
					    PA_SOURCE_MESSAGE_SET_FIXED_LATENCY,
 | 
				
			||||||
 | 
					    PA_SOURCE_MESSAGE_GET_FIXED_LATENCY,
 | 
				
			||||||
    PA_SOURCE_MESSAGE_GET_MAX_REWIND,
 | 
					    PA_SOURCE_MESSAGE_GET_MAX_REWIND,
 | 
				
			||||||
    PA_SOURCE_MESSAGE_SET_MAX_REWIND,
 | 
					    PA_SOURCE_MESSAGE_SET_MAX_REWIND,
 | 
				
			||||||
    PA_SOURCE_MESSAGE_MAX
 | 
					    PA_SOURCE_MESSAGE_MAX
 | 
				
			||||||
| 
						 | 
					@ -250,6 +252,7 @@ int pa_source_sync_suspend(pa_source *s);
 | 
				
			||||||
pa_usec_t pa_source_get_latency(pa_source *s);
 | 
					pa_usec_t pa_source_get_latency(pa_source *s);
 | 
				
			||||||
pa_usec_t pa_source_get_requested_latency(pa_source *s);
 | 
					pa_usec_t pa_source_get_requested_latency(pa_source *s);
 | 
				
			||||||
void pa_source_get_latency_range(pa_source *s, pa_usec_t *min_latency, pa_usec_t *max_latency);
 | 
					void pa_source_get_latency_range(pa_source *s, pa_usec_t *min_latency, pa_usec_t *max_latency);
 | 
				
			||||||
 | 
					pa_usec_t pa_source_get_fixed_latency(pa_source *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
size_t pa_source_get_max_rewind(pa_source *s);
 | 
					size_t pa_source_get_max_rewind(pa_source *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -259,6 +262,7 @@ int pa_source_suspend_all(pa_core *c, pa_bool_t suspend, pa_suspend_cause_t caus
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_source_set_volume(pa_source *source, const pa_cvolume *volume, pa_bool_t save);
 | 
					void pa_source_set_volume(pa_source *source, const pa_cvolume *volume, pa_bool_t save);
 | 
				
			||||||
const pa_cvolume *pa_source_get_volume(pa_source *source, pa_bool_t force_refresh);
 | 
					const pa_cvolume *pa_source_get_volume(pa_source *source, pa_bool_t force_refresh);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_source_set_mute(pa_source *source, pa_bool_t mute, pa_bool_t save);
 | 
					void pa_source_set_mute(pa_source *source, pa_bool_t mute, pa_bool_t save);
 | 
				
			||||||
pa_bool_t pa_source_get_mute(pa_source *source, pa_bool_t force_refresh);
 | 
					pa_bool_t pa_source_get_mute(pa_source *source, pa_bool_t force_refresh);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -290,11 +294,13 @@ void pa_source_detach_within_thread(pa_source *s);
 | 
				
			||||||
pa_usec_t pa_source_get_requested_latency_within_thread(pa_source *s);
 | 
					pa_usec_t pa_source_get_requested_latency_within_thread(pa_source *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_source_set_max_rewind_within_thread(pa_source *s, size_t max_rewind);
 | 
					void pa_source_set_max_rewind_within_thread(pa_source *s, size_t max_rewind);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_source_set_latency_range_within_thread(pa_source *s, pa_usec_t min_latency, pa_usec_t max_latency);
 | 
					void pa_source_set_latency_range_within_thread(pa_source *s, pa_usec_t min_latency, pa_usec_t max_latency);
 | 
				
			||||||
 | 
					void pa_source_set_fixed_latency_within_thread(pa_source *s, pa_usec_t latency);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*** To be called exclusively by source output drivers, from IO context */
 | 
					/*** To be called exclusively by source output drivers, from IO context */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void pa_source_invalidate_requested_latency(pa_source *s);
 | 
					void pa_source_invalidate_requested_latency(pa_source *s, pa_bool_t dynamic);
 | 
				
			||||||
pa_usec_t pa_source_get_latency_within_thread(pa_source *s);
 | 
					pa_usec_t pa_source_get_latency_within_thread(pa_source *s);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define pa_source_assert_io_context(s) \
 | 
					#define pa_source_assert_io_context(s) \
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue