/* * dbus.c - awesome dbus support * * Copyright © 2008 Julien Danjou * * This program 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. * * This program 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 this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * */ #include "config.h" #ifdef HAVE_DBUS #include #include "dbus.h" #include "statusbar.h" #include "widget.h" extern awesome_t globalconf; static DBusError err; static DBusConnection *dbus_connection = NULL; /** Check a dbus object path format and its number of element. * \param path The path. * \param nelem The number of element it should have. * \return true if the path is ok, false otherwise. */ static bool a_dbus_path_check(char **path, int nelem) { int i; for(i = 0; path[i]; i++); if(i != nelem) return false; return (!a_strcmp(path[0], "org")&& !a_strcmp(path[1], "awesome")); } /** Process widget.set method call. * \param req The dbus message. */ static void a_dbus_process_widget_set(DBusMessage *req) { char *arg, **path; int i; DBusMessageIter iter; widget_t *widget; widget_tell_status_t status; if(!dbus_message_get_path_decomposed(req, &path) || !a_dbus_path_check(path, 6) || a_strcmp(path[2], "widget") || a_strcmp(path[4], "property")) { warn("invalid object path."); dbus_error_free(&err); return; } if(!dbus_message_iter_init(req, &iter)) { warn("message has no argument: %s", err.message); dbus_error_free(&err); return; } else if(DBUS_TYPE_STRING != dbus_message_iter_get_arg_type(&iter)) { warn("argument must be a string"); dbus_error_free(&err); return; } else dbus_message_iter_get_basic(&iter, &arg); if(!(widget = widget_getbyname(path[3]))) return warn("no such widget: %s.", path[3]); status = widget->tell(widget, path[5], arg); widget_tell_managestatus(widget, status, path[5]); for(i = 0; path[i]; i++) p_delete(&path[i]); p_delete(&path); } void a_dbus_process_requests(int *fd) { DBusMessage *msg; int nmsg = 0; if(!dbus_connection && !a_dbus_init(fd)) return; while(true) { dbus_connection_read_write(dbus_connection, 0); if(!(msg = dbus_connection_pop_message(dbus_connection))) break; if(dbus_message_is_method_call(msg, "org.awesome.widget", "set")) a_dbus_process_widget_set(msg); else if(dbus_message_is_signal(msg, DBUS_INTERFACE_LOCAL, "Disconnected")) { a_dbus_cleanup(); dbus_message_unref(msg); return; } dbus_message_unref(msg); nmsg++; } if(nmsg) dbus_connection_flush(dbus_connection); } bool a_dbus_init(int *fd) { bool ret; dbus_error_init(&err); dbus_connection = dbus_bus_get(DBUS_BUS_SESSION, &err); if(dbus_error_is_set(&err)) { warn("DBus system bus connection failed: %s", err.message); dbus_connection = NULL; dbus_error_free(&err); return false; } dbus_connection_set_exit_on_disconnect(dbus_connection, FALSE); ret = dbus_bus_request_name(dbus_connection, "org.awesome", 0, &err); if(dbus_error_is_set(&err)) { warn("failed to request DBus name: %s", err.message); a_dbus_cleanup(); return false; } if(ret != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER) { warn("not primary DBus name owner"); a_dbus_cleanup(); return false; } if(!dbus_connection_get_unix_fd(dbus_connection, fd)) { warn("cannot get DBus connection file descriptor"); a_dbus_cleanup(); return false; } return true; } void a_dbus_cleanup(void) { if(!dbus_connection) return; dbus_error_free(&err); /* This is a shared connection owned by libdbus * Do not close it, only unref */ dbus_connection_unref(dbus_connection); dbus_connection = NULL; } #else /* HAVE_DBUS */ #include "dbus.h" void a_dbus_process_requests(int *fd __attribute__((unused))) { /* empty */ } bool a_dbus_init(int *fd) { *fd = -1; return false; } void a_dbus_cleanup(void) { /* empty */ } #endif // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80