Incorporate siginfo-1
[ladish.git] / proxies / jack_proxy.c
blob2c0b6d01813bd348f3ac61b32abf4dde97a1414d
1 /* -*- Mode: C ; c-basic-offset: 2 -*- */
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;
34 static
35 void
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();
47 static
48 void
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)
62 if (appeared)
64 log_debug("JACK serivce appeared");
65 if (g_on_server_appeared != NULL)
67 g_on_server_appeared();
70 else
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},
86 {NULL, NULL}
89 bool
90 jack_proxy_init(
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");
104 return false;
107 if (!cdbus_register_object_signal_hooks(
108 cdbus_g_dbus_connection,
109 JACKDBUS_SERVICE_NAME,
110 JACKDBUS_OBJECT_PATH,
111 JACKDBUS_IFACE_CONTROL,
112 NULL,
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");
117 return false;
121 bool started;
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();
137 return true;
140 void
141 jack_proxy_uninit(
142 void)
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);
148 bool
149 jack_proxy_is_started(
150 bool * started_ptr)
152 dbus_bool_t started;
154 if (!cdbus_call(0, JACKDBUS_SERVICE_NAME, JACKDBUS_OBJECT_PATH, JACKDBUS_IFACE_CONTROL, "IsStarted", "", "b", &started))
156 return false;
159 *started_ptr = started;
160 return true;
163 bool
164 jack_proxy_connect_ports(
165 uint64_t UNUSED(port1_id),
166 uint64_t UNUSED(port2_id))
168 /* not implemented */
169 return false;
172 static
173 bool
174 add_address(
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.");
184 return false;
187 if (address != NULL)
189 component = address;
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.");
195 return false;
198 component += strlen(component) + 1;
202 dbus_message_iter_close_container(iter_ptr, &array_iter);
204 return true;
207 bool
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) */
219 char * child;
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.");
225 return false;
228 dbus_message_iter_init_append(request_ptr, &top_iter);
230 if (!add_address(&top_iter, address))
232 dbus_message_unref(request_ptr);
233 return false;
236 reply_ptr = cdbus_call_raw(0, request_ptr);
237 dbus_message_unref(request_ptr);
238 if (reply_ptr == NULL)
240 return false;
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);
249 return false;
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))
265 break;
268 dbus_message_iter_next(&array_iter);
271 dbus_message_unref(reply_ptr);
273 return true;
276 bool
277 get_variant(
278 DBusMessageIter * iter_ptr,
279 struct jack_parameter_variant * parameter_ptr)
281 DBusMessageIter variant_iter;
282 int type;
283 dbus_bool_t boolean;
284 dbus_int32_t int32;
285 dbus_uint32_t uint32;
286 char * string;
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);
292 switch (type)
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;
298 return true;
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;
303 return true;
304 case DBUS_TYPE_BYTE:
305 dbus_message_iter_get_basic(&variant_iter, &parameter_ptr->value.byte);
306 parameter_ptr->type = jack_byte;
307 return true;
308 case DBUS_TYPE_STRING:
309 dbus_message_iter_get_basic(&variant_iter, &string);
310 string = strdup(string);
311 if (string == NULL)
313 log_error("strdup failed.");
314 return false;
317 parameter_ptr->value.string = string;
318 parameter_ptr->type = jack_string;
319 return true;
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;
324 return true;
327 log_error("Unknown D-Bus parameter type %i", (int)type);
328 return false;
331 bool
332 jack_proxy_get_parameter_value(
333 const char * address,
334 bool * is_set_ptr,
335 struct jack_parameter_variant * parameter_ptr)
337 DBusMessage * request_ptr;
338 DBusMessage * reply_ptr;
339 DBusMessageIter top_iter;
340 const char * reply_signature;
341 dbus_bool_t is_set;
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.");
348 return false;
351 dbus_message_iter_init_append(request_ptr, &top_iter);
353 if (!add_address(&top_iter, address))
355 dbus_message_unref(request_ptr);
356 return false;
359 reply_ptr = cdbus_call_raw(0, request_ptr);
360 dbus_message_unref(request_ptr);
361 if (reply_ptr == NULL)
363 return false;
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);
372 return false;
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);
383 return false;
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);
396 return false;
399 dbus_message_unref(reply_ptr);
401 *is_set_ptr = is_set;
403 return true;
406 bool
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;
415 int type;
416 const void * value_ptr;
417 dbus_bool_t boolean;
419 switch (parameter_ptr->type)
421 case jack_int32:
422 type = DBUS_TYPE_INT32;
423 value_ptr = &parameter_ptr->value.int32;
424 break;
425 case jack_uint32:
426 type = DBUS_TYPE_UINT32;
427 value_ptr = &parameter_ptr->value.uint32;
428 break;
429 case jack_byte:
430 type = DBUS_TYPE_BYTE;
431 value_ptr = &parameter_ptr->value.byte;
432 break;
433 case jack_string:
434 type = DBUS_TYPE_STRING;
435 value_ptr = &parameter_ptr->value.string;
436 break;
437 case jack_boolean:
438 type = DBUS_TYPE_BOOLEAN;
439 boolean = parameter_ptr->value.boolean;
440 value_ptr = &boolean;
441 break;
442 default:
443 log_error("Unknown jack parameter type %i", (int)parameter_ptr->type);
444 return false;
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.");
451 return false;
454 dbus_message_iter_init_append(request_ptr, &top_iter);
456 if (!add_address(&top_iter, address))
458 dbus_message_unref(request_ptr);
459 return false;
462 if (!cdbus_iter_append_variant(&top_iter, type, value_ptr))
464 dbus_message_unref(request_ptr);
465 return false;
468 reply_ptr = cdbus_call_raw(0, request_ptr);
469 dbus_message_unref(request_ptr);
470 if (reply_ptr == NULL)
472 return false;
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);
482 return false;
485 return true;
488 bool
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.");
501 return false;
504 dbus_message_iter_init_append(request_ptr, &top_iter);
506 if (!add_address(&top_iter, address))
508 dbus_message_unref(request_ptr);
509 return false;
512 reply_ptr = cdbus_call_raw(0, request_ptr);
513 dbus_message_unref(request_ptr);
514 if (reply_ptr == NULL)
516 return false;
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);
526 return false;
529 return true;
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))
548 return false;
551 *realtime_ptr = realtime;
552 return true;
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", "", "");
585 static
586 bool
587 reset_callback(
588 void * context,
589 bool leaf,
590 const char * address,
591 char * child)
593 const char * component;
594 char * dst;
595 size_t len;
597 component = address;
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);
608 dst[len] = 0;
610 if (leaf)
612 if (!jack_proxy_reset_parameter_value(address))
614 log_error("cannot reset value of parameter");
615 return false;
618 else
620 if (!jack_proxy_read_conf_container(address, context, reset_callback))
622 log_error("cannot read container");
623 return false;
627 *dst = 0;
629 return true;
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
641 void * context;
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;
653 const char * uuid;
654 const char * client_name;
655 const char * commandline;
656 uint32_t flags;
658 if (reply_ptr == NULL)
660 cookie_ptr->callback(cookie_ptr->context, NULL);
661 return;
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);
669 return;
672 dbus_message_iter_init(reply_ptr, &top_iter);
674 commandline = NULL;
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");
683 return;
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");
708 return;
711 cookie_ptr->callback(cookie_ptr->context, commandline);
714 #undef cookie_ptr
716 bool
717 jack_proxy_session_save_one(
718 bool queue,
719 const char * target,
720 const char * path,
721 void * callback_context,
722 void (* completion_callback)(
723 void * context,
724 const char * commandline))
726 bool ret;
727 dbus_bool_t dbus_bool;
728 dbus_uint32_t type;
729 DBusMessage * request_ptr;
730 struct jack_proxy_session_save_one_reply_cookie cookie;
732 dbus_bool = queue;
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,
739 "Notify",
740 "bsus",
741 &dbus_bool,
742 &target,
743 &type,
744 &path,
745 NULL);
746 if (request_ptr == NULL)
748 return false;
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);
758 return ret;
761 bool
762 jack_proxy_session_has_callback(
763 const char * client,
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))
770 return false;
773 *has_callback_ptr = has_callback;
774 return true;
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.");
782 return false;
785 return true;