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 "sink-input.h"
32 #include "sample-util.h"
34 #include "subscribe.h"
37 #define CONVERT_BUFFER_LENGTH 4096
39 struct pa_sink_input
* pa_sink_input_new(struct pa_sink
*s
, const char *name
, const struct pa_sample_spec
*spec
, int variable_rate
, int resample_method
) {
40 struct pa_sink_input
*i
;
41 struct pa_resampler
*resampler
= NULL
;
44 assert(s
&& spec
&& s
->state
== PA_SINK_RUNNING
);
46 if (pa_idxset_ncontents(s
->inputs
) >= PA_MAX_INPUTS_PER_SINK
) {
47 pa_log(__FILE__
": Failed to create sink input: too many inputs per sink.\n");
51 if (resample_method
< 0)
52 resample_method
= s
->core
->resample_method
;
54 if (variable_rate
|| !pa_sample_spec_equal(spec
, &s
->sample_spec
))
55 if (!(resampler
= pa_resampler_new(spec
, &s
->sample_spec
, s
->core
->memblock_stat
, resample_method
)))
58 i
= pa_xmalloc(sizeof(struct pa_sink_input
));
60 i
->state
= PA_SINK_INPUT_RUNNING
;
61 i
->name
= pa_xstrdup(name
);
65 i
->sample_spec
= *spec
;
70 i
->get_latency
= NULL
;
74 i
->volume
= PA_VOLUME_NORM
;
77 i
->resampled_chunk
.memblock
= NULL
;
78 i
->resampled_chunk
.index
= i
->resampled_chunk
.length
= 0;
79 i
->resampler
= resampler
;
82 r
= pa_idxset_put(s
->core
->sink_inputs
, i
, &i
->index
);
83 assert(r
== 0 && i
->index
!= PA_IDXSET_INVALID
);
84 r
= pa_idxset_put(s
->inputs
, i
, NULL
);
87 pa_sample_spec_snprint(st
, sizeof(st
), spec
);
88 pa_log(__FILE__
": created %u \"%s\" on %u with sample spec \"%s\"\n", i
->index
, i
->name
, s
->index
, st
);
90 pa_subscription_post(s
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_NEW
, i
->index
);
95 void pa_sink_input_disconnect(struct pa_sink_input
*i
) {
96 assert(i
&& i
->state
!= PA_SINK_INPUT_DISCONNECTED
&& i
->sink
&& i
->sink
->core
);
98 pa_idxset_remove_by_data(i
->sink
->core
->sink_inputs
, i
, NULL
);
99 pa_idxset_remove_by_data(i
->sink
->inputs
, i
, NULL
);
101 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_REMOVE
, i
->index
);
107 i
->get_latency
= NULL
;
109 i
->state
= PA_SINK_INPUT_DISCONNECTED
;
112 static void sink_input_free(struct pa_sink_input
* i
) {
115 if (i
->state
!= PA_SINK_INPUT_DISCONNECTED
)
116 pa_sink_input_disconnect(i
);
118 if (i
->resampled_chunk
.memblock
)
119 pa_memblock_unref(i
->resampled_chunk
.memblock
);
121 pa_resampler_free(i
->resampler
);
127 void pa_sink_input_unref(struct pa_sink_input
*i
) {
128 assert(i
&& i
->ref
>= 1);
134 struct pa_sink_input
* pa_sink_input_ref(struct pa_sink_input
*i
) {
135 assert(i
&& i
->ref
>= 1);
140 void pa_sink_input_kill(struct pa_sink_input
*i
) {
141 assert(i
&& i
->ref
>= 1);
147 pa_usec_t
pa_sink_input_get_latency(struct pa_sink_input
*i
) {
149 assert(i
&& i
->ref
>= 1);
152 r
+= i
->get_latency(i
);
154 if (i
->resampled_chunk
.memblock
)
155 r
+= pa_bytes_to_usec(i
->resampled_chunk
.length
, &i
->sample_spec
);
160 int pa_sink_input_peek(struct pa_sink_input
*i
, struct pa_memchunk
*chunk
) {
162 assert(i
&& chunk
&& i
->ref
>= 1);
164 pa_sink_input_ref(i
);
166 if (!i
->peek
|| !i
->drop
|| i
->state
== PA_SINK_INPUT_CORKED
)
170 ret
= i
->peek(i
, chunk
);
174 while (!i
->resampled_chunk
.memblock
) {
175 struct pa_memchunk tchunk
;
178 if ((ret
= i
->peek(i
, &tchunk
)) < 0)
181 assert(tchunk
.length
);
183 l
= pa_resampler_request(i
->resampler
, CONVERT_BUFFER_LENGTH
);
185 if (l
> tchunk
.length
)
188 i
->drop(i
, &tchunk
, l
);
191 pa_resampler_run(i
->resampler
, &tchunk
, &i
->resampled_chunk
);
192 pa_memblock_unref(tchunk
.memblock
);
195 assert(i
->resampled_chunk
.memblock
&& i
->resampled_chunk
.length
);
196 *chunk
= i
->resampled_chunk
;
197 pa_memblock_ref(i
->resampled_chunk
.memblock
);
203 if (ret
< 0 && i
->playing
&& i
->underrun
)
206 i
->playing
= ret
>= 0;
208 pa_sink_input_unref(i
);
213 void pa_sink_input_drop(struct pa_sink_input
*i
, const struct pa_memchunk
*chunk
, size_t length
) {
214 assert(i
&& length
&& i
->ref
>= 1);
218 i
->drop(i
, chunk
, length
);
222 assert(i
->resampled_chunk
.memblock
&& i
->resampled_chunk
.length
>= length
);
224 i
->resampled_chunk
.index
+= length
;
225 i
->resampled_chunk
.length
-= length
;
227 if (!i
->resampled_chunk
.length
) {
228 pa_memblock_unref(i
->resampled_chunk
.memblock
);
229 i
->resampled_chunk
.memblock
= NULL
;
230 i
->resampled_chunk
.index
= i
->resampled_chunk
.length
= 0;
234 void pa_sink_input_set_volume(struct pa_sink_input
*i
, pa_volume_t volume
) {
235 assert(i
&& i
->sink
&& i
->sink
->core
&& i
->ref
>= 1);
237 if (i
->volume
!= volume
) {
239 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_CHANGE
, i
->index
);
243 void pa_sink_input_cork(struct pa_sink_input
*i
, int b
) {
245 assert(i
&& i
->ref
>= 1);
247 if (i
->state
== PA_SINK_INPUT_DISCONNECTED
)
250 n
= i
->state
== PA_SINK_INPUT_CORKED
&& !b
;
251 i
->state
= b
? PA_SINK_INPUT_CORKED
: PA_SINK_INPUT_RUNNING
;
254 pa_sink_notify(i
->sink
);
257 void pa_sink_input_set_rate(struct pa_sink_input
*i
, uint32_t rate
) {
258 assert(i
&& i
->resampler
&& i
->ref
>= 1);
260 if (i
->sample_spec
.rate
== rate
)
263 i
->sample_spec
.rate
= rate
;
264 pa_resampler_set_input_rate(i
->resampler
, rate
);
267 void pa_sink_input_set_name(struct pa_sink_input
*i
, const char *name
) {
268 assert(i
&& i
->ref
>= 1);
271 i
->name
= pa_xstrdup(name
);
273 pa_subscription_post(i
->sink
->core
, PA_SUBSCRIPTION_EVENT_SINK_INPUT
|PA_SUBSCRIPTION_EVENT_CHANGE
, i
->index
);