dbus.c revision 18fa6b2705aaef42041942f47f013e426640b6a4
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright 2010 Lennart Poettering
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.
systemd 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 systemd; If not, see <http://www.gnu.org/licenses/>.
***/
#include <errno.h>
#include <unistd.h>
#include "dbus.h"
#include "log.h"
#include "strv.h"
#include "cgroup.h"
#include "dbus-unit.h"
#include "dbus-job.h"
#include "dbus-manager.h"
#include "dbus-service.h"
#include "dbus-socket.h"
#include "dbus-target.h"
#include "dbus-device.h"
#include "dbus-mount.h"
#include "dbus-automount.h"
#include "dbus-snapshot.h"
#include "dbus-swap.h"
#include "dbus-timer.h"
#include "dbus-path.h"
#include "bus-errors.h"
#include "special.h"
#include "dbus-common.h"
#define CONNECTIONS_MAX 52
static const char bus_properties_interface[] = BUS_PROPERTIES_INTERFACE;
static const char bus_introspectable_interface[] = BUS_INTROSPECTABLE_INTERFACE;
const char *const bus_interface_table[] = {
"org.freedesktop.DBus.Properties", bus_properties_interface,
"org.freedesktop.DBus.Introspectable", bus_introspectable_interface,
"org.freedesktop.systemd1.Manager", bus_manager_interface,
"org.freedesktop.systemd1.Job", bus_job_interface,
"org.freedesktop.systemd1.Unit", bus_unit_interface,
"org.freedesktop.systemd1.Service", bus_service_interface,
"org.freedesktop.systemd1.Socket", bus_socket_interface,
"org.freedesktop.systemd1.Target", bus_target_interface,
"org.freedesktop.systemd1.Device", bus_device_interface,
"org.freedesktop.systemd1.Mount", bus_mount_interface,
"org.freedesktop.systemd1.Automount", bus_automount_interface,
"org.freedesktop.systemd1.Snapshot", bus_snapshot_interface,
"org.freedesktop.systemd1.Swap", bus_swap_interface,
"org.freedesktop.systemd1.Timer", bus_timer_interface,
"org.freedesktop.systemd1.Path", bus_path_interface,
};
static void bus_done_api(Manager *m);
static void bus_done_system(Manager *m);
static void bus_done_private(Manager *m);
assert(m);
/* We maintain two sets, one for those connections where we
* requested a dispatch, and another where we didn't. And then,
* we move the connections between the two sets. */
if (status == DBUS_DISPATCH_COMPLETE)
else
}
assert(m);
assert(w);
/* This is called by the event loop whenever there is
* something happening on D-Bus' file handles. */
return;
}
Watch *w;
struct epoll_event ev;
assert(m);
return FALSE;
w->type = WATCH_DBUS_WATCH;
free(w);
return FALSE;
}
/* Hmm, bloody D-Bus creates multiple watches on the
* same fd. epoll() does not like that. As a dirty
* hack we simply dup() the fd and hence get a second
* one we can safely add to the epoll(). */
free(w);
return FALSE;
}
close_nointr_nofail(w->fd);
free(w);
return FALSE;
}
w->fd_is_dupped = true;
}
return TRUE;
}
Watch *w;
assert(m);
w = dbus_watch_get_data(bus_watch);
if (!w)
return;
if (w->fd_is_dupped)
close_nointr_nofail(w->fd);
free(w);
}
Watch *w;
struct epoll_event ev;
assert(m);
w = dbus_watch_get_data(bus_watch);
if (!w)
return;
}
struct itimerspec its;
assert(m);
assert(w);
}
return -errno;
return 0;
}
assert(m);
assert(w);
/* This is called by the event loop whenever there is
* something happening on D-Bus' file handles. */
return;
}
Watch *w;
struct epoll_event ev;
assert(m);
return FALSE;
goto fail;
w->type = WATCH_DBUS_TIMEOUT;
if (bus_timeout_arm(m, w) < 0)
goto fail;
goto fail;
return TRUE;
fail:
if (w->fd >= 0)
close_nointr_nofail(w->fd);
free(w);
return FALSE;
}
Watch *w;
assert(m);
w = dbus_timeout_get_data(timeout);
if (!w)
return;
close_nointr_nofail(w->fd);
free(w);
}
Watch *w;
int r;
assert(m);
w = dbus_timeout_get_data(timeout);
if (!w)
return;
if ((r = bus_timeout_arm(m, w)) < 0)
}
static DBusHandlerResult api_bus_message_filter(DBusConnection *connection, DBusMessage *message, void *data) {
assert(m);
log_debug("Got D-Bus request: %s.%s() on %s",
log_debug("API D-Bus connection terminated.");
bus_done_api(m);
else {
log_debug("Subscription client vanished: %s (left: %u)", name, set_size(BUS_CONNECTION_SUBSCRIBED(m, connection)));
if (old_owner[0] == 0)
if (new_owner[0] == 0)
}
} else if (dbus_message_is_signal(message, "org.freedesktop.systemd1.Activator", "ActivationRequest")) {
const char *name;
else {
int r;
Unit *u;
r = -EADDRNOTAVAIL;
} else {
if (r >= 0 && u->meta.refuse_manual_start)
r = -EPERM;
if (r >= 0)
}
if (r < 0) {
if (!(reply = dbus_message_new_signal("/org/freedesktop/systemd1", "org.freedesktop.systemd1.Activator", "ActivationFailure")))
goto oom;
goto oom;
}
/* On success we don't do anything, the service will be spawned now */
}
}
if (reply) {
goto oom;
}
oom:
if (reply)
return DBUS_HANDLER_RESULT_NEED_MEMORY;
}
static DBusHandlerResult system_bus_message_filter(DBusConnection *connection, DBusMessage *message, void *data) {
assert(m);
if (m->api_bus != m->system_bus &&
log_debug("Got D-Bus request on system bus: %s.%s() on %s",
log_debug("System D-Bus connection terminated.");
bus_done_system(m);
} else if (m->running_as != MANAGER_SYSTEM &&
const char *cgroup;
else
}
}
static DBusHandlerResult private_bus_message_filter(DBusConnection *connection, DBusMessage *message, void *data) {
assert(m);
log_debug("Got D-Bus request: %s.%s() on %s",
else if (m->running_as == MANAGER_SYSTEM &&
const char *cgroup;
else
/* Forward the message to the system bus, so that user
* instances are notified as well */
if (m->system_bus)
}
}
unsigned bus_dispatch(Manager *m) {
DBusConnection *c;
assert(m);
if (m->queued_message) {
/* If we cannot get rid of this message we won't
* dispatch any D-Bus messages, so that we won't end
* up wanting to queue another message. */
if (m->queued_message_connection)
return 0;
m->queued_message = NULL;
}
if ((c = set_first(m->bus_connections_for_dispatch))) {
if (dbus_connection_dispatch(c) == DBUS_DISPATCH_COMPLETE)
return 1;
}
return 0;
}
switch (dbus_message_get_type(reply)) {
case DBUS_MESSAGE_TYPE_ERROR:
break;
case DBUS_MESSAGE_TYPE_METHOD_RETURN: {
uint32_t r;
if (!dbus_message_get_args(reply,
&error,
DBUS_TYPE_UINT32, &r,
break;
}
if (r == 1)
log_debug("Successfully acquired name.");
else
log_error("Name already owned.");
break;
}
default:
assert_not_reached("Invalid reply message");
}
}
static int request_name(Manager *m) {
const char *name = "org.freedesktop.systemd1";
/* Allow replacing of our name, to ease implementation of
* reexecution, where we keep the old connection open until
* after the new connection is set up and the name installed
* to allow clients to synchronously wait for reexecution to
* finish */
if (!(message = dbus_message_new_method_call(
"RequestName")))
goto oom;
if (!dbus_message_append_args(
goto oom;
goto oom;
goto oom;
/* We simple ask for the name and don't wait for it. Sooner or
* later we'll have it. */
return 0;
oom:
if (pending) {
}
if (message)
return -ENOMEM;
}
assert(m);
switch (dbus_message_get_type(reply)) {
case DBUS_MESSAGE_TYPE_ERROR:
break;
case DBUS_MESSAGE_TYPE_METHOD_RETURN: {
int r;
char **l;
if ((r = bus_parse_strv(reply, &l)) < 0)
else {
char **t;
STRV_FOREACH(t, l)
/* This is a bit hacky, we say the
* owner of the name is the name
* itself, because we don't want the
* extra traffic to figure out the
* real owner. */
manager_dispatch_bus_name_owner_changed(m, *t, NULL, *t);
strv_free(l);
}
break;
}
default:
assert_not_reached("Invalid reply message");
}
}
static int query_name_list(Manager *m) {
/* Asks for the currently installed bus names */
if (!(message = dbus_message_new_method_call(
"ListNames")))
goto oom;
goto oom;
goto oom;
/* We simple ask for the list and don't wait for it. Sooner or
* later we'll get it. */
return 0;
oom:
if (pending) {
}
if (message)
return -ENOMEM;
}
assert(m);
if (!dbus_connection_set_watch_functions(bus, bus_add_watch, bus_remove_watch, bus_toggle_watch, m, NULL) ||
!dbus_connection_set_timeout_functions(bus, bus_add_timeout, bus_remove_timeout, bus_toggle_timeout, m, NULL)) {
log_error("Not enough memory");
return -ENOMEM;
}
log_error("Not enough memory");
return -ENOMEM;
}
return 0;
}
return uid == 0;
}
static void bus_new_connection(
void *data) {
assert(m);
log_error("Too many concurrent connections.");
return;
}
if (bus_setup_loop(m, new_connection) < 0)
return;
if (!dbus_connection_register_object_path(new_connection, "/org/freedesktop/systemd1", &bus_manager_vtable, m) ||
!dbus_connection_register_fallback(new_connection, "/org/freedesktop/systemd1/unit", &bus_unit_vtable, m) ||
!dbus_connection_register_fallback(new_connection, "/org/freedesktop/systemd1/job", &bus_job_vtable, m) ||
log_error("Not enough memory.");
return;
}
log_debug("Accepted connection on private bus.");
}
static int bus_init_system(Manager *m) {
int r;
assert(m);
if (m->system_bus)
return 0;
m->system_bus = m->api_bus;
else {
r = 0;
goto fail;
}
if ((r = bus_setup_loop(m, m->system_bus)) < 0)
goto fail;
}
log_error("Not enough memory");
r = -ENOMEM;
goto fail;
}
if (m->running_as != MANAGER_SYSTEM) {
"type='signal',"
"interface='org.freedesktop.systemd1.Agent',"
"member='Released',"
"path='/org/freedesktop/systemd1/agent'",
&error);
if (dbus_error_is_set(&error)) {
r = -EIO;
goto fail;
}
}
if (m->api_bus != m->system_bus) {
char *id;
log_debug("Successfully connected to system D-Bus bus %s as %s",
}
return 0;
fail:
bus_done_system(m);
return r;
}
static int bus_init_api(Manager *m) {
int r;
assert(m);
if (m->api_bus)
return 0;
m->api_bus = m->system_bus;
else {
if (!(m->api_bus = dbus_bus_get_private(m->running_as == MANAGER_USER ? DBUS_BUS_SESSION : DBUS_BUS_SYSTEM, &error))) {
r = 0;
goto fail;
}
if ((r = bus_setup_loop(m, m->api_bus)) < 0)
goto fail;
}
if (!dbus_connection_register_object_path(m->api_bus, "/org/freedesktop/systemd1", &bus_manager_vtable, m) ||
!dbus_connection_register_fallback(m->api_bus, "/org/freedesktop/systemd1/unit", &bus_unit_vtable, m) ||
!dbus_connection_register_fallback(m->api_bus, "/org/freedesktop/systemd1/job", &bus_job_vtable, m) ||
log_error("Not enough memory");
r = -ENOMEM;
goto fail;
}
/* Get NameOwnerChange messages */
"type='signal',"
"member='NameOwnerChanged',"
&error);
if (dbus_error_is_set(&error)) {
r = -EIO;
goto fail;
}
/* Get activation requests */
"type='signal',"
"interface='org.freedesktop.systemd1.Activator',"
"member='ActivationRequest',"
&error);
if (dbus_error_is_set(&error)) {
r = -EIO;
goto fail;
}
if ((r = request_name(m)) < 0)
goto fail;
if ((r = query_name_list(m)) < 0)
goto fail;
if (m->api_bus != m->system_bus) {
char *id;
log_debug("Successfully connected to API D-Bus bus %s as %s",
}
return 0;
fail:
bus_done_api(m);
return r;
}
static int bus_init_private(Manager *m) {
int r;
const char *const external_only[] = {
"EXTERNAL",
};
assert(m);
if (m->private_bus)
return 0;
/* We want the private bus only when running as init */
if (getpid() != 1)
return 0;
r = -EIO;
goto fail;
}
!dbus_server_set_watch_functions(m->private_bus, bus_add_watch, bus_remove_watch, bus_toggle_watch, m, NULL) ||
!dbus_server_set_timeout_functions(m->private_bus, bus_add_timeout, bus_remove_timeout, bus_toggle_timeout, m, NULL)) {
log_error("Not enough memory");
r = -ENOMEM;
goto fail;
}
log_debug("Successfully created private D-Bus server.");
return 0;
fail:
bus_done_private(m);
return r;
}
int r;
set_ensure_allocated(&m->bus_connections_for_dispatch, trivial_hash_func, trivial_compare_func) < 0) {
log_error("Not enough memory");
return -ENOMEM;
}
if (m->name_data_slot < 0)
if (!dbus_pending_call_allocate_data_slot(&m->name_data_slot)) {
log_error("Not enough memory");
return -ENOMEM;
}
if (m->subscribed_data_slot < 0)
if (!dbus_connection_allocate_data_slot(&m->subscribed_data_slot)) {
log_error("Not enough memory");
return -ENOMEM;
}
if (try_bus_connect) {
if ((r = bus_init_system(m)) < 0 ||
(r = bus_init_api(m)) < 0)
return r;
}
if ((r = bus_init_private(m)) < 0)
return r;
return 0;
}
Set *s;
Job *j;
Iterator i;
HASHMAP_FOREACH(j, m->jobs, i)
if (j->bus == c) {
free(j->bus_client);
j->bus_client = NULL;
}
set_remove(m->bus_connections, c);
if ((s = BUS_CONNECTION_SUBSCRIBED(m, c))) {
char *t;
while ((t = set_steal_first(s)))
free(t);
set_free(s);
}
if (m->queued_message_connection == c) {
if (m->queued_message) {
m->queued_message = NULL;
}
}
}
static void bus_done_api(Manager *m) {
assert(m);
if (m->api_bus) {
if (m->system_bus == m->api_bus)
m->system_bus = NULL;
shutdown_connection(m, m->api_bus);
}
if (m->queued_message) {
m->queued_message = NULL;
}
}
static void bus_done_system(Manager *m) {
assert(m);
if (m->system_bus == m->api_bus)
bus_done_api(m);
if (m->system_bus) {
shutdown_connection(m, m->system_bus);
m->system_bus = NULL;
}
}
static void bus_done_private(Manager *m) {
if (m->private_bus) {
m->private_bus = NULL;
}
}
DBusConnection *c;
bus_done_api(m);
bus_done_system(m);
bus_done_private(m);
while ((c = set_steal_first(m->bus_connections)))
shutdown_connection(m, c);
while ((c = set_steal_first(m->bus_connections_for_dispatch)))
shutdown_connection(m, c);
set_free(m->bus_connections);
if (m->name_data_slot >= 0)
if (m->subscribed_data_slot >= 0)
}
const char *name;
switch (dbus_message_get_type(reply)) {
case DBUS_MESSAGE_TYPE_ERROR:
break;
case DBUS_MESSAGE_TYPE_METHOD_RETURN: {
uint32_t r;
if (!dbus_message_get_args(reply,
&error,
DBUS_TYPE_UINT32, &r,
break;
}
break;
}
default:
assert_not_reached("Invalid reply message");
}
}
char *n = NULL;
assert(m);
if (!(message = dbus_message_new_method_call(
"GetConnectionUnixProcessID")))
goto oom;
if (!(dbus_message_append_args(
goto oom;
goto oom;
goto oom;
goto oom;
n = NULL;
goto oom;
return 0;
oom:
free(n);
if (pending) {
}
if (message)
return -ENOMEM;
}
bool oom = false;
Iterator i;
DBusConnection *c;
assert(m);
SET_FOREACH(c, m->bus_connections_for_dispatch, i)
SET_FOREACH(c, m->bus_connections, i)
}
bool bus_has_subscriber(Manager *m) {
Iterator i;
DBusConnection *c;
assert(m);
SET_FOREACH(c, m->bus_connections_for_dispatch, i)
if (bus_connection_has_subscriber(m, c))
return true;
SET_FOREACH(c, m->bus_connections, i)
if (bus_connection_has_subscriber(m, c))
return true;
return false;
}
assert(m);
assert(c);
return !set_isempty(BUS_CONNECTION_SUBSCRIBED(m, c));
}
Iterator i;
DBusConnection *c;
assert(m);
/* When we are about to reexecute we add all D-Bus fds to the
* set to pass over to the newly executed systemd. They won't
* be used there however, except that they are closed at the
* very end of deserialization, those making it possible for
* clients to synchronously wait for systemd to reexec by
* simply waiting for disconnection */
SET_FOREACH(c, m->bus_connections_for_dispatch, i) {
int fd;
if (dbus_connection_get_unix_fd(c, &fd)) {
if (fd < 0)
return fd;
}
}
SET_FOREACH(c, m->bus_connections, i) {
int fd;
if (dbus_connection_get_unix_fd(c, &fd)) {
if (fd < 0)
return fd;
}
}
return 0;
}
void bus_broadcast_finished(
Manager *m,
usec_t total_usec) {
assert(m);
message = dbus_message_new_signal("/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager", "StartupFinished");
if (!message) {
log_error("Out of memory.");
return;
}
log_error("Out of memory.");
goto finish;
}
if (bus_broadcast(m, message) < 0) {
log_error("Out of memory.");
goto finish;
}
if (m)
}