1 /* -*- Mode: C ; c-basic-offset: 2 -*- */
3 * LADI Session Handler (ladish)
5 * Copyright (C) 2009, 2010 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 //#define LADISH_DEBUG
29 #include "jack_proxy.h"
31 jack_proxy_callback_server_started g_on_server_started
;
32 jack_proxy_callback_server_stopped g_on_server_stopped
;
33 jack_proxy_callback_server_appeared g_on_server_appeared
;
34 jack_proxy_callback_server_disappeared g_on_server_disappeared
;
36 static void on_jack_server_started(void * context
, DBusMessage
* message_ptr
)
38 log_debug("JACK server start signal received.");
39 if (g_on_server_started
!= NULL
)
41 g_on_server_started();
45 static void on_jack_server_stopped(void * context
, DBusMessage
* message_ptr
)
47 log_debug("JACK server stop signal received.");
48 if (g_on_server_stopped
!= NULL
)
50 g_on_server_stopped();
54 static void on_jack_life_status_changed(bool appeared
)
58 log_debug("JACK serivce appeared");
59 if (g_on_server_appeared
!= NULL
)
61 g_on_server_appeared();
66 log_debug("JACK serivce disappeared");
67 if (g_on_server_disappeared
!= NULL
)
69 g_on_server_disappeared();
74 /* this must be static because it is referenced by the
75 * dbus helper layer when hooks are active */
76 static struct dbus_signal_hook g_control_signal_hooks
[] =
78 {"ServerStarted", on_jack_server_started
},
79 {"ServerStopped", on_jack_server_stopped
},
85 jack_proxy_callback_server_started server_started
,
86 jack_proxy_callback_server_stopped server_stopped
,
87 jack_proxy_callback_server_appeared server_appeared
,
88 jack_proxy_callback_server_disappeared server_disappeared
)
90 g_on_server_started
= server_started
;
91 g_on_server_stopped
= server_stopped
;
92 g_on_server_appeared
= server_appeared
;
93 g_on_server_disappeared
= server_disappeared
;
95 if (!dbus_register_service_lifetime_hook(g_dbus_connection
, JACKDBUS_SERVICE_NAME
, on_jack_life_status_changed
))
97 log_error("dbus_register_service_lifetime_hook() failed for jackdbus service");
101 if (!dbus_register_object_signal_hooks(
103 JACKDBUS_SERVICE_NAME
,
104 JACKDBUS_OBJECT_PATH
,
105 JACKDBUS_IFACE_CONTROL
,
107 g_control_signal_hooks
))
109 dbus_unregister_service_lifetime_hook(g_dbus_connection
, JACKDBUS_SERVICE_NAME
);
110 log_error("dbus_register_object_signal_hooks() failed for jackdbus control interface");
117 if (jack_proxy_is_started(&started
))
119 if (g_on_server_appeared
!= NULL
)
121 g_on_server_appeared();
124 if (g_on_server_started
!= NULL
&& started
)
126 g_on_server_started();
138 dbus_unregister_object_signal_hooks(g_dbus_connection
, JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
);
139 dbus_unregister_service_lifetime_hook(g_dbus_connection
, JACKDBUS_SERVICE_NAME
);
143 jack_proxy_is_started(
148 if (!dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "IsStarted", "", "b", &started
))
153 *started_ptr
= started
;
158 jack_proxy_get_client_pid(
162 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_PATCHBAY
, "GetClientPID", "t", &client_id
, "x", pid_ptr
);
166 jack_proxy_connect_ports(
176 DBusMessageIter
* iter_ptr
,
177 const char * address
)
179 DBusMessageIter array_iter
;
180 const char * component
;
182 if (!dbus_message_iter_open_container(iter_ptr
, DBUS_TYPE_ARRAY
, "s", &array_iter
))
184 log_error("dbus_message_iter_open_container() failed.");
191 while (*component
!= 0)
193 if (!dbus_message_iter_append_basic(&array_iter
, DBUS_TYPE_STRING
, &component
))
195 log_error("dbus_message_iter_append_basic() failed.");
199 component
+= strlen(component
) + 1;
203 dbus_message_iter_close_container(iter_ptr
, &array_iter
);
209 jack_proxy_read_conf_container(
210 const char * address
,
211 void * callback_context
,
212 bool (* callback
)(void * context
, bool leaf
, const char * address
, char * child
))
214 DBusMessage
* request_ptr
;
215 DBusMessage
* reply_ptr
;
216 DBusMessageIter top_iter
;
217 DBusMessageIter array_iter
;
218 const char * reply_signature
;
219 dbus_bool_t leaf
; /* Whether children are parameters (true) or containers (false) */
222 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "ReadContainer");
223 if (request_ptr
== NULL
)
225 log_error("dbus_message_new_method_call() failed.");
229 dbus_message_iter_init_append(request_ptr
, &top_iter
);
231 if (!add_address(&top_iter
, address
))
233 dbus_message_unref(request_ptr
);
237 // send message and get a handle for a reply
238 reply_ptr
= dbus_connection_send_with_reply_and_block(
241 DBUS_CALL_DEFAULT_TIMEOUT
,
244 dbus_message_unref(request_ptr
);
246 if (reply_ptr
== NULL
)
248 log_error("no reply from JACK server, error is '%s'", g_dbus_error
.message
);
249 dbus_error_free(&g_dbus_error
);
253 reply_signature
= dbus_message_get_signature(reply_ptr
);
255 if (strcmp(reply_signature
, "bas") != 0)
257 log_error("ReadContainer() reply signature mismatch. '%s'", reply_signature
);
258 dbus_message_unref(reply_ptr
);
262 dbus_message_iter_init(reply_ptr
, &top_iter
);
264 dbus_message_iter_get_basic(&top_iter
, &leaf
);
265 dbus_message_iter_next(&top_iter
);
267 dbus_message_iter_recurse(&top_iter
, &array_iter
);
269 while (dbus_message_iter_get_arg_type(&array_iter
) != DBUS_TYPE_INVALID
)
271 dbus_message_iter_get_basic(&array_iter
, &child
);
273 if (!callback(callback_context
, leaf
, address
, child
))
278 dbus_message_iter_next(&array_iter
);
281 dbus_message_unref(reply_ptr
);
288 DBusMessageIter
* iter_ptr
,
289 struct jack_parameter_variant
* parameter_ptr
)
291 DBusMessageIter variant_iter
;
295 dbus_uint32_t uint32
;
298 dbus_message_iter_recurse(iter_ptr
, &variant_iter
);
299 log_debug("variant signature: '%s'", dbus_message_iter_get_signature(&variant_iter
));
301 type
= dbus_message_iter_get_arg_type(&variant_iter
);
304 case DBUS_TYPE_INT32
:
305 dbus_message_iter_get_basic(&variant_iter
, &int32
);
306 parameter_ptr
->value
.int32
= int32
;
307 parameter_ptr
->type
= jack_int32
;
309 case DBUS_TYPE_UINT32
:
310 dbus_message_iter_get_basic(&variant_iter
, &uint32
);
311 parameter_ptr
->value
.uint32
= uint32
;
312 parameter_ptr
->type
= jack_uint32
;
315 dbus_message_iter_get_basic(&variant_iter
, ¶meter_ptr
->value
.byte
);
316 parameter_ptr
->type
= jack_byte
;
318 case DBUS_TYPE_STRING
:
319 dbus_message_iter_get_basic(&variant_iter
, &string
);
320 string
= strdup(string
);
323 log_error("strdup failed.");
327 parameter_ptr
->value
.string
= string
;
328 parameter_ptr
->type
= jack_string
;
330 case DBUS_TYPE_BOOLEAN
:
331 dbus_message_iter_get_basic(&variant_iter
, &boolean
);
332 parameter_ptr
->value
.boolean
= boolean
;
333 parameter_ptr
->type
= jack_boolean
;
337 log_error("Unknown D-Bus parameter type %i", (int)type
);
342 jack_proxy_get_parameter_value(
343 const char * address
,
345 struct jack_parameter_variant
* parameter_ptr
)
347 DBusMessage
* request_ptr
;
348 DBusMessage
* reply_ptr
;
349 DBusMessageIter top_iter
;
350 const char * reply_signature
;
352 struct jack_parameter_variant default_value
;
354 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "GetParameterValue");
355 if (request_ptr
== NULL
)
357 log_error("dbus_message_new_method_call() failed.");
361 dbus_message_iter_init_append(request_ptr
, &top_iter
);
363 if (!add_address(&top_iter
, address
))
365 dbus_message_unref(request_ptr
);
369 // send message and get a handle for a reply
370 reply_ptr
= dbus_connection_send_with_reply_and_block(
373 DBUS_CALL_DEFAULT_TIMEOUT
,
376 dbus_message_unref(request_ptr
);
378 if (reply_ptr
== NULL
)
380 log_error("no reply from JACK server, error is '%s'", g_dbus_error
.message
);
381 dbus_error_free(&g_dbus_error
);
385 reply_signature
= dbus_message_get_signature(reply_ptr
);
387 if (strcmp(reply_signature
, "bvv") != 0)
389 log_error("GetParameterValue() reply signature mismatch. '%s'", reply_signature
);
390 dbus_message_unref(reply_ptr
);
394 dbus_message_iter_init(reply_ptr
, &top_iter
);
396 dbus_message_iter_get_basic(&top_iter
, &is_set
);
397 dbus_message_iter_next(&top_iter
);
399 if (!get_variant(&top_iter
, &default_value
))
401 dbus_message_unref(reply_ptr
);
405 if (default_value
.type
== jack_string
)
407 free(default_value
.value
.string
);
410 dbus_message_iter_next(&top_iter
);
412 if (!get_variant(&top_iter
, parameter_ptr
))
414 dbus_message_unref(reply_ptr
);
418 dbus_message_unref(reply_ptr
);
420 *is_set_ptr
= is_set
;
426 jack_proxy_set_parameter_value(
427 const char * address
,
428 const struct jack_parameter_variant
* parameter_ptr
)
430 DBusMessage
* request_ptr
;
431 DBusMessage
* reply_ptr
;
432 DBusMessageIter top_iter
;
433 const char * reply_signature
;
435 const void * value_ptr
;
438 switch (parameter_ptr
->type
)
441 type
= DBUS_TYPE_INT32
;
442 value_ptr
= ¶meter_ptr
->value
.int32
;
445 type
= DBUS_TYPE_UINT32
;
446 value_ptr
= ¶meter_ptr
->value
.uint32
;
449 type
= DBUS_TYPE_BYTE
;
450 value_ptr
= ¶meter_ptr
->value
.byte
;
453 type
= DBUS_TYPE_STRING
;
454 value_ptr
= ¶meter_ptr
->value
.string
;
457 type
= DBUS_TYPE_BOOLEAN
;
458 boolean
= parameter_ptr
->value
.boolean
;
459 value_ptr
= &boolean
;
462 log_error("Unknown jack parameter type %i", (int)parameter_ptr
->type
);
466 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "SetParameterValue");
467 if (request_ptr
== NULL
)
469 log_error("dbus_message_new_method_call() failed.");
473 dbus_message_iter_init_append(request_ptr
, &top_iter
);
475 if (!add_address(&top_iter
, address
))
477 dbus_message_unref(request_ptr
);
481 if (!dbus_iter_append_variant(&top_iter
, type
, value_ptr
))
483 dbus_message_unref(request_ptr
);
487 // send message and get a handle for a reply
488 reply_ptr
= dbus_connection_send_with_reply_and_block(
491 DBUS_CALL_DEFAULT_TIMEOUT
,
494 dbus_message_unref(request_ptr
);
496 if (reply_ptr
== NULL
)
498 log_error("no reply from JACK server, error is '%s'", g_dbus_error
.message
);
499 dbus_error_free(&g_dbus_error
);
503 reply_signature
= dbus_message_get_signature(reply_ptr
);
505 dbus_message_unref(reply_ptr
);
507 if (strcmp(reply_signature
, "") != 0)
509 log_error("SetParameterValue() reply signature mismatch. '%s'", reply_signature
);
517 jack_proxy_reset_parameter_value(
518 const char * address
)
520 DBusMessage
* request_ptr
;
521 DBusMessage
* reply_ptr
;
522 DBusMessageIter top_iter
;
523 const char * reply_signature
;
525 request_ptr
= dbus_message_new_method_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONFIGURE
, "ResetParameterValue");
526 if (request_ptr
== NULL
)
528 log_error("dbus_message_new_method_call() failed.");
532 dbus_message_iter_init_append(request_ptr
, &top_iter
);
534 if (!add_address(&top_iter
, address
))
536 dbus_message_unref(request_ptr
);
540 // send message and get a handle for a reply
541 reply_ptr
= dbus_connection_send_with_reply_and_block(
544 DBUS_CALL_DEFAULT_TIMEOUT
,
547 dbus_message_unref(request_ptr
);
549 if (reply_ptr
== NULL
)
551 log_error("no reply from JACK server, error is '%s'", g_dbus_error
.message
);
552 dbus_error_free(&g_dbus_error
);
556 reply_signature
= dbus_message_get_signature(reply_ptr
);
558 dbus_message_unref(reply_ptr
);
560 if (strcmp(reply_signature
, "") != 0)
562 log_error("ResetParameterValue() reply signature mismatch. '%s'", reply_signature
);
569 bool jack_proxy_start_server(void)
571 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "StartServer", "", "");
574 bool jack_proxy_stop_server(void)
576 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "StopServer", "", "");
579 bool jack_proxy_is_realtime(bool * realtime_ptr
)
581 dbus_bool_t realtime
;
583 if (!dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "IsStarted", "", "b", &realtime
))
588 *realtime_ptr
= realtime
;
592 bool jack_proxy_sample_rate(uint32_t * sample_rate_ptr
)
594 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetSampleRate", "", "u", sample_rate_ptr
);
597 bool jack_proxy_get_xruns(uint32_t * xruns_ptr
)
599 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetXruns", "", "u", xruns_ptr
);
602 bool jack_proxy_get_dsp_load(double * dsp_load_ptr
)
604 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetLoad", "", "d", dsp_load_ptr
);
607 bool jack_proxy_get_buffer_size(uint32_t * size_ptr
)
609 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "GetBufferSize", "", "u", size_ptr
);
612 bool jack_proxy_set_buffer_size(uint32_t size
)
614 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "SetBufferSize", "u", &size
, "");
617 bool jack_proxy_reset_xruns(void)
619 return dbus_call(JACKDBUS_SERVICE_NAME
, JACKDBUS_OBJECT_PATH
, JACKDBUS_IFACE_CONTROL
, "ResetXruns", "", "");
627 const char * address
,
630 const char * component
;
635 while (*component
!= 0)
637 component
+= strlen(component
) + 1;
640 /* address always is same buffer as the one in the jack_reset_all_params() stack */
641 dst
= (char *)component
;
643 len
= strlen(child
) + 1;
644 memcpy(dst
, child
, len
);
649 if (!jack_proxy_reset_parameter_value(address
))
651 log_error("cannot reset value of parameter");
657 if (!jack_proxy_read_conf_container(address
, context
, reset_callback
))
659 log_error("cannot read container");
669 bool jack_reset_all_params(void)
671 char address
[1024] = "";
673 return jack_proxy_read_conf_container(address
, NULL
, reset_callback
);