153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2004-2006 Lennart Poettering 553a5a1b3Sopenharmony_ci 653a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 753a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as published 853a5a1b3Sopenharmony_ci by the Free Software Foundation; either version 2.1 of the License, 953a5a1b3Sopenharmony_ci or (at your option) any later version. 1053a5a1b3Sopenharmony_ci 1153a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1253a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1353a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1453a5a1b3Sopenharmony_ci General Public License for more details. 1553a5a1b3Sopenharmony_ci 1653a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public License 1753a5a1b3Sopenharmony_ci along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1853a5a1b3Sopenharmony_ci***/ 1953a5a1b3Sopenharmony_ci 2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2153a5a1b3Sopenharmony_ci#include <config.h> 2253a5a1b3Sopenharmony_ci#endif 2353a5a1b3Sopenharmony_ci 2453a5a1b3Sopenharmony_ci#include <stdlib.h> 2553a5a1b3Sopenharmony_ci#include <stdio.h> 2653a5a1b3Sopenharmony_ci 2753a5a1b3Sopenharmony_ci#include <pulse/utf8.h> 2853a5a1b3Sopenharmony_ci#include <pulse/fork-detect.h> 2953a5a1b3Sopenharmony_ci 3053a5a1b3Sopenharmony_ci#include <pulsecore/pstream-util.h> 3153a5a1b3Sopenharmony_ci#include <pulsecore/macro.h> 3253a5a1b3Sopenharmony_ci#include <pulsecore/proplist-util.h> 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_ci#include "internal.h" 3553a5a1b3Sopenharmony_ci#include "scache.h" 3653a5a1b3Sopenharmony_ci 3753a5a1b3Sopenharmony_ciint pa_stream_connect_upload(pa_stream *s, size_t length) { 3853a5a1b3Sopenharmony_ci pa_tagstruct *t; 3953a5a1b3Sopenharmony_ci uint32_t tag; 4053a5a1b3Sopenharmony_ci const char *name; 4153a5a1b3Sopenharmony_ci 4253a5a1b3Sopenharmony_ci pa_assert(s); 4353a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(s) >= 1); 4453a5a1b3Sopenharmony_ci 4553a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, !pa_detect_fork(), PA_ERR_FORKED); 4653a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, s->state == PA_STREAM_UNCONNECTED, PA_ERR_BADSTATE); 4753a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, length > 0, PA_ERR_INVALID); 4853a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, length == (size_t) (uint32_t) length, PA_ERR_INVALID); 4953a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, s->context->state == PA_CONTEXT_READY, PA_ERR_BADSTATE); 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci if (!(name = pa_proplist_gets(s->proplist, PA_PROP_EVENT_ID))) 5253a5a1b3Sopenharmony_ci name = pa_proplist_gets(s->proplist, PA_PROP_MEDIA_NAME); 5353a5a1b3Sopenharmony_ci 5453a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, name && *name && pa_utf8_valid(name), PA_ERR_INVALID); 5553a5a1b3Sopenharmony_ci 5653a5a1b3Sopenharmony_ci pa_stream_ref(s); 5753a5a1b3Sopenharmony_ci 5853a5a1b3Sopenharmony_ci s->direction = PA_STREAM_UPLOAD; 5953a5a1b3Sopenharmony_ci s->flags = 0; 6053a5a1b3Sopenharmony_ci 6153a5a1b3Sopenharmony_ci t = pa_tagstruct_command(s->context, PA_COMMAND_CREATE_UPLOAD_STREAM, &tag); 6253a5a1b3Sopenharmony_ci 6353a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, name); 6453a5a1b3Sopenharmony_ci pa_tagstruct_put_sample_spec(t, &s->sample_spec); 6553a5a1b3Sopenharmony_ci pa_tagstruct_put_channel_map(t, &s->channel_map); 6653a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, (uint32_t) length); 6753a5a1b3Sopenharmony_ci 6853a5a1b3Sopenharmony_ci if (s->context->version >= 13) 6953a5a1b3Sopenharmony_ci pa_tagstruct_put_proplist(t, s->proplist); 7053a5a1b3Sopenharmony_ci 7153a5a1b3Sopenharmony_ci pa_pstream_send_tagstruct(s->context->pstream, t); 7253a5a1b3Sopenharmony_ci pa_pdispatch_register_reply(s->context->pdispatch, tag, DEFAULT_TIMEOUT, pa_create_stream_callback, s, NULL); 7353a5a1b3Sopenharmony_ci 7453a5a1b3Sopenharmony_ci pa_stream_set_state(s, PA_STREAM_CREATING); 7553a5a1b3Sopenharmony_ci 7653a5a1b3Sopenharmony_ci pa_stream_unref(s); 7753a5a1b3Sopenharmony_ci return 0; 7853a5a1b3Sopenharmony_ci} 7953a5a1b3Sopenharmony_ci 8053a5a1b3Sopenharmony_ciint pa_stream_finish_upload(pa_stream *s) { 8153a5a1b3Sopenharmony_ci pa_tagstruct *t; 8253a5a1b3Sopenharmony_ci uint32_t tag; 8353a5a1b3Sopenharmony_ci 8453a5a1b3Sopenharmony_ci pa_assert(s); 8553a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(s) >= 1); 8653a5a1b3Sopenharmony_ci 8753a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, !pa_detect_fork(), PA_ERR_FORKED); 8853a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, s->channel_valid, PA_ERR_BADSTATE); 8953a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY(s->context, s->context->state == PA_CONTEXT_READY, PA_ERR_BADSTATE); 9053a5a1b3Sopenharmony_ci 9153a5a1b3Sopenharmony_ci pa_stream_ref(s); 9253a5a1b3Sopenharmony_ci 9353a5a1b3Sopenharmony_ci t = pa_tagstruct_command(s->context, PA_COMMAND_FINISH_UPLOAD_STREAM, &tag); 9453a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, s->channel); 9553a5a1b3Sopenharmony_ci pa_pstream_send_tagstruct(s->context->pstream, t); 9653a5a1b3Sopenharmony_ci pa_pdispatch_register_reply(s->context->pdispatch, tag, DEFAULT_TIMEOUT, pa_stream_disconnect_callback, s, NULL); 9753a5a1b3Sopenharmony_ci 9853a5a1b3Sopenharmony_ci pa_stream_unref(s); 9953a5a1b3Sopenharmony_ci return 0; 10053a5a1b3Sopenharmony_ci} 10153a5a1b3Sopenharmony_ci 10253a5a1b3Sopenharmony_cistatic void play_sample_ack_callback(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { 10353a5a1b3Sopenharmony_ci pa_operation *o = userdata; 10453a5a1b3Sopenharmony_ci int success = 1; 10553a5a1b3Sopenharmony_ci uint32_t idx = PA_INVALID_INDEX; 10653a5a1b3Sopenharmony_ci 10753a5a1b3Sopenharmony_ci pa_assert(pd); 10853a5a1b3Sopenharmony_ci pa_assert(o); 10953a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(o) >= 1); 11053a5a1b3Sopenharmony_ci 11153a5a1b3Sopenharmony_ci if (!o->context) 11253a5a1b3Sopenharmony_ci goto finish; 11353a5a1b3Sopenharmony_ci 11453a5a1b3Sopenharmony_ci if (command != PA_COMMAND_REPLY) { 11553a5a1b3Sopenharmony_ci if (pa_context_handle_error(o->context, command, t, false) < 0) 11653a5a1b3Sopenharmony_ci goto finish; 11753a5a1b3Sopenharmony_ci 11853a5a1b3Sopenharmony_ci success = 0; 11953a5a1b3Sopenharmony_ci } else if ((o->context->version >= 13 && pa_tagstruct_getu32(t, &idx) < 0) || 12053a5a1b3Sopenharmony_ci !pa_tagstruct_eof(t)) { 12153a5a1b3Sopenharmony_ci pa_context_fail(o->context, PA_ERR_PROTOCOL); 12253a5a1b3Sopenharmony_ci goto finish; 12353a5a1b3Sopenharmony_ci } else if (o->context->version >= 13 && idx == PA_INVALID_INDEX) 12453a5a1b3Sopenharmony_ci success = 0; 12553a5a1b3Sopenharmony_ci 12653a5a1b3Sopenharmony_ci if (o->callback) { 12753a5a1b3Sopenharmony_ci pa_context_success_cb_t cb = (pa_context_success_cb_t) o->callback; 12853a5a1b3Sopenharmony_ci cb(o->context, success, o->userdata); 12953a5a1b3Sopenharmony_ci } 13053a5a1b3Sopenharmony_ci 13153a5a1b3Sopenharmony_cifinish: 13253a5a1b3Sopenharmony_ci pa_operation_done(o); 13353a5a1b3Sopenharmony_ci pa_operation_unref(o); 13453a5a1b3Sopenharmony_ci} 13553a5a1b3Sopenharmony_ci 13653a5a1b3Sopenharmony_cistatic void play_sample_with_proplist_ack_callback(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { 13753a5a1b3Sopenharmony_ci pa_operation *o = userdata; 13853a5a1b3Sopenharmony_ci uint32_t idx; 13953a5a1b3Sopenharmony_ci 14053a5a1b3Sopenharmony_ci pa_assert(pd); 14153a5a1b3Sopenharmony_ci pa_assert(o); 14253a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(o) >= 1); 14353a5a1b3Sopenharmony_ci 14453a5a1b3Sopenharmony_ci if (!o->context) 14553a5a1b3Sopenharmony_ci goto finish; 14653a5a1b3Sopenharmony_ci 14753a5a1b3Sopenharmony_ci if (command != PA_COMMAND_REPLY) { 14853a5a1b3Sopenharmony_ci if (pa_context_handle_error(o->context, command, t, false) < 0) 14953a5a1b3Sopenharmony_ci goto finish; 15053a5a1b3Sopenharmony_ci 15153a5a1b3Sopenharmony_ci idx = PA_INVALID_INDEX; 15253a5a1b3Sopenharmony_ci } else if (pa_tagstruct_getu32(t, &idx) < 0 || 15353a5a1b3Sopenharmony_ci !pa_tagstruct_eof(t)) { 15453a5a1b3Sopenharmony_ci pa_context_fail(o->context, PA_ERR_PROTOCOL); 15553a5a1b3Sopenharmony_ci goto finish; 15653a5a1b3Sopenharmony_ci } 15753a5a1b3Sopenharmony_ci 15853a5a1b3Sopenharmony_ci if (o->callback) { 15953a5a1b3Sopenharmony_ci pa_context_play_sample_cb_t cb = (pa_context_play_sample_cb_t) o->callback; 16053a5a1b3Sopenharmony_ci cb(o->context, idx, o->userdata); 16153a5a1b3Sopenharmony_ci } 16253a5a1b3Sopenharmony_ci 16353a5a1b3Sopenharmony_cifinish: 16453a5a1b3Sopenharmony_ci pa_operation_done(o); 16553a5a1b3Sopenharmony_ci pa_operation_unref(o); 16653a5a1b3Sopenharmony_ci} 16753a5a1b3Sopenharmony_ci 16853a5a1b3Sopenharmony_cipa_operation *pa_context_play_sample(pa_context *c, const char *name, const char *dev, pa_volume_t volume, pa_context_success_cb_t cb, void *userdata) { 16953a5a1b3Sopenharmony_ci pa_operation *o; 17053a5a1b3Sopenharmony_ci pa_tagstruct *t; 17153a5a1b3Sopenharmony_ci uint32_t tag; 17253a5a1b3Sopenharmony_ci 17353a5a1b3Sopenharmony_ci pa_assert(c); 17453a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(c) >= 1); 17553a5a1b3Sopenharmony_ci 17653a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED); 17753a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE); 17853a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, name && *name, PA_ERR_INVALID); 17953a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, !dev || *dev, PA_ERR_INVALID); 18053a5a1b3Sopenharmony_ci 18153a5a1b3Sopenharmony_ci o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata); 18253a5a1b3Sopenharmony_ci 18353a5a1b3Sopenharmony_ci if (!dev) 18453a5a1b3Sopenharmony_ci dev = c->conf->default_sink; 18553a5a1b3Sopenharmony_ci 18653a5a1b3Sopenharmony_ci t = pa_tagstruct_command(c, PA_COMMAND_PLAY_SAMPLE, &tag); 18753a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, PA_INVALID_INDEX); 18853a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, dev); 18953a5a1b3Sopenharmony_ci 19053a5a1b3Sopenharmony_ci if (!PA_VOLUME_IS_VALID(volume) && c->version < 15) 19153a5a1b3Sopenharmony_ci volume = PA_VOLUME_NORM; 19253a5a1b3Sopenharmony_ci 19353a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, volume); 19453a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, name); 19553a5a1b3Sopenharmony_ci 19653a5a1b3Sopenharmony_ci if (c->version >= 13) { 19753a5a1b3Sopenharmony_ci pa_proplist *p = pa_proplist_new(); 19853a5a1b3Sopenharmony_ci pa_tagstruct_put_proplist(t, p); 19953a5a1b3Sopenharmony_ci pa_proplist_free(p); 20053a5a1b3Sopenharmony_ci } 20153a5a1b3Sopenharmony_ci 20253a5a1b3Sopenharmony_ci pa_pstream_send_tagstruct(c->pstream, t); 20353a5a1b3Sopenharmony_ci pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, play_sample_ack_callback, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref); 20453a5a1b3Sopenharmony_ci 20553a5a1b3Sopenharmony_ci return o; 20653a5a1b3Sopenharmony_ci} 20753a5a1b3Sopenharmony_ci 20853a5a1b3Sopenharmony_cipa_operation *pa_context_play_sample_with_proplist(pa_context *c, const char *name, const char *dev, pa_volume_t volume, const pa_proplist *p, pa_context_play_sample_cb_t cb, void *userdata) { 20953a5a1b3Sopenharmony_ci pa_operation *o; 21053a5a1b3Sopenharmony_ci pa_tagstruct *t; 21153a5a1b3Sopenharmony_ci uint32_t tag; 21253a5a1b3Sopenharmony_ci 21353a5a1b3Sopenharmony_ci pa_assert(c); 21453a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(c) >= 1); 21553a5a1b3Sopenharmony_ci 21653a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED); 21753a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE); 21853a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, name && *name, PA_ERR_INVALID); 21953a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, !dev || *dev, PA_ERR_INVALID); 22053a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, c->version >= 13, PA_ERR_NOTSUPPORTED); 22153a5a1b3Sopenharmony_ci 22253a5a1b3Sopenharmony_ci o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata); 22353a5a1b3Sopenharmony_ci 22453a5a1b3Sopenharmony_ci if (!dev) 22553a5a1b3Sopenharmony_ci dev = c->conf->default_sink; 22653a5a1b3Sopenharmony_ci 22753a5a1b3Sopenharmony_ci t = pa_tagstruct_command(c, PA_COMMAND_PLAY_SAMPLE, &tag); 22853a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, PA_INVALID_INDEX); 22953a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, dev); 23053a5a1b3Sopenharmony_ci 23153a5a1b3Sopenharmony_ci if (!PA_VOLUME_IS_VALID(volume) && c->version < 15) 23253a5a1b3Sopenharmony_ci volume = PA_VOLUME_NORM; 23353a5a1b3Sopenharmony_ci 23453a5a1b3Sopenharmony_ci pa_tagstruct_putu32(t, volume); 23553a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, name); 23653a5a1b3Sopenharmony_ci 23753a5a1b3Sopenharmony_ci if (p) 23853a5a1b3Sopenharmony_ci pa_tagstruct_put_proplist(t, p); 23953a5a1b3Sopenharmony_ci else { 24053a5a1b3Sopenharmony_ci pa_proplist *empty_proplist = pa_proplist_new(); 24153a5a1b3Sopenharmony_ci pa_tagstruct_put_proplist(t, empty_proplist); 24253a5a1b3Sopenharmony_ci pa_proplist_free(empty_proplist); 24353a5a1b3Sopenharmony_ci } 24453a5a1b3Sopenharmony_ci 24553a5a1b3Sopenharmony_ci pa_pstream_send_tagstruct(c->pstream, t); 24653a5a1b3Sopenharmony_ci pa_pdispatch_register_reply(c->pdispatch, tag, DEFAULT_TIMEOUT, play_sample_with_proplist_ack_callback, pa_operation_ref(o), (pa_free_cb_t) pa_operation_unref); 24753a5a1b3Sopenharmony_ci 24853a5a1b3Sopenharmony_ci return o; 24953a5a1b3Sopenharmony_ci} 25053a5a1b3Sopenharmony_ci 25153a5a1b3Sopenharmony_cipa_operation* pa_context_remove_sample(pa_context *c, const char *name, pa_context_success_cb_t cb, void *userdata) { 25253a5a1b3Sopenharmony_ci pa_operation *o; 25353a5a1b3Sopenharmony_ci pa_tagstruct *t; 25453a5a1b3Sopenharmony_ci uint32_t tag; 25553a5a1b3Sopenharmony_ci 25653a5a1b3Sopenharmony_ci pa_assert(c); 25753a5a1b3Sopenharmony_ci pa_assert(PA_REFCNT_VALUE(c) >= 1); 25853a5a1b3Sopenharmony_ci 25953a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, !pa_detect_fork(), PA_ERR_FORKED); 26053a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, c->state == PA_CONTEXT_READY, PA_ERR_BADSTATE); 26153a5a1b3Sopenharmony_ci PA_CHECK_VALIDITY_RETURN_NULL(c, name && *name, PA_ERR_INVALID); 26253a5a1b3Sopenharmony_ci 26353a5a1b3Sopenharmony_ci o = pa_operation_new(c, NULL, (pa_operation_cb_t) cb, userdata); 26453a5a1b3Sopenharmony_ci 26553a5a1b3Sopenharmony_ci t = pa_tagstruct_command(c, PA_COMMAND_REMOVE_SAMPLE, &tag); 26653a5a1b3Sopenharmony_ci pa_tagstruct_puts(t, name); 26753a5a1b3Sopenharmony_ci 26853a5a1b3Sopenharmony_ci pa_pstream_send_tagstruct(c->pstream, t); 26953a5a1b3Sopenharmony_ci 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); 27053a5a1b3Sopenharmony_ci 27153a5a1b3Sopenharmony_ci return o; 27253a5a1b3Sopenharmony_ci} 273