mirror of
				https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
				synced 2025-10-29 05:40:23 -04: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,18 +2199,18 @@ 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)) | ||||||
|  |         s->thread_info.requested_latency_valid = FALSE; | ||||||
|  |     else if (dynamic) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     s->thread_info.requested_latency_valid = FALSE; |  | ||||||
| 
 |  | ||||||
|     if (PA_SINK_IS_LINKED(s->thread_info.state)) { |     if (PA_SINK_IS_LINKED(s->thread_info.state)) { | ||||||
| 
 | 
 | ||||||
|         if (s->update_requested_latency) |         if (s->update_requested_latency) | ||||||
|  | @ -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,18 +1300,18 @@ 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)) | ||||||
|  |         s->thread_info.requested_latency_valid = FALSE; | ||||||
|  |     else if (dynamic) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     s->thread_info.requested_latency_valid = FALSE; |  | ||||||
| 
 |  | ||||||
|     if (PA_SOURCE_IS_LINKED(s->thread_info.state)) { |     if (PA_SOURCE_IS_LINKED(s->thread_info.state)) { | ||||||
| 
 | 
 | ||||||
|         if (s->update_requested_latency) |         if (s->update_requested_latency) | ||||||
|  | @ -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,7 +151,9 @@ 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 */ | ||||||
|     } thread_info; | 
 | ||||||
|  |         pa_usec_t fixed_latency; /* for sources with PA_SOURCE_DYNAMIC_LATENCY this is 0 */ | ||||||
|  |  } 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
	
	 Lennart Poettering
						Lennart Poettering