/* -*- Mode: C ; c-basic-offset: 2 -*- */ /* * LADI Session Handler (ladish) * * Copyright (C) 2008, 2009 Nedko Arnaudov * Copyright (C) 2008 Juuso Alasuutari * ************************************************************************** * This file contains code of the D-Bus control interface helpers ************************************************************************** * * LADI Session Handler is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * LADI Session Handler is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with LADI Session Handler. If not, see * or write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "common.h" #include "../dbus/interface.h" #include "../dbus/error.h" #include "control.h" #include "../dbus_constants.h" #define INTERFACE_NAME IFACE_CONTROL static void ladish_is_studio_loaded(method_call_t * call_ptr) { DBusMessageIter iter; dbus_bool_t is_loaded; is_loaded = studio_is_loaded(); call_ptr->reply = dbus_message_new_method_return(call_ptr->message); if (call_ptr->reply == NULL) { goto fail; } dbus_message_iter_init_append(call_ptr->reply, &iter); if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &is_loaded)) { goto fail_unref; } return; fail_unref: dbus_message_unref(call_ptr->reply); call_ptr->reply = NULL; fail: lash_error("Ran out of memory trying to construct method return"); } #define array_iter_ptr ((DBusMessageIter *)context) static bool get_studio_list_callback(void * call_ptr, void * context, const char * studio, uint32_t modtime) { DBusMessageIter struct_iter; DBusMessageIter dict_iter; bool ret; ret = false; if (!dbus_message_iter_open_container(array_iter_ptr, DBUS_TYPE_STRUCT, NULL, &struct_iter)) goto exit; if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &studio)) goto close_struct; if (!dbus_message_iter_open_container(&struct_iter, DBUS_TYPE_ARRAY, "{sv}", &dict_iter)) goto close_struct; /* if (!maybe_add_dict_entry_string(&dict_iter, "Description", xxx)) */ /* goto close_dict; */ if (!dbus_add_dict_entry_uint32(&dict_iter, "Modification Time", modtime)) goto close_dict; ret = true; close_dict: if (!dbus_message_iter_close_container(&struct_iter, &dict_iter)) ret = false; close_struct: if (!dbus_message_iter_close_container(array_iter_ptr, &struct_iter)) ret = false; exit: return ret; } static void ladish_get_studio_list(method_call_t * call_ptr) { DBusMessageIter iter, array_iter; call_ptr->reply = dbus_message_new_method_return(call_ptr->message); if (call_ptr->reply == NULL) { goto fail; } dbus_message_iter_init_append(call_ptr->reply, &iter); if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "(sa{sv})", &array_iter)) { goto fail_unref; } if (!studios_iterate(call_ptr, &array_iter, get_studio_list_callback)) { dbus_message_iter_close_container(&iter, &array_iter); if (call_ptr->reply == NULL) goto fail_unref; /* studios_iterate or get_studio_list_callback() composed error reply */ return; } if (!dbus_message_iter_close_container(&iter, &array_iter)) { goto fail_unref; } return; fail_unref: dbus_message_unref(call_ptr->reply); call_ptr->reply = NULL; fail: lash_error("Ran out of memory trying to construct method return"); } static void ladish_load_studio(method_call_t * call_ptr) { const char * name; dbus_error_init(&g_dbus_error); if (!dbus_message_get_args(call_ptr->message, &g_dbus_error, DBUS_TYPE_STRING, &name, DBUS_TYPE_INVALID)) { lash_dbus_error(call_ptr, LASH_DBUS_ERROR_INVALID_ARGS, "Invalid arguments to method \"%s\": %s", call_ptr->method_name, g_dbus_error.message); dbus_error_free(&g_dbus_error); return; } if (studio_load(call_ptr, name)) { method_return_new_void(call_ptr); } } static void ladish_delete_studio(method_call_t * call_ptr) { const char * name; dbus_error_init(&g_dbus_error); if (!dbus_message_get_args(call_ptr->message, &g_dbus_error, DBUS_TYPE_STRING, &name, DBUS_TYPE_INVALID)) { lash_dbus_error(call_ptr, LASH_DBUS_ERROR_INVALID_ARGS, "Invalid arguments to method \"%s\": %s", call_ptr->method_name, g_dbus_error.message); dbus_error_free(&g_dbus_error); return; } if (studio_delete(call_ptr, name)) { method_return_new_void(call_ptr); } } static void ladish_new_studio(method_call_t * call_ptr) { const char * name; dbus_error_init(&g_dbus_error); if (!dbus_message_get_args(call_ptr->message, &g_dbus_error, DBUS_TYPE_STRING, &name, DBUS_TYPE_INVALID)) { lash_dbus_error(call_ptr, LASH_DBUS_ERROR_INVALID_ARGS, "Invalid arguments to method \"%s\": %s", call_ptr->method_name, g_dbus_error.message); dbus_error_free(&g_dbus_error); return; } if (studio_new(call_ptr, name)) { method_return_new_void(call_ptr); } } static void ladish_get_application_list(method_call_t * call_ptr) { DBusMessageIter iter; DBusMessageIter array_iter; #if 0 DBusMessageIter struct_iter; DBusMessageIter dict_iter; struct list_head * node_ptr; struct lash_appdb_entry * entry_ptr; #endif lash_info("Getting applications list"); call_ptr->reply = dbus_message_new_method_return(call_ptr->message); if (call_ptr->reply == NULL) { goto fail; } dbus_message_iter_init_append(call_ptr->reply, &iter); if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "(sa{sv})", &array_iter)) { goto fail_unref; } #if 0 list_for_each(node_ptr, &g_server->appdb) { entry_ptr = list_entry(node_ptr, struct lash_appdb_entry, siblings); if (!dbus_message_iter_open_container(&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter)) goto fail_unref; if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, (const void *) &entry_ptr->name)) { dbus_message_iter_close_container(&iter, &array_iter); goto fail_unref; } if (!dbus_message_iter_open_container(&struct_iter, DBUS_TYPE_ARRAY, "{sv}", &dict_iter)) goto fail_unref; if (!maybe_add_dict_entry_string(&dict_iter, "GenericName", entry_ptr->generic_name)) goto fail_unref; if (!maybe_add_dict_entry_string(&dict_iter, "Comment", entry_ptr->comment)) goto fail_unref; if (!maybe_add_dict_entry_string(&dict_iter, "Icon", entry_ptr->icon)) goto fail_unref; if (!dbus_message_iter_close_container(&struct_iter, &dict_iter)) goto fail_unref; if (!dbus_message_iter_close_container(&array_iter, &struct_iter)) goto fail_unref; } #endif if (!dbus_message_iter_close_container(&iter, &array_iter)) { goto fail_unref; } return; fail_unref: dbus_message_unref(call_ptr->reply); call_ptr->reply = NULL; fail: return; } static void ladish_exit(method_call_t * call_ptr) { lash_info("Exit command received through D-Bus"); g_quit = true; } void emit_studio_appeared() { signal_new_valist(g_dbus_connection, CONTROL_OBJECT_PATH, INTERFACE_NAME, "StudioAppeared", DBUS_TYPE_INVALID); } void emit_studio_disappeared() { signal_new_valist(g_dbus_connection, CONTROL_OBJECT_PATH, INTERFACE_NAME, "StudioDisappeared", DBUS_TYPE_INVALID); } METHOD_ARGS_BEGIN(IsStudioLoaded, "Check whether studio D-Bus object is present") METHOD_ARG_DESCRIBE_OUT("present", "b", "Whether studio D-Bus object is present") METHOD_ARGS_END METHOD_ARGS_BEGIN(GetStudioList, "Get list of studios") METHOD_ARG_DESCRIBE_OUT("studio_list", "a(sa{sv})", "List of studios, name and properties") METHOD_ARGS_END METHOD_ARGS_BEGIN(NewStudio, "New studio") METHOD_ARG_DESCRIBE_IN("studio_name", "s", "Name of studio, if empty name will be generated") METHOD_ARGS_END METHOD_ARGS_BEGIN(LoadStudio, "Load studio") METHOD_ARG_DESCRIBE_IN("studio_name", "s", "Name of studio to load") METHOD_ARG_DESCRIBE_IN("options", "a{sv}", "Load options") METHOD_ARGS_END METHOD_ARGS_BEGIN(DeleteStudio, "Delete studio") METHOD_ARG_DESCRIBE_IN("studio_name", "s", "Name of studio to delete") METHOD_ARGS_END METHOD_ARGS_BEGIN(GetApplicationList, "Get list of applications that can be launched") METHOD_ARG_DESCRIBE_OUT("applications", "a(sa{sv})", "List of applications, name and properties") METHOD_ARGS_END METHOD_ARGS_BEGIN(Exit, "Tell ladish D-Bus service to exit") METHOD_ARGS_END METHODS_BEGIN METHOD_DESCRIBE(IsStudioLoaded, ladish_is_studio_loaded) METHOD_DESCRIBE(GetStudioList, ladish_get_studio_list) METHOD_DESCRIBE(NewStudio, ladish_new_studio) METHOD_DESCRIBE(LoadStudio, ladish_load_studio) METHOD_DESCRIBE(DeleteStudio, ladish_delete_studio) METHOD_DESCRIBE(GetApplicationList, ladish_get_application_list) METHOD_DESCRIBE(Exit, ladish_exit) METHODS_END SIGNAL_ARGS_BEGIN(StudioAppeared, "Studio D-Bus object appeared") SIGNAL_ARGS_END SIGNAL_ARGS_BEGIN(StudioDisappeared, "Studio D-Bus object disappeared") SIGNAL_ARGS_END SIGNALS_BEGIN SIGNAL_DESCRIBE(StudioAppeared) SIGNAL_DESCRIBE(StudioDisappeared) SIGNALS_END /* * Interface description. */ INTERFACE_BEGIN(g_lashd_interface_control, INTERFACE_NAME) INTERFACE_DEFAULT_HANDLER INTERFACE_EXPOSE_METHODS INTERFACE_EXPOSE_SIGNALS INTERFACE_END