]> code.delx.au - pulseaudio/blob - src/pulse/ext-stream-restore.c
Merge commit 'origin/master-tx'
[pulseaudio] / src / pulse / ext-stream-restore.c
1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2008 Lennart Poettering
5
6 PulseAudio 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.1 of the License,
9 or (at your option) any later version.
10
11 PulseAudio 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.
15
16 You should have received a copy of the GNU Lesser General Public License
17 along with PulseAudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19 USA.
20 ***/
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <pulse/context.h>
27 #include <pulse/gccmacro.h>
28
29 #include <pulsecore/macro.h>
30 #include <pulsecore/pstream-util.h>
31
32 #include "internal.h"
33 #include "operation.h"
34 #include "fork-detect.h"
35
36 #include "ext-stream-restore.h"
37
38 enum {
39 SUBCOMMAND_TEST,
40 SUBCOMMAND_READ,
41 SUBCOMMAND_WRITE,
42 SUBCOMMAND_DELETE,
43 SUBCOMMAND_SUBSCRIBE,
44 SUBCOMMAND_EVENT
45 };
46
47 static void ext_stream_restore_test_cb(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) {
48 pa_operation *o = userdata;
49 uint32_t version = PA_INVALID_INDEX;
50
51 pa_assert(pd);
52 pa_assert(o);
53 pa_assert(PA_REFCNT_VALUE(o) >= 1);
54
55 if (!o->context)
56 goto finish;
57
58 if (command != PA_COMMAND_REPLY) {
59 if (pa_context_handle_error(o->context, command, t, FALSE) < 0)
60 goto finish;
61
62 } else if (pa_tagstruct_getu32(t, &version) < 0 ||
63 !pa_tagstruct_eof(t)) {
64
65 pa_context_fail(o->context, PA_ERR_PROTOCOL);
66 goto finish;
67 }
68
69 if (o->callback) {
70 pa_ext_stream_restore_test_cb_t cb = (pa_ext_stream_restore_test_cb_t) o->callback;
71 cb(o->context, version, o->userdata);
72 }
73
74 finish:
75 pa_operation_done(o);
76 pa_operation_unref(o);
77 }
78
79 pa_operation *pa_ext_stream_restore_test(
80 pa_context *c,
81 pa_ext_stream_restore_test_cb_t cb,
82 void *userdata) {
83
84 uint32_t tag;
85 pa_operation *o;
86 pa_tagstruct *t;
87
88 pa_assert(c);
89 pa_assert(PA_REFCNT_VALUE(c) >= 1);
90
91 PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED);
92 PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE);
93 PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 14, PA_ERR_NOTSUPPORTED);
94
95 o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata);
96
97 t = pa_tagstruct_command(c, PA_COMMAND_EXTENSION, &tag);
98 pa_tagstruct_putu32(t, PA_INVALID_INDEX);
99 pa_tagstruct_puts(t, "module-stream-restore");
100 pa_tagstruct_putu32(t, SUBCOMMAND_TEST);
101 pa_pstream_send_tagstruct(c->pstream, t);
102 pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, ext_stream_restore_test_cb, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref);
103
104 return o;
105 }
106
107 static void ext_stream_restore_read_cb(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) {
108 pa_operation *o = userdata;
109 int eol = 1;
110
111 pa_assert(pd);
112 pa_assert(o);
113 pa_assert(PA_REFCNT_VALUE(o) >= 1);
114
115 if (!o->context)
116 goto finish;
117
118 if (command != PA_COMMAND_REPLY) {
119 if (pa_context_handle_error(o->context, command, t, FALSE) < 0)
120 goto finish;
121
122 eol = -1;
123 } else {
124
125 while (!pa_tagstruct_eof(t)) {
126 pa_ext_stream_restore_info i;
127 pa_bool_t mute = FALSE;
128
129 memset(&i, 0, sizeof(i));
130
131 if (pa_tagstruct_gets(t, &i.name) < 0 ||
132 pa_tagstruct_get_channel_map(t, &i.channel_map) < 0 ||
133 pa_tagstruct_get_cvolume(t, &i.volume) < 0 ||
134 pa_tagstruct_gets(t, &i.device) < 0 ||
135 pa_tagstruct_get_boolean(t, &mute) < 0) {
136
137 pa_context_fail(o->context, PA_ERR_PROTOCOL);
138 goto finish;
139 }
140
141 i.mute = (int) mute;
142
143 if (o->callback) {
144 pa_ext_stream_restore_read_cb_t cb = (pa_ext_stream_restore_read_cb_t) o->callback;
145 cb(o->context, &i, 0, o->userdata);
146 }
147 }
148 }
149
150 if (o->callback) {
151 pa_ext_stream_restore_read_cb_t cb = (pa_ext_stream_restore_read_cb_t) o->callback;
152 cb(o->context, NULL, eol, o->userdata);
153 }
154
155 finish:
156 pa_operation_done(o);
157 pa_operation_unref(o);
158 }
159
160 pa_operation *pa_ext_stream_restore_read(
161 pa_context *c,
162 pa_ext_stream_restore_read_cb_t cb,
163 void *userdata) {
164
165 uint32_t tag;
166 pa_operation *o;
167 pa_tagstruct *t;
168
169 pa_assert(c);
170 pa_assert(PA_REFCNT_VALUE(c) >= 1);
171
172 PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED);
173 PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE);
174 PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 14, PA_ERR_NOTSUPPORTED);
175
176 o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata);
177
178 t = pa_tagstruct_command(c, PA_COMMAND_EXTENSION, &tag);
179 pa_tagstruct_putu32(t, PA_INVALID_INDEX);
180 pa_tagstruct_puts(t, "module-stream-restore");
181 pa_tagstruct_putu32(t, SUBCOMMAND_READ);
182 pa_pstream_send_tagstruct(c->pstream, t);
183 pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, ext_stream_restore_read_cb, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref);
184
185 return o;
186 }
187
188 pa_operation *pa_ext_stream_restore_write(
189 pa_context *c,
190 pa_update_mode_t mode,
191 const pa_ext_stream_restore_info data[],
192 unsigned n,
193 int apply_immediately,
194 pa_context_success_cb_t cb,
195 void *userdata) {
196
197 uint32_t tag;
198 pa_operation *o = NULL;
199 pa_tagstruct *t = NULL;
200
201 pa_assert(c);
202 pa_assert(PA_REFCNT_VALUE(c) >= 1);
203 pa_assert(mode == PA_UPDATE_MERGE || mode == PA_UPDATE_REPLACE || mode == PA_UPDATE_SET);
204 pa_assert(data);
205
206 PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED);
207 PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE);
208 PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 14, PA_ERR_NOTSUPPORTED);
209
210 o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata);
211
212 t = pa_tagstruct_command(c, PA_COMMAND_EXTENSION, &tag);
213 pa_tagstruct_putu32(t, PA_INVALID_INDEX);
214 pa_tagstruct_puts(t, "module-stream-restore");
215 pa_tagstruct_putu32(t, SUBCOMMAND_WRITE);
216
217 pa_tagstruct_putu32(t, mode);
218 pa_tagstruct_put_boolean(t, apply_immediately);
219
220 for (; n > 0; n--, data++) {
221 if (!data->name || !*data->name)
222 goto fail;
223
224 pa_tagstruct_puts(t, data->name);
225
226 if (data->volume.channels > 0 &&
227 !pa_cvolume_compatible_with_channel_map(&data->volume, &data->channel_map))
228 goto fail;
229
230 pa_tagstruct_put_channel_map(t, &data->channel_map);
231 pa_tagstruct_put_cvolume(t, &data->volume);
232 pa_tagstruct_puts(t, data->device);
233 pa_tagstruct_put_boolean(t, data->mute);
234 }
235
236 pa_pstream_send_tagstruct(c->pstream, t);
237 pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, pa_context_simple_ack_callback, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref);
238
239 return o;
240
241 fail:
242 if (o) {
243 pa_operation_cancel(o);
244 pa_operation_unref(o);
245 }
246
247 if (t)
248 pa_tagstruct_free(t);
249
250 pa_context_set_error(c, PA_ERR_INVALID);
251 return NULL;
252 }
253
254 pa_operation *pa_ext_stream_restore_delete(
255 pa_context *c,
256 const char *const s[],
257 pa_context_success_cb_t cb,
258 void *userdata) {
259
260 uint32_t tag;
261 pa_operation *o = NULL;
262 pa_tagstruct *t = NULL;
263 const char *const *k;
264
265 pa_assert(c);
266 pa_assert(PA_REFCNT_VALUE(c) >= 1);
267 pa_assert(s);
268
269 PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED);
270 PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE);
271 PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 14, PA_ERR_NOTSUPPORTED);
272
273 o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata);
274
275 t = pa_tagstruct_command(c, PA_COMMAND_EXTENSION, &tag);
276 pa_tagstruct_putu32(t, PA_INVALID_INDEX);
277 pa_tagstruct_puts(t, "module-stream-restore");
278 pa_tagstruct_putu32(t, SUBCOMMAND_DELETE);
279
280 for (k = s; *k; k++) {
281 if (!*k || !**k)
282 goto fail;
283
284 pa_tagstruct_puts(t, *k);
285 }
286
287 pa_pstream_send_tagstruct(c->pstream, t);
288 pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, pa_context_simple_ack_callback, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref);
289
290 return o;
291
292 fail:
293 if (o) {
294 pa_operation_cancel(o);
295 pa_operation_unref(o);
296 }
297
298 if (t)
299 pa_tagstruct_free(t);
300
301 pa_context_set_error(c, PA_ERR_INVALID);
302 return NULL;
303 }
304
305 pa_operation *pa_ext_stream_restore_subscribe(
306 pa_context *c,
307 int enable,
308 pa_context_success_cb_t cb,
309 void *userdata) {
310
311 uint32_t tag;
312 pa_operation *o;
313 pa_tagstruct *t;
314
315 pa_assert(c);
316 pa_assert(PA_REFCNT_VALUE(c) >= 1);
317
318 PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED);
319 PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE);
320 PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 14, PA_ERR_NOTSUPPORTED);
321
322 o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata);
323
324 t = pa_tagstruct_command(c, PA_COMMAND_EXTENSION, &tag);
325 pa_tagstruct_putu32(t, PA_INVALID_INDEX);
326 pa_tagstruct_puts(t, "module-stream-restore");
327 pa_tagstruct_putu32(t, SUBCOMMAND_SUBSCRIBE);
328 pa_tagstruct_put_boolean(t, enable);
329 pa_pstream_send_tagstruct(c->pstream, t);
330 pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, pa_context_simple_ack_callback, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref);
331
332 return o;
333 }
334
335 void pa_ext_stream_restore_set_subscribe_cb(
336 pa_context *c,
337 pa_ext_stream_restore_subscribe_cb_t cb,
338 void *userdata) {
339
340 pa_assert(c);
341 pa_assert(PA_REFCNT_VALUE(c) >= 1);
342
343 if (pa_detect_fork())
344 return;
345
346 c->ext_stream_restore.callback = cb;
347 c->ext_stream_restore.userdata = userdata;
348 }
349
350 void pa_ext_stream_restore_command(pa_context *c, uint32_t tag, pa_tagstruct *t) {
351 uint32_t subcommand;
352
353 pa_assert(c);
354 pa_assert(PA_REFCNT_VALUE(c) >= 1);
355 pa_assert(t);
356
357 if (pa_tagstruct_getu32(t, &subcommand) < 0 ||
358 !pa_tagstruct_eof(t)) {
359
360 pa_context_fail(c, PA_ERR_PROTOCOL);
361 return;
362 }
363
364 if (subcommand != SUBCOMMAND_EVENT) {
365 pa_context_fail(c, PA_ERR_PROTOCOL);
366 return;
367 }
368
369 if (c->ext_stream_restore.callback)
370 c->ext_stream_restore.callback(c, c->ext_stream_restore.userdata);
371 }