]> code.delx.au - pulseaudio/blob - src/modules/dbus/iface-core.c
f2e34685f5c67b3dfa7ff45aedb8ab8e8c79a04d
[pulseaudio] / src / modules / dbus / iface-core.c
1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2009 Tanu Kaskinen
5
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as published
8 by the Free Software Foundation; either version 2.1 of the License,
9 or (at your option) any later version.
10
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 General Public License for more details.
15
16 You should have received a copy of the GNU Lesser General Public License
17 along with PulseAudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19 USA.
20 ***/
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <ctype.h>
27
28 #include <dbus/dbus.h>
29
30 #include <pulse/utf8.h>
31 #include <pulse/xmalloc.h>
32
33 #include <pulsecore/core-util.h>
34 #include <pulsecore/dbus-util.h>
35 #include <pulsecore/macro.h>
36 #include <pulsecore/namereg.h>
37 #include <pulsecore/protocol-dbus.h>
38 #include <pulsecore/socket-util.h>
39 #include <pulsecore/strbuf.h>
40
41 #include "iface-card.h"
42 #include "iface-client.h"
43 #include "iface-device.h"
44 #include "iface-memstats.h"
45 #include "iface-module.h"
46 #include "iface-sample.h"
47 #include "iface-stream.h"
48
49 #include "iface-core.h"
50
51 #define INTERFACE_REVISION 0
52
53 static void handle_get_interface_revision(DBusConnection *conn, DBusMessage *msg, void *userdata);
54 static void handle_get_name(DBusConnection *conn, DBusMessage *msg, void *userdata);
55 static void handle_get_version(DBusConnection *conn, DBusMessage *msg, void *userdata);
56 static void handle_get_is_local(DBusConnection *conn, DBusMessage *msg, void *userdata);
57 static void handle_get_username(DBusConnection *conn, DBusMessage *msg, void *userdata);
58 static void handle_get_hostname(DBusConnection *conn, DBusMessage *msg, void *userdata);
59 static void handle_get_default_channels(DBusConnection *conn, DBusMessage *msg, void *userdata);
60 static void handle_set_default_channels(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata);
61 static void handle_get_default_sample_format(DBusConnection *conn, DBusMessage *msg, void *userdata);
62 static void handle_set_default_sample_format(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata);
63 static void handle_get_default_sample_rate(DBusConnection *conn, DBusMessage *msg, void *userdata);
64 static void handle_set_default_sample_rate(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata);
65 static void handle_get_cards(DBusConnection *conn, DBusMessage *msg, void *userdata);
66 static void handle_get_sinks(DBusConnection *conn, DBusMessage *msg, void *userdata);
67 static void handle_get_fallback_sink(DBusConnection *conn, DBusMessage *msg, void *userdata);
68 static void handle_set_fallback_sink(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata);
69 static void handle_get_sources(DBusConnection *conn, DBusMessage *msg, void *userdata);
70 static void handle_get_fallback_source(DBusConnection *conn, DBusMessage *msg, void *userdata);
71 static void handle_set_fallback_source(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata);
72 static void handle_get_playback_streams(DBusConnection *conn, DBusMessage *msg, void *userdata);
73 static void handle_get_record_streams(DBusConnection *conn, DBusMessage *msg, void *userdata);
74 static void handle_get_samples(DBusConnection *conn, DBusMessage *msg, void *userdata);
75 static void handle_get_modules(DBusConnection *conn, DBusMessage *msg, void *userdata);
76 static void handle_get_clients(DBusConnection *conn, DBusMessage *msg, void *userdata);
77 static void handle_get_my_client(DBusConnection *conn, DBusMessage *msg, void *userdata);
78 static void handle_get_extensions(DBusConnection *conn, DBusMessage *msg, void *userdata);
79
80 static void handle_get_all(DBusConnection *conn, DBusMessage *msg, void *userdata);
81
82 static void handle_get_card_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata);
83 static void handle_get_sink_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata);
84 static void handle_get_source_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata);
85 static void handle_get_sample_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata);
86 static void handle_upload_sample(DBusConnection *conn, DBusMessage *msg, void *userdata);
87 static void handle_load_module(DBusConnection *conn, DBusMessage *msg, void *userdata);
88 static void handle_exit(DBusConnection *conn, DBusMessage *msg, void *userdata);
89 static void handle_listen_for_signal(DBusConnection *conn, DBusMessage *msg, void *userdata);
90 static void handle_stop_listening_for_signal(DBusConnection *conn, DBusMessage *msg, void *userdata);
91
92 struct pa_dbusiface_core {
93 pa_core *core;
94 pa_subscription *subscription;
95
96 pa_dbus_protocol *dbus_protocol;
97
98 pa_hashmap *cards;
99 pa_hashmap *sinks_by_index;
100 pa_hashmap *sinks_by_path;
101 pa_hashmap *sources_by_index;
102 pa_hashmap *sources_by_path;
103 pa_hashmap *playback_streams;
104 pa_hashmap *record_streams;
105 pa_hashmap *samples;
106 pa_hashmap *modules;
107 pa_hashmap *clients;
108
109 pa_sink *fallback_sink;
110 pa_source *fallback_source;
111
112 pa_hook_slot *sink_put_slot;
113 pa_hook_slot *sink_unlink_slot;
114 pa_hook_slot *source_put_slot;
115 pa_hook_slot *source_unlink_slot;
116 pa_hook_slot *extension_registered_slot;
117 pa_hook_slot *extension_unregistered_slot;
118
119 pa_dbusiface_memstats *memstats;
120 };
121
122 enum property_handler_index {
123 PROPERTY_HANDLER_INTERFACE_REVISION,
124 PROPERTY_HANDLER_NAME,
125 PROPERTY_HANDLER_VERSION,
126 PROPERTY_HANDLER_IS_LOCAL,
127 PROPERTY_HANDLER_USERNAME,
128 PROPERTY_HANDLER_HOSTNAME,
129 PROPERTY_HANDLER_DEFAULT_CHANNELS,
130 PROPERTY_HANDLER_DEFAULT_SAMPLE_FORMAT,
131 PROPERTY_HANDLER_DEFAULT_SAMPLE_RATE,
132 PROPERTY_HANDLER_CARDS,
133 PROPERTY_HANDLER_SINKS,
134 PROPERTY_HANDLER_FALLBACK_SINK,
135 PROPERTY_HANDLER_SOURCES,
136 PROPERTY_HANDLER_FALLBACK_SOURCE,
137 PROPERTY_HANDLER_PLAYBACK_STREAMS,
138 PROPERTY_HANDLER_RECORD_STREAMS,
139 PROPERTY_HANDLER_SAMPLES,
140 PROPERTY_HANDLER_MODULES,
141 PROPERTY_HANDLER_CLIENTS,
142 PROPERTY_HANDLER_MY_CLIENT,
143 PROPERTY_HANDLER_EXTENSIONS,
144 PROPERTY_HANDLER_MAX
145 };
146
147 static pa_dbus_property_handler property_handlers[PROPERTY_HANDLER_MAX] = {
148 [PROPERTY_HANDLER_INTERFACE_REVISION] = { .property_name = "InterfaceRevision", .type = "u", .get_cb = handle_get_interface_revision, .set_cb = NULL },
149 [PROPERTY_HANDLER_NAME] = { .property_name = "Name", .type = "s", .get_cb = handle_get_name, .set_cb = NULL },
150 [PROPERTY_HANDLER_VERSION] = { .property_name = "Version", .type = "s", .get_cb = handle_get_version, .set_cb = NULL },
151 [PROPERTY_HANDLER_IS_LOCAL] = { .property_name = "IsLocal", .type = "b", .get_cb = handle_get_is_local, .set_cb = NULL },
152 [PROPERTY_HANDLER_USERNAME] = { .property_name = "Username", .type = "s", .get_cb = handle_get_username, .set_cb = NULL },
153 [PROPERTY_HANDLER_HOSTNAME] = { .property_name = "Hostname", .type = "s", .get_cb = handle_get_hostname, .set_cb = NULL },
154 [PROPERTY_HANDLER_DEFAULT_CHANNELS] = { .property_name = "DefaultChannels", .type = "au", .get_cb = handle_get_default_channels, .set_cb = handle_set_default_channels },
155 [PROPERTY_HANDLER_DEFAULT_SAMPLE_FORMAT] = { .property_name = "DefaultSampleFormat", .type = "u", .get_cb = handle_get_default_sample_format, .set_cb = handle_set_default_sample_format },
156 [PROPERTY_HANDLER_DEFAULT_SAMPLE_RATE] = { .property_name = "DefaultSampleRate", .type = "u", .get_cb = handle_get_default_sample_rate, .set_cb = handle_set_default_sample_rate },
157 [PROPERTY_HANDLER_CARDS] = { .property_name = "Cards", .type = "ao", .get_cb = handle_get_cards, .set_cb = NULL },
158 [PROPERTY_HANDLER_SINKS] = { .property_name = "Sinks", .type = "ao", .get_cb = handle_get_sinks, .set_cb = NULL },
159 [PROPERTY_HANDLER_FALLBACK_SINK] = { .property_name = "FallbackSink", .type = "o", .get_cb = handle_get_fallback_sink, .set_cb = handle_set_fallback_sink },
160 [PROPERTY_HANDLER_SOURCES] = { .property_name = "Sources", .type = "ao", .get_cb = handle_get_sources, .set_cb = NULL },
161 [PROPERTY_HANDLER_FALLBACK_SOURCE] = { .property_name = "FallbackSource", .type = "o", .get_cb = handle_get_fallback_source, .set_cb = handle_set_fallback_source },
162 [PROPERTY_HANDLER_PLAYBACK_STREAMS] = { .property_name = "PlaybackStreams", .type = "ao", .get_cb = handle_get_playback_streams, .set_cb = NULL },
163 [PROPERTY_HANDLER_RECORD_STREAMS] = { .property_name = "RecordStreams", .type = "ao", .get_cb = handle_get_record_streams, .set_cb = NULL },
164 [PROPERTY_HANDLER_SAMPLES] = { .property_name = "Samples", .type = "ao", .get_cb = handle_get_samples, .set_cb = NULL },
165 [PROPERTY_HANDLER_MODULES] = { .property_name = "Modules", .type = "ao", .get_cb = handle_get_modules, .set_cb = NULL },
166 [PROPERTY_HANDLER_CLIENTS] = { .property_name = "Clients", .type = "ao", .get_cb = handle_get_clients, .set_cb = NULL },
167 [PROPERTY_HANDLER_MY_CLIENT] = { .property_name = "MyClient", .type = "o", .get_cb = handle_get_my_client, .set_cb = NULL },
168 [PROPERTY_HANDLER_EXTENSIONS] = { .property_name = "Extensions", .type = "as", .get_cb = handle_get_extensions, .set_cb = NULL }
169 };
170
171 enum method_handler_index {
172 METHOD_HANDLER_GET_CARD_BY_NAME,
173 METHOD_HANDLER_GET_SINK_BY_NAME,
174 METHOD_HANDLER_GET_SOURCE_BY_NAME,
175 METHOD_HANDLER_GET_SAMPLE_BY_NAME,
176 METHOD_HANDLER_UPLOAD_SAMPLE,
177 METHOD_HANDLER_LOAD_MODULE,
178 METHOD_HANDLER_EXIT,
179 METHOD_HANDLER_LISTEN_FOR_SIGNAL,
180 METHOD_HANDLER_STOP_LISTENING_FOR_SIGNAL,
181 METHOD_HANDLER_MAX
182 };
183
184 static pa_dbus_arg_info get_card_by_name_args[] = { { "name", "s", "in" }, { "card", "o", "out" } };
185 static pa_dbus_arg_info get_sink_by_name_args[] = { { "name", "s", "in" }, { "sink", "o", "out" } };
186 static pa_dbus_arg_info get_source_by_name_args[] = { { "name", "s", "in" }, { "source", "o", "out" } };
187 static pa_dbus_arg_info get_sample_by_name_args[] = { { "name", "s", "in" }, { "sample", "o", "out" } };
188 static pa_dbus_arg_info upload_sample_args[] = { { "name", "s", "in" },
189 { "sample_format", "u", "in" },
190 { "sample_rate", "u", "in" },
191 { "channels", "au", "in" },
192 { "default_volume", "au", "in" },
193 { "property_list", "a{say}", "in" },
194 { "data", "ay", "in" },
195 { "sample", "o", "out" } };
196 static pa_dbus_arg_info load_module_args[] = { { "name", "s", "in" }, { "arguments", "a{ss}", "in" }, { "module", "o", "out" } };
197 static pa_dbus_arg_info listen_for_signal_args[] = { { "signal", "s", "in" }, { "objects", "ao", "in" } };
198 static pa_dbus_arg_info stop_listening_for_signal_args[] = { { "signal", "s", "in" } };
199
200 static pa_dbus_method_handler method_handlers[METHOD_HANDLER_MAX] = {
201 [METHOD_HANDLER_GET_CARD_BY_NAME] = {
202 .method_name = "GetCardByName",
203 .arguments = get_card_by_name_args,
204 .n_arguments = sizeof(get_card_by_name_args) / sizeof(pa_dbus_arg_info),
205 .receive_cb = handle_get_card_by_name },
206 [METHOD_HANDLER_GET_SINK_BY_NAME] = {
207 .method_name = "GetSinkByName",
208 .arguments = get_sink_by_name_args,
209 .n_arguments = sizeof(get_sink_by_name_args) / sizeof(pa_dbus_arg_info),
210 .receive_cb = handle_get_sink_by_name },
211 [METHOD_HANDLER_GET_SOURCE_BY_NAME] = {
212 .method_name = "GetSourceByName",
213 .arguments = get_source_by_name_args,
214 .n_arguments = sizeof(get_source_by_name_args) / sizeof(pa_dbus_arg_info),
215 .receive_cb = handle_get_source_by_name },
216 [METHOD_HANDLER_GET_SAMPLE_BY_NAME] = {
217 .method_name = "GetSampleByName",
218 .arguments = get_sample_by_name_args,
219 .n_arguments = sizeof(get_sample_by_name_args) / sizeof(pa_dbus_arg_info),
220 .receive_cb = handle_get_sample_by_name },
221 [METHOD_HANDLER_UPLOAD_SAMPLE] = {
222 .method_name = "UploadSample",
223 .arguments = upload_sample_args,
224 .n_arguments = sizeof(upload_sample_args) / sizeof(pa_dbus_arg_info),
225 .receive_cb = handle_upload_sample },
226 [METHOD_HANDLER_LOAD_MODULE] = {
227 .method_name = "LoadModule",
228 .arguments = load_module_args,
229 .n_arguments = sizeof(load_module_args) / sizeof(pa_dbus_arg_info),
230 .receive_cb = handle_load_module },
231 [METHOD_HANDLER_EXIT] = {
232 .method_name = "Exit",
233 .arguments = NULL,
234 .n_arguments = 0,
235 .receive_cb = handle_exit },
236 [METHOD_HANDLER_LISTEN_FOR_SIGNAL] = {
237 .method_name = "ListenForSignal",
238 .arguments = listen_for_signal_args,
239 .n_arguments = sizeof(listen_for_signal_args) / sizeof(pa_dbus_arg_info),
240 .receive_cb = handle_listen_for_signal },
241 [METHOD_HANDLER_STOP_LISTENING_FOR_SIGNAL] = {
242 .method_name = "StopListeningForSignal",
243 .arguments = stop_listening_for_signal_args,
244 .n_arguments = sizeof(stop_listening_for_signal_args) / sizeof(pa_dbus_arg_info),
245 .receive_cb = handle_stop_listening_for_signal }
246 };
247
248 enum signal_index {
249 SIGNAL_NEW_CARD,
250 SIGNAL_CARD_REMOVED,
251 SIGNAL_NEW_SINK,
252 SIGNAL_SINK_REMOVED,
253 SIGNAL_FALLBACK_SINK_UPDATED,
254 SIGNAL_FALLBACK_SINK_UNSET,
255 SIGNAL_NEW_SOURCE,
256 SIGNAL_SOURCE_REMOVED,
257 SIGNAL_FALLBACK_SOURCE_UPDATED,
258 SIGNAL_FALLBACK_SOURCE_UNSET,
259 SIGNAL_NEW_PLAYBACK_STREAM,
260 SIGNAL_PLAYBACK_STREAM_REMOVED,
261 SIGNAL_NEW_RECORD_STREAM,
262 SIGNAL_RECORD_STREAM_REMOVED,
263 SIGNAL_NEW_SAMPLE,
264 SIGNAL_SAMPLE_REMOVED,
265 SIGNAL_NEW_MODULE,
266 SIGNAL_MODULE_REMOVED,
267 SIGNAL_NEW_CLIENT,
268 SIGNAL_CLIENT_REMOVED,
269 SIGNAL_NEW_EXTENSION,
270 SIGNAL_EXTENSION_REMOVED,
271 SIGNAL_MAX
272 };
273
274 static pa_dbus_arg_info new_card_args[] = { { "card", "o", NULL } };
275 static pa_dbus_arg_info card_removed_args[] = { { "card", "o", NULL } };
276 static pa_dbus_arg_info new_sink_args[] = { { "sink", "o", NULL } };
277 static pa_dbus_arg_info sink_removed_args[] = { { "sink", "o", NULL } };
278 static pa_dbus_arg_info fallback_sink_updated_args[] = { { "sink", "o", NULL } };
279 static pa_dbus_arg_info new_source_args[] = { { "source", "o", NULL } };
280 static pa_dbus_arg_info source_removed_args[] = { { "source", "o", NULL } };
281 static pa_dbus_arg_info fallback_source_updated_args[] = { { "source", "o", NULL } };
282 static pa_dbus_arg_info new_playback_stream_args[] = { { "playback_stream", "o", NULL } };
283 static pa_dbus_arg_info playback_stream_removed_args[] = { { "playback_stream", "o", NULL } };
284 static pa_dbus_arg_info new_record_stream_args[] = { { "record_stream", "o", NULL } };
285 static pa_dbus_arg_info record_stream_removed_args[] = { { "record_stream", "o", NULL } };
286 static pa_dbus_arg_info new_sample_args[] = { { "sample", "o", NULL } };
287 static pa_dbus_arg_info sample_removed_args[] = { { "sample", "o", NULL } };
288 static pa_dbus_arg_info new_module_args[] = { { "module", "o", NULL } };
289 static pa_dbus_arg_info module_removed_args[] = { { "module", "o", NULL } };
290 static pa_dbus_arg_info new_client_args[] = { { "client", "o", NULL } };
291 static pa_dbus_arg_info client_removed_args[] = { { "client", "o", NULL } };
292 static pa_dbus_arg_info new_extension_args[] = { { "extension", "s", NULL } };
293 static pa_dbus_arg_info extension_removed_args[] = { { "extension", "s", NULL } };
294
295 static pa_dbus_signal_info signals[SIGNAL_MAX] = {
296 [SIGNAL_NEW_CARD] = { .name = "NewCard", .arguments = new_card_args, .n_arguments = 1 },
297 [SIGNAL_CARD_REMOVED] = { .name = "CardRemoved", .arguments = card_removed_args, .n_arguments = 1 },
298 [SIGNAL_NEW_SINK] = { .name = "NewSink", .arguments = new_sink_args, .n_arguments = 1 },
299 [SIGNAL_SINK_REMOVED] = { .name = "SinkRemoved", .arguments = sink_removed_args, .n_arguments = 1 },
300 [SIGNAL_FALLBACK_SINK_UPDATED] = { .name = "FallbackSinkUpdated", .arguments = fallback_sink_updated_args, .n_arguments = 1 },
301 [SIGNAL_FALLBACK_SINK_UNSET] = { .name = "FallbackSinkUnset", .arguments = NULL, .n_arguments = 0 },
302 [SIGNAL_NEW_SOURCE] = { .name = "NewSource", .arguments = new_source_args, .n_arguments = 1 },
303 [SIGNAL_SOURCE_REMOVED] = { .name = "SourceRemoved", .arguments = source_removed_args, .n_arguments = 1 },
304 [SIGNAL_FALLBACK_SOURCE_UPDATED] = { .name = "FallbackSourceUpdated", .arguments = fallback_source_updated_args, .n_arguments = 1 },
305 [SIGNAL_FALLBACK_SOURCE_UNSET] = { .name = "FallbackSourceUnset", .arguments = NULL, .n_arguments = 0 },
306 [SIGNAL_NEW_PLAYBACK_STREAM] = { .name = "NewPlaybackStream", .arguments = new_playback_stream_args, .n_arguments = 1 },
307 [SIGNAL_PLAYBACK_STREAM_REMOVED] = { .name = "PlaybackStreamRemoved", .arguments = playback_stream_removed_args, .n_arguments = 1 },
308 [SIGNAL_NEW_RECORD_STREAM] = { .name = "NewRecordStream", .arguments = new_record_stream_args, .n_arguments = 1 },
309 [SIGNAL_RECORD_STREAM_REMOVED] = { .name = "RecordStreamRemoved", .arguments = record_stream_removed_args, .n_arguments = 1 },
310 [SIGNAL_NEW_SAMPLE] = { .name = "NewSample", .arguments = new_sample_args, .n_arguments = 1 },
311 [SIGNAL_SAMPLE_REMOVED] = { .name = "SampleRemoved", .arguments = sample_removed_args, .n_arguments = 1 },
312 [SIGNAL_NEW_MODULE] = { .name = "NewModule", .arguments = new_module_args, .n_arguments = 1 },
313 [SIGNAL_MODULE_REMOVED] = { .name = "ModuleRemoved", .arguments = module_removed_args, .n_arguments = 1 },
314 [SIGNAL_NEW_CLIENT] = { .name = "NewClient", .arguments = new_client_args, .n_arguments = 1 },
315 [SIGNAL_CLIENT_REMOVED] = { .name = "ClientRemoved", .arguments = client_removed_args, .n_arguments = 1 },
316 [SIGNAL_NEW_EXTENSION] = { .name = "NewExtension", .arguments = new_extension_args, .n_arguments = 1 },
317 [SIGNAL_EXTENSION_REMOVED] = { .name = "ExtensionRemoved", .arguments = extension_removed_args, .n_arguments = 1 }
318 };
319
320 static pa_dbus_interface_info core_interface_info = {
321 .name = PA_DBUS_CORE_INTERFACE,
322 .method_handlers = method_handlers,
323 .n_method_handlers = METHOD_HANDLER_MAX,
324 .property_handlers = property_handlers,
325 .n_property_handlers = PROPERTY_HANDLER_MAX,
326 .get_all_properties_cb = handle_get_all,
327 .signals = signals,
328 .n_signals = SIGNAL_MAX
329 };
330
331 static void handle_get_interface_revision(DBusConnection *conn, DBusMessage *msg, void *userdata) {
332 dbus_uint32_t interface_revision = INTERFACE_REVISION;
333
334 pa_assert(conn);
335 pa_assert(msg);
336
337 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_UINT32, &interface_revision);
338 }
339
340 static void handle_get_name(DBusConnection *conn, DBusMessage *msg, void *userdata) {
341 const char *server_name = PACKAGE_NAME;
342
343 pa_assert(conn);
344 pa_assert(msg);
345
346 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_STRING, &server_name);
347 }
348
349 static void handle_get_version(DBusConnection *conn, DBusMessage *msg, void *userdata) {
350 const char *version = PACKAGE_VERSION;
351
352 pa_assert(conn);
353 pa_assert(msg);
354
355 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_STRING, &version);
356 }
357
358 static dbus_bool_t get_is_local(DBusConnection *conn) {
359 int conn_fd;
360
361 pa_assert(conn);
362
363 if (!dbus_connection_get_socket(conn, &conn_fd))
364 return FALSE;
365
366 return pa_socket_is_local(conn_fd);
367 }
368
369 static void handle_get_is_local(DBusConnection *conn, DBusMessage *msg, void *userdata) {
370 dbus_bool_t is_local;
371
372 pa_assert(conn);
373 pa_assert(msg);
374
375 is_local = get_is_local(conn);
376
377 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_BOOLEAN, &is_local);
378 }
379
380 static void handle_get_username(DBusConnection *conn, DBusMessage *msg, void *userdata) {
381 char *username = NULL;
382
383 pa_assert(conn);
384 pa_assert(msg);
385
386 username = pa_get_user_name_malloc();
387
388 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_STRING, &username);
389
390 pa_xfree(username);
391 }
392
393 static void handle_get_hostname(DBusConnection *conn, DBusMessage *msg, void *userdata) {
394 char *hostname = NULL;
395
396 pa_assert(conn);
397 pa_assert(msg);
398
399 hostname = pa_get_host_name_malloc();
400
401 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_STRING, &hostname);
402
403 pa_xfree(hostname);
404 }
405
406 /* Caller frees the returned array. */
407 static dbus_uint32_t *get_default_channels(pa_dbusiface_core *c, unsigned *n) {
408 dbus_uint32_t *default_channels = NULL;
409 unsigned i;
410
411 pa_assert(c);
412 pa_assert(n);
413
414 *n = c->core->default_channel_map.channels;
415 default_channels = pa_xnew(dbus_uint32_t, *n);
416
417 for (i = 0; i < *n; ++i)
418 default_channels[i] = c->core->default_channel_map.map[i];
419
420 return default_channels;
421 }
422
423 static void handle_get_default_channels(DBusConnection *conn, DBusMessage *msg, void *userdata) {
424 pa_dbusiface_core *c = userdata;
425 dbus_uint32_t *default_channels = NULL;
426 unsigned n;
427
428 pa_assert(conn);
429 pa_assert(msg);
430 pa_assert(c);
431
432 default_channels = get_default_channels(c, &n);
433
434 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_UINT32, default_channels, n);
435
436 pa_xfree(default_channels);
437 }
438
439 static void handle_set_default_channels(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata) {
440 pa_dbusiface_core *c = userdata;
441 DBusMessageIter array_iter;
442 pa_channel_map new_channel_map;
443 const dbus_uint32_t *default_channels;
444 int n_channels;
445 unsigned i;
446
447 pa_assert(conn);
448 pa_assert(msg);
449 pa_assert(iter);
450 pa_assert(c);
451
452 pa_channel_map_init(&new_channel_map);
453
454 dbus_message_iter_recurse(iter, &array_iter);
455 dbus_message_iter_get_fixed_array(&array_iter, &default_channels, &n_channels);
456
457 if (n_channels <= 0) {
458 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Empty channel array.");
459 return;
460 }
461
462 if (n_channels > (int) PA_CHANNELS_MAX) {
463 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS,
464 "Too many channels: %i. The maximum number of channels is %u.", n_channels, PA_CHANNELS_MAX);
465 return;
466 }
467
468 new_channel_map.channels = n_channels;
469
470 for (i = 0; i < new_channel_map.channels; ++i) {
471 if (default_channels[i] >= PA_CHANNEL_POSITION_MAX) {
472 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid channel position: %u.", default_channels[i]);
473 return;
474 }
475
476 new_channel_map.map[i] = default_channels[i];
477 }
478
479 c->core->default_channel_map = new_channel_map;
480 c->core->default_sample_spec.channels = n_channels;
481
482 pa_dbus_send_empty_reply(conn, msg);
483 }
484
485 static void handle_get_default_sample_format(DBusConnection *conn, DBusMessage *msg, void *userdata) {
486 pa_dbusiface_core *c = userdata;
487 dbus_uint32_t default_sample_format;
488
489 pa_assert(conn);
490 pa_assert(msg);
491 pa_assert(c);
492
493 default_sample_format = c->core->default_sample_spec.format;
494
495 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_UINT32, &default_sample_format);
496 }
497
498 static void handle_set_default_sample_format(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata) {
499 pa_dbusiface_core *c = userdata;
500 dbus_uint32_t default_sample_format;
501
502 pa_assert(conn);
503 pa_assert(msg);
504 pa_assert(iter);
505 pa_assert(c);
506
507 dbus_message_iter_get_basic(iter, &default_sample_format);
508
509 if (default_sample_format >= PA_SAMPLE_MAX) {
510 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid sample format.");
511 return;
512 }
513
514 c->core->default_sample_spec.format = default_sample_format;
515
516 pa_dbus_send_empty_reply(conn, msg);
517 }
518
519 static void handle_get_default_sample_rate(DBusConnection *conn, DBusMessage *msg, void *userdata) {
520 pa_dbusiface_core *c = userdata;
521 dbus_uint32_t default_sample_rate;
522
523 pa_assert(conn);
524 pa_assert(msg);
525 pa_assert(c);
526
527 default_sample_rate = c->core->default_sample_spec.rate;
528
529 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_UINT32, &default_sample_rate);
530 }
531
532 static void handle_set_default_sample_rate(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata) {
533 pa_dbusiface_core *c = userdata;
534 dbus_uint32_t default_sample_rate;
535
536 pa_assert(conn);
537 pa_assert(msg);
538 pa_assert(iter);
539 pa_assert(c);
540
541 dbus_message_iter_get_basic(iter, &default_sample_rate);
542
543 if (default_sample_rate <= 0 || default_sample_rate > PA_RATE_MAX) {
544 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid sample rate.");
545 return;
546 }
547
548 c->core->default_sample_spec.rate = default_sample_rate;
549
550 pa_dbus_send_empty_reply(conn, msg);
551 }
552
553 /* The caller frees the array, but not the strings. */
554 static const char **get_cards(pa_dbusiface_core *c, unsigned *n) {
555 const char **cards;
556 unsigned i = 0;
557 void *state = NULL;
558 pa_dbusiface_card *card;
559
560 pa_assert(c);
561 pa_assert(n);
562
563 *n = pa_hashmap_size(c->cards);
564
565 if (*n == 0)
566 return NULL;
567
568 cards = pa_xnew(const char *, *n);
569
570 PA_HASHMAP_FOREACH(card, c->cards, state)
571 cards[i++] = pa_dbusiface_card_get_path(card);
572
573 return cards;
574 }
575
576 static void handle_get_cards(DBusConnection *conn, DBusMessage *msg, void *userdata) {
577 pa_dbusiface_core *c = userdata;
578 const char **cards;
579 unsigned n;
580
581 pa_assert(conn);
582 pa_assert(msg);
583 pa_assert(c);
584
585 cards = get_cards(c, &n);
586
587 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, cards, n);
588
589 pa_xfree(cards);
590 }
591
592 /* The caller frees the array, but not the strings. */
593 static const char **get_sinks(pa_dbusiface_core *c, unsigned *n) {
594 const char **sinks;
595 unsigned i = 0;
596 void *state = NULL;
597 pa_dbusiface_device *sink;
598
599 pa_assert(c);
600 pa_assert(n);
601
602 *n = pa_hashmap_size(c->sinks_by_index);
603
604 if (*n == 0)
605 return NULL;
606
607 sinks = pa_xnew(const char *, *n);
608
609 PA_HASHMAP_FOREACH(sink, c->sinks_by_index, state)
610 sinks[i++] = pa_dbusiface_device_get_path(sink);
611
612 return sinks;
613 }
614
615 static void handle_get_sinks(DBusConnection *conn, DBusMessage *msg, void *userdata) {
616 pa_dbusiface_core *c = userdata;
617 const char **sinks;
618 unsigned n;
619
620 pa_assert(conn);
621 pa_assert(msg);
622 pa_assert(c);
623
624 sinks = get_sinks(c, &n);
625
626 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, sinks, n);
627
628 pa_xfree(sinks);
629 }
630
631 static void handle_get_fallback_sink(DBusConnection *conn, DBusMessage *msg, void *userdata) {
632 pa_dbusiface_core *c = userdata;
633 pa_dbusiface_device *fallback_sink;
634 const char *object_path;
635
636 pa_assert(conn);
637 pa_assert(msg);
638 pa_assert(c);
639
640 if (!c->fallback_sink) {
641 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NO_SUCH_PROPERTY,
642 "There are no sinks, and therefore no fallback sink either.");
643 return;
644 }
645
646 pa_assert_se((fallback_sink = pa_hashmap_get(c->sinks_by_index, PA_UINT32_TO_PTR(c->fallback_sink->index))));
647 object_path = pa_dbusiface_device_get_path(fallback_sink);
648
649 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
650 }
651
652 static void handle_set_fallback_sink(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata) {
653 pa_dbusiface_core *c = userdata;
654 pa_dbusiface_device *fallback_sink;
655 const char *object_path;
656
657 pa_assert(conn);
658 pa_assert(msg);
659 pa_assert(iter);
660 pa_assert(c);
661
662 if (!c->fallback_sink) {
663 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NO_SUCH_PROPERTY,
664 "There are no sinks, and therefore no fallback sink either.");
665 return;
666 }
667
668 dbus_message_iter_get_basic(iter, &object_path);
669
670 if (!(fallback_sink = pa_hashmap_get(c->sinks_by_path, object_path))) {
671 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "%s: No such sink.", object_path);
672 return;
673 }
674
675 pa_namereg_set_default_sink(c->core, pa_dbusiface_device_get_sink(fallback_sink));
676
677 pa_dbus_send_empty_reply(conn, msg);
678 }
679
680 /* The caller frees the array, but not the strings. */
681 static const char **get_sources(pa_dbusiface_core *c, unsigned *n) {
682 const char **sources;
683 unsigned i = 0;
684 void *state = NULL;
685 pa_dbusiface_device *source;
686
687 pa_assert(c);
688 pa_assert(n);
689
690 *n = pa_hashmap_size(c->sources_by_index);
691
692 if (*n == 0)
693 return NULL;
694
695 sources = pa_xnew(const char *, *n);
696
697 PA_HASHMAP_FOREACH(source, c->sources_by_index, state)
698 sources[i++] = pa_dbusiface_device_get_path(source);
699
700 return sources;
701 }
702
703 static void handle_get_sources(DBusConnection *conn, DBusMessage *msg, void *userdata) {
704 pa_dbusiface_core *c = userdata;
705 const char **sources;
706 unsigned n;
707
708 pa_assert(conn);
709 pa_assert(msg);
710 pa_assert(c);
711
712 sources = get_sources(c, &n);
713
714 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, sources, n);
715
716 pa_xfree(sources);
717 }
718
719 static void handle_get_fallback_source(DBusConnection *conn, DBusMessage *msg, void *userdata) {
720 pa_dbusiface_core *c = userdata;
721 pa_dbusiface_device *fallback_source;
722 const char *object_path;
723
724 pa_assert(conn);
725 pa_assert(msg);
726 pa_assert(c);
727
728 if (!c->fallback_source) {
729 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NO_SUCH_PROPERTY,
730 "There are no sources, and therefore no fallback source either.");
731 return;
732 }
733
734 pa_assert_se((fallback_source = pa_hashmap_get(c->sources_by_index, PA_UINT32_TO_PTR(c->fallback_source->index))));
735 object_path = pa_dbusiface_device_get_path(fallback_source);
736
737 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
738 }
739
740 static void handle_set_fallback_source(DBusConnection *conn, DBusMessage *msg, DBusMessageIter *iter, void *userdata) {
741 pa_dbusiface_core *c = userdata;
742 pa_dbusiface_device *fallback_source;
743 const char *object_path;
744
745 pa_assert(conn);
746 pa_assert(msg);
747 pa_assert(iter);
748 pa_assert(c);
749
750 if (!c->fallback_source) {
751 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NO_SUCH_PROPERTY,
752 "There are no sources, and therefore no fallback source either.");
753 return;
754 }
755
756 dbus_message_iter_get_basic(iter, &object_path);
757
758 if (!(fallback_source = pa_hashmap_get(c->sources_by_path, object_path))) {
759 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "%s: No such source.", object_path);
760 return;
761 }
762
763 pa_namereg_set_default_source(c->core, pa_dbusiface_device_get_source(fallback_source));
764
765 pa_dbus_send_empty_reply(conn, msg);
766 }
767
768 /* The caller frees the array, but not the strings. */
769 static const char **get_playback_streams(pa_dbusiface_core *c, unsigned *n) {
770 const char **streams;
771 unsigned i = 0;
772 void *state = NULL;
773 pa_dbusiface_stream *stream;
774
775 pa_assert(c);
776 pa_assert(n);
777
778 *n = pa_hashmap_size(c->playback_streams);
779
780 if (*n == 0)
781 return NULL;
782
783 streams = pa_xnew(const char *, *n);
784
785 PA_HASHMAP_FOREACH(stream, c->playback_streams, state)
786 streams[i++] = pa_dbusiface_stream_get_path(stream);
787
788 return streams;
789 }
790
791 static void handle_get_playback_streams(DBusConnection *conn, DBusMessage *msg, void *userdata) {
792 pa_dbusiface_core *c = userdata;
793 const char **playback_streams;
794 unsigned n;
795
796 pa_assert(conn);
797 pa_assert(msg);
798 pa_assert(c);
799
800 playback_streams = get_playback_streams(c, &n);
801
802 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, playback_streams, n);
803
804 pa_xfree(playback_streams);
805 }
806
807 /* The caller frees the array, but not the strings. */
808 static const char **get_record_streams(pa_dbusiface_core *c, unsigned *n) {
809 const char **streams;
810 unsigned i = 0;
811 void *state = NULL;
812 pa_dbusiface_stream *stream;
813
814 pa_assert(c);
815 pa_assert(n);
816
817 *n = pa_hashmap_size(c->record_streams);
818
819 if (*n == 0)
820 return NULL;
821
822 streams = pa_xnew(const char *, *n);
823
824 PA_HASHMAP_FOREACH(stream, c->record_streams, state)
825 streams[i++] = pa_dbusiface_stream_get_path(stream);
826
827 return streams;
828 }
829
830 static void handle_get_record_streams(DBusConnection *conn, DBusMessage *msg, void *userdata) {
831 pa_dbusiface_core *c = userdata;
832 const char **record_streams;
833 unsigned n;
834
835 pa_assert(conn);
836 pa_assert(msg);
837 pa_assert(c);
838
839 record_streams = get_record_streams(c, &n);
840
841 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, record_streams, n);
842
843 pa_xfree(record_streams);
844 }
845
846 /* The caller frees the array, but not the strings. */
847 static const char **get_samples(pa_dbusiface_core *c, unsigned *n) {
848 const char **samples;
849 unsigned i = 0;
850 void *state = NULL;
851 pa_dbusiface_sample *sample;
852
853 pa_assert(c);
854 pa_assert(n);
855
856 *n = pa_hashmap_size(c->samples);
857
858 if (*n == 0)
859 return NULL;
860
861 samples = pa_xnew(const char *, *n);
862
863 PA_HASHMAP_FOREACH(sample, c->samples, state)
864 samples[i++] = pa_dbusiface_sample_get_path(sample);
865
866 return samples;
867 }
868
869 static void handle_get_samples(DBusConnection *conn, DBusMessage *msg, void *userdata) {
870 pa_dbusiface_core *c = userdata;
871 const char **samples;
872 unsigned n;
873
874 pa_assert(conn);
875 pa_assert(msg);
876 pa_assert(c);
877
878 samples = get_samples(c, &n);
879
880 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, samples, n);
881
882 pa_xfree(samples);
883 }
884
885 /* The caller frees the array, but not the strings. */
886 static const char **get_modules(pa_dbusiface_core *c, unsigned *n) {
887 const char **modules;
888 unsigned i = 0;
889 void *state = NULL;
890 pa_dbusiface_module *module;
891
892 pa_assert(c);
893 pa_assert(n);
894
895 *n = pa_hashmap_size(c->modules);
896
897 if (*n == 0)
898 return NULL;
899
900 modules = pa_xnew(const char *, *n);
901
902 PA_HASHMAP_FOREACH(module, c->modules, state)
903 modules[i++] = pa_dbusiface_module_get_path(module);
904
905 return modules;
906 }
907
908 static void handle_get_modules(DBusConnection *conn, DBusMessage *msg, void *userdata) {
909 pa_dbusiface_core *c = userdata;
910 const char **modules;
911 unsigned n;
912
913 pa_assert(conn);
914 pa_assert(msg);
915 pa_assert(c);
916
917 modules = get_modules(c, &n);
918
919 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, modules, n);
920
921 pa_xfree(modules);
922 }
923
924 /* The caller frees the array, but not the strings. */
925 static const char **get_clients(pa_dbusiface_core *c, unsigned *n) {
926 const char **clients;
927 unsigned i = 0;
928 void *state = NULL;
929 pa_dbusiface_client *client;
930
931 pa_assert(c);
932 pa_assert(n);
933
934 *n = pa_hashmap_size(c->clients);
935
936 if (*n == 0)
937 return NULL;
938
939 clients = pa_xnew(const char *, *n);
940
941 PA_HASHMAP_FOREACH(client, c->clients, state)
942 clients[i++] = pa_dbusiface_client_get_path(client);
943
944 return clients;
945 }
946
947 static void handle_get_clients(DBusConnection *conn, DBusMessage *msg, void *userdata) {
948 pa_dbusiface_core *c = userdata;
949 const char **clients;
950 unsigned n;
951
952 pa_assert(conn);
953 pa_assert(msg);
954 pa_assert(c);
955
956 clients = get_clients(c, &n);
957
958 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, clients, n);
959
960 pa_xfree(clients);
961 }
962
963 static const char *get_my_client(pa_dbusiface_core *c, DBusConnection *conn) {
964 pa_client *my_client;
965
966 pa_assert(c);
967 pa_assert(conn);
968
969 pa_assert_se((my_client = pa_dbus_protocol_get_client(c->dbus_protocol, conn)));
970
971 return pa_dbusiface_client_get_path(pa_hashmap_get(c->clients, PA_UINT32_TO_PTR(my_client->index)));
972 }
973
974 static void handle_get_my_client(DBusConnection *conn, DBusMessage *msg, void *userdata) {
975 pa_dbusiface_core *c = userdata;
976 const char *my_client;
977
978 pa_assert(conn);
979 pa_assert(msg);
980 pa_assert(c);
981
982 my_client = get_my_client(c, conn);
983
984 pa_dbus_send_basic_variant_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &my_client);
985 }
986
987 static void handle_get_extensions(DBusConnection *conn, DBusMessage *msg, void *userdata) {
988 pa_dbusiface_core *c = userdata;
989 const char **extensions;
990 unsigned n;
991
992 pa_assert(conn);
993 pa_assert(msg);
994 pa_assert(c);
995
996 extensions = pa_dbus_protocol_get_extensions(c->dbus_protocol, &n);
997
998 pa_dbus_send_basic_array_variant_reply(conn, msg, DBUS_TYPE_STRING, extensions, n);
999
1000 pa_xfree(extensions);
1001 }
1002
1003 static void handle_get_all(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1004 pa_dbusiface_core *c = userdata;
1005 DBusMessage *reply = NULL;
1006 DBusMessageIter msg_iter;
1007 DBusMessageIter dict_iter;
1008 dbus_uint32_t interface_revision;
1009 const char *server_name;
1010 const char *version;
1011 dbus_bool_t is_local;
1012 char *username;
1013 char *hostname;
1014 dbus_uint32_t *default_channels;
1015 unsigned n_default_channels;
1016 dbus_uint32_t default_sample_format;
1017 dbus_uint32_t default_sample_rate;
1018 const char **cards;
1019 unsigned n_cards;
1020 const char **sinks;
1021 unsigned n_sinks;
1022 const char *fallback_sink;
1023 const char **sources;
1024 unsigned n_sources;
1025 const char *fallback_source;
1026 const char **playback_streams;
1027 unsigned n_playback_streams;
1028 const char **record_streams;
1029 unsigned n_record_streams;
1030 const char **samples;
1031 unsigned n_samples;
1032 const char **modules;
1033 unsigned n_modules;
1034 const char **clients;
1035 unsigned n_clients;
1036 const char *my_client;
1037 const char **extensions;
1038 unsigned n_extensions;
1039
1040 pa_assert(conn);
1041 pa_assert(msg);
1042 pa_assert(c);
1043
1044 interface_revision = INTERFACE_REVISION;
1045 server_name = PACKAGE_NAME;
1046 version = PACKAGE_VERSION;
1047 is_local = get_is_local(conn);
1048 username = pa_get_user_name_malloc();
1049 hostname = pa_get_host_name_malloc();
1050 default_channels = get_default_channels(c, &n_default_channels);
1051 default_sample_format = c->core->default_sample_spec.format;
1052 default_sample_rate = c->core->default_sample_spec.rate;
1053 cards = get_cards(c, &n_cards);
1054 sinks = get_sinks(c, &n_sinks);
1055 fallback_sink = c->fallback_sink
1056 ? pa_dbusiface_device_get_path(pa_hashmap_get(c->sinks_by_index, PA_UINT32_TO_PTR(c->fallback_sink->index)))
1057 : NULL;
1058 sources = get_sources(c, &n_sources);
1059 fallback_source = c->fallback_source
1060 ? pa_dbusiface_device_get_path(pa_hashmap_get(c->sources_by_index,
1061 PA_UINT32_TO_PTR(c->fallback_source->index)))
1062 : NULL;
1063 playback_streams = get_playback_streams(c, &n_playback_streams);
1064 record_streams = get_record_streams(c, &n_record_streams);
1065 samples = get_samples(c, &n_samples);
1066 modules = get_modules(c, &n_modules);
1067 clients = get_clients(c, &n_clients);
1068 my_client = get_my_client(c, conn);
1069 extensions = pa_dbus_protocol_get_extensions(c->dbus_protocol, &n_extensions);
1070
1071 pa_assert_se((reply = dbus_message_new_method_return(msg)));
1072
1073 dbus_message_iter_init_append(reply, &msg_iter);
1074 pa_assert_se(dbus_message_iter_open_container(&msg_iter, DBUS_TYPE_ARRAY, "{sv}", &dict_iter));
1075
1076 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_INTERFACE_REVISION].property_name, DBUS_TYPE_UINT32, &interface_revision);
1077 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_NAME].property_name, DBUS_TYPE_STRING, &server_name);
1078 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_VERSION].property_name, DBUS_TYPE_STRING, &version);
1079 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_IS_LOCAL].property_name, DBUS_TYPE_BOOLEAN, &is_local);
1080 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_USERNAME].property_name, DBUS_TYPE_STRING, &username);
1081 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_HOSTNAME].property_name, DBUS_TYPE_STRING, &hostname);
1082 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_DEFAULT_CHANNELS].property_name, DBUS_TYPE_UINT32, default_channels, n_default_channels);
1083 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_DEFAULT_SAMPLE_FORMAT].property_name, DBUS_TYPE_UINT32, &default_sample_format);
1084 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_DEFAULT_SAMPLE_RATE].property_name, DBUS_TYPE_UINT32, &default_sample_rate);
1085 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_CARDS].property_name, DBUS_TYPE_OBJECT_PATH, cards, n_cards);
1086 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_SINKS].property_name, DBUS_TYPE_OBJECT_PATH, sinks, n_sinks);
1087
1088 if (fallback_sink)
1089 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_FALLBACK_SINK].property_name, DBUS_TYPE_OBJECT_PATH, &fallback_sink);
1090
1091 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_SOURCES].property_name, DBUS_TYPE_OBJECT_PATH, sources, n_sources);
1092
1093 if (fallback_source)
1094 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_FALLBACK_SOURCE].property_name, DBUS_TYPE_OBJECT_PATH, &fallback_source);
1095
1096 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_PLAYBACK_STREAMS].property_name, DBUS_TYPE_OBJECT_PATH, playback_streams, n_playback_streams);
1097 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_RECORD_STREAMS].property_name, DBUS_TYPE_OBJECT_PATH, record_streams, n_record_streams);
1098 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_SAMPLES].property_name, DBUS_TYPE_OBJECT_PATH, samples, n_samples);
1099 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_MODULES].property_name, DBUS_TYPE_OBJECT_PATH, modules, n_modules);
1100 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_CLIENTS].property_name, DBUS_TYPE_OBJECT_PATH, clients, n_clients);
1101 pa_dbus_append_basic_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_MY_CLIENT].property_name, DBUS_TYPE_OBJECT_PATH, &my_client);
1102 pa_dbus_append_basic_array_variant_dict_entry(&dict_iter, property_handlers[PROPERTY_HANDLER_EXTENSIONS].property_name, DBUS_TYPE_STRING, extensions, n_extensions);
1103
1104 pa_assert_se(dbus_message_iter_close_container(&msg_iter, &dict_iter));
1105
1106 pa_assert_se(dbus_connection_send(conn, reply, NULL));
1107
1108 dbus_message_unref(reply);
1109
1110 pa_xfree(username);
1111 pa_xfree(hostname);
1112 pa_xfree(default_channels);
1113 pa_xfree(cards);
1114 pa_xfree(sinks);
1115 pa_xfree(sources);
1116 pa_xfree(playback_streams);
1117 pa_xfree(record_streams);
1118 pa_xfree(samples);
1119 pa_xfree(modules);
1120 pa_xfree(clients);
1121 pa_xfree(extensions);
1122 }
1123
1124 static void handle_get_card_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1125 pa_dbusiface_core *c = userdata;
1126 char *card_name;
1127 pa_card *card;
1128 pa_dbusiface_card *dbus_card;
1129 const char *object_path;
1130
1131 pa_assert(conn);
1132 pa_assert(msg);
1133 pa_assert(c);
1134
1135 pa_assert_se(dbus_message_get_args(msg, NULL, DBUS_TYPE_STRING, &card_name, DBUS_TYPE_INVALID));
1136
1137 if (!(card = pa_namereg_get(c->core, card_name, PA_NAMEREG_CARD))) {
1138 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "No such card.");
1139 return;
1140 }
1141
1142 pa_assert_se((dbus_card = pa_hashmap_get(c->cards, PA_UINT32_TO_PTR(card->index))));
1143
1144 object_path = pa_dbusiface_card_get_path(dbus_card);
1145
1146 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1147 }
1148
1149 static void handle_get_sink_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1150 pa_dbusiface_core *c = userdata;
1151 char *sink_name;
1152 pa_sink *sink;
1153 pa_dbusiface_device *dbus_sink;
1154 const char *object_path;
1155
1156 pa_assert(conn);
1157 pa_assert(msg);
1158 pa_assert(c);
1159
1160 pa_assert_se(dbus_message_get_args(msg, NULL, DBUS_TYPE_STRING, &sink_name, DBUS_TYPE_INVALID));
1161
1162 if (!(sink = pa_namereg_get(c->core, sink_name, PA_NAMEREG_SINK))) {
1163 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "%s: No such sink.", sink_name);
1164 return;
1165 }
1166
1167 pa_assert_se((dbus_sink = pa_hashmap_get(c->sinks_by_index, PA_UINT32_TO_PTR(sink->index))));
1168
1169 object_path = pa_dbusiface_device_get_path(dbus_sink);
1170
1171 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1172 }
1173
1174 static void handle_get_source_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1175 pa_dbusiface_core *c = userdata;
1176 char *source_name;
1177 pa_source *source;
1178 pa_dbusiface_device *dbus_source;
1179 const char *object_path;
1180
1181 pa_assert(conn);
1182 pa_assert(msg);
1183 pa_assert(c);
1184
1185 pa_assert_se(dbus_message_get_args(msg, NULL, DBUS_TYPE_STRING, &source_name, DBUS_TYPE_INVALID));
1186
1187 if (!(source = pa_namereg_get(c->core, source_name, PA_NAMEREG_SOURCE))) {
1188 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "%s: No such source.", source_name);
1189 return;
1190 }
1191
1192 pa_assert_se((dbus_source = pa_hashmap_get(c->sources_by_index, PA_UINT32_TO_PTR(source->index))));
1193
1194 object_path = pa_dbusiface_device_get_path(dbus_source);
1195
1196 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1197 }
1198
1199 static void handle_get_sample_by_name(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1200 pa_dbusiface_core *c = userdata;
1201 char *sample_name;
1202 pa_scache_entry *sample;
1203 pa_dbusiface_sample *dbus_sample;
1204 const char *object_path;
1205
1206 pa_assert(conn);
1207 pa_assert(msg);
1208 pa_assert(c);
1209
1210 pa_assert_se(dbus_message_get_args(msg, NULL, DBUS_TYPE_STRING, &sample_name, DBUS_TYPE_INVALID));
1211
1212 if (!(sample = pa_namereg_get(c->core, sample_name, PA_NAMEREG_SAMPLE))) {
1213 pa_dbus_send_error(conn, msg, PA_DBUS_ERROR_NOT_FOUND, "No such sample.");
1214 return;
1215 }
1216
1217 pa_assert_se((dbus_sample = pa_hashmap_get(c->samples, PA_UINT32_TO_PTR(sample->index))));
1218
1219 object_path = pa_dbusiface_sample_get_path(dbus_sample);
1220
1221 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1222 }
1223
1224 static void handle_upload_sample(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1225 pa_dbusiface_core *c = userdata;
1226 DBusMessageIter msg_iter;
1227 DBusMessageIter array_iter;
1228 const char *name;
1229 dbus_uint32_t sample_format;
1230 dbus_uint32_t sample_rate;
1231 const dbus_uint32_t *channels;
1232 int n_channels;
1233 const dbus_uint32_t *default_volume;
1234 int n_volume_entries;
1235 pa_proplist *property_list;
1236 const uint8_t *data;
1237 int data_length;
1238 int i;
1239 pa_sample_spec ss;
1240 pa_channel_map map;
1241 pa_memchunk chunk;
1242 uint32_t idx;
1243 pa_dbusiface_sample *dbus_sample = NULL;
1244 pa_scache_entry *sample = NULL;
1245 const char *object_path;
1246
1247 pa_assert(conn);
1248 pa_assert(msg);
1249 pa_assert(c);
1250
1251 chunk.memblock = NULL;
1252
1253 pa_assert_se(dbus_message_iter_init(msg, &msg_iter));
1254 dbus_message_iter_get_basic(&msg_iter, &name);
1255
1256 pa_assert_se(dbus_message_iter_next(&msg_iter));
1257 dbus_message_iter_get_basic(&msg_iter, &sample_format);
1258
1259 pa_assert_se(dbus_message_iter_next(&msg_iter));
1260 dbus_message_iter_get_basic(&msg_iter, &sample_rate);
1261
1262 pa_assert_se(dbus_message_iter_next(&msg_iter));
1263 dbus_message_iter_recurse(&msg_iter, &array_iter);
1264 dbus_message_iter_get_fixed_array(&array_iter, &channels, &n_channels);
1265
1266 pa_assert_se(dbus_message_iter_next(&msg_iter));
1267 dbus_message_iter_recurse(&msg_iter, &array_iter);
1268 dbus_message_iter_get_fixed_array(&array_iter, &default_volume, &n_volume_entries);
1269
1270 pa_assert_se(dbus_message_iter_next(&msg_iter));
1271 if (!(property_list = pa_dbus_get_proplist_arg(conn, msg, &msg_iter)))
1272 return;
1273
1274 dbus_message_iter_recurse(&msg_iter, &array_iter);
1275 dbus_message_iter_get_fixed_array(&array_iter, &data, &data_length);
1276
1277 if (sample_format >= PA_SAMPLE_MAX) {
1278 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid sample format.");
1279 goto finish;
1280 }
1281
1282 if (sample_rate <= 0 || sample_rate > PA_RATE_MAX) {
1283 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid sample rate.");
1284 goto finish;
1285 }
1286
1287 if (n_channels <= 0) {
1288 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Empty channel map.");
1289 goto finish;
1290 }
1291
1292 if (n_channels > (int) PA_CHANNELS_MAX) {
1293 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS,
1294 "Too many channels: %i. The maximum is %u.", n_channels, PA_CHANNELS_MAX);
1295 goto finish;
1296 }
1297
1298 for (i = 0; i < n_channels; ++i) {
1299 if (channels[i] >= PA_CHANNEL_POSITION_MAX) {
1300 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid channel position.");
1301 goto finish;
1302 }
1303 }
1304
1305 if (n_volume_entries != 0 && n_volume_entries != n_channels) {
1306 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS,
1307 "The channels and default_volume arguments have different number of elements (%i and %i, resp).",
1308 n_channels, n_volume_entries);
1309 goto finish;
1310 }
1311
1312 for (i = 0; i < n_volume_entries; ++i) {
1313 if (!PA_VOLUME_IS_VALID(default_volume[i])) {
1314 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid volume: %u.", default_volume[i]);
1315 goto finish;
1316 }
1317 }
1318
1319 if (data_length == 0) {
1320 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Empty data.");
1321 goto finish;
1322 }
1323
1324 if (data_length > PA_SCACHE_ENTRY_SIZE_MAX) {
1325 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS,
1326 "Too big sample: %i bytes. The maximum sample length is %u bytes.",
1327 data_length, PA_SCACHE_ENTRY_SIZE_MAX);
1328 goto finish;
1329 }
1330
1331 ss.format = sample_format;
1332 ss.rate = sample_rate;
1333 ss.channels = n_channels;
1334
1335 pa_assert(pa_sample_spec_valid(&ss));
1336
1337 if (!pa_frame_aligned(data_length, &ss)) {
1338 char buf[PA_SAMPLE_SPEC_SNPRINT_MAX];
1339 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS,
1340 "The sample length (%i bytes) doesn't align with the sample format and channels (%s).",
1341 data_length, pa_sample_spec_snprint(buf, sizeof(buf), &ss));
1342 goto finish;
1343 }
1344
1345 map.channels = n_channels;
1346 for (i = 0; i < n_channels; ++i)
1347 map.map[i] = channels[i];
1348
1349 chunk.memblock = pa_memblock_new(c->core->mempool, data_length);
1350 chunk.index = 0;
1351 chunk.length = data_length;
1352
1353 memcpy(pa_memblock_acquire(chunk.memblock), data, data_length);
1354 pa_memblock_release(chunk.memblock);
1355
1356 if (pa_scache_add_item(c->core, name, &ss, &map, &chunk, property_list, &idx) < 0) {
1357 pa_dbus_send_error(conn, msg, DBUS_ERROR_FAILED, "Adding the sample failed.");
1358 goto finish;
1359 }
1360
1361 sample = pa_idxset_get_by_index(c->core->scache, idx);
1362
1363 if (n_volume_entries > 0) {
1364 sample->volume.channels = n_channels;
1365 for (i = 0; i < n_volume_entries; ++i)
1366 sample->volume.values[i] = default_volume[i];
1367 sample->volume_is_set = TRUE;
1368 } else {
1369 sample->volume_is_set = FALSE;
1370 }
1371
1372 dbus_sample = pa_dbusiface_sample_new(c, sample);
1373 pa_hashmap_put(c->samples, PA_UINT32_TO_PTR(idx), dbus_sample);
1374
1375 object_path = pa_dbusiface_sample_get_path(dbus_sample);
1376
1377 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1378
1379 finish:
1380 if (property_list)
1381 pa_proplist_free(property_list);
1382
1383 if (chunk.memblock)
1384 pa_memblock_unref(chunk.memblock);
1385 }
1386
1387 static pa_bool_t contains_space(const char *string) {
1388 const char *p;
1389
1390 pa_assert(string);
1391
1392 for (p = string; *p; ++p) {
1393 if (isspace(*p))
1394 return TRUE;
1395 }
1396
1397 return FALSE;
1398 }
1399
1400 static void handle_load_module(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1401 pa_dbusiface_core *c = userdata;
1402 DBusMessageIter msg_iter;
1403 DBusMessageIter dict_iter;
1404 DBusMessageIter dict_entry_iter;
1405 char *name = NULL;
1406 const char *key = NULL;
1407 const char *value = NULL;
1408 char *escaped_value = NULL;
1409 pa_strbuf *arg_buffer = NULL;
1410 char *arg_string = NULL;
1411 pa_module *module = NULL;
1412 pa_dbusiface_module *dbus_module = NULL;
1413 const char *object_path = NULL;
1414
1415 pa_assert(conn);
1416 pa_assert(msg);
1417 pa_assert(c);
1418
1419 if (c->core->disallow_module_loading) {
1420 pa_dbus_send_error(conn, msg, DBUS_ERROR_ACCESS_DENIED, "The server is configured to disallow module loading.");
1421 return;
1422 }
1423
1424 pa_assert_se(dbus_message_iter_init(msg, &msg_iter));
1425 dbus_message_iter_get_basic(&msg_iter, &name);
1426
1427 arg_buffer = pa_strbuf_new();
1428
1429 pa_assert_se(dbus_message_iter_next(&msg_iter));
1430 dbus_message_iter_recurse(&msg_iter, &dict_iter);
1431
1432 while (dbus_message_iter_get_arg_type(&dict_iter) != DBUS_TYPE_INVALID) {
1433 if (!pa_strbuf_isempty(arg_buffer))
1434 pa_strbuf_putc(arg_buffer, ' ');
1435
1436 dbus_message_iter_recurse(&dict_iter, &dict_entry_iter);
1437
1438 dbus_message_iter_get_basic(&dict_entry_iter, &key);
1439
1440 if (strlen(key) <= 0 || !pa_ascii_valid(key) || contains_space(key)) {
1441 pa_dbus_send_error(conn, msg, DBUS_ERROR_INVALID_ARGS, "Invalid module argument name: %s", key);
1442 goto finish;
1443 }
1444
1445 pa_assert_se(dbus_message_iter_next(&dict_entry_iter));
1446 dbus_message_iter_get_basic(&dict_entry_iter, &value);
1447
1448 escaped_value = pa_escape(value, "\"");
1449 pa_strbuf_printf(arg_buffer, "%s=\"%s\"", key, escaped_value);
1450 pa_xfree(escaped_value);
1451
1452 dbus_message_iter_next(&dict_iter);
1453 }
1454
1455 arg_string = pa_strbuf_tostring(arg_buffer);
1456
1457 if (!(module = pa_module_load(c->core, name, arg_string))) {
1458 pa_dbus_send_error(conn, msg, DBUS_ERROR_FAILED, "Failed to load module.");
1459 goto finish;
1460 }
1461
1462 dbus_module = pa_dbusiface_module_new(module);
1463 pa_hashmap_put(c->modules, PA_UINT32_TO_PTR(module->index), dbus_module);
1464
1465 object_path = pa_dbusiface_module_get_path(dbus_module);
1466
1467 pa_dbus_send_basic_value_reply(conn, msg, DBUS_TYPE_OBJECT_PATH, &object_path);
1468
1469 finish:
1470 if (arg_buffer)
1471 pa_strbuf_free(arg_buffer);
1472
1473 pa_xfree(arg_string);
1474 }
1475
1476 static void handle_exit(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1477 pa_dbusiface_core *c = userdata;
1478
1479 pa_assert(conn);
1480 pa_assert(msg);
1481 pa_assert(c);
1482
1483 if (c->core->disallow_exit) {
1484 pa_dbus_send_error(conn, msg, DBUS_ERROR_ACCESS_DENIED, "The server is configured to disallow exiting.");
1485 return;
1486 }
1487
1488 pa_dbus_send_empty_reply(conn, msg);
1489
1490 pa_core_exit(c->core, FALSE, 0);
1491 }
1492
1493 static void handle_listen_for_signal(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1494 pa_dbusiface_core *c = userdata;
1495 const char *signal_str;
1496 char **objects = NULL;
1497 int n_objects;
1498
1499 pa_assert(conn);
1500 pa_assert(msg);
1501 pa_assert(c);
1502
1503 pa_assert_se(dbus_message_get_args(msg, NULL,
1504 DBUS_TYPE_STRING, &signal_str,
1505 DBUS_TYPE_ARRAY, DBUS_TYPE_OBJECT_PATH, &objects, &n_objects,
1506 DBUS_TYPE_INVALID));
1507
1508 pa_dbus_protocol_add_signal_listener(c->dbus_protocol, conn, *signal_str ? signal_str : NULL, objects, n_objects);
1509
1510 pa_dbus_send_empty_reply(conn, msg);
1511
1512 dbus_free_string_array(objects);
1513 }
1514
1515 static void handle_stop_listening_for_signal(DBusConnection *conn, DBusMessage *msg, void *userdata) {
1516 pa_dbusiface_core *c = userdata;
1517 const char *signal_str;
1518
1519 pa_assert(conn);
1520 pa_assert(msg);
1521 pa_assert(c);
1522
1523 pa_assert_se(dbus_message_get_args(msg, NULL, DBUS_TYPE_STRING, &signal_str, DBUS_TYPE_INVALID));
1524
1525 pa_dbus_protocol_remove_signal_listener(c->dbus_protocol, conn, *signal_str ? signal_str : NULL);
1526
1527 pa_dbus_send_empty_reply(conn, msg);
1528 }
1529
1530 static void subscription_cb(pa_core *core, pa_subscription_event_type_t t, uint32_t idx, void *userdata) {
1531 pa_dbusiface_core *c = userdata;
1532 pa_dbusiface_card *card_iface = NULL;
1533 pa_dbusiface_device *device_iface = NULL;
1534 pa_dbusiface_stream *stream_iface = NULL;
1535 pa_dbusiface_sample *sample_iface = NULL;
1536 pa_dbusiface_module *module_iface = NULL;
1537 pa_dbusiface_client *client_iface = NULL;
1538 DBusMessage *signal_msg = NULL;
1539 const char *object_path = NULL;
1540 pa_sink *new_fallback_sink = NULL;
1541 pa_source *new_fallback_source = NULL;
1542
1543 pa_assert(c);
1544
1545 switch (t & PA_SUBSCRIPTION_EVENT_FACILITY_MASK) {
1546 case PA_SUBSCRIPTION_EVENT_SERVER:
1547 new_fallback_sink = pa_namereg_get_default_sink(core);
1548 new_fallback_source = pa_namereg_get_default_source(core);
1549
1550 if (c->fallback_sink != new_fallback_sink) {
1551 if (c->fallback_sink)
1552 pa_sink_unref(c->fallback_sink);
1553 c->fallback_sink = new_fallback_sink ? pa_sink_ref(new_fallback_sink) : NULL;
1554
1555 if (c->fallback_sink) {
1556 pa_assert_se(device_iface = pa_hashmap_get(c->sinks_by_index, PA_UINT32_TO_PTR(c->fallback_sink->index)));
1557 object_path = pa_dbusiface_device_get_path(device_iface);
1558
1559 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1560 PA_DBUS_CORE_INTERFACE,
1561 signals[SIGNAL_FALLBACK_SINK_UPDATED].name)));
1562 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1563 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1564 dbus_message_unref(signal_msg);
1565 signal_msg = NULL;
1566
1567 } else {
1568 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1569 PA_DBUS_CORE_INTERFACE,
1570 signals[SIGNAL_FALLBACK_SINK_UNSET].name)));
1571 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1572 dbus_message_unref(signal_msg);
1573 signal_msg = NULL;
1574 }
1575 }
1576
1577 if (c->fallback_source != new_fallback_source) {
1578 if (c->fallback_source)
1579 pa_source_unref(c->fallback_source);
1580 c->fallback_source = new_fallback_source ? pa_source_ref(new_fallback_source) : NULL;
1581
1582 if (c->fallback_source) {
1583 pa_assert_se(device_iface = pa_hashmap_get(c->sources_by_index, PA_UINT32_TO_PTR(c->fallback_source->index)));
1584 object_path = pa_dbusiface_device_get_path(device_iface);
1585
1586 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1587 PA_DBUS_CORE_INTERFACE,
1588 signals[SIGNAL_FALLBACK_SOURCE_UPDATED].name)));
1589 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1590 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1591 dbus_message_unref(signal_msg);
1592 signal_msg = NULL;
1593
1594 } else {
1595 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1596 PA_DBUS_CORE_INTERFACE,
1597 signals[SIGNAL_FALLBACK_SOURCE_UNSET].name)));
1598 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1599 dbus_message_unref(signal_msg);
1600 signal_msg = NULL;
1601 }
1602 }
1603 break;
1604
1605 case PA_SUBSCRIPTION_EVENT_CARD:
1606 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1607 if (!(card_iface = pa_hashmap_get(c->cards, PA_UINT32_TO_PTR(idx)))) {
1608 pa_card *card = NULL;
1609
1610 if (!(card = pa_idxset_get_by_index(core->cards, idx)))
1611 return; /* The card was removed immediately after creation. */
1612
1613 card_iface = pa_dbusiface_card_new(c, card);
1614 pa_hashmap_put(c->cards, PA_UINT32_TO_PTR(idx), card_iface);
1615 }
1616
1617 object_path = pa_dbusiface_card_get_path(card_iface);
1618
1619 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1620 PA_DBUS_CORE_INTERFACE,
1621 signals[SIGNAL_NEW_CARD].name)));
1622 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1623
1624 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1625 if (!(card_iface = pa_hashmap_remove(c->cards, PA_UINT32_TO_PTR(idx))))
1626 return;
1627
1628 object_path = pa_dbusiface_card_get_path(card_iface);
1629
1630 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1631 PA_DBUS_CORE_INTERFACE,
1632 signals[SIGNAL_CARD_REMOVED].name)));
1633 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1634
1635 pa_dbusiface_card_free(card_iface);
1636 }
1637 break;
1638
1639 case PA_SUBSCRIPTION_EVENT_SINK_INPUT:
1640 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1641 pa_sink_input *sink_input = NULL;
1642
1643 if (!(sink_input = pa_idxset_get_by_index(core->sink_inputs, idx)))
1644 return; /* The sink input was removed immediately after creation. */
1645
1646 if (!(stream_iface = pa_hashmap_get(c->playback_streams, PA_UINT32_TO_PTR(idx)))) {
1647 stream_iface = pa_dbusiface_stream_new_playback(c, sink_input);
1648 pa_hashmap_put(c->playback_streams, PA_UINT32_TO_PTR(idx), stream_iface);
1649 }
1650
1651 object_path = pa_dbusiface_stream_get_path(stream_iface);
1652
1653 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1654 PA_DBUS_CORE_INTERFACE,
1655 signals[SIGNAL_NEW_PLAYBACK_STREAM].name)));
1656 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1657
1658 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1659 if (!(stream_iface = pa_hashmap_remove(c->playback_streams, PA_UINT32_TO_PTR(idx))))
1660 return;
1661
1662 object_path = pa_dbusiface_stream_get_path(stream_iface);
1663
1664 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1665 PA_DBUS_CORE_INTERFACE,
1666 signals[SIGNAL_PLAYBACK_STREAM_REMOVED].name)));
1667 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1668
1669 pa_dbusiface_stream_free(stream_iface);
1670 }
1671 break;
1672
1673 case PA_SUBSCRIPTION_EVENT_SOURCE_OUTPUT:
1674 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1675 pa_source_output *source_output = NULL;
1676
1677 if (!(source_output = pa_idxset_get_by_index(core->source_outputs, idx)))
1678 return; /* The source output was removed immediately after creation. */
1679
1680 if (!(stream_iface = pa_hashmap_get(c->record_streams, PA_UINT32_TO_PTR(idx)))) {
1681 stream_iface = pa_dbusiface_stream_new_record(c, source_output);
1682 pa_hashmap_put(c->record_streams, PA_UINT32_TO_PTR(idx), stream_iface);
1683 }
1684
1685 object_path = pa_dbusiface_stream_get_path(stream_iface);
1686
1687 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1688 PA_DBUS_CORE_INTERFACE,
1689 signals[SIGNAL_NEW_RECORD_STREAM].name)));
1690 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1691
1692 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1693 if (!(stream_iface = pa_hashmap_remove(c->record_streams, PA_UINT32_TO_PTR(idx))))
1694 return;
1695
1696 object_path = pa_dbusiface_stream_get_path(stream_iface);
1697
1698 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1699 PA_DBUS_CORE_INTERFACE,
1700 signals[SIGNAL_RECORD_STREAM_REMOVED].name)));
1701 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1702
1703 pa_dbusiface_stream_free(stream_iface);
1704 }
1705 break;
1706
1707 case PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE:
1708 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1709 pa_scache_entry *sample = NULL;
1710
1711 if (!(sample = pa_idxset_get_by_index(core->scache, idx)))
1712 return; /* The sample was removed immediately after creation. */
1713
1714 if (!(sample_iface = pa_hashmap_get(c->samples, PA_UINT32_TO_PTR(idx)))) {
1715 sample_iface = pa_dbusiface_sample_new(c, sample);
1716 pa_hashmap_put(c->samples, PA_UINT32_TO_PTR(idx), sample_iface);
1717 }
1718
1719 object_path = pa_dbusiface_sample_get_path(sample_iface);
1720
1721 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1722 PA_DBUS_CORE_INTERFACE,
1723 signals[SIGNAL_NEW_SAMPLE].name)));
1724 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1725
1726 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1727 if (!(sample_iface = pa_hashmap_remove(c->samples, PA_UINT32_TO_PTR(idx))))
1728 return;
1729
1730 object_path = pa_dbusiface_sample_get_path(sample_iface);
1731
1732 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1733 PA_DBUS_CORE_INTERFACE,
1734 signals[SIGNAL_SAMPLE_REMOVED].name)));
1735 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1736
1737 pa_dbusiface_sample_free(sample_iface);
1738 }
1739 break;
1740
1741 case PA_SUBSCRIPTION_EVENT_MODULE:
1742 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1743 pa_module *module = NULL;
1744
1745 if (!(module = pa_idxset_get_by_index(core->modules, idx)))
1746 return; /* The module was removed immediately after creation. */
1747
1748 if (!(module_iface = pa_hashmap_get(c->modules, PA_UINT32_TO_PTR(idx)))) {
1749 module_iface = pa_dbusiface_module_new(module);
1750 pa_hashmap_put(c->modules, PA_UINT32_TO_PTR(idx), module_iface);
1751 }
1752
1753 object_path = pa_dbusiface_module_get_path(module_iface);
1754
1755 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1756 PA_DBUS_CORE_INTERFACE,
1757 signals[SIGNAL_NEW_MODULE].name)));
1758 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1759
1760 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1761 if (!(module_iface = pa_hashmap_remove(c->modules, PA_UINT32_TO_PTR(idx))))
1762 return;
1763
1764 object_path = pa_dbusiface_module_get_path(module_iface);
1765
1766 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1767 PA_DBUS_CORE_INTERFACE,
1768 signals[SIGNAL_MODULE_REMOVED].name)));
1769 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1770
1771 pa_dbusiface_module_free(module_iface);
1772 }
1773 break;
1774
1775 case PA_SUBSCRIPTION_EVENT_CLIENT:
1776 if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_NEW) {
1777 pa_client *client = NULL;
1778
1779 if (!(client = pa_idxset_get_by_index(core->clients, idx)))
1780 return; /* The client was removed immediately after creation. */
1781
1782 if (!(client_iface = pa_hashmap_get(c->clients, PA_UINT32_TO_PTR(idx)))) {
1783 client_iface = pa_dbusiface_client_new(c, client);
1784 pa_hashmap_put(c->clients, PA_UINT32_TO_PTR(idx), client_iface);
1785 }
1786
1787 object_path = pa_dbusiface_client_get_path(client_iface);
1788
1789 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1790 PA_DBUS_CORE_INTERFACE,
1791 signals[SIGNAL_NEW_CLIENT].name)));
1792 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1793
1794 } else if ((t & PA_SUBSCRIPTION_EVENT_TYPE_MASK) == PA_SUBSCRIPTION_EVENT_REMOVE) {
1795 if (!(client_iface = pa_hashmap_remove(c->clients, PA_UINT32_TO_PTR(idx))))
1796 return;
1797
1798 object_path = pa_dbusiface_client_get_path(client_iface);
1799
1800 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1801 PA_DBUS_CORE_INTERFACE,
1802 signals[SIGNAL_CLIENT_REMOVED].name)));
1803 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1804
1805 pa_dbusiface_client_free(client_iface);
1806 }
1807 break;
1808 }
1809
1810 if (signal_msg) {
1811 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1812 dbus_message_unref(signal_msg);
1813 }
1814 }
1815
1816 static pa_hook_result_t sink_put_cb(void *hook_data, void *call_data, void *slot_data) {
1817 pa_dbusiface_core *c = slot_data;
1818 pa_sink *s = call_data;
1819 pa_dbusiface_device *d = NULL;
1820 const char *object_path = NULL;
1821 DBusMessage *signal_msg = NULL;
1822
1823 pa_assert(c);
1824 pa_assert(s);
1825
1826 d = pa_dbusiface_device_new_sink(c, s);
1827 object_path = pa_dbusiface_device_get_path(d);
1828
1829 pa_assert_se(pa_hashmap_put(c->sinks_by_index, PA_UINT32_TO_PTR(s->index), d) >= 0);
1830 pa_assert_se(pa_hashmap_put(c->sinks_by_path, object_path, d) >= 0);
1831
1832 pa_assert_se(signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1833 PA_DBUS_CORE_INTERFACE,
1834 signals[SIGNAL_NEW_SINK].name));
1835 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1836
1837 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1838 dbus_message_unref(signal_msg);
1839
1840 return PA_HOOK_OK;
1841 }
1842
1843 static pa_hook_result_t sink_unlink_cb(void *hook_data, void *call_data, void *slot_data) {
1844 pa_dbusiface_core *c = slot_data;
1845 pa_sink *s = call_data;
1846 pa_dbusiface_device *d = NULL;
1847 const char *object_path = NULL;
1848 DBusMessage *signal_msg = NULL;
1849
1850 pa_assert(c);
1851 pa_assert(s);
1852
1853 pa_assert_se(d = pa_hashmap_remove(c->sinks_by_index, PA_UINT32_TO_PTR(s->index)));
1854 object_path = pa_dbusiface_device_get_path(d);
1855 pa_assert_se(pa_hashmap_remove(c->sinks_by_path, object_path));
1856
1857 pa_assert_se(signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1858 PA_DBUS_CORE_INTERFACE,
1859 signals[SIGNAL_SINK_REMOVED].name));
1860 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1861
1862 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1863 dbus_message_unref(signal_msg);
1864
1865 pa_dbusiface_device_free(d);
1866
1867 return PA_HOOK_OK;
1868 }
1869
1870 static pa_hook_result_t source_put_cb(void *hook_data, void *call_data, void *slot_data) {
1871 pa_dbusiface_core *c = slot_data;
1872 pa_source *s = call_data;
1873 pa_dbusiface_device *d = NULL;
1874 const char *object_path = NULL;
1875 DBusMessage *signal_msg = NULL;
1876
1877 pa_assert(c);
1878 pa_assert(s);
1879
1880 d = pa_dbusiface_device_new_source(c, s);
1881 object_path = pa_dbusiface_device_get_path(d);
1882
1883 pa_assert_se(pa_hashmap_put(c->sources_by_index, PA_UINT32_TO_PTR(s->index), d) >= 0);
1884 pa_assert_se(pa_hashmap_put(c->sources_by_path, object_path, d) >= 0);
1885
1886 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1887 PA_DBUS_CORE_INTERFACE,
1888 signals[SIGNAL_NEW_SOURCE].name)));
1889 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1890
1891 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1892 dbus_message_unref(signal_msg);
1893
1894 return PA_HOOK_OK;
1895 }
1896
1897 static pa_hook_result_t source_unlink_cb(void *hook_data, void *call_data, void *slot_data) {
1898 pa_dbusiface_core *c = slot_data;
1899 pa_source *s = call_data;
1900 pa_dbusiface_device *d = NULL;
1901 const char *object_path = NULL;
1902 DBusMessage *signal_msg = NULL;
1903
1904 pa_assert(c);
1905 pa_assert(s);
1906
1907 pa_assert_se(d = pa_hashmap_remove(c->sources_by_index, PA_UINT32_TO_PTR(s->index)));
1908 object_path = pa_dbusiface_device_get_path(d);
1909 pa_assert_se(pa_hashmap_remove(c->sources_by_path, object_path));
1910
1911 pa_assert_se(signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1912 PA_DBUS_CORE_INTERFACE,
1913 signals[SIGNAL_SOURCE_REMOVED].name));
1914 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_OBJECT_PATH, &object_path, DBUS_TYPE_INVALID));
1915
1916 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1917 dbus_message_unref(signal_msg);
1918
1919 pa_dbusiface_device_free(d);
1920
1921 return PA_HOOK_OK;
1922 }
1923
1924 static pa_hook_result_t extension_registered_cb(void *hook_data, void *call_data, void *slot_data) {
1925 pa_dbusiface_core *c = slot_data;
1926 const char *ext_name = call_data;
1927 DBusMessage *signal_msg = NULL;
1928
1929 pa_assert(c);
1930 pa_assert(ext_name);
1931
1932 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1933 PA_DBUS_CORE_INTERFACE,
1934 signals[SIGNAL_NEW_EXTENSION].name)));
1935 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_STRING, &ext_name, DBUS_TYPE_INVALID));
1936
1937 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1938 dbus_message_unref(signal_msg);
1939
1940 return PA_HOOK_OK;
1941 }
1942
1943 static pa_hook_result_t extension_unregistered_cb(void *hook_data, void *call_data, void *slot_data) {
1944 pa_dbusiface_core *c = slot_data;
1945 const char *ext_name = call_data;
1946 DBusMessage *signal_msg = NULL;
1947
1948 pa_assert(c);
1949 pa_assert(ext_name);
1950
1951 pa_assert_se((signal_msg = dbus_message_new_signal(PA_DBUS_CORE_OBJECT_PATH,
1952 PA_DBUS_CORE_INTERFACE,
1953 signals[SIGNAL_EXTENSION_REMOVED].name)));
1954 pa_assert_se(dbus_message_append_args(signal_msg, DBUS_TYPE_STRING, &ext_name, DBUS_TYPE_INVALID));
1955
1956 pa_dbus_protocol_send_signal(c->dbus_protocol, signal_msg);
1957 dbus_message_unref(signal_msg);
1958
1959 return PA_HOOK_OK;
1960 }
1961
1962 pa_dbusiface_core *pa_dbusiface_core_new(pa_core *core) {
1963 pa_dbusiface_core *c;
1964 pa_card *card;
1965 pa_sink *sink;
1966 pa_source *source;
1967 pa_dbusiface_device *device;
1968 pa_sink_input *sink_input;
1969 pa_source_output *source_output;
1970 pa_scache_entry *sample;
1971 pa_module *module;
1972 pa_client *client;
1973 uint32_t idx;
1974
1975 pa_assert(core);
1976
1977 c = pa_xnew(pa_dbusiface_core, 1);
1978 c->core = core;
1979 c->subscription = pa_subscription_new(core, PA_SUBSCRIPTION_MASK_ALL, subscription_cb, c);
1980 c->dbus_protocol = pa_dbus_protocol_get(core);
1981 c->cards = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1982 c->sinks_by_index = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1983 c->sinks_by_path = pa_hashmap_new(pa_idxset_string_hash_func, pa_idxset_string_compare_func);
1984 c->sources_by_index = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1985 c->sources_by_path = pa_hashmap_new(pa_idxset_string_hash_func, pa_idxset_string_compare_func);
1986 c->playback_streams = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1987 c->record_streams = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1988 c->samples = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1989 c->modules = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1990 c->clients = pa_hashmap_new(pa_idxset_trivial_hash_func, pa_idxset_trivial_compare_func);
1991 c->fallback_sink = pa_namereg_get_default_sink(core);
1992 c->fallback_source = pa_namereg_get_default_source(core);
1993 c->sink_put_slot = pa_hook_connect(&core->hooks[PA_CORE_HOOK_SINK_PUT], PA_HOOK_NORMAL, sink_put_cb, c);
1994 c->sink_unlink_slot = pa_hook_connect(&core->hooks[PA_CORE_HOOK_SINK_UNLINK], PA_HOOK_NORMAL, sink_unlink_cb, c);
1995 c->source_put_slot = pa_hook_connect(&core->hooks[PA_CORE_HOOK_SOURCE_PUT], PA_HOOK_NORMAL, source_put_cb, c);
1996 c->source_unlink_slot = pa_hook_connect(&core->hooks[PA_CORE_HOOK_SOURCE_UNLINK], PA_HOOK_NORMAL, source_unlink_cb, c);
1997 c->extension_registered_slot = pa_dbus_protocol_hook_connect(c->dbus_protocol,
1998 PA_DBUS_PROTOCOL_HOOK_EXTENSION_REGISTERED,
1999 PA_HOOK_NORMAL,
2000 extension_registered_cb,
2001 c);
2002 c->extension_unregistered_slot = pa_dbus_protocol_hook_connect(c->dbus_protocol,
2003 PA_DBUS_PROTOCOL_HOOK_EXTENSION_UNREGISTERED,
2004 PA_HOOK_NORMAL,
2005 extension_unregistered_cb,
2006 c);
2007 c->memstats = pa_dbusiface_memstats_new(c, core);
2008
2009 if (c->fallback_sink)
2010 pa_sink_ref(c->fallback_sink);
2011 if (c->fallback_source)
2012 pa_source_ref(c->fallback_source);
2013
2014 PA_IDXSET_FOREACH(card, core->cards, idx)
2015 pa_hashmap_put(c->cards, PA_UINT32_TO_PTR(idx), pa_dbusiface_card_new(c, card));
2016
2017 PA_IDXSET_FOREACH(sink, core->sinks, idx) {
2018 device = pa_dbusiface_device_new_sink(c, sink);
2019 pa_hashmap_put(c->sinks_by_index, PA_UINT32_TO_PTR(idx), device);
2020 pa_hashmap_put(c->sinks_by_path, pa_dbusiface_device_get_path(device), device);
2021 }
2022
2023 PA_IDXSET_FOREACH(source, core->sources, idx) {
2024 device = pa_dbusiface_device_new_source(c, source);
2025 pa_hashmap_put(c->sources_by_index, PA_UINT32_TO_PTR(idx), device);
2026 pa_hashmap_put(c->sources_by_path, pa_dbusiface_device_get_path(device), device);
2027 }
2028
2029 PA_IDXSET_FOREACH(sink_input, core->sink_inputs, idx)
2030 pa_hashmap_put(c->playback_streams, PA_UINT32_TO_PTR(idx), pa_dbusiface_stream_new_playback(c, sink_input));
2031
2032 PA_IDXSET_FOREACH(source_output, core->source_outputs, idx)
2033 pa_hashmap_put(c->record_streams, PA_UINT32_TO_PTR(idx), pa_dbusiface_stream_new_record(c, source_output));
2034
2035 PA_IDXSET_FOREACH(sample, core->scache, idx)
2036 pa_hashmap_put(c->samples, PA_UINT32_TO_PTR(idx), pa_dbusiface_sample_new(c, sample));
2037
2038 PA_IDXSET_FOREACH(module, core->modules, idx)
2039 pa_hashmap_put(c->modules, PA_UINT32_TO_PTR(idx), pa_dbusiface_module_new(module));
2040
2041 PA_IDXSET_FOREACH(client, core->clients, idx)
2042 pa_hashmap_put(c->clients, PA_UINT32_TO_PTR(idx), pa_dbusiface_client_new(c, client));
2043
2044 pa_assert_se(pa_dbus_protocol_add_interface(c->dbus_protocol, PA_DBUS_CORE_OBJECT_PATH, &core_interface_info, c) >= 0);
2045
2046 return c;
2047 }
2048
2049 static void free_card_cb(void *p, void *userdata) {
2050 pa_dbusiface_card *c = p;
2051
2052 pa_assert(c);
2053
2054 pa_dbusiface_card_free(c);
2055 }
2056
2057 static void free_device_cb(void *p, void *userdata) {
2058 pa_dbusiface_device *d = p;
2059
2060 pa_assert(d);
2061
2062 pa_dbusiface_device_free(d);
2063 }
2064
2065 static void free_stream_cb(void *p, void *userdata) {
2066 pa_dbusiface_stream *s = p;
2067
2068 pa_assert(s);
2069
2070 pa_dbusiface_stream_free(s);
2071 }
2072
2073 static void free_sample_cb(void *p, void *userdata) {
2074 pa_dbusiface_sample *s = p;
2075
2076 pa_assert(s);
2077
2078 pa_dbusiface_sample_free(s);
2079 }
2080
2081 static void free_module_cb(void *p, void *userdata) {
2082 pa_dbusiface_module *m = p;
2083
2084 pa_assert(m);
2085
2086 pa_dbusiface_module_free(m);
2087 }
2088
2089 static void free_client_cb(void *p, void *userdata) {
2090 pa_dbusiface_client *c = p;
2091
2092 pa_assert(c);
2093
2094 pa_dbusiface_client_free(c);
2095 }
2096
2097 void pa_dbusiface_core_free(pa_dbusiface_core *c) {
2098 pa_assert(c);
2099
2100 pa_assert_se(pa_dbus_protocol_remove_interface(c->dbus_protocol, PA_DBUS_CORE_OBJECT_PATH, core_interface_info.name) >= 0);
2101
2102 /* Note that the order of freeing is important below.
2103 * Do not change it for the sake of tidiness without checking! */
2104 pa_subscription_free(c->subscription);
2105 pa_hashmap_free(c->cards, free_card_cb, NULL);
2106 pa_hashmap_free(c->sinks_by_path, NULL, NULL);
2107 pa_hashmap_free(c->sinks_by_index, free_device_cb, NULL);
2108 pa_hashmap_free(c->sources_by_path, NULL, NULL);
2109 pa_hashmap_free(c->sources_by_index, free_device_cb, NULL);
2110 pa_hashmap_free(c->playback_streams, free_stream_cb, NULL);
2111 pa_hashmap_free(c->record_streams, free_stream_cb, NULL);
2112 pa_hashmap_free(c->samples, free_sample_cb, NULL);
2113 pa_hashmap_free(c->modules, free_module_cb, NULL);
2114 pa_hashmap_free(c->clients, free_client_cb, NULL);
2115 pa_hook_slot_free(c->sink_put_slot);
2116 pa_hook_slot_free(c->sink_unlink_slot);
2117 pa_hook_slot_free(c->source_put_slot);
2118 pa_hook_slot_free(c->source_unlink_slot);
2119 pa_hook_slot_free(c->extension_registered_slot);
2120 pa_hook_slot_free(c->extension_unregistered_slot);
2121 pa_dbusiface_memstats_free(c->memstats);
2122
2123 if (c->fallback_sink)
2124 pa_sink_unref(c->fallback_sink);
2125 if (c->fallback_source)
2126 pa_source_unref(c->fallback_source);
2127
2128 pa_dbus_protocol_unref(c->dbus_protocol);
2129
2130 pa_xfree(c);
2131 }
2132
2133 const char *pa_dbusiface_core_get_card_path(pa_dbusiface_core *c, const pa_card *card) {
2134 pa_assert(c);
2135 pa_assert(card);
2136
2137 return pa_dbusiface_card_get_path(pa_hashmap_get(c->cards, PA_UINT32_TO_PTR(card->index)));
2138 }
2139
2140 const char *pa_dbusiface_core_get_sink_path(pa_dbusiface_core *c, const pa_sink *sink) {
2141 pa_assert(c);
2142 pa_assert(sink);
2143
2144 return pa_dbusiface_device_get_path(pa_hashmap_get(c->sinks_by_index, PA_UINT32_TO_PTR(sink->index)));
2145 }
2146
2147 const char *pa_dbusiface_core_get_source_path(pa_dbusiface_core *c, const pa_source *source) {
2148 pa_assert(c);
2149 pa_assert(source);
2150
2151 return pa_dbusiface_device_get_path(pa_hashmap_get(c->sources_by_index, PA_UINT32_TO_PTR(source->index)));
2152 }
2153
2154 const char *pa_dbusiface_core_get_playback_stream_path(pa_dbusiface_core *c, const pa_sink_input *sink_input) {
2155 pa_assert(c);
2156 pa_assert(sink_input);
2157
2158 return pa_dbusiface_stream_get_path(pa_hashmap_get(c->playback_streams, PA_UINT32_TO_PTR(sink_input->index)));
2159 }
2160
2161 const char *pa_dbusiface_core_get_record_stream_path(pa_dbusiface_core *c, const pa_source_output *source_output) {
2162 pa_assert(c);
2163 pa_assert(source_output);
2164
2165 return pa_dbusiface_stream_get_path(pa_hashmap_get(c->record_streams, PA_UINT32_TO_PTR(source_output->index)));
2166 }
2167
2168 const char *pa_dbusiface_core_get_module_path(pa_dbusiface_core *c, const pa_module *module) {
2169 pa_assert(c);
2170 pa_assert(module);
2171
2172 return pa_dbusiface_module_get_path(pa_hashmap_get(c->modules, PA_UINT32_TO_PTR(module->index)));
2173 }
2174
2175 const char *pa_dbusiface_core_get_client_path(pa_dbusiface_core *c, const pa_client *client) {
2176 pa_assert(c);
2177 pa_assert(client);
2178
2179 return pa_dbusiface_client_get_path(pa_hashmap_get(c->clients, PA_UINT32_TO_PTR(client->index)));
2180 }
2181
2182 pa_sink *pa_dbusiface_core_get_sink(pa_dbusiface_core *c, const char *object_path) {
2183 pa_dbusiface_device *device = NULL;
2184
2185 pa_assert(c);
2186 pa_assert(object_path);
2187
2188 device = pa_hashmap_get(c->sinks_by_path, object_path);
2189
2190 if (device)
2191 return pa_dbusiface_device_get_sink(device);
2192 else
2193 return NULL;
2194 }
2195
2196 pa_source *pa_dbusiface_core_get_source(pa_dbusiface_core *c, const char *object_path) {
2197 pa_dbusiface_device *device = NULL;
2198
2199 pa_assert(c);
2200 pa_assert(object_path);
2201
2202 device = pa_hashmap_get(c->sources_by_path, object_path);
2203
2204 if (device)
2205 return pa_dbusiface_device_get_source(device);
2206 else
2207 return NULL;
2208 }