4 This file is part of polypaudio.
6 polypaudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as published
8 by the Free Software Foundation; either version 2 of the License,
9 or (at your option) any later version.
11 polypaudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public License
17 along with polypaudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
31 #include <polyp/utf8.h>
32 #include <polyp/xmalloc.h>
34 #include <polypcore/sample-util.h>
35 #include <polypcore/core-subscribe.h>
36 #include <polypcore/log.h>
38 #include "sink-input.h"
40 #define CONVERT_BUFFER_LENGTH 4096
42 #define CHECK_VALIDITY_RETURN_NULL(condition) \
48 pa_sink_input
* pa_sink_input_new(
52 const pa_sample_spec
*spec
,
53 const pa_channel_map
*map
,
54 const pa_cvolume
*volume
,
56 int resample_method
) {
59 pa_resampler
*resampler
= NULL
;
67 assert(s
->state
== PA_SINK_RUNNING
);
69 CHECK_VALIDITY_RETURN_NULL(pa_sample_spec_valid(spec
));
72 map
= pa_channel_map_init_auto(&tmap
, spec
->channels
, PA_CHANNEL_MAP_DEFAULT
);
74 volume
= pa_cvolume_reset(&tvol
, spec
->channels
);
76 CHECK_VALIDITY_RETURN_NULL(map
&& pa_channel_map_valid(map
));
77 CHECK_VALIDITY_RETURN_NULL(volume
&& pa_cvolume_valid(volume
));
78 CHECK_VALIDITY_RETURN_NULL(map
->channels
== spec
->channels
);
79 CHECK_VALIDITY_RETURN_NULL(volume
->channels
== spec
->channels
);
80 CHECK_VALIDITY_RETURN_NULL(!driver
|| pa_utf8_valid(driver
));
81 CHECK_VALIDITY_RETURN_NULL(pa_utf8_valid(name
));
83 if (pa_idxset_size(s
->inputs
) >= PA_MAX_INPUTS_PER_SINK
) {
84 pa_log_warn(__FILE__
": Failed to create sink input: too many inputs per sink.");
88 if (resample_method
== PA_RESAMPLER_INVALID
)
89 resample_method
= s
->core
->resample_method
;
91 if (variable_rate
|| !pa_sample_spec_equal(spec
, &s
->sample_spec
) || !pa_channel_map_equal(map
, &s
->channel_map
))
92 if (!(resampler
= pa_resampler_new(spec
, map
, &s
->sample_spec
, &s
->channel_map
, s
->core
->memblock_stat
, resample_method
)))
95 i
= pa_xnew(pa_sink_input
, 1);
97 i
->state
= PA_SINK_INPUT_RUNNING
;
98 i
->name
= pa_xstrdup(name
);
99 i
->driver
= pa_xstrdup(driver
);
104 i
->sample_spec
= *spec
;
105 i
->channel_map
= *map
;
111 i
->get_latency
= NULL
;
117 pa_memchunk_reset(&i
->resampled_chunk
);
118 i
->resampler
= resampler
;
121 r
= pa_idxset_put(s
->core
->sink_inputs
, i
, &i
->index
);
122 assert(r
== 0 && i
->index
!= PA_IDXSET_INVALID
);
123 r
= pa_idxset_put(s
->inputs
, i
, NULL
);
126 pa_sample_spec_snprint(st
, sizeof(st
), spec
);
127 pa_log_info(__FILE__
": created %u \"%s\" on %u with sample spec \"%s\"", i
->index
, i
->name
, s
->index
, st
);
129 pa_subscription_post(s
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_NEW
, i
->index
);
134 void pa_sink_input_disconnect(pa_sink_input
*i
) {
136 assert(i
->state
!= PA_SINK_INPUT_DISCONNECTED
);
138 assert(i
->sink
->core
);
140 pa_idxset_remove_by_data(i
->sink
->core
->sink_inputs
, i
, NULL
);
141 pa_idxset_remove_by_data(i
->sink
->inputs
, i
, NULL
);
143 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_REMOVE
, i
->index
);
149 i
->get_latency
= NULL
;
153 i
->state
= PA_SINK_INPUT_DISCONNECTED
;
156 static void sink_input_free(pa_sink_input
* i
) {
159 if (i
->state
!= PA_SINK_INPUT_DISCONNECTED
)
160 pa_sink_input_disconnect(i
);
162 pa_log_info(__FILE__
": freed %u \"%s\"", i
->index
, i
->name
);
164 if (i
->resampled_chunk
.memblock
)
165 pa_memblock_unref(i
->resampled_chunk
.memblock
);
168 pa_resampler_free(i
->resampler
);
175 void pa_sink_input_unref(pa_sink_input
*i
) {
183 pa_sink_input
* pa_sink_input_ref(pa_sink_input
*i
) {
191 void pa_sink_input_kill(pa_sink_input
*i
) {
199 pa_usec_t
pa_sink_input_get_latency(pa_sink_input
*i
) {
206 r
+= i
->get_latency(i
);
208 if (i
->resampled_chunk
.memblock
)
209 r
+= pa_bytes_to_usec(i
->resampled_chunk
.length
, &i
->sample_spec
);
214 int pa_sink_input_peek(pa_sink_input
*i
, pa_memchunk
*chunk
, pa_cvolume
*volume
) {
216 int do_volume_adj_here
;
223 pa_sink_input_ref(i
);
225 if (!i
->peek
|| !i
->drop
|| i
->state
== PA_SINK_INPUT_CORKED
)
229 do_volume_adj_here
= 0;
230 ret
= i
->peek(i
, chunk
);
234 do_volume_adj_here
= !pa_channel_map_equal(&i
->channel_map
, &i
->sink
->channel_map
);
236 while (!i
->resampled_chunk
.memblock
) {
240 if ((ret
= i
->peek(i
, &tchunk
)) < 0)
243 assert(tchunk
.length
);
245 l
= pa_resampler_request(i
->resampler
, CONVERT_BUFFER_LENGTH
);
247 if (l
> tchunk
.length
)
250 i
->drop(i
, &tchunk
, l
);
253 /* It might be necessary to adjust the volume here */
254 if (do_volume_adj_here
) {
255 pa_memchunk_make_writable(&tchunk
, i
->sink
->core
->memblock_stat
, 0);
256 pa_volume_memchunk(&tchunk
, &i
->sample_spec
, &i
->volume
);
259 pa_resampler_run(i
->resampler
, &tchunk
, &i
->resampled_chunk
);
260 pa_memblock_unref(tchunk
.memblock
);
263 assert(i
->resampled_chunk
.memblock
);
264 assert(i
->resampled_chunk
.length
);
266 *chunk
= i
->resampled_chunk
;
267 pa_memblock_ref(i
->resampled_chunk
.memblock
);
273 if (ret
< 0 && i
->playing
&& i
->underrun
)
276 i
->playing
= ret
>= 0;
279 /* Let's see if we had to apply the volume adjustment
280 * ourselves, or if this can be done by the sink for us */
282 if (do_volume_adj_here
)
283 /* We had different channel maps, so we already did the adjustment */
284 pa_cvolume_reset(volume
, i
->sink
->sample_spec
.channels
);
286 /* We've both the same channel map, so let's have the sink do the adjustment for us*/
290 pa_sink_input_unref(i
);
295 void pa_sink_input_drop(pa_sink_input
*i
, const pa_memchunk
*chunk
, size_t length
) {
302 i
->drop(i
, chunk
, length
);
306 assert(i
->resampled_chunk
.memblock
);
307 assert(i
->resampled_chunk
.length
>= length
);
309 i
->resampled_chunk
.index
+= length
;
310 i
->resampled_chunk
.length
-= length
;
312 if (i
->resampled_chunk
.length
<= 0) {
313 pa_memblock_unref(i
->resampled_chunk
.memblock
);
314 i
->resampled_chunk
.memblock
= NULL
;
315 i
->resampled_chunk
.index
= i
->resampled_chunk
.length
= 0;
319 void pa_sink_input_set_volume(pa_sink_input
*i
, const pa_cvolume
*volume
) {
323 assert(i
->sink
->core
);
325 if (pa_cvolume_equal(&i
->volume
, volume
))
329 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_CHANGE
, i
->index
);
332 const pa_cvolume
* pa_sink_input_get_volume(pa_sink_input
*i
) {
339 void pa_sink_input_cork(pa_sink_input
*i
, int b
) {
345 if (i
->state
== PA_SINK_INPUT_DISCONNECTED
)
348 n
= i
->state
== PA_SINK_INPUT_CORKED
&& !b
;
350 i
->state
= b
? PA_SINK_INPUT_CORKED
: PA_SINK_INPUT_RUNNING
;
353 pa_sink_notify(i
->sink
);
356 void pa_sink_input_set_rate(pa_sink_input
*i
, uint32_t rate
) {
358 assert(i
->resampler
);
361 if (i
->sample_spec
.rate
== rate
)
364 i
->sample_spec
.rate
= rate
;
365 pa_resampler_set_input_rate(i
->resampler
, rate
);
368 void pa_sink_input_set_name(pa_sink_input
*i
, const char *name
) {
373 i
->name
= pa_xstrdup(name
);
375 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_CHANGE
, i
->index
);
378 pa_resample_method_t
pa_sink_input_get_resample_method(pa_sink_input
*i
) {
383 return PA_RESAMPLER_INVALID
;
385 return pa_resampler_get_method(i
->resampler
);