| 
									
										
										
										
											2004-07-16 19:56:36 +00:00
										 |  |  | /* $Id$ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***
 | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  |   This file is part of PulseAudio. | 
					
						
							| 
									
										
										
										
											2004-07-16 19:56:36 +00:00
										 |  |  |   | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  |   PulseAudio is free software; you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2004-11-14 14:58:54 +00:00
										 |  |  |   it under the terms of the GNU Lesser General Public License as published | 
					
						
							| 
									
										
										
										
											2004-07-16 19:56:36 +00:00
										 |  |  |   by the Free Software Foundation; either version 2 of the License, | 
					
						
							|  |  |  |   or (at your option) any later version. | 
					
						
							|  |  |  |   | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  |   PulseAudio is distributed in the hope that it will be useful, but | 
					
						
							| 
									
										
										
										
											2004-07-16 19:56:36 +00:00
										 |  |  |   WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 
					
						
							|  |  |  |   General Public License for more details. | 
					
						
							|  |  |  |   | 
					
						
							| 
									
										
										
										
											2004-11-14 14:58:54 +00:00
										 |  |  |   You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  |   along with PulseAudio; if not, write to the Free Software | 
					
						
							| 
									
										
										
										
											2004-07-16 19:56:36 +00:00
										 |  |  |   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 | 
					
						
							|  |  |  |   USA. | 
					
						
							|  |  |  | ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-16 19:16:42 +00:00
										 |  |  | #ifdef HAVE_CONFIG_H
 | 
					
						
							|  |  |  | #include <config.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  | #include <pulse/timeval.h>
 | 
					
						
							|  |  |  | #include <pulse/version.h>
 | 
					
						
							|  |  |  | #include <pulse/utf8.h>
 | 
					
						
							|  |  |  | #include <pulse/util.h>
 | 
					
						
							|  |  |  | #include <pulse/xmalloc.h>
 | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-19 21:53:48 +00:00
										 |  |  | #include <pulsecore/native-common.h>
 | 
					
						
							|  |  |  | #include <pulsecore/packet.h>
 | 
					
						
							|  |  |  | #include <pulsecore/client.h>
 | 
					
						
							|  |  |  | #include <pulsecore/source-output.h>
 | 
					
						
							|  |  |  | #include <pulsecore/sink-input.h>
 | 
					
						
							|  |  |  | #include <pulsecore/pstream.h>
 | 
					
						
							|  |  |  | #include <pulsecore/tagstruct.h>
 | 
					
						
							|  |  |  | #include <pulsecore/pdispatch.h>
 | 
					
						
							|  |  |  | #include <pulsecore/pstream-util.h>
 | 
					
						
							|  |  |  | #include <pulsecore/authkey.h>
 | 
					
						
							|  |  |  | #include <pulsecore/namereg.h>
 | 
					
						
							|  |  |  | #include <pulsecore/core-scache.h>
 | 
					
						
							|  |  |  | #include <pulsecore/core-subscribe.h>
 | 
					
						
							|  |  |  | #include <pulsecore/log.h>
 | 
					
						
							|  |  |  | #include <pulsecore/autoload.h>
 | 
					
						
							|  |  |  | #include <pulsecore/authkey-prop.h>
 | 
					
						
							|  |  |  | #include <pulsecore/strlist.h>
 | 
					
						
							|  |  |  | #include <pulsecore/props.h>
 | 
					
						
							|  |  |  | #include <pulsecore/sample-util.h>
 | 
					
						
							|  |  |  | #include <pulsecore/llist.h>
 | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #include <pulsecore/creds.h>
 | 
					
						
							|  |  |  | #include <pulsecore/core-util.h>
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  | #include <pulsecore/ipacl.h>
 | 
					
						
							| 
									
										
										
										
											2006-02-17 12:10:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | #include "protocol-native.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  | /* Kick a client if it doesn't authenticate within this time */ | 
					
						
							| 
									
										
										
										
											2006-04-23 19:49:01 +00:00
										 |  |  | #define AUTH_TIMEOUT 60
 | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Don't accept more connection than this */ | 
					
						
							| 
									
										
										
										
											2006-06-21 16:36:58 +00:00
										 |  |  | #define MAX_CONNECTIONS 64
 | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 21:02:27 +00:00
										 |  |  | #define MAX_MEMBLOCKQ_LENGTH (4*1024*1024) /* 4MB */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | struct connection; | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  | struct pa_protocol_native; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct record_stream { | 
					
						
							|  |  |  |     struct connection *connection; | 
					
						
							|  |  |  |     uint32_t index; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_source_output *source_output; | 
					
						
							|  |  |  |     pa_memblockq *memblockq; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     size_t fragment_size; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct playback_stream { | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     int type; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *connection; | 
					
						
							|  |  |  |     uint32_t index; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sink_input *sink_input; | 
					
						
							|  |  |  |     pa_memblockq *memblockq; | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |     size_t requested_bytes; | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     int drain_request; | 
					
						
							|  |  |  |     uint32_t drain_tag; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     uint32_t syncid; | 
					
						
							|  |  |  |     int underrun; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Sync group members */ | 
					
						
							|  |  |  |     PA_LLIST_FIELDS(struct playback_stream); | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | struct upload_stream { | 
					
						
							|  |  |  |     int type; | 
					
						
							|  |  |  |     struct connection *connection; | 
					
						
							|  |  |  |     uint32_t index; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_memchunk memchunk; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     size_t length; | 
					
						
							|  |  |  |     char *name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sample_spec sample_spec; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_channel_map channel_map; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct output_stream { | 
					
						
							|  |  |  |     int type; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     UPLOAD_STREAM, | 
					
						
							|  |  |  |     PLAYBACK_STREAM | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | struct connection { | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     int authorized; | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  |     uint32_t version; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_protocol_native *protocol; | 
					
						
							|  |  |  |     pa_client *client; | 
					
						
							|  |  |  |     pa_pstream *pstream; | 
					
						
							|  |  |  |     pa_pdispatch *pdispatch; | 
					
						
							|  |  |  |     pa_idxset *record_streams, *output_streams; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     uint32_t rrobin_index; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_subscription *subscription; | 
					
						
							|  |  |  |     pa_time_event *auth_timeout_event; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  | struct pa_protocol_native { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_module *module; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     int public; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_core *core; | 
					
						
							|  |  |  |     pa_socket_server *server; | 
					
						
							|  |  |  |     pa_idxset *connections; | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     uint8_t auth_cookie[PA_NATIVE_COOKIE_LENGTH]; | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  |     int auth_cookie_in_property; | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |     char *auth_group; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     pa_ip_acl *auth_ip_acl; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static int sink_input_peek_cb(pa_sink_input *i, pa_memchunk *chunk); | 
					
						
							|  |  |  | static void sink_input_drop_cb(pa_sink_input *i, const pa_memchunk *chunk, size_t length); | 
					
						
							|  |  |  | static void sink_input_kill_cb(pa_sink_input *i); | 
					
						
							|  |  |  | static pa_usec_t sink_input_get_latency_cb(pa_sink_input *i); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void request_bytes(struct playback_stream*s); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void source_output_kill_cb(pa_source_output *o); | 
					
						
							|  |  |  | static void source_output_push_cb(pa_source_output *o, const pa_memchunk *chunk); | 
					
						
							|  |  |  | static pa_usec_t source_output_get_latency_cb(pa_source_output *o); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void command_exit(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_create_playback_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_drain_playback_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_create_record_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_delete_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_auth(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_set_client_name(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_lookup(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_stat(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_playback_latency(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_record_latency(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_create_upload_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_finish_upload_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_play_sample(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_remove_sample(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_info(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_info_list(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_server_info(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_subscribe(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_set_volume(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  | static void command_set_mute(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_cork_playback_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | static void command_flush_playback_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_trigger_or_prebuf_playback_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_set_default_sink_or_source(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_set_stream_name(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_kill(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_load_module(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_unload_module(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_add_autoload(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_remove_autoload(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_autoload_info(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_get_autoload_info_list(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_cork_record_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | static void command_flush_record_stream(pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 22:41:02 +00:00
										 |  |  | static const pa_pdispatch_cb_t command_table[PA_COMMAND_MAX] = { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     [PA_COMMAND_ERROR] = NULL, | 
					
						
							|  |  |  |     [PA_COMMAND_TIMEOUT] = NULL, | 
					
						
							|  |  |  |     [PA_COMMAND_REPLY] = NULL, | 
					
						
							|  |  |  |     [PA_COMMAND_CREATE_PLAYBACK_STREAM] = command_create_playback_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_DELETE_PLAYBACK_STREAM] = command_delete_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_DRAIN_PLAYBACK_STREAM] = command_drain_playback_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_CREATE_RECORD_STREAM] = command_create_record_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_DELETE_RECORD_STREAM] = command_delete_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_AUTH] = command_auth, | 
					
						
							|  |  |  |     [PA_COMMAND_REQUEST] = NULL, | 
					
						
							|  |  |  |     [PA_COMMAND_EXIT] = command_exit, | 
					
						
							|  |  |  |     [PA_COMMAND_SET_CLIENT_NAME] = command_set_client_name, | 
					
						
							|  |  |  |     [PA_COMMAND_LOOKUP_SINK] = command_lookup, | 
					
						
							|  |  |  |     [PA_COMMAND_LOOKUP_SOURCE] = command_lookup, | 
					
						
							|  |  |  |     [PA_COMMAND_STAT] = command_stat, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_PLAYBACK_LATENCY] = command_get_playback_latency, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_RECORD_LATENCY] = command_get_record_latency, | 
					
						
							|  |  |  |     [PA_COMMAND_CREATE_UPLOAD_STREAM] = command_create_upload_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_DELETE_UPLOAD_STREAM] = command_delete_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_FINISH_UPLOAD_STREAM] = command_finish_upload_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_PLAY_SAMPLE] = command_play_sample, | 
					
						
							|  |  |  |     [PA_COMMAND_REMOVE_SAMPLE] = command_remove_sample, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SINK_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SOURCE_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_CLIENT_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_MODULE_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SINK_INPUT_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SOURCE_OUTPUT_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SAMPLE_INFO] = command_get_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SINK_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SOURCE_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_MODULE_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_CLIENT_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SINK_INPUT_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SOURCE_OUTPUT_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SAMPLE_INFO_LIST] = command_get_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_SERVER_INFO] = command_get_server_info, | 
					
						
							|  |  |  |     [PA_COMMAND_SUBSCRIBE] = command_subscribe, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [PA_COMMAND_SET_SINK_VOLUME] = command_set_volume, | 
					
						
							|  |  |  |     [PA_COMMAND_SET_SINK_INPUT_VOLUME] = command_set_volume, | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |     [PA_COMMAND_SET_SOURCE_VOLUME] = command_set_volume, | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  |     [PA_COMMAND_SET_SINK_MUTE] = command_set_mute, | 
					
						
							|  |  |  |     [PA_COMMAND_SET_SOURCE_MUTE] = command_set_mute, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     [PA_COMMAND_CORK_PLAYBACK_STREAM] = command_cork_playback_stream, | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     [PA_COMMAND_FLUSH_PLAYBACK_STREAM] = command_flush_playback_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_TRIGGER_PLAYBACK_STREAM] = command_trigger_or_prebuf_playback_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_PREBUF_PLAYBACK_STREAM] = command_trigger_or_prebuf_playback_stream, | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     [PA_COMMAND_CORK_RECORD_STREAM] = command_cork_record_stream, | 
					
						
							|  |  |  |     [PA_COMMAND_FLUSH_RECORD_STREAM] = command_flush_record_stream, | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     [PA_COMMAND_SET_DEFAULT_SINK] = command_set_default_sink_or_source, | 
					
						
							|  |  |  |     [PA_COMMAND_SET_DEFAULT_SOURCE] = command_set_default_sink_or_source, | 
					
						
							|  |  |  |     [PA_COMMAND_SET_PLAYBACK_STREAM_NAME] = command_set_stream_name,  | 
					
						
							|  |  |  |     [PA_COMMAND_SET_RECORD_STREAM_NAME] = command_set_stream_name, | 
					
						
							|  |  |  |     [PA_COMMAND_KILL_CLIENT] = command_kill, | 
					
						
							|  |  |  |     [PA_COMMAND_KILL_SINK_INPUT] = command_kill, | 
					
						
							|  |  |  |     [PA_COMMAND_KILL_SOURCE_OUTPUT] = command_kill, | 
					
						
							|  |  |  |     [PA_COMMAND_LOAD_MODULE] = command_load_module, | 
					
						
							|  |  |  |     [PA_COMMAND_UNLOAD_MODULE] = command_unload_module, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_AUTOLOAD_INFO] = command_get_autoload_info, | 
					
						
							|  |  |  |     [PA_COMMAND_GET_AUTOLOAD_INFO_LIST] = command_get_autoload_info_list, | 
					
						
							|  |  |  |     [PA_COMMAND_ADD_AUTOLOAD] = command_add_autoload, | 
					
						
							| 
									
										
										
										
											2006-02-22 21:02:27 +00:00
										 |  |  |     [PA_COMMAND_REMOVE_AUTOLOAD] = command_remove_autoload | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* structure management */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  | static struct upload_stream* upload_stream_new( | 
					
						
							|  |  |  |     struct connection *c, | 
					
						
							|  |  |  |     const pa_sample_spec *ss, | 
					
						
							|  |  |  |     const pa_channel_map *map, | 
					
						
							|  |  |  |     const char *name, size_t length) { | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct upload_stream *s; | 
					
						
							|  |  |  |     assert(c && ss && name && length); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s = pa_xnew(struct upload_stream, 1); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     s->type = UPLOAD_STREAM; | 
					
						
							|  |  |  |     s->connection = c; | 
					
						
							|  |  |  |     s->sample_spec = *ss; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     s->channel_map = *map; | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     s->name = pa_xstrdup(name); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s->memchunk.memblock = NULL; | 
					
						
							|  |  |  |     s->memchunk.index = 0; | 
					
						
							|  |  |  |     s->memchunk.length = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->length = length; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     pa_idxset_put(c->output_streams, s, &s->index); | 
					
						
							|  |  |  |     return s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void upload_stream_free(struct upload_stream *o) { | 
					
						
							|  |  |  |     assert(o && o->connection); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_idxset_remove_by_data(o->connection->output_streams, o, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(o->name); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     if (o->memchunk.memblock) | 
					
						
							|  |  |  |         pa_memblock_unref(o->memchunk.memblock); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(o); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  | static struct record_stream* record_stream_new( | 
					
						
							|  |  |  |     struct connection *c, | 
					
						
							|  |  |  |     pa_source *source, | 
					
						
							|  |  |  |     const pa_sample_spec *ss, | 
					
						
							|  |  |  |     const pa_channel_map *map, | 
					
						
							|  |  |  |     const char *name, | 
					
						
							|  |  |  |     size_t maxlength, | 
					
						
							|  |  |  |     size_t fragment_size) { | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_source_output *source_output; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     size_t base; | 
					
						
							|  |  |  |     assert(c && source && ss && name && maxlength); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     if (!(source_output = pa_source_output_new(source, __FILE__, name, ss, map, -1))) | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s = pa_xnew(struct record_stream, 1); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     s->connection = c; | 
					
						
							|  |  |  |     s->source_output = source_output; | 
					
						
							|  |  |  |     s->source_output->push = source_output_push_cb; | 
					
						
							|  |  |  |     s->source_output->kill = source_output_kill_cb; | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     s->source_output->get_latency = source_output_get_latency_cb; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     s->source_output->userdata = s; | 
					
						
							| 
									
										
										
										
											2004-07-10 20:56:38 +00:00
										 |  |  |     s->source_output->owner = c->protocol->module; | 
					
						
							|  |  |  |     s->source_output->client = c->client; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s->memblockq = pa_memblockq_new( | 
					
						
							|  |  |  |             0, | 
					
						
							|  |  |  |             maxlength, | 
					
						
							|  |  |  |             0, | 
					
						
							|  |  |  |             base = pa_frame_size(ss), | 
					
						
							|  |  |  |             1, | 
					
						
							|  |  |  |             0, | 
					
						
							|  |  |  |             NULL, | 
					
						
							|  |  |  |             c->protocol->core->memblock_stat); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     assert(s->memblockq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->fragment_size = (fragment_size/base)*base; | 
					
						
							|  |  |  |     if (!s->fragment_size) | 
					
						
							|  |  |  |         s->fragment_size = base; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_idxset_put(c->record_streams, s, &s->index); | 
					
						
							|  |  |  |     return s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | static void record_stream_free(struct record_stream* r) { | 
					
						
							|  |  |  |     assert(r && r->connection); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_idxset_remove_by_data(r->connection->record_streams, r, NULL); | 
					
						
							| 
									
										
										
										
											2004-09-14 20:53:25 +00:00
										 |  |  |     pa_source_output_disconnect(r->source_output); | 
					
						
							|  |  |  |     pa_source_output_unref(r->source_output); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_memblockq_free(r->memblockq); | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(r); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  | static struct playback_stream* playback_stream_new( | 
					
						
							|  |  |  |     struct connection *c, | 
					
						
							|  |  |  |     pa_sink *sink, | 
					
						
							|  |  |  |     const pa_sample_spec *ss, | 
					
						
							|  |  |  |     const pa_channel_map *map, | 
					
						
							|  |  |  |     const char *name, | 
					
						
							|  |  |  |     size_t maxlength, | 
					
						
							|  |  |  |     size_t tlength, | 
					
						
							|  |  |  |     size_t prebuf, | 
					
						
							|  |  |  |     size_t minreq, | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     pa_cvolume *volume, | 
					
						
							|  |  |  |     uint32_t syncid) { | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     struct playback_stream *s, *ssync; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sink_input *sink_input; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     pa_memblock *silence; | 
					
						
							|  |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     int64_t start_index; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     assert(c && sink && ss && name && maxlength); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     /* Find syncid group */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = pa_idxset_first(c->output_streams, &idx); ssync; ssync = pa_idxset_next(c->output_streams, &idx)) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         if (ssync->type != PLAYBACK_STREAM) | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         if (ssync->syncid == syncid) | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Synced streams must connect to the same sink */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     if (ssync && ssync->sink_input->sink != sink) | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-05-14 00:41:18 +00:00
										 |  |  |     if (!(sink_input = pa_sink_input_new(sink, __FILE__, name, ss, map, volume, 0, -1))) | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s = pa_xnew(struct playback_stream, 1); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     s->type = PLAYBACK_STREAM; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     s->connection = c; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s->syncid = syncid; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     s->sink_input = sink_input; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     s->underrun = 1; | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     s->sink_input->peek = sink_input_peek_cb; | 
					
						
							|  |  |  |     s->sink_input->drop = sink_input_drop_cb; | 
					
						
							|  |  |  |     s->sink_input->kill = sink_input_kill_cb; | 
					
						
							|  |  |  |     s->sink_input->get_latency = sink_input_get_latency_cb; | 
					
						
							|  |  |  |     s->sink_input->userdata = s; | 
					
						
							| 
									
										
										
										
											2004-07-10 20:56:38 +00:00
										 |  |  |     s->sink_input->owner = c->protocol->module; | 
					
						
							|  |  |  |     s->sink_input->client = c->client; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     if (ssync) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         /* Sync id found, now find head of list */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         PA_LLIST_FIND_HEAD(struct playback_stream, ssync, &ssync); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* Prepend ourselves */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         PA_LLIST_PREPEND(struct playback_stream, ssync, s); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* Set our start index to the current read index of the other grozp member(s) */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         assert(ssync->next); | 
					
						
							|  |  |  |         start_index = pa_memblockq_get_read_index(ssync->next->memblockq); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         /* This ia a new sync group */ | 
					
						
							|  |  |  |         PA_LLIST_INIT(struct playback_stream, s); | 
					
						
							|  |  |  |         start_index = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     silence = pa_silence_memblock_new(ss, 0, c->protocol->core->memblock_stat); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     s->memblockq = pa_memblockq_new( | 
					
						
							|  |  |  |             start_index, | 
					
						
							|  |  |  |             maxlength, | 
					
						
							|  |  |  |             tlength, | 
					
						
							|  |  |  |             pa_frame_size(ss), | 
					
						
							|  |  |  |             prebuf, | 
					
						
							|  |  |  |             minreq, | 
					
						
							|  |  |  |             silence, | 
					
						
							|  |  |  |             c->protocol->core->memblock_stat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_memblock_unref(silence); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |     s->requested_bytes = 0; | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     s->drain_request = 0; | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     pa_idxset_put(c->output_streams, s, &s->index); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     return s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void playback_stream_free(struct playback_stream* p) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     struct playback_stream *head; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(p && p->connection); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     if (p->drain_request) | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(p->connection->pstream, p->drain_tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PA_LLIST_FIND_HEAD(struct playback_stream, p, &head); | 
					
						
							|  |  |  |     PA_LLIST_REMOVE(struct playback_stream, head, p); | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     pa_idxset_remove_by_data(p->connection->output_streams, p, NULL); | 
					
						
							| 
									
										
										
										
											2004-09-14 20:53:25 +00:00
										 |  |  |     pa_sink_input_disconnect(p->sink_input); | 
					
						
							|  |  |  |     pa_sink_input_unref(p->sink_input); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_memblockq_free(p->memblockq); | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(p); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void connection_free(struct connection *c) { | 
					
						
							|  |  |  |     struct record_stream *r; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct output_stream *o; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(c && c->protocol); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_idxset_remove_by_data(c->protocol->connections, c, NULL); | 
					
						
							|  |  |  |     while ((r = pa_idxset_first(c->record_streams, NULL))) | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |         record_stream_free(r); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_idxset_free(c->record_streams, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     while ((o = pa_idxset_first(c->output_streams, NULL))) | 
					
						
							|  |  |  |         if (o->type == PLAYBACK_STREAM) | 
					
						
							|  |  |  |             playback_stream_free((struct playback_stream*) o); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             upload_stream_free((struct upload_stream*) o); | 
					
						
							|  |  |  |     pa_idxset_free(c->output_streams, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-15 00:02:26 +00:00
										 |  |  |     pa_pdispatch_unref(c->pdispatch); | 
					
						
							|  |  |  |     pa_pstream_close(c->pstream); | 
					
						
							|  |  |  |     pa_pstream_unref(c->pstream); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_client_free(c->client); | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (c->subscription) | 
					
						
							|  |  |  |         pa_subscription_free(c->subscription); | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (c->auth_timeout_event) | 
					
						
							|  |  |  |         c->protocol->core->mainloop->time_free(c->auth_timeout_event); | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(c); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | static void request_bytes(struct playback_stream *s) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *t; | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     size_t l; | 
					
						
							|  |  |  |     assert(s); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     if (!(l = pa_memblockq_missing(s->memblockq))) | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2004-09-10 22:35:12 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |     if (l <= s->requested_bytes) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     l -= s->requested_bytes; | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (l < pa_memblockq_get_minreq(s->memblockq)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |     s->requested_bytes += l; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     t = pa_tagstruct_new(NULL, 0); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(t); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_tagstruct_putu32(t, PA_COMMAND_REQUEST); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */ | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, s->index); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, l); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(s->connection->pstream, t); | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*     pa_log(__FILE__": Requesting %u bytes", l);  */ | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | static void send_memblock(struct connection *c) { | 
					
						
							|  |  |  |     uint32_t start; | 
					
						
							|  |  |  |     struct record_stream *r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     start = PA_IDXSET_INVALID; | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_memchunk chunk; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |          | 
					
						
							|  |  |  |         if (!(r = pa_idxset_rrobin(c->record_streams, &c->rrobin_index))) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (start == PA_IDXSET_INVALID) | 
					
						
							|  |  |  |             start = c->rrobin_index; | 
					
						
							|  |  |  |         else if (start == c->rrobin_index) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (pa_memblockq_peek(r->memblockq,  &chunk) >= 0) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             pa_memchunk schunk = chunk; | 
					
						
							| 
									
										
										
										
											2004-09-06 18:55:47 +00:00
										 |  |  |              | 
					
						
							|  |  |  |             if (schunk.length > r->fragment_size) | 
					
						
							|  |  |  |                 schunk.length = r->fragment_size; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             pa_pstream_send_memblock(c->pstream, r->index, 0, PA_SEEK_RELATIVE, &schunk); | 
					
						
							| 
									
										
										
										
											2004-09-06 18:55:47 +00:00
										 |  |  |             pa_memblockq_drop(r->memblockq, &chunk, schunk.length); | 
					
						
							|  |  |  |             pa_memblock_unref(schunk.memblock); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |              | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 19:04:21 +00:00
										 |  |  | static void send_playback_stream_killed(struct playback_stream *p) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *t; | 
					
						
							| 
									
										
										
										
											2004-07-10 19:04:21 +00:00
										 |  |  |     assert(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     t = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, PA_COMMAND_PLAYBACK_STREAM_KILLED); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */ | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, p->index); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(p->connection->pstream, t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void send_record_stream_killed(struct record_stream *r) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *t; | 
					
						
							| 
									
										
										
										
											2004-07-10 19:04:21 +00:00
										 |  |  |     assert(r); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     t = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, PA_COMMAND_RECORD_STREAM_KILLED); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */ | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, r->index); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(r->connection->pstream, t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | /*** sinkinput callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static int sink_input_peek_cb(pa_sink_input *i, pa_memchunk *chunk) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							|  |  |  |     assert(i && i->userdata && chunk); | 
					
						
							|  |  |  |     s = i->userdata; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     if (pa_memblockq_get_length(s->memblockq) <= 0 && !s->underrun) { | 
					
						
							|  |  |  |         pa_tagstruct *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Report that we're empty */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(t, PA_COMMAND_UNDERFLOW); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */ | 
					
						
							|  |  |  |         pa_tagstruct_putu32(t, s->index); | 
					
						
							|  |  |  |         pa_pstream_send_tagstruct(s->connection->pstream, t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         s->underrun = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     if (pa_memblockq_peek(s->memblockq, chunk) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*         pa_log(__FILE__": peek: failure");    */ | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*     pa_log(__FILE__": peek: %u", chunk->length);    */ | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void sink_input_drop_cb(pa_sink_input *i, const pa_memchunk *chunk, size_t length) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							|  |  |  |     assert(i && i->userdata && length); | 
					
						
							|  |  |  |     s = i->userdata; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     pa_memblockq_drop(s->memblockq, chunk, length); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     request_bytes(s); | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (s->drain_request && !pa_memblockq_is_readable(s->memblockq)) { | 
					
						
							|  |  |  |         pa_pstream_send_simple_ack(s->connection->pstream, s->drain_tag); | 
					
						
							|  |  |  |         s->drain_request = 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-08-27 01:29:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*     pa_log(__FILE__": after_drop: %u %u", pa_memblockq_get_length(s->memblockq), pa_memblockq_is_readable(s->memblockq));   */ | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void sink_input_kill_cb(pa_sink_input *i) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     assert(i && i->userdata); | 
					
						
							| 
									
										
										
										
											2004-07-10 19:04:21 +00:00
										 |  |  |     send_playback_stream_killed((struct playback_stream *) i->userdata); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     playback_stream_free((struct playback_stream *) i->userdata); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static pa_usec_t sink_input_get_latency_cb(pa_sink_input *i) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							|  |  |  |     assert(i && i->userdata); | 
					
						
							|  |  |  |     s = i->userdata; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |     /*pa_log(__FILE__": get_latency: %u", pa_memblockq_get_length(s->memblockq));*/ | 
					
						
							| 
									
										
										
										
											2004-08-27 01:29:49 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     return pa_bytes_to_usec(pa_memblockq_get_length(s->memblockq), &s->sink_input->sample_spec); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | /*** source_output callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void source_output_push_cb(pa_source_output *o, const pa_memchunk *chunk) { | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							|  |  |  |     assert(o && o->userdata && chunk); | 
					
						
							|  |  |  |     s = o->userdata; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     if (pa_memblockq_push_align(s->memblockq, chunk) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |         pa_log_warn(__FILE__": Failed to push data into output queue."); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     }  | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     if (!pa_pstream_is_pending(s->connection->pstream)) | 
					
						
							|  |  |  |         send_memblock(s->connection); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void source_output_kill_cb(pa_source_output *o) { | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     assert(o && o->userdata); | 
					
						
							| 
									
										
										
										
											2004-07-10 19:04:21 +00:00
										 |  |  |     send_record_stream_killed((struct record_stream *) o->userdata); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     record_stream_free((struct record_stream *) o->userdata); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static pa_usec_t source_output_get_latency_cb(pa_source_output *o) { | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							|  |  |  |     assert(o && o->userdata); | 
					
						
							|  |  |  |     s = o->userdata; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |     /*pa_log(__FILE__": get_latency: %u", pa_memblockq_get_length(s->memblockq));*/ | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     return pa_bytes_to_usec(pa_memblockq_get_length(s->memblockq), &o->sample_spec); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | /*** pdispatch callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  | static void protocol_error(struct connection *c) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |     pa_log(__FILE__": protocol error, kicking client"); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     connection_free(c); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  | #define CHECK_VALIDITY(pstream, expression, tag, error) do { \
 | 
					
						
							|  |  |  | if (!(expression)) { \ | 
					
						
							|  |  |  |     pa_pstream_send_error((pstream), (tag), (error)); \ | 
					
						
							|  |  |  |     return; \ | 
					
						
							|  |  |  | } \ | 
					
						
							|  |  |  | } while(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static pa_tagstruct *reply_new(uint32_t tag) { | 
					
						
							|  |  |  |     pa_tagstruct *reply; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     reply = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, PA_COMMAND_REPLY); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, tag); | 
					
						
							|  |  |  |     return reply; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_create_playback_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     struct playback_stream *s; | 
					
						
							| 
									
										
										
										
											2006-05-30 00:25:51 +00:00
										 |  |  |     uint32_t maxlength, tlength, prebuf, minreq, sink_index, syncid; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     const char *name, *sink_name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sample_spec ss; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_channel_map map; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							|  |  |  |     pa_sink *sink; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_cvolume volume; | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     int corked; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(c && t && c->protocol && c->protocol->core); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-01-28 01:07:09 +00:00
										 |  |  |     if (pa_tagstruct_get( | 
					
						
							|  |  |  |             t, | 
					
						
							|  |  |  |             PA_TAG_STRING, &name, | 
					
						
							|  |  |  |             PA_TAG_SAMPLE_SPEC, &ss, | 
					
						
							|  |  |  |             PA_TAG_CHANNEL_MAP, &map, | 
					
						
							|  |  |  |             PA_TAG_U32, &sink_index, | 
					
						
							|  |  |  |             PA_TAG_STRING, &sink_name, | 
					
						
							|  |  |  |             PA_TAG_U32, &maxlength, | 
					
						
							|  |  |  |             PA_TAG_BOOLEAN, &corked, | 
					
						
							|  |  |  |             PA_TAG_U32, &tlength, | 
					
						
							|  |  |  |             PA_TAG_U32, &prebuf, | 
					
						
							|  |  |  |             PA_TAG_U32, &minreq, | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             PA_TAG_U32, &syncid, | 
					
						
							| 
									
										
										
										
											2006-01-28 01:07:09 +00:00
										 |  |  |             PA_TAG_CVOLUME, &volume, | 
					
						
							|  |  |  |             PA_TAG_INVALID) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t) || | 
					
						
							|  |  |  |         !name) { | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, sink_index != PA_INVALID_INDEX || !sink_name || (*sink_name && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, pa_channel_map_valid(&map), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, pa_sample_spec_valid(&ss), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, pa_cvolume_valid(&volume), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, map.channels == ss.channels && volume.channels == ss.channels, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-05-30 00:25:51 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, maxlength > 0 && maxlength <= MAX_MEMBLOCKQ_LENGTH, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     if (sink_index != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |         sink = pa_idxset_get_by_index(c->protocol->core->sinks, sink_index); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |         sink = pa_namereg_get(c->protocol->core, sink_name, PA_NAMEREG_SINK, 1); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, sink, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s = playback_stream_new(c, sink, &ss, &map, name, maxlength, tlength, prebuf, minreq, &volume, syncid); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_sink_input_cork(s->sink_input, corked); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, s->index); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     assert(s->sink_input); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, s->sink_input->index); | 
					
						
							| 
									
										
										
										
											2004-09-10 22:35:12 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, s->requested_bytes = pa_memblockq_missing(s->memblockq)); | 
					
						
							| 
									
										
										
										
											2006-05-25 23:20:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (c->version >= 9) { | 
					
						
							|  |  |  |         /* Since 0.9 we support sending the buffer metrics back to the client */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) pa_memblockq_get_maxlength(s->memblockq)); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) pa_memblockq_get_tlength(s->memblockq)); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) pa_memblockq_get_prebuf(s->memblockq)); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) pa_memblockq_get_minreq(s->memblockq)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							| 
									
										
										
										
											2004-06-27 17:50:02 +00:00
										 |  |  |     request_bytes(s); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_delete_stream(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     uint32_t channel; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &channel) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     if (command == PA_COMMAND_DELETE_PLAYBACK_STREAM) { | 
					
						
							|  |  |  |         struct playback_stream *s; | 
					
						
							|  |  |  |         if (!(s = pa_idxset_get_by_index(c->output_streams, channel)) || (s->type != PLAYBACK_STREAM)) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             pa_pstream_send_error(c->pstream, tag, PA_ERR_EXIST); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         playback_stream_free(s); | 
					
						
							|  |  |  |     } else if (command == PA_COMMAND_DELETE_RECORD_STREAM) { | 
					
						
							|  |  |  |         struct record_stream *s; | 
					
						
							|  |  |  |         if (!(s = pa_idxset_get_by_index(c->record_streams, channel))) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             pa_pstream_send_error(c->pstream, tag, PA_ERR_EXIST); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         record_stream_free(s); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         struct upload_stream *s; | 
					
						
							|  |  |  |         assert(command == PA_COMMAND_DELETE_UPLOAD_STREAM); | 
					
						
							|  |  |  |         if (!(s = pa_idxset_get_by_index(c->output_streams, channel)) || (s->type != UPLOAD_STREAM)) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             pa_pstream_send_error(c->pstream, tag, PA_ERR_EXIST); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         upload_stream_free(s); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_create_record_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     struct record_stream *s; | 
					
						
							| 
									
										
										
										
											2006-02-27 09:09:15 +00:00
										 |  |  |     uint32_t maxlength, fragment_size; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     uint32_t source_index; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     const char *name, *source_name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sample_spec ss; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_channel_map map; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							|  |  |  |     pa_source *source; | 
					
						
							| 
									
										
										
										
											2004-09-29 17:38:45 +00:00
										 |  |  |     int corked; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     assert(c && t && c->protocol && c->protocol->core); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         pa_tagstruct_get_sample_spec(t, &ss) < 0 || | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |         pa_tagstruct_get_channel_map(t, &map) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         pa_tagstruct_getu32(t, &source_index) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         pa_tagstruct_gets(t, &source_name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         pa_tagstruct_getu32(t, &maxlength) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-29 17:38:45 +00:00
										 |  |  |         pa_tagstruct_get_boolean(t, &corked) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         pa_tagstruct_getu32(t, &fragment_size) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, pa_sample_spec_valid(&ss), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, pa_channel_map_valid(&map), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, source_index != PA_INVALID_INDEX || !source_name || (*source_name && pa_utf8_valid(source_name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, map.channels == ss.channels, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 21:02:27 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, maxlength <= MAX_MEMBLOCKQ_LENGTH, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (source_index != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |         source = pa_idxset_get_by_index(c->protocol->core->sources, source_index); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |         source = pa_namereg_get(c->protocol->core, source_name, PA_NAMEREG_SOURCE, 1); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, source, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     s = record_stream_new(c, source, &ss, &map, name, maxlength, fragment_size); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-09-29 17:38:45 +00:00
										 |  |  |     pa_source_output_cork(s->source_output, corked); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, s->index); | 
					
						
							|  |  |  |     assert(s->source_output); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, s->source_output->index); | 
					
						
							| 
									
										
										
										
											2006-05-25 23:20:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (c->version >= 9) { | 
					
						
							|  |  |  |         /* Since 0.9 we support sending the buffer metrics back to the client */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) pa_memblockq_get_maxlength(s->memblockq)); | 
					
						
							|  |  |  |         pa_tagstruct_putu32(reply, (uint32_t) s->fragment_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_exit(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     if (!pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     assert(c->protocol && c->protocol->core && c->protocol->core->mainloop); | 
					
						
							|  |  |  |     c->protocol->core->mainloop->quit(c->protocol->core->mainloop, 0); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); /* nonsense */ | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_auth(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const void*cookie; | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &c->version) < 0 || | 
					
						
							|  |  |  |         pa_tagstruct_get_arbitrary(t, &cookie, PA_NATIVE_COOKIE_LENGTH) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-09-12 23:40:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  |     /* Minimum supported version */ | 
					
						
							|  |  |  |     if (c->version < 8) { | 
					
						
							|  |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_VERSION); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-09-12 23:40:53 +00:00
										 |  |  |     if (!c->authorized) { | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |         int success = 0; | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							|  |  |  |         const pa_creds *creds; | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  |         if ((creds = pa_pdispatch_creds(pd))) { | 
					
						
							|  |  |  |             if (creds->uid == getuid()) | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |                 success = 1; | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |             else if (c->protocol->auth_group) { | 
					
						
							|  |  |  |                 int r; | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  |                 gid_t gid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if ((gid = pa_get_gid_of_group(c->protocol->auth_group)) == (gid_t) -1) | 
					
						
							|  |  |  |                     pa_log_warn(__FILE__": failed to get GID of group '%s'", c->protocol->auth_group); | 
					
						
							|  |  |  |                 else if (gid == creds->gid) | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |                     success = 1; | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  |                      | 
					
						
							|  |  |  |                 if (!success) { | 
					
						
							|  |  |  |                     if ((r = pa_uid_in_group(creds->uid, c->protocol->auth_group)) < 0) | 
					
						
							|  |  |  |                         pa_log_warn(__FILE__": failed to check group membership."); | 
					
						
							|  |  |  |                     else if (r > 0) | 
					
						
							|  |  |  |                         success = 1; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |                  | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  |             pa_log_info(__FILE__": Got credentials: uid=%lu gid=%lu success=%i", | 
					
						
							|  |  |  |                         (unsigned long) creds->uid, | 
					
						
							|  |  |  |                         (unsigned long) creds->gid, | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |                         success); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |         if (!success && memcmp(c->protocol->auth_cookie, cookie, PA_NATIVE_COOKIE_LENGTH) == 0) | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |             success = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!success) { | 
					
						
							|  |  |  |             pa_log_warn(__FILE__": Denied access to client with invalid authorization data."); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |             pa_pstream_send_error(c->pstream, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-09-12 23:40:53 +00:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2004-09-12 23:40:53 +00:00
										 |  |  |         c->authorized = 1; | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  |         if (c->auth_timeout_event) { | 
					
						
							|  |  |  |             c->protocol->core->mainloop->time_free(c->auth_timeout_event); | 
					
						
							|  |  |  |             c->auth_timeout_event = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-03-02 21:56:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     reply = reply_new(tag); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, PA_PROTOCOL_VERSION); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_set_client_name(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     pa_client_set_name(c->client, name); | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_lookup(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx = PA_IDXSET_INVALID; | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     if (command == PA_COMMAND_LOOKUP_SINK) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_sink *sink; | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |         if ((sink = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SINK, 1))) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             idx = sink->index; | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_source *source; | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         assert(command == PA_COMMAND_LOOKUP_SOURCE); | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |         if ((source = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SOURCE, 1))) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             idx = source->index; | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (idx == PA_IDXSET_INVALID) | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_tagstruct_putu32(reply, idx); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_drain_playback_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     s->drain_request = 0; | 
					
						
							| 
									
										
										
										
											2004-08-17 18:53:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_memblockq_prebuf_disable(s->memblockq); | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-09-06 17:47:04 +00:00
										 |  |  |     if (!pa_memblockq_is_readable(s->memblockq)) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*         pa_log("immediate drain: %u", pa_memblockq_get_length(s->memblockq));  */ | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |         pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2004-09-06 17:47:04 +00:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*         pa_log("slow drain triggered");  */ | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |         s->drain_request = 1; | 
					
						
							|  |  |  |         s->drain_tag = tag; | 
					
						
							| 
									
										
										
										
											2004-08-17 18:53:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pa_sink_notify(s->sink_input->sink); | 
					
						
							| 
									
										
										
										
											2004-07-07 22:02:15 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | }  | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_stat(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-15 20:51:55 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     assert(c && t); | 
					
						
							| 
									
										
										
										
											2004-07-15 20:51:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-07-15 20:51:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-08-17 19:37:29 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, c->protocol->core->memblock_stat->total); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, c->protocol->core->memblock_stat->total_size); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, c->protocol->core->memblock_stat->allocated); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, c->protocol->core->memblock_stat->allocated_size); | 
					
						
							| 
									
										
										
										
											2004-09-01 22:46:27 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, pa_scache_total_size(c->protocol->core)); | 
					
						
							| 
									
										
										
										
											2004-07-15 20:51:55 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_playback_latency(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-15 21:18:18 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-07-15 21:18:18 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							| 
									
										
										
										
											2004-09-10 22:35:12 +00:00
										 |  |  |     struct timeval tv, now; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2006-04-13 18:27:35 +00:00
										 |  |  |     pa_usec_t latency; | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     assert(c && t); | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-10 22:35:12 +00:00
										 |  |  |         pa_tagstruct_get_timeval(t, &tv) < 0 || | 
					
						
							| 
									
										
										
										
											2004-07-15 21:18:18 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-07-15 21:18:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2006-04-13 18:27:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     latency = pa_sink_get_latency(s->sink_input->sink); | 
					
						
							|  |  |  |     if (s->sink_input->resampled_chunk.memblock) | 
					
						
							|  |  |  |         latency += pa_bytes_to_usec(s->sink_input->resampled_chunk.length, &s->sink_input->sample_spec); | 
					
						
							|  |  |  |     pa_tagstruct_put_usec(reply, latency); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     pa_tagstruct_put_usec(reply, 0); | 
					
						
							| 
									
										
										
										
											2006-07-28 23:29:37 +00:00
										 |  |  |     pa_tagstruct_put_boolean(reply, s->sink_input->state == PA_SINK_INPUT_RUNNING); | 
					
						
							| 
									
										
										
										
											2004-09-10 22:35:12 +00:00
										 |  |  |     pa_tagstruct_put_timeval(reply, &tv); | 
					
						
							| 
									
										
										
										
											2006-04-07 00:23:38 +00:00
										 |  |  |     pa_tagstruct_put_timeval(reply, pa_gettimeofday(&now)); | 
					
						
							| 
									
										
										
										
											2006-03-02 14:22:25 +00:00
										 |  |  |     pa_tagstruct_puts64(reply, pa_memblockq_get_write_index(s->memblockq)); | 
					
						
							|  |  |  |     pa_tagstruct_puts64(reply, pa_memblockq_get_read_index(s->memblockq)); | 
					
						
							| 
									
										
										
										
											2004-07-15 21:18:18 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_record_latency(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							|  |  |  |     struct timeval tv, now; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |         pa_tagstruct_get_timeval(t, &tv) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->record_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     pa_tagstruct_put_usec(reply, s->source_output->source->monitor_of ? pa_sink_get_latency(s->source_output->source->monitor_of) : 0); | 
					
						
							|  |  |  |     pa_tagstruct_put_usec(reply, pa_source_get_latency(s->source_output->source)); | 
					
						
							|  |  |  |     pa_tagstruct_put_boolean(reply, 0); | 
					
						
							|  |  |  |     pa_tagstruct_put_timeval(reply, &tv); | 
					
						
							| 
									
										
										
										
											2006-04-07 00:23:38 +00:00
										 |  |  |     pa_tagstruct_put_timeval(reply, pa_gettimeofday(&now)); | 
					
						
							| 
									
										
										
										
											2006-03-02 14:22:25 +00:00
										 |  |  |     pa_tagstruct_puts64(reply, pa_memblockq_get_write_index(s->memblockq)); | 
					
						
							|  |  |  |     pa_tagstruct_puts64(reply, pa_memblockq_get_read_index(s->memblockq)); | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_create_upload_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     struct upload_stream *s; | 
					
						
							| 
									
										
										
										
											2006-02-27 09:09:15 +00:00
										 |  |  |     uint32_t length; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sample_spec ss; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_channel_map map; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     assert(c && t && c->protocol && c->protocol->core); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 ||  | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         pa_tagstruct_get_sample_spec(t, &ss) < 0 || | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |         pa_tagstruct_get_channel_map(t, &map) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         pa_tagstruct_getu32(t, &length) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, pa_sample_spec_valid(&ss), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, pa_channel_map_valid(&map), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, map.channels == ss.channels, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, (length % pa_frame_size(&ss)) == 0 && length > 0, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, length <= PA_SCACHE_ENTRY_SIZE_MAX, tag, PA_ERR_TOOLARGE); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     s = upload_stream_new(c, &ss, &map, name, length); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, s->index); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, length); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_finish_upload_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     uint32_t channel; | 
					
						
							|  |  |  |     struct upload_stream *s; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     if (pa_tagstruct_getu32(t, &channel) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     s = pa_idxset_get_by_index(c->output_streams, channel); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == UPLOAD_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     if (pa_scache_add_item(c->protocol->core, s->name, &s->sample_spec, &s->channel_map, &s->memchunk, &idx) < 0) | 
					
						
							|  |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_INTERNAL); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     upload_stream_free(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_play_sample(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     uint32_t sink_index; | 
					
						
							| 
									
										
										
										
											2006-04-23 19:49:01 +00:00
										 |  |  |     pa_volume_t volume; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sink *sink; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     const char *name, *sink_name; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pa_tagstruct_getu32(t, &sink_index) < 0 || | 
					
						
							|  |  |  |         pa_tagstruct_gets(t, &sink_name) < 0 || | 
					
						
							| 
									
										
										
										
											2006-04-23 19:49:01 +00:00
										 |  |  |         pa_tagstruct_getu32(t, &volume) < 0 || | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, sink_index != PA_INVALID_INDEX || !sink_name || (*sink_name && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (sink_index != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         sink = pa_idxset_get_by_index(c->protocol->core->sinks, sink_index); | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |         sink = pa_namereg_get(c->protocol->core, sink_name, PA_NAMEREG_SINK, 1); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, sink, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-23 19:49:01 +00:00
										 |  |  |     if (pa_scache_play_item(c->protocol->core, name, sink, volume) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_remove_sample(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (pa_scache_remove_item(c->protocol->core, name) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void sink_fill_tagstruct(pa_tagstruct *t, pa_sink *sink) { | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     assert(t && sink); | 
					
						
							| 
									
										
										
										
											2006-01-28 01:07:09 +00:00
										 |  |  |     pa_tagstruct_put( | 
					
						
							|  |  |  |         t, | 
					
						
							|  |  |  |         PA_TAG_U32, sink->index, | 
					
						
							|  |  |  |         PA_TAG_STRING, sink->name, | 
					
						
							|  |  |  |         PA_TAG_STRING, sink->description, | 
					
						
							|  |  |  |         PA_TAG_SAMPLE_SPEC, &sink->sample_spec, | 
					
						
							|  |  |  |         PA_TAG_CHANNEL_MAP, &sink->channel_map, | 
					
						
							|  |  |  |         PA_TAG_U32, sink->owner ? sink->owner->index : PA_INVALID_INDEX, | 
					
						
							|  |  |  |         PA_TAG_CVOLUME, pa_sink_get_volume(sink, PA_MIXER_HARDWARE), | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  |         PA_TAG_BOOLEAN, pa_sink_get_mute(sink, PA_MIXER_HARDWARE), | 
					
						
							| 
									
										
										
										
											2006-01-28 01:07:09 +00:00
										 |  |  |         PA_TAG_U32, sink->monitor_source->index, | 
					
						
							|  |  |  |         PA_TAG_STRING, sink->monitor_source->name, | 
					
						
							|  |  |  |         PA_TAG_USEC, pa_sink_get_latency(sink), | 
					
						
							|  |  |  |         PA_TAG_STRING, sink->driver, | 
					
						
							| 
									
										
										
										
											2006-07-16 17:28:10 +00:00
										 |  |  |         PA_TAG_U32, | 
					
						
							|  |  |  |         (sink->get_hw_volume ? PA_SINK_HW_VOLUME_CTRL : 0) | | 
					
						
							|  |  |  |         (sink->get_latency ? PA_SINK_LATENCY : 0) | | 
					
						
							|  |  |  |         (sink->is_hardware ? PA_SINK_HARDWARE : 0), | 
					
						
							| 
									
										
										
										
											2006-01-28 01:07:09 +00:00
										 |  |  |         PA_TAG_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void source_fill_tagstruct(pa_tagstruct *t, pa_source *source) { | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     assert(t && source); | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |     pa_tagstruct_put( | 
					
						
							|  |  |  |         t, | 
					
						
							|  |  |  |         PA_TAG_U32, source->index, | 
					
						
							|  |  |  |         PA_TAG_STRING, source->name, | 
					
						
							|  |  |  |         PA_TAG_STRING, source->description, | 
					
						
							|  |  |  |         PA_TAG_SAMPLE_SPEC, &source->sample_spec, | 
					
						
							|  |  |  |         PA_TAG_CHANNEL_MAP, &source->channel_map, | 
					
						
							|  |  |  |         PA_TAG_U32, source->owner ? source->owner->index : PA_INVALID_INDEX, | 
					
						
							|  |  |  |         PA_TAG_CVOLUME, pa_source_get_volume(source, PA_MIXER_HARDWARE), | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  |         PA_TAG_BOOLEAN, pa_source_get_mute(source, PA_MIXER_HARDWARE), | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |         PA_TAG_U32, source->monitor_of ? source->monitor_of->index : PA_INVALID_INDEX, | 
					
						
							|  |  |  |         PA_TAG_STRING, source->monitor_of ? source->monitor_of->name : NULL, | 
					
						
							|  |  |  |         PA_TAG_USEC, pa_source_get_latency(source), | 
					
						
							|  |  |  |         PA_TAG_STRING, source->driver, | 
					
						
							| 
									
										
										
										
											2006-07-16 17:28:10 +00:00
										 |  |  |         PA_TAG_U32, | 
					
						
							|  |  |  |         (source->get_hw_volume ? PA_SOURCE_HW_VOLUME_CTRL : 0) | | 
					
						
							|  |  |  |         (source->get_latency ? PA_SOURCE_LATENCY : 0) | | 
					
						
							|  |  |  |         (source->is_hardware ? PA_SOURCE_HARDWARE : 0), | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |         PA_TAG_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void client_fill_tagstruct(pa_tagstruct *t, pa_client *client) { | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     assert(t && client); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, client->index); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, client->name); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     pa_tagstruct_putu32(t, client->owner ? client->owner->index : PA_INVALID_INDEX); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_puts(t, client->driver); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void module_fill_tagstruct(pa_tagstruct *t, pa_module *module) { | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     assert(t && module); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, module->index); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, module->name); | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |     pa_tagstruct_puts(t, module->argument); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     pa_tagstruct_putu32(t, module->n_used); | 
					
						
							| 
									
										
										
										
											2004-09-04 00:27:36 +00:00
										 |  |  |     pa_tagstruct_put_boolean(t, module->auto_unload); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void sink_input_fill_tagstruct(pa_tagstruct *t, pa_sink_input *s) { | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     assert(t && s); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, s->index); | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |     pa_tagstruct_puts(t, s->name); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     pa_tagstruct_putu32(t, s->owner ? s->owner->index : PA_INVALID_INDEX); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, s->client ? s->client->index : PA_INVALID_INDEX); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     pa_tagstruct_putu32(t, s->sink->index); | 
					
						
							|  |  |  |     pa_tagstruct_put_sample_spec(t, &s->sample_spec); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_put_channel_map(t, &s->channel_map); | 
					
						
							|  |  |  |     pa_tagstruct_put_cvolume(t, &s->volume); | 
					
						
							| 
									
										
										
										
											2004-09-12 13:14:49 +00:00
										 |  |  |     pa_tagstruct_put_usec(t, pa_sink_input_get_latency(s)); | 
					
						
							|  |  |  |     pa_tagstruct_put_usec(t, pa_sink_get_latency(s->sink)); | 
					
						
							| 
									
										
										
										
											2004-11-20 16:23:53 +00:00
										 |  |  |     pa_tagstruct_puts(t, pa_resample_method_to_string(pa_sink_input_get_resample_method(s))); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_puts(t, s->driver); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void source_output_fill_tagstruct(pa_tagstruct *t, pa_source_output *s) { | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     assert(t && s); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, s->index); | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |     pa_tagstruct_puts(t, s->name); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     pa_tagstruct_putu32(t, s->owner ? s->owner->index : PA_INVALID_INDEX); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, s->client ? s->client->index : PA_INVALID_INDEX); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     pa_tagstruct_putu32(t, s->source->index); | 
					
						
							|  |  |  |     pa_tagstruct_put_sample_spec(t, &s->sample_spec); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_put_channel_map(t, &s->channel_map); | 
					
						
							| 
									
										
										
										
											2004-09-16 00:05:56 +00:00
										 |  |  |     pa_tagstruct_put_usec(t, pa_source_output_get_latency(s)); | 
					
						
							|  |  |  |     pa_tagstruct_put_usec(t, pa_source_get_latency(s->source)); | 
					
						
							| 
									
										
										
										
											2004-11-20 16:23:53 +00:00
										 |  |  |     pa_tagstruct_puts(t, pa_resample_method_to_string(pa_source_output_get_resample_method(s))); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_puts(t, s->driver); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void scache_fill_tagstruct(pa_tagstruct *t, pa_scache_entry *e) { | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     assert(t && e); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, e->index); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, e->name); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_put_cvolume(t, &e->volume); | 
					
						
							| 
									
										
										
										
											2004-09-12 13:14:49 +00:00
										 |  |  |     pa_tagstruct_put_usec(t, pa_bytes_to_usec(e->memchunk.length, &e->sample_spec)); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     pa_tagstruct_put_sample_spec(t, &e->sample_spec); | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_tagstruct_put_channel_map(t, &e->channel_map); | 
					
						
							| 
									
										
										
										
											2004-09-01 22:46:27 +00:00
										 |  |  |     pa_tagstruct_putu32(t, e->memchunk.length); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     pa_tagstruct_put_boolean(t, e->lazy); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, e->filename); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_info(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     pa_sink *sink = NULL; | 
					
						
							|  |  |  |     pa_source *source = NULL; | 
					
						
							|  |  |  |     pa_client *client = NULL; | 
					
						
							|  |  |  |     pa_module *module = NULL; | 
					
						
							|  |  |  |     pa_sink_input *si = NULL; | 
					
						
							|  |  |  |     pa_source_output *so = NULL; | 
					
						
							|  |  |  |     pa_scache_entry *sce = NULL; | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |         (command != PA_COMMAND_GET_CLIENT_INFO && | 
					
						
							|  |  |  |          command != PA_COMMAND_GET_MODULE_INFO && | 
					
						
							|  |  |  |          command != PA_COMMAND_GET_SINK_INPUT_INFO && | 
					
						
							|  |  |  |          command != PA_COMMAND_GET_SOURCE_OUTPUT_INFO && | 
					
						
							|  |  |  |          pa_tagstruct_gets(t, &name) < 0) || | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX || !name || (*name && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_GET_SINK_INFO) { | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         if (idx != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             sink = pa_idxset_get_by_index(c->protocol->core->sinks, idx); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |             sink = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SINK, 1); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     } else if (command == PA_COMMAND_GET_SOURCE_INFO) { | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         if (idx != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             source = pa_idxset_get_by_index(c->protocol->core->sources, idx); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |             source = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SOURCE, 1); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     } else if (command == PA_COMMAND_GET_CLIENT_INFO) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         client = pa_idxset_get_by_index(c->protocol->core->clients, idx); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_MODULE_INFO)  | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         module = pa_idxset_get_by_index(c->protocol->core->modules, idx); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_SINK_INPUT_INFO) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         si = pa_idxset_get_by_index(c->protocol->core->sink_inputs, idx); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_SOURCE_OUTPUT_INFO) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         so = pa_idxset_get_by_index(c->protocol->core->source_outputs, idx); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |         assert(command == PA_COMMAND_GET_SAMPLE_INFO); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         if (idx != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             sce = pa_idxset_get_by_index(c->protocol->core->scache, idx); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |         else | 
					
						
							|  |  |  |             sce = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SAMPLE, 0); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |              | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     if (!sink && !source && !client && !module && !si && !so && !sce) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     if (sink) | 
					
						
							|  |  |  |         sink_fill_tagstruct(reply, sink); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     else if (source) | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         source_fill_tagstruct(reply, source); | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     else if (client) | 
					
						
							|  |  |  |         client_fill_tagstruct(reply, client); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (module) | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |         module_fill_tagstruct(reply, module); | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (si) | 
					
						
							|  |  |  |         sink_input_fill_tagstruct(reply, si); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else if (so) | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |         source_output_fill_tagstruct(reply, so); | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else | 
					
						
							|  |  |  |         scache_fill_tagstruct(reply, sce); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_info_list(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_idxset *i; | 
					
						
							|  |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     void *p; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_GET_SINK_INFO_LIST) | 
					
						
							|  |  |  |         i = c->protocol->core->sinks; | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_SOURCE_INFO_LIST) | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         i = c->protocol->core->sources; | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_CLIENT_INFO_LIST) | 
					
						
							|  |  |  |         i = c->protocol->core->clients; | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_MODULE_INFO_LIST) | 
					
						
							| 
									
										
										
										
											2004-08-12 23:25:28 +00:00
										 |  |  |         i = c->protocol->core->modules; | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_SINK_INPUT_INFO_LIST) | 
					
						
							|  |  |  |         i = c->protocol->core->sink_inputs; | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else if (command == PA_COMMAND_GET_SOURCE_OUTPUT_INFO_LIST) | 
					
						
							| 
									
										
										
										
											2004-08-16 19:55:02 +00:00
										 |  |  |         i = c->protocol->core->source_outputs; | 
					
						
							| 
									
										
										
										
											2004-08-19 23:14:59 +00:00
										 |  |  |     else { | 
					
						
							|  |  |  |         assert(command == PA_COMMAND_GET_SAMPLE_INFO_LIST); | 
					
						
							|  |  |  |         i = c->protocol->core->scache; | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-08-20 10:54:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (i) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         for (p = pa_idxset_first(i, &idx); p; p = pa_idxset_next(i, &idx)) { | 
					
						
							| 
									
										
										
										
											2004-08-20 10:54:31 +00:00
										 |  |  |             if (command == PA_COMMAND_GET_SINK_INFO_LIST) | 
					
						
							|  |  |  |                 sink_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else if (command == PA_COMMAND_GET_SOURCE_INFO_LIST) | 
					
						
							|  |  |  |                 source_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else if (command == PA_COMMAND_GET_CLIENT_INFO_LIST) | 
					
						
							|  |  |  |                 client_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else if (command == PA_COMMAND_GET_MODULE_INFO_LIST) | 
					
						
							|  |  |  |                 module_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else if (command == PA_COMMAND_GET_SINK_INPUT_INFO_LIST) | 
					
						
							|  |  |  |                 sink_input_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else if (command == PA_COMMAND_GET_SOURCE_OUTPUT_INFO_LIST)  | 
					
						
							|  |  |  |                 source_output_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 assert(command == PA_COMMAND_GET_SAMPLE_INFO_LIST); | 
					
						
							|  |  |  |                 scache_fill_tagstruct(reply, p); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-08-20 10:54:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-08-05 19:53:57 +00:00
										 |  |  |      | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_server_info(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     char txt[256]; | 
					
						
							| 
									
										
										
										
											2004-09-07 14:58:42 +00:00
										 |  |  |     const char *n; | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     pa_tagstruct_puts(reply, PACKAGE_NAME); | 
					
						
							|  |  |  |     pa_tagstruct_puts(reply, PACKAGE_VERSION); | 
					
						
							|  |  |  |     pa_tagstruct_puts(reply, pa_get_user_name(txt, sizeof(txt))); | 
					
						
							| 
									
										
										
										
											2004-11-17 00:05:25 +00:00
										 |  |  |     pa_tagstruct_puts(reply, pa_get_fqdn(txt, sizeof(txt))); | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     pa_tagstruct_put_sample_spec(reply, &c->protocol->core->default_sample_spec); | 
					
						
							| 
									
										
										
										
											2004-09-07 14:58:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     n = pa_namereg_get_default_sink_name(c->protocol->core); | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |     pa_tagstruct_puts(reply, n); | 
					
						
							| 
									
										
										
										
											2004-09-07 14:58:42 +00:00
										 |  |  |     n = pa_namereg_get_default_source_name(c->protocol->core); | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |     pa_tagstruct_puts(reply, n); | 
					
						
							| 
									
										
										
										
											2004-12-12 22:58:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_tagstruct_putu32(reply, c->protocol->core->cookie); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-10 13:00:12 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  | static void subscription_cb(pa_core *core, pa_subscription_event_type_t e, uint32_t idx, void *userdata) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *t; | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     assert(c && core); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     t = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, PA_COMMAND_SUBSCRIBE_EVENT); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, (uint32_t) -1); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, e); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct_putu32(t, idx); | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_subscribe(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     pa_subscription_mask_t m; | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pa_tagstruct_getu32(t, &m) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, (m & ~PA_SUBSCRIPTION_MASK_ALL) == 0, tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     if (c->subscription) | 
					
						
							|  |  |  |         pa_subscription_free(c->subscription); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m != 0) { | 
					
						
							|  |  |  |         c->subscription = pa_subscription_new(c->protocol->core, m, subscription_cb, c); | 
					
						
							|  |  |  |         assert(c->subscription); | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         c->subscription = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  | static void command_set_volume( | 
					
						
							|  |  |  |         PA_GCC_UNUSED pa_pdispatch *pd, | 
					
						
							|  |  |  |         uint32_t command, | 
					
						
							|  |  |  |         uint32_t tag, | 
					
						
							|  |  |  |         pa_tagstruct *t, | 
					
						
							|  |  |  |         void *userdata) { | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     pa_cvolume volume; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sink *sink = NULL; | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |     pa_source *source = NULL; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_sink_input *si = NULL; | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |     const char *name = NULL; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |         (command == PA_COMMAND_SET_SINK_VOLUME && pa_tagstruct_gets(t, &name) < 0) || | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |         (command == PA_COMMAND_SET_SOURCE_VOLUME && pa_tagstruct_gets(t, &name) < 0) || | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |         pa_tagstruct_get_cvolume(t, &volume) || | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX || !name || (*name && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-04-23 21:01:44 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, pa_cvolume_valid(&volume), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_SET_SINK_VOLUME) { | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         if (idx != PA_INVALID_INDEX) | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             sink = pa_idxset_get_by_index(c->protocol->core->sinks, idx); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2004-09-19 23:12:41 +00:00
										 |  |  |             sink = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SINK, 1); | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |     } else if (command == PA_COMMAND_SET_SOURCE_VOLUME) { | 
					
						
							|  |  |  |         if (idx != (uint32_t) -1) | 
					
						
							|  |  |  |             source = pa_idxset_get_by_index(c->protocol->core->sources, idx); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             source = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SOURCE, 1); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |     }  else { | 
					
						
							|  |  |  |         assert(command == PA_COMMAND_SET_SINK_INPUT_VOLUME); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         si = pa_idxset_get_by_index(c->protocol->core->sink_inputs, idx); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, si || sink || source, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (sink) | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |         pa_sink_set_volume(sink, PA_MIXER_HARDWARE, &volume); | 
					
						
							| 
									
										
										
										
											2006-02-22 14:11:23 +00:00
										 |  |  |     else if (source) | 
					
						
							|  |  |  |         pa_source_set_volume(source, PA_MIXER_HARDWARE, &volume); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  |     else if (si) | 
					
						
							| 
									
										
										
										
											2006-01-27 16:25:31 +00:00
										 |  |  |         pa_sink_input_set_volume(si, &volume); | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  | static void command_set_mute( | 
					
						
							|  |  |  |         PA_GCC_UNUSED pa_pdispatch *pd, | 
					
						
							|  |  |  |         uint32_t command, | 
					
						
							|  |  |  |         uint32_t tag, | 
					
						
							|  |  |  |         pa_tagstruct *t, | 
					
						
							|  |  |  |         void *userdata) { | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     int mute; | 
					
						
							|  |  |  |     pa_sink *sink = NULL; | 
					
						
							|  |  |  |     pa_source *source = NULL; | 
					
						
							|  |  |  |     const char *name = NULL; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							|  |  |  |         pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							|  |  |  |         pa_tagstruct_get_boolean(t, &mute) || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX || !name || (*name && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-23 12:04:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_SET_SINK_MUTE) { | 
					
						
							|  |  |  |         if (idx != PA_INVALID_INDEX) | 
					
						
							|  |  |  |             sink = pa_idxset_get_by_index(c->protocol->core->sinks, idx); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             sink = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SINK, 1); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         assert(command == PA_COMMAND_SET_SOURCE_MUTE); | 
					
						
							|  |  |  |         if (idx != (uint32_t) -1) | 
					
						
							|  |  |  |             source = pa_idxset_get_by_index(c->protocol->core->sources, idx); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             source = pa_namereg_get(c->protocol->core, name, PA_NAMEREG_SOURCE, 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, sink || source, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sink) | 
					
						
							|  |  |  |         pa_sink_set_mute(sink, PA_MIXER_HARDWARE, mute); | 
					
						
							|  |  |  |     else if (source) | 
					
						
							|  |  |  |         pa_source_set_mute(source, PA_MIXER_HARDWARE, mute); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_cork_playback_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     int b; | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     struct playback_stream *s, *ssync; | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |         pa_tagstruct_get_boolean(t, &b) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX, tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     pa_sink_input_cork(s->sink_input, b); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     pa_memblockq_prebuf_force(s->memblockq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Do the same for all other members in the sync group */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->prev; ssync; ssync = ssync->prev) { | 
					
						
							|  |  |  |         pa_sink_input_cork(ssync->sink_input, b); | 
					
						
							|  |  |  |         pa_memblockq_prebuf_force(ssync->memblockq); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->next; ssync; ssync = ssync->next) { | 
					
						
							|  |  |  |         pa_sink_input_cork(ssync->sink_input, b); | 
					
						
							|  |  |  |         pa_memblockq_prebuf_force(ssync->memblockq); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | }  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void command_flush_playback_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							|  |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     struct playback_stream *s, *ssync; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX, tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_memblockq_flush(s->memblockq); | 
					
						
							|  |  |  |     s->underrun = 0; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     /* Do the same for all other members in the sync group */ | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->prev; ssync; ssync = ssync->prev) { | 
					
						
							|  |  |  |         pa_memblockq_flush(ssync->memblockq); | 
					
						
							|  |  |  |         ssync->underrun = 0; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->next; ssync; ssync = ssync->next) { | 
					
						
							|  |  |  |         pa_memblockq_flush(ssync->memblockq); | 
					
						
							|  |  |  |         ssync->underrun = 0; | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     pa_sink_notify(s->sink_input->sink); | 
					
						
							|  |  |  |     request_bytes(s); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->prev; ssync; ssync = ssync->prev) | 
					
						
							|  |  |  |         request_bytes(ssync); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     for (ssync = s->next; ssync; ssync = ssync->next) | 
					
						
							|  |  |  |         request_bytes(ssync); | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | static void command_trigger_or_prebuf_playback_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     struct playback_stream *s; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, idx != PA_INVALID_INDEX, tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     switch (command) { | 
					
						
							|  |  |  |         case PA_COMMAND_PREBUF_PLAYBACK_STREAM: | 
					
						
							|  |  |  |             pa_memblockq_prebuf_force(s->memblockq); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |         case PA_COMMAND_TRIGGER_PLAYBACK_STREAM: | 
					
						
							|  |  |  |             pa_memblockq_prebuf_disable(s->memblockq); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             abort(); | 
					
						
							| 
									
										
										
										
											2004-08-27 01:29:49 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_sink_notify(s->sink_input->sink); | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							| 
									
										
										
										
											2004-08-27 16:24:22 +00:00
										 |  |  |     request_bytes(s); | 
					
						
							| 
									
										
										
										
											2004-08-22 21:13:58 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2004-08-15 13:15:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_cork_record_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							|  |  |  |     int b; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |         pa_tagstruct_get_boolean(t, &b) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->record_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_source_output_cork(s->source_output, b); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |     pa_memblockq_prebuf_force(s->memblockq); | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_flush_record_stream(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |     struct record_stream *s; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     s = pa_idxset_get_by_index(c->record_streams, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-26 17:02:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_memblockq_flush(s->memblockq); | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_set_default_sink_or_source(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-06 21:55:09 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const char *s; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &s) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-06 21:55:09 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, !s || (*s && pa_utf8_valid(s)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-09-06 21:55:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_namereg_set_default(c->protocol->core, s, command == PA_COMMAND_SET_DEFAULT_SOURCE ? PA_NAMEREG_SOURCE : PA_NAMEREG_SINK); | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_set_stream_name(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     const char *name; | 
					
						
							|  |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_SET_PLAYBACK_STREAM_NAME) { | 
					
						
							|  |  |  |         struct playback_stream *s; | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         s = pa_idxset_get_by_index(c->output_streams, idx); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, s->type == PLAYBACK_STREAM, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pa_sink_input_set_name(s->sink_input, name); | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         struct record_stream *s; | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         s = pa_idxset_get_by_index(c->record_streams, idx); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pa_source_output_set_name(s->source_output, name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_kill(PA_GCC_UNUSED pa_pdispatch *pd, uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (command == PA_COMMAND_KILL_CLIENT) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_client *client; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         client = pa_idxset_get_by_index(c->protocol->core->clients, idx); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, client, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         pa_client_kill(client); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     } else if (command == PA_COMMAND_KILL_SINK_INPUT) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_sink_input *s; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         s = pa_idxset_get_by_index(c->protocol->core->sink_inputs, idx); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pa_sink_input_kill(s); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_source_output *s; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         assert(command == PA_COMMAND_KILL_SOURCE_OUTPUT); | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |         s = pa_idxset_get_by_index(c->protocol->core->source_outputs, idx); | 
					
						
							|  |  |  |         CHECK_VALIDITY(c->pstream, s, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pa_source_output_kill(s); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_load_module(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_module *m; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     const char *name, *argument; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         pa_tagstruct_gets(t, &argument) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name) && !strchr(name, '/'), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, !argument || pa_utf8_valid(argument), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     if (!(m = pa_module_load(c->protocol->core, name, argument))) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_MODINITFAILED); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, m->index); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_unload_module(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     pa_module *m; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_tagstruct_getu32(t, &idx) < 0 || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     m = pa_idxset_get_by_index(c->protocol->core->modules, idx); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, m, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_module_unload_request(m); | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_add_autoload(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     const char *name, *module, *argument; | 
					
						
							|  |  |  |     uint32_t type; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t idx; | 
					
						
							|  |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     if (pa_tagstruct_gets(t, &name) < 0 ||  | 
					
						
							|  |  |  |         pa_tagstruct_getu32(t, &type) < 0 ||  | 
					
						
							|  |  |  |         pa_tagstruct_gets(t, &module) < 0 ||  | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         pa_tagstruct_gets(t, &argument) < 0 || | 
					
						
							|  |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, name && *name && pa_utf8_valid(name), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, type == 0 || type == 1, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, module && *module && pa_utf8_valid(module), tag, PA_ERR_INVALID); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, !argument || pa_utf8_valid(argument), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_autoload_add(c->protocol->core, name, type == 0 ? PA_NAMEREG_SINK : PA_NAMEREG_SOURCE, module, argument, &idx) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_pstream_send_error(c->pstream, tag, PA_ERR_EXIST); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct_putu32(reply, idx); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_remove_autoload(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |     const char *name = NULL; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     uint32_t type, idx = PA_IDXSET_INVALID; | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |     int r; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if ((pa_tagstruct_getu32(t, &idx) < 0 && | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |         (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							|  |  |  |          pa_tagstruct_getu32(t, &type) < 0)) || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, name || idx != PA_IDXSET_INVALID, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, !name || (*name && pa_utf8_valid(name) && (type == 0 || type == 1)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |     if (name)  | 
					
						
							|  |  |  |         r = pa_autoload_remove_by_name(c->protocol->core, name, type == 0 ? PA_NAMEREG_SINK : PA_NAMEREG_SOURCE); | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         r = pa_autoload_remove_by_index(c->protocol->core, idx); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, r >= 0, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_pstream_send_simple_ack(c->pstream, tag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void autoload_fill_tagstruct(pa_tagstruct *t, const pa_autoload_entry *e) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(t && e); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, e->index); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     pa_tagstruct_puts(t, e->name); | 
					
						
							|  |  |  |     pa_tagstruct_putu32(t, e->type == PA_NAMEREG_SINK ? 0 : 1); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, e->module); | 
					
						
							|  |  |  |     pa_tagstruct_puts(t, e->argument); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_autoload_info(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     const pa_autoload_entry *a = NULL; | 
					
						
							|  |  |  |     uint32_t type, idx; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if ((pa_tagstruct_getu32(t, &idx) < 0 && | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  |         (pa_tagstruct_gets(t, &name) < 0 || | 
					
						
							|  |  |  |          pa_tagstruct_getu32(t, &type) < 0)) || | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         !pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							|  |  |  |     CHECK_VALIDITY(c->pstream, name || idx != PA_IDXSET_INVALID, tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2006-05-14 16:02:40 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, !name || (*name && (type == 0 || type == 1) && pa_utf8_valid(name)), tag, PA_ERR_INVALID); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (name) | 
					
						
							|  |  |  |         a = pa_autoload_get_by_name(c->protocol->core, name, type == 0 ? PA_NAMEREG_SINK : PA_NAMEREG_SOURCE); | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         a = pa_autoload_get_by_index(c->protocol->core, idx); | 
					
						
							| 
									
										
										
										
											2004-10-27 16:23:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, a, tag, PA_ERR_NOENTITY); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     autoload_fill_tagstruct(reply, a); | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void command_get_autoload_info_list(PA_GCC_UNUSED pa_pdispatch *pd, PA_GCC_UNUSED uint32_t command, uint32_t tag, pa_tagstruct *t, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_tagstruct *reply; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |     assert(c && t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!pa_tagstruct_eof(t)) { | 
					
						
							|  |  |  |         protocol_error(c); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     CHECK_VALIDITY(c->pstream, c->authorized, tag, PA_ERR_ACCESS); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     reply = reply_new(tag); | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (c->protocol->core->autoload_hashmap) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_autoload_entry *a; | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |         void *state = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-30 01:55:16 +00:00
										 |  |  |         while ((a = pa_hashmap_iterate(c->protocol->core->autoload_hashmap, &state, NULL))) | 
					
						
							| 
									
										
										
										
											2004-09-15 19:16:57 +00:00
										 |  |  |             autoload_fill_tagstruct(reply, a); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     pa_pstream_send_tagstruct(c->pstream, reply); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  | /*** pstream callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | static void pstream_packet_callback(pa_pstream *p, pa_packet *packet, const pa_creds *creds, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     assert(p && packet && packet->data && c); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     if (pa_pdispatch_run(c->pdispatch, packet, creds, c) < 0) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |         pa_log(__FILE__": invalid packet."); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         connection_free(c); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | static void pstream_memblock_callback(pa_pstream *p, uint32_t channel, int64_t offset, pa_seek_mode_t seek, const pa_memchunk *chunk, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     struct output_stream *stream; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(p && chunk && userdata); | 
					
						
							| 
									
										
										
										
											2004-09-26 22:27:04 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     if (!(stream = pa_idxset_get_by_index(c->output_streams, channel))) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |         pa_log(__FILE__": client sent block for invalid stream."); | 
					
						
							| 
									
										
										
										
											2004-07-07 00:22:46 +00:00
										 |  |  |         connection_free(c); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     if (stream->type == PLAYBACK_STREAM) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         struct playback_stream *ps = (struct playback_stream*) stream; | 
					
						
							|  |  |  |         if (chunk->length >= ps->requested_bytes) | 
					
						
							|  |  |  |             ps->requested_bytes = 0; | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             ps->requested_bytes -= chunk->length; | 
					
						
							| 
									
										
										
										
											2004-08-27 01:29:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  |         pa_memblockq_seek(ps->memblockq, offset, seek); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (pa_memblockq_push_align(ps->memblockq, chunk) < 0) { | 
					
						
							|  |  |  |             pa_tagstruct *t; | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |             pa_log_warn(__FILE__": failed to push data into queue"); | 
					
						
							| 
									
										
										
										
											2006-02-20 04:05:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* Pushing this block into the queue failed, so we simulate
 | 
					
						
							|  |  |  |              * it by skipping ahead */ | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             pa_memblockq_seek(ps->memblockq, chunk->length, PA_SEEK_RELATIVE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* Notify the user */ | 
					
						
							|  |  |  |             t = pa_tagstruct_new(NULL, 0); | 
					
						
							|  |  |  |             pa_tagstruct_putu32(t, PA_COMMAND_OVERFLOW); | 
					
						
							|  |  |  |             pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */ | 
					
						
							|  |  |  |             pa_tagstruct_putu32(t, ps->index); | 
					
						
							|  |  |  |             pa_pstream_send_tagstruct(p, t); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ps->underrun = 0; | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_sink_notify(ps->sink_input->sink); | 
					
						
							| 
									
										
										
										
											2004-08-27 01:29:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         struct upload_stream *u = (struct upload_stream*) stream; | 
					
						
							|  |  |  |         size_t l; | 
					
						
							|  |  |  |         assert(u->type == UPLOAD_STREAM); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!u->memchunk.memblock) { | 
					
						
							|  |  |  |             if (u->length == chunk->length) { | 
					
						
							|  |  |  |                 u->memchunk = *chunk; | 
					
						
							|  |  |  |                 pa_memblock_ref(u->memchunk.memblock); | 
					
						
							|  |  |  |                 u->length = 0; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2004-08-17 19:37:29 +00:00
										 |  |  |                 u->memchunk.memblock = pa_memblock_new(u->length, c->protocol->core->memblock_stat); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |                 u->memchunk.index = u->memchunk.length = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         assert(u->memchunk.memblock); | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         l = u->length;  | 
					
						
							|  |  |  |         if (l > chunk->length) | 
					
						
							|  |  |  |             l = chunk->length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (l > 0) { | 
					
						
							| 
									
										
										
										
											2004-09-01 00:23:51 +00:00
										 |  |  |             memcpy((uint8_t*) u->memchunk.memblock->data + u->memchunk.index + u->memchunk.length, | 
					
						
							|  |  |  |                    (uint8_t*) chunk->memblock->data+chunk->index, l); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |             u->memchunk.length += l; | 
					
						
							|  |  |  |             u->length -= l; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void pstream_die_callback(pa_pstream *p, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     assert(p && c); | 
					
						
							|  |  |  |     connection_free(c); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  | /*    pa_log(__FILE__": connection died.");*/ | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void pstream_drain_callback(pa_pstream *p, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     assert(p && c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     send_memblock(c); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  | /*** client callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void client_kill_cb(pa_client *c) { | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     assert(c && c->userdata); | 
					
						
							|  |  |  |     connection_free(c->userdata); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | /*** socket server callbacks ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void auth_timeout(pa_mainloop_api*m, pa_time_event *e, const struct timeval *tv, void *userdata) { | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  |     struct connection *c = userdata; | 
					
						
							|  |  |  |     assert(m && tv && c && c->auth_timeout_event == e); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!c->authorized) | 
					
						
							|  |  |  |         connection_free(c); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static void on_connection(PA_GCC_UNUSED pa_socket_server*s, pa_iochannel *io, void *userdata) { | 
					
						
							|  |  |  |     pa_protocol_native *p = userdata; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *c; | 
					
						
							| 
									
										
										
										
											2006-06-13 09:33:55 +00:00
										 |  |  |     char cname[256], pname[128]; | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  |     assert(io && p); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     if (pa_idxset_size(p->connections)+1 > MAX_CONNECTIONS) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |         pa_log_warn(__FILE__": Warning! Too many connections (%u), dropping incoming connection.", MAX_CONNECTIONS); | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  |         pa_iochannel_free(io); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     c = pa_xmalloc(sizeof(struct connection)); | 
					
						
							| 
									
										
										
										
											2004-09-14 23:08:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     c->authorized = !!p->public; | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     if (!c->authorized && p->auth_ip_acl && pa_ip_acl_check(p->auth_ip_acl, pa_iochannel_get_recv_fd(io)) > 0) { | 
					
						
							|  |  |  |         pa_log_info(__FILE__": Client authenticated by IP ACL."); | 
					
						
							|  |  |  |         c->authorized = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  |     if (!c->authorized) { | 
					
						
							|  |  |  |         struct timeval tv; | 
					
						
							| 
									
										
										
										
											2006-01-10 17:51:06 +00:00
										 |  |  |         pa_gettimeofday(&tv); | 
					
						
							| 
									
										
										
										
											2004-11-18 00:28:26 +00:00
										 |  |  |         tv.tv_sec += AUTH_TIMEOUT; | 
					
						
							|  |  |  |         c->auth_timeout_event = p->core->mainloop->time_new(p->core->mainloop, &tv, auth_timeout, c); | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         c->auth_timeout_event = NULL; | 
					
						
							| 
									
										
										
										
											2006-05-25 23:20:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     c->version = 8; | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     c->protocol = p; | 
					
						
							| 
									
										
										
										
											2006-06-13 09:33:55 +00:00
										 |  |  |     pa_iochannel_socket_peer_to_string(io, pname, sizeof(pname)); | 
					
						
							|  |  |  |     snprintf(cname, sizeof(cname), "Native client (%s)", pname); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(p->core); | 
					
						
							| 
									
										
										
										
											2006-06-13 09:33:55 +00:00
										 |  |  |     c->client = pa_client_new(p->core, __FILE__, cname); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(c->client); | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     c->client->kill = client_kill_cb; | 
					
						
							|  |  |  |     c->client->userdata = c; | 
					
						
							| 
									
										
										
										
											2004-07-10 20:56:38 +00:00
										 |  |  |     c->client->owner = p->module; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-08-17 19:37:29 +00:00
										 |  |  |     c->pstream = pa_pstream_new(p->core->mainloop, io, p->core->memblock_stat); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     assert(c->pstream); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     pa_pstream_set_recieve_packet_callback(c->pstream, pstream_packet_callback, c); | 
					
						
							|  |  |  |     pa_pstream_set_recieve_memblock_callback(c->pstream, pstream_memblock_callback, c); | 
					
						
							|  |  |  |     pa_pstream_set_die_callback(c->pstream, pstream_die_callback, c); | 
					
						
							|  |  |  |     pa_pstream_set_drain_callback(c->pstream, pstream_drain_callback, c); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     c->pdispatch = pa_pdispatch_new(p->core->mainloop, command_table, PA_COMMAND_MAX); | 
					
						
							| 
									
										
										
										
											2004-06-23 23:17:30 +00:00
										 |  |  |     assert(c->pdispatch); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     c->record_streams = pa_idxset_new(NULL, NULL); | 
					
						
							| 
									
										
										
										
											2004-08-03 19:26:56 +00:00
										 |  |  |     c->output_streams = pa_idxset_new(NULL, NULL); | 
					
						
							|  |  |  |     assert(c->record_streams && c->output_streams); | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  |     c->rrobin_index = PA_IDXSET_INVALID; | 
					
						
							| 
									
										
										
										
											2004-08-11 15:11:26 +00:00
										 |  |  |     c->subscription = NULL; | 
					
						
							| 
									
										
										
										
											2004-07-10 16:50:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_idxset_put(p->connections, c, NULL); | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							| 
									
										
										
										
											2006-02-24 15:12:42 +00:00
										 |  |  |     if (pa_iochannel_creds_supported(io)) | 
					
						
							|  |  |  |         pa_iochannel_creds_enable(io); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*** module entry points ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static int load_key(pa_protocol_native*p, const char*fn) { | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  |     assert(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     p->auth_cookie_in_property = 0; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     if (!fn && pa_authkey_prop_get(p->core, PA_NATIVE_COOKIE_PROPERTY_NAME, p->auth_cookie, sizeof(p->auth_cookie)) >= 0) { | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |         pa_log_info(__FILE__": using already loaded auth cookie."); | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  |         pa_authkey_prop_ref(p->core, PA_NATIVE_COOKIE_PROPERTY_NAME); | 
					
						
							|  |  |  |         p->auth_cookie_in_property = 1; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     if (!fn) | 
					
						
							|  |  |  |         fn = PA_NATIVE_COOKIE_FILE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-21 21:31:28 +00:00
										 |  |  |     if (pa_authkey_load_auto(fn, p->auth_cookie, sizeof(p->auth_cookie)) < 0) | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-23 02:27:19 +00:00
										 |  |  |     pa_log_info(__FILE__": loading cookie from disk."); | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (pa_authkey_prop_put(p->core, PA_NATIVE_COOKIE_PROPERTY_NAME, p->auth_cookie, sizeof(p->auth_cookie)) >= 0) | 
					
						
							|  |  |  |         p->auth_cookie_in_property = 1; | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | static pa_protocol_native* protocol_new_internal(pa_core *c, pa_module *m, pa_modargs *ma) { | 
					
						
							|  |  |  |     pa_protocol_native *p; | 
					
						
							| 
									
										
										
										
											2004-09-14 23:08:39 +00:00
										 |  |  |     int public = 0; | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     const char *acl; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     assert(c); | 
					
						
							|  |  |  |     assert(ma); | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |     if (pa_modargs_get_value_boolean(ma, "auth-anonymous", &public) < 0) { | 
					
						
							|  |  |  |         pa_log(__FILE__": auth-anonymous= expects a boolean argument."); | 
					
						
							| 
									
										
										
										
											2004-07-11 22:20:08 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-02-22 18:54:21 +00:00
										 |  |  |     p = pa_xnew(pa_protocol_native, 1); | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  |     p->core = c; | 
					
						
							|  |  |  |     p->module = m; | 
					
						
							|  |  |  |     p->public = public; | 
					
						
							|  |  |  |     p->server = NULL; | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     p->auth_ip_acl = NULL; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         int a = 1; | 
					
						
							|  |  |  |         if (pa_modargs_get_value_boolean(ma, "auth-group-enabled", &a) < 0) { | 
					
						
							|  |  |  |             pa_log(__FILE__": auth-group-enabled= expects a boolean argument."); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         p->auth_group = a ? pa_xstrdup(pa_modargs_get_value(ma, "auth-group", c->is_system_instance ? PA_ACCESS_GROUP : NULL)) : NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (p->auth_group) | 
					
						
							|  |  |  |             pa_log_info(__FILE__": Allowing access to group '%s'.", p->auth_group); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((acl = pa_modargs_get_value(ma, "auth-ip-acl", NULL))) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!(p->auth_ip_acl = pa_ip_acl_new(acl))) { | 
					
						
							|  |  |  |             pa_log(__FILE__": Failed to parse IP ACL '%s'", acl); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-07-06 00:08:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-07-10 20:56:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     if (load_key(p, pa_modargs_get_value(ma, "cookie", NULL)) < 0) | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     p->connections = pa_idxset_new(NULL, NULL); | 
					
						
							| 
									
										
										
										
											2004-06-29 16:48:37 +00:00
										 |  |  |     assert(p->connections); | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | fail: | 
					
						
							| 
									
										
										
										
											2006-07-20 22:46:41 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     pa_xfree(p->auth_group); | 
					
						
							| 
									
										
										
										
											2006-07-20 22:46:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     if (p->auth_ip_acl) | 
					
						
							|  |  |  |         pa_ip_acl_free(p->auth_ip_acl); | 
					
						
							|  |  |  |     pa_xfree(p); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | pa_protocol_native* pa_protocol_native_new(pa_core *core, pa_socket_server *server, pa_module *m, pa_modargs *ma) { | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  |     char t[256]; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |     pa_protocol_native *p; | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!(p = protocol_new_internal(core, m, ma))) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     p->server = server; | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_socket_server_set_callback(p->server, on_connection, p); | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (pa_socket_server_get_address(p->server, t, sizeof(t))) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |         pa_strlist *l; | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  |         l = pa_property_get(core, PA_NATIVE_SERVER_PROPERTY_NAME); | 
					
						
							|  |  |  |         l = pa_strlist_prepend(l, t); | 
					
						
							|  |  |  |         pa_property_replace(core, PA_NATIVE_SERVER_PROPERTY_NAME, l); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  |     return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | void pa_protocol_native_free(pa_protocol_native *p) { | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |     struct connection *c; | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  |     assert(p); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     while ((c = pa_idxset_first(p->connections, NULL))) | 
					
						
							| 
									
										
										
										
											2004-06-20 01:12:13 +00:00
										 |  |  |         connection_free(c); | 
					
						
							| 
									
										
										
										
											2004-07-03 23:35:12 +00:00
										 |  |  |     pa_idxset_free(p->connections, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  |     if (p->server) { | 
					
						
							|  |  |  |         char t[256]; | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         if (pa_socket_server_get_address(p->server, t, sizeof(t))) { | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  |             pa_strlist *l; | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  |             l = pa_property_get(p->core, PA_NATIVE_SERVER_PROPERTY_NAME); | 
					
						
							|  |  |  |             l = pa_strlist_remove(l, t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (l) | 
					
						
							|  |  |  |                 pa_property_replace(p->core, PA_NATIVE_SERVER_PROPERTY_NAME, l); | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 pa_property_remove(p->core, PA_NATIVE_SERVER_PROPERTY_NAME); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  |         pa_socket_server_unref(p->server); | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-11-07 20:48:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (p->auth_cookie_in_property) | 
					
						
							|  |  |  |         pa_authkey_prop_unref(p->core, PA_NATIVE_COOKIE_PROPERTY_NAME); | 
					
						
							| 
									
										
										
										
											2004-11-11 21:18:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 18:43:20 +00:00
										 |  |  |     if (p->auth_ip_acl) | 
					
						
							|  |  |  |         pa_ip_acl_free(p->auth_ip_acl); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2006-07-19 21:48:35 +00:00
										 |  |  | #ifdef HAVE_CREDS
 | 
					
						
							| 
									
										
										
										
											2006-02-24 17:14:23 +00:00
										 |  |  |     pa_xfree(p->auth_group); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-08-04 16:39:30 +00:00
										 |  |  |     pa_xfree(p); | 
					
						
							| 
									
										
										
										
											2004-06-08 23:54:24 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:17:39 +00:00
										 |  |  | pa_protocol_native* pa_protocol_native_new_iochannel(pa_core*core, pa_iochannel *io, pa_module *m, pa_modargs *ma) { | 
					
						
							|  |  |  |     pa_protocol_native *p; | 
					
						
							| 
									
										
										
										
											2004-09-01 21:12:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!(p = protocol_new_internal(core, m, ma))) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     on_connection(NULL, io, p); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     return p; | 
					
						
							|  |  |  | } |