1 /* -*- Mode: C ; c-basic-offset: 2 -*- */
3 * LADI Session Handler (ladish)
5 * Copyright (C) 2009,2010,2011,2012 Nedko Arnaudov <nedko@arnaudov.name>
7 **************************************************************************
8 * This file contains helper functionality for accessing JACK through D-Bus
9 **************************************************************************
11 * LADI Session Handler is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
16 * LADI Session Handler is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License
22 * along with LADI Session Handler. If not, see <http://www.gnu.org/licenses/>
23 * or write to the Free Software Foundation, Inc.,
24 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
27 #include "jack_proxy.h"
29 jack_proxy_callback_server_started g_on_server_started
;
30 jack_proxy_callback_server_stopped g_on_server_stopped
;
31 jack_proxy_callback_server_appeared g_on_server_appeared
;
32 jack_proxy_callback_server_disappeared g_on_server_disappeared
;
36 on_jack_server_started(
37 void * UNUSED(context
),
38 DBusMessage
* UNUSED(message_ptr
))
40 log_debug("JACK server start signal received.");
41 if (g_on_server_started
!= NULL
)
43 g_on_server_started();
49 on_jack_server_stopped(
50 void * UNUSED(context
),
51 DBusMessage
* UNUSED(message_ptr
))
53 log_debug("JACK server stop signal received.");
54 if (g_on_server_stopped
!= NULL
)
56 g_on_server_stopped();
60 static void on_jack_life_status_changed(bool appeared
)
64 log_debug("JACK serivce appeared");
65 if (g_on_server_appeared
!= NULL
)
67 g_on_server_appeared();
72 log_debug("JACK serivce disappeared");
73 if (g_on_server_disappeared
!= NULL
)
75 g_on_server_disappeared();
80 /* this must be static because it is referenced by the
81 * dbus helper layer when hooks are active */
82 static struct cdbus_signal_hook g_control_signal_hooks
[] =
84 {"ServerStarted", on_jack_server_started
},
85 {"ServerStopped", on_jack_server_stopped
},
91 jack_proxy_callback_server_started server_started
,
92 jack_proxy_callback_server_stopped server_stopped
,
93 jack_proxy_callback_server_appeared server_appeared
,
94 jack_proxy_callback_server_disappeared server_disappeared
)
96 g_on_server_started
= server_started
;
97 g_on_server_stopped
= server_stopped
;
98 g_on_server_appeared
= server_appeared
;
99 g_on_server_disappeared
= server_disappeared
;
101 if (!cdbus_register_service_lifetime_hook(cdbus_g_dbus_connection
, JACKDBUS_SERVICE_NAME
, on_jack_life_status_changed
))
103 log_error("dbus_register_service_lifetime_hook() failed for jackdbus service");
107 if (!cdbus_register_object_signal_hooks(
108 cdbus_g_dbus_connection
,
109 JACKDBUS_SERVICE_NAME
,
110 JACKDBUS_OBJECT_PATH
,
111 JACKDBUS_IFACE_CONTROL
,
113 g_control_signal_hooks
))
115 cdbus_unregister_service_lifetime_hook(cdbus_g_dbus_connection
, JACKDBUS_SERVICE_NAME
);
116 log_error("dbus_register_object_signal_hooks() failed for jackdbus control interface");
123 if (jack_proxy_is_started(&started
))
125 if (g_on_server_appeared
!= NULL
)
127 g_on_server_appeared();
130 if (g_on_server_started
!= NULL
&& started
)
132 g_on_server_started();
144 cdbus_unregister_object_signal_hooks(cdbus_g_dbus_connection
, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
);
145 cdbus_unregister_service_lifetime_hook(cdbus_g_dbus_connection
, JACKDBUS_SERVICE_NAME
);
149 jack_proxy_is_started(
154 if (!cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "IsStarted", "", "b", &started
))
159 *started_ptr
= started
;
164 jack_proxy_connect_ports(
165 uint64_t UNUSED(port1_id
),
166 uint64_t UNUSED(port2_id
))
168 /* not implemented */
175 DBusMessageIter
* iter_ptr
,
176 const char * address
)
178 DBusMessageIter array_iter
;
179 const char * component
;
181 if (!dbus_message_iter_open_container(iter_ptr
, DBUS_TYPE_ARRAY
, "s", &array_iter
))
183 log_error("dbus_message_iter_open_container() failed.");
190 while (*component
!= 0)
192 if (!dbus_message_iter_append_basic(&array_iter
, DBUS_TYPE_STRING
, &component
))
194 log_error("dbus_message_iter_append_basic() failed.");
198 component
+= strlen(component
) + 1;
202 dbus_message_iter_close_container(iter_ptr
, &array_iter
);
208 jack_proxy_read_conf_container(
209 const char * address
,
210 void * callback_context
,
211 bool (* callback
)(void * context
, bool leaf
, const char * address
, char * child
))
213 DBusMessage
* request_ptr
;
214 DBusMessage
* reply_ptr
;
215 DBusMessageIter top_iter
;
216 DBusMessageIter array_iter
;
217 const char * reply_signature
;
218 dbus_bool_t leaf
; /* Whether children are parameters (true) or containers (false) */
221 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "ReadContainer");
222 if (request_ptr
== NULL
)
224 log_error("dbus_message_new_method_call() failed.");
228 dbus_message_iter_init_append(request_ptr
, &top_iter
);
230 if (!add_address(&top_iter
, address
))
232 dbus_message_unref(request_ptr
);
236 reply_ptr
= cdbus_call_raw(0, request_ptr
);
237 dbus_message_unref(request_ptr
);
238 if (reply_ptr
== NULL
)
243 reply_signature
= dbus_message_get_signature(reply_ptr
);
245 if (strcmp(reply_signature
, "bas") != 0)
247 log_error("ReadContainer() reply signature mismatch. '%s'", reply_signature
);
248 dbus_message_unref(reply_ptr
);
252 dbus_message_iter_init(reply_ptr
, &top_iter
);
254 dbus_message_iter_get_basic(&top_iter
, &leaf
);
255 dbus_message_iter_next(&top_iter
);
257 dbus_message_iter_recurse(&top_iter
, &array_iter
);
259 while (dbus_message_iter_get_arg_type(&array_iter
) != DBUS_TYPE_INVALID
)
261 dbus_message_iter_get_basic(&array_iter
, &child
);
263 if (!callback(callback_context
, leaf
, address
, child
))
268 dbus_message_iter_next(&array_iter
);
271 dbus_message_unref(reply_ptr
);
278 DBusMessageIter
* iter_ptr
,
279 struct jack_parameter_variant
* parameter_ptr
)
281 DBusMessageIter variant_iter
;
285 dbus_uint32_t uint32
;
288 dbus_message_iter_recurse(iter_ptr
, &variant_iter
);
289 log_debug("variant signature: '%s'", dbus_message_iter_get_signature(&variant_iter
));
291 type
= dbus_message_iter_get_arg_type(&variant_iter
);
294 case DBUS_TYPE_INT32
:
295 dbus_message_iter_get_basic(&variant_iter
, &int32
);
296 parameter_ptr
->value
.int32
= int32
;
297 parameter_ptr
->type
= jack_int32
;
299 case DBUS_TYPE_UINT32
:
300 dbus_message_iter_get_basic(&variant_iter
, &uint32
);
301 parameter_ptr
->value
.uint32
= uint32
;
302 parameter_ptr
->type
= jack_uint32
;
305 dbus_message_iter_get_basic(&variant_iter
, ¶meter_ptr
->value
.byte
);
306 parameter_ptr
->type
= jack_byte
;
308 case DBUS_TYPE_STRING
:
309 dbus_message_iter_get_basic(&variant_iter
, &string
);
310 string
= strdup(string
);
313 log_error("strdup failed.");
317 parameter_ptr
->value
.string
= string
;
318 parameter_ptr
->type
= jack_string
;
320 case DBUS_TYPE_BOOLEAN
:
321 dbus_message_iter_get_basic(&variant_iter
, &boolean
);
322 parameter_ptr
->value
.boolean
= boolean
;
323 parameter_ptr
->type
= jack_boolean
;
327 log_error("Unknown D-Bus parameter type %i", (int)type
);
332 jack_proxy_get_parameter_value(
333 const char * address
,
335 struct jack_parameter_variant
* parameter_ptr
)
337 DBusMessage
* request_ptr
;
338 DBusMessage
* reply_ptr
;
339 DBusMessageIter top_iter
;
340 const char * reply_signature
;
342 struct jack_parameter_variant default_value
;
344 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "GetParameterValue");
345 if (request_ptr
== NULL
)
347 log_error("dbus_message_new_method_call() failed.");
351 dbus_message_iter_init_append(request_ptr
, &top_iter
);
353 if (!add_address(&top_iter
, address
))
355 dbus_message_unref(request_ptr
);
359 reply_ptr
= cdbus_call_raw(0, request_ptr
);
360 dbus_message_unref(request_ptr
);
361 if (reply_ptr
== NULL
)
366 reply_signature
= dbus_message_get_signature(reply_ptr
);
368 if (strcmp(reply_signature
, "bvv") != 0)
370 log_error("GetParameterValue() reply signature mismatch. '%s'", reply_signature
);
371 dbus_message_unref(reply_ptr
);
375 dbus_message_iter_init(reply_ptr
, &top_iter
);
377 dbus_message_iter_get_basic(&top_iter
, &is_set
);
378 dbus_message_iter_next(&top_iter
);
380 if (!get_variant(&top_iter
, &default_value
))
382 dbus_message_unref(reply_ptr
);
386 if (default_value
.type
== jack_string
)
388 free(default_value
.value
.string
);
391 dbus_message_iter_next(&top_iter
);
393 if (!get_variant(&top_iter
, parameter_ptr
))
395 dbus_message_unref(reply_ptr
);
399 dbus_message_unref(reply_ptr
);
401 *is_set_ptr
= is_set
;
407 jack_proxy_set_parameter_value(
408 const char * address
,
409 const struct jack_parameter_variant
* parameter_ptr
)
411 DBusMessage
* request_ptr
;
412 DBusMessage
* reply_ptr
;
413 DBusMessageIter top_iter
;
414 const char * reply_signature
;
416 const void * value_ptr
;
419 switch (parameter_ptr
->type
)
422 type
= DBUS_TYPE_INT32
;
423 value_ptr
= ¶meter_ptr
->value
.int32
;
426 type
= DBUS_TYPE_UINT32
;
427 value_ptr
= ¶meter_ptr
->value
.uint32
;
430 type
= DBUS_TYPE_BYTE
;
431 value_ptr
= ¶meter_ptr
->value
.byte
;
434 type
= DBUS_TYPE_STRING
;
435 value_ptr
= ¶meter_ptr
->value
.string
;
438 type
= DBUS_TYPE_BOOLEAN
;
439 boolean
= parameter_ptr
->value
.boolean
;
440 value_ptr
= &boolean
;
443 log_error("Unknown jack parameter type %i", (int)parameter_ptr
->type
);
447 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "SetParameterValue");
448 if (request_ptr
== NULL
)
450 log_error("dbus_message_new_method_call() failed.");
454 dbus_message_iter_init_append(request_ptr
, &top_iter
);
456 if (!add_address(&top_iter
, address
))
458 dbus_message_unref(request_ptr
);
462 if (!cdbus_iter_append_variant(&top_iter
, type
, value_ptr
))
464 dbus_message_unref(request_ptr
);
468 reply_ptr
= cdbus_call_raw(0, request_ptr
);
469 dbus_message_unref(request_ptr
);
470 if (reply_ptr
== NULL
)
475 reply_signature
= dbus_message_get_signature(reply_ptr
);
477 dbus_message_unref(reply_ptr
);
479 if (strcmp(reply_signature
, "") != 0)
481 log_error("SetParameterValue() reply signature mismatch. '%s'", reply_signature
);
489 jack_proxy_reset_parameter_value(
490 const char * address
)
492 DBusMessage
* request_ptr
;
493 DBusMessage
* reply_ptr
;
494 DBusMessageIter top_iter
;
495 const char * reply_signature
;
497 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "ResetParameterValue");
498 if (request_ptr
== NULL
)
500 log_error("dbus_message_new_method_call() failed.");
504 dbus_message_iter_init_append(request_ptr
, &top_iter
);
506 if (!add_address(&top_iter
, address
))
508 dbus_message_unref(request_ptr
);
512 reply_ptr
= cdbus_call_raw(0, request_ptr
);
513 dbus_message_unref(request_ptr
);
514 if (reply_ptr
== NULL
)
519 reply_signature
= dbus_message_get_signature(reply_ptr
);
521 dbus_message_unref(reply_ptr
);
523 if (strcmp(reply_signature
, "") != 0)
525 log_error("ResetParameterValue() reply signature mismatch. '%s'", reply_signature
);
532 bool jack_proxy_start_server(void)
534 return cdbus_call(7000, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "StartServer", "", "");
537 bool jack_proxy_stop_server(void)
539 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "StopServer", "", "");
542 bool jack_proxy_is_realtime(bool * realtime_ptr
)
544 dbus_bool_t realtime
;
546 if (!cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "IsStarted", "", "b", &realtime
))
551 *realtime_ptr
= realtime
;
555 bool jack_proxy_sample_rate(uint32_t * sample_rate_ptr
)
557 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetSampleRate", "", "u", sample_rate_ptr
);
560 bool jack_proxy_get_xruns(uint32_t * xruns_ptr
)
562 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetXruns", "", "u", xruns_ptr
);
565 bool jack_proxy_get_dsp_load(double * dsp_load_ptr
)
567 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetLoad", "", "d", dsp_load_ptr
);
570 bool jack_proxy_get_buffer_size(uint32_t * size_ptr
)
572 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetBufferSize", "", "u", size_ptr
);
575 bool jack_proxy_set_buffer_size(uint32_t size
)
577 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "SetBufferSize", "u", &size
, "");
580 bool jack_proxy_reset_xruns(void)
582 return cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "ResetXruns", "", "");
590 const char * address
,
593 const char * component
;
598 while (*component
!= 0)
600 component
+= strlen(component
) + 1;
603 /* address always is same buffer as the one in the jack_reset_all_params() stack */
604 dst
= (char *)component
;
606 len
= strlen(child
) + 1;
607 memcpy(dst
, child
, len
);
612 if (!jack_proxy_reset_parameter_value(address
))
614 log_error("cannot reset value of parameter");
620 if (!jack_proxy_read_conf_container(address
, context
, reset_callback
))
622 log_error("cannot read container");
632 bool jack_reset_all_params(void)
634 char address
[1024] = "";
636 return jack_proxy_read_conf_container(address
, NULL
, reset_callback
);
639 struct jack_proxy_session_save_one_reply_cookie
642 void (* callback
)(void * context
, const char * commandline
);
645 #define cookie_ptr ((struct jack_proxy_session_save_one_reply_cookie *)void_cookie)
647 static void jack_proxy_session_save_one_handle_reply(void * UNUSED(context
), void * void_cookie
, DBusMessage
* reply_ptr
)
649 const char * reply_signature
;
650 DBusMessageIter top_iter
;
651 DBusMessageIter array_iter
;
652 DBusMessageIter struct_iter
;
654 const char * client_name
;
655 const char * commandline
;
658 if (reply_ptr
== NULL
)
660 cookie_ptr
->callback(cookie_ptr
->context
, NULL
);
664 reply_signature
= dbus_message_get_signature(reply_ptr
);
666 if (strcmp(reply_signature
, "a(sssu)") != 0)
668 log_error(JACKDBUS_IFACE_SESSMGR
".Notify() reply signature mismatch. '%s'", reply_signature
);
672 dbus_message_iter_init(reply_ptr
, &top_iter
);
676 for (dbus_message_iter_recurse(&top_iter
, &array_iter
);
677 dbus_message_iter_get_arg_type(&array_iter
) != DBUS_TYPE_INVALID
;
678 dbus_message_iter_next(&array_iter
))
680 if (commandline
!= NULL
)
682 log_error(JACKDBUS_IFACE_SESSMGR
".Notify() save returned more than one command");
686 dbus_message_iter_recurse(&array_iter
, &struct_iter
);
688 dbus_message_iter_get_basic(&struct_iter
, &uuid
);
689 dbus_message_iter_next(&struct_iter
);
691 dbus_message_iter_get_basic(&struct_iter
, &client_name
);
692 dbus_message_iter_next(&struct_iter
);
694 dbus_message_iter_get_basic(&struct_iter
, &commandline
);
695 dbus_message_iter_next(&struct_iter
);
697 ASSERT(commandline
!= NULL
);
699 dbus_message_iter_get_basic(&struct_iter
, &flags
);
700 dbus_message_iter_next(&struct_iter
);
702 dbus_message_iter_next(&struct_iter
);
705 if (commandline
== NULL
)
707 log_error(JACKDBUS_IFACE_SESSMGR
".Notify() save returned no commands");
711 cookie_ptr
->callback(cookie_ptr
->context
, commandline
);
717 jack_proxy_session_save_one(
721 void * callback_context
,
722 void (* completion_callback
)(
724 const char * commandline
))
727 dbus_bool_t dbus_bool
;
729 DBusMessage
* request_ptr
;
730 struct jack_proxy_session_save_one_reply_cookie cookie
;
733 type
= JACKDBUS_SESSION_NOTIFY_TYPE_SAVE
;
735 request_ptr
= cdbus_new_method_call_message(
736 JACKDBUS_SERVICE_NAME
,
737 JACKDBUS_OBJECT_PATH
,
738 JACKDBUS_IFACE_SESSMGR
,
746 if (request_ptr
== NULL
)
751 cookie
.context
= callback_context
;
752 cookie
.callback
= completion_callback
;
754 ret
= cdbus_call_async(request_ptr
, callback_context
, &cookie
, sizeof(cookie
), jack_proxy_session_save_one_handle_reply
);
756 dbus_message_unref(request_ptr
);
762 jack_proxy_session_has_callback(
764 bool * has_callback_ptr
)
766 dbus_bool_t has_callback
;
768 if (!cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_SESSMGR
, "HasSessionCallback", "s", &client
, "b", &has_callback
))
773 *has_callback_ptr
= has_callback
;
777 bool jack_proxy_exit(void)
779 if (!cdbus_call(0, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "Exit", "", ""))
781 log_error("Exit() failed.");