1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: t -*-*/
4 Copyright 2009 Lennart Poettering
6 Permission is hereby granted, free of charge, to any person
7 obtaining a copy of this software and associated documentation files
8 (the "Software"), to deal in the Software without restriction,
9 including without limitation the rights to use, copy, modify, merge,
10 publish, distribute, sublicense, and/or sell copies of the Software,
11 and to permit persons to whom the Software is furnished to do so,
12 subject to the following conditions:
14 The above copyright notice and this permission notice shall be
15 included in all copies or substantial portions of the Software.
17 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
21 BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
22 ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23 CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
40 char *application_name;
41 char *application_device_name;
46 DBusConnection *connection;
49 unsigned registered:1;
53 rd_request_cb_t request_cb;
57 #define SERVICE_PREFIX "org.freedesktop.ReserveDevice1."
58 #define OBJECT_PREFIX "/org/freedesktop/ReserveDevice1/"
60 static const char introspection[] =
61 DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE
63 " <!-- If you are looking for documentation make sure to check out\n"
64 " http://git.0pointer.de/?p=reserve.git;a=blob;f=reserve.txt -->\n"
65 " <interface name=\"org.freedesktop.ReserveDevice1\">"
66 " <method name=\"RequestRelease\">"
67 " <arg name=\"priority\" type=\"i\" direction=\"in\"/>"
68 " <arg name=\"result\" type=\"b\" direction=\"out\"/>"
70 " <property name=\"Priority\" type=\"i\" access=\"read\"/>"
71 " <property name=\"ApplicationName\" type=\"s\" access=\"read\"/>"
72 " <property name=\"ApplicationDeviceName\" type=\"s\" access=\"read\"/>"
74 " <interface name=\"org.freedesktop.DBus.Properties\">"
75 " <method name=\"Get\">"
76 " <arg name=\"interface\" direction=\"in\" type=\"s\"/>"
77 " <arg name=\"property\" direction=\"in\" type=\"s\"/>"
78 " <arg name=\"value\" direction=\"out\" type=\"v\"/>"
81 " <interface name=\"org.freedesktop.DBus.Introspectable\">"
82 " <method name=\"Introspect\">"
83 " <arg name=\"data\" type=\"s\" direction=\"out\"/>"
88 static dbus_bool_t add_variant(
93 DBusMessageIter iter, sub;
99 dbus_message_iter_init_append(m, &iter);
101 if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_VARIANT, t, &sub))
104 if (!dbus_message_iter_append_basic(&sub, type, data))
107 if (!dbus_message_iter_close_container(&iter, &sub))
113 static DBusHandlerResult object_handler(
120 DBusMessage *reply = NULL;
122 dbus_error_init(&error);
127 if (dbus_message_is_method_call(
129 "org.freedesktop.ReserveDevice1",
135 if (!dbus_message_get_args(
138 DBUS_TYPE_INT32, &priority,
144 if (priority > d->priority && d->request_cb) {
147 if (d->request_cb(d, 0) > 0) {
155 if (!(reply = dbus_message_new_method_return(m)))
158 if (!dbus_message_append_args(
160 DBUS_TYPE_BOOLEAN, &ret,
164 if (!dbus_connection_send(c, reply, NULL))
167 dbus_message_unref(reply);
169 return DBUS_HANDLER_RESULT_HANDLED;
171 } else if (dbus_message_is_method_call(
173 "org.freedesktop.DBus.Properties",
176 const char *interface, *property;
178 if (!dbus_message_get_args(
181 DBUS_TYPE_STRING, &interface,
182 DBUS_TYPE_STRING, &property,
186 if (strcmp(interface, "org.freedesktop.ReserveDevice1") == 0) {
187 const char *empty = "";
189 if (strcmp(property, "ApplicationName") == 0 && d->application_name) {
190 if (!(reply = dbus_message_new_method_return(m)))
196 d->application_name ? (const char**) &d->application_name : &empty))
199 } else if (strcmp(property, "ApplicationDeviceName") == 0) {
200 if (!(reply = dbus_message_new_method_return(m)))
206 d->application_device_name ? (const char**) &d->application_device_name : &empty))
209 } else if (strcmp(property, "Priority") == 0) {
210 if (!(reply = dbus_message_new_method_return(m)))
219 if (!(reply = dbus_message_new_error_printf(
221 DBUS_ERROR_UNKNOWN_METHOD,
222 "Unknown property %s",
227 if (!dbus_connection_send(c, reply, NULL))
230 dbus_message_unref(reply);
232 return DBUS_HANDLER_RESULT_HANDLED;
235 } else if (dbus_message_is_method_call(
237 "org.freedesktop.DBus.Introspectable",
239 const char *i = introspection;
241 if (!(reply = dbus_message_new_method_return(m)))
244 if (!dbus_message_append_args(
251 if (!dbus_connection_send(c, reply, NULL))
254 dbus_message_unref(reply);
256 return DBUS_HANDLER_RESULT_HANDLED;
259 return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
263 dbus_message_unref(reply);
265 if (!(reply = dbus_message_new_error(
267 DBUS_ERROR_INVALID_ARGS,
268 "Invalid arguments")))
271 if (!dbus_connection_send(c, reply, NULL))
274 dbus_message_unref(reply);
276 dbus_error_free(&error);
278 return DBUS_HANDLER_RESULT_HANDLED;
282 dbus_message_unref(reply);
284 dbus_error_free(&error);
286 return DBUS_HANDLER_RESULT_NEED_MEMORY;
289 static DBusHandlerResult filter_handler(
297 dbus_error_init(&error);
302 if (dbus_message_is_signal(m, "org.freedesktop.DBus", "NameLost")) {
305 if (!dbus_message_get_args(
308 DBUS_TYPE_STRING, &name,
312 if (strcmp(name, d->service_name) == 0 && d->owning) {
329 dbus_error_free(&error);
331 return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
335 static const struct DBusObjectPathVTable vtable ={
336 .message_function = object_handler
341 DBusConnection *connection,
342 const char *device_name,
343 const char *application_name,
345 rd_request_cb_t request_cb,
351 DBusMessage *m = NULL, *reply = NULL;
357 dbus_error_init(error);
368 if (!request_cb && priority != INT32_MAX)
371 if (!(d = calloc(sizeof(rd_device), 1)))
376 if (!(d->device_name = strdup(device_name))) {
381 if (!(d->application_name = strdup(application_name))) {
386 d->priority = priority;
387 d->connection = dbus_connection_ref(connection);
388 d->request_cb = request_cb;
390 if (!(d->service_name = malloc(sizeof(SERVICE_PREFIX) + strlen(device_name)))) {
394 sprintf(d->service_name, SERVICE_PREFIX "%s", d->device_name);
396 if (!(d->object_path = malloc(sizeof(OBJECT_PREFIX) + strlen(device_name)))) {
400 sprintf(d->object_path, OBJECT_PREFIX "%s", d->device_name);
402 if ((k = dbus_bus_request_name(
405 DBUS_NAME_FLAG_DO_NOT_QUEUE|
406 (priority < INT32_MAX ? DBUS_NAME_FLAG_ALLOW_REPLACEMENT : 0),
412 if (k == DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER)
415 if (k != DBUS_REQUEST_NAME_REPLY_EXISTS) {
420 if (priority <= INT32_MIN) {
425 if (!(m = dbus_message_new_method_call(
428 "org.freedesktop.ReserveDevice1",
429 "RequestRelease"))) {
434 if (!dbus_message_append_args(
436 DBUS_TYPE_INT32, &d->priority,
437 DBUS_TYPE_INVALID)) {
442 if (!(reply = dbus_connection_send_with_reply_and_block(
448 if (dbus_error_has_name(error, DBUS_ERROR_TIMED_OUT) ||
449 dbus_error_has_name(error, DBUS_ERROR_UNKNOWN_METHOD) ||
450 dbus_error_has_name(error, DBUS_ERROR_NO_REPLY)) {
451 /* This must be treated as denied. */
460 if (!dbus_message_get_args(
463 DBUS_TYPE_BOOLEAN, &good,
464 DBUS_TYPE_INVALID)) {
474 if ((k = dbus_bus_request_name(
477 DBUS_NAME_FLAG_DO_NOT_QUEUE|
478 (priority < INT32_MAX ? DBUS_NAME_FLAG_ALLOW_REPLACEMENT : 0)|
479 DBUS_NAME_FLAG_REPLACE_EXISTING,
485 if (k != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER) {
493 if (!(dbus_connection_register_object_path(
504 if (!dbus_connection_add_filter(
520 dbus_message_unref(m);
523 dbus_message_unref(reply);
525 if (&_error == error)
526 dbus_error_free(&_error);
547 dbus_connection_remove_filter(
553 dbus_connection_unregister_object_path(
558 dbus_bus_release_name(
563 free(d->device_name);
564 free(d->application_name);
565 free(d->application_device_name);
566 free(d->service_name);
567 free(d->object_path);
570 dbus_connection_unref(d->connection);
575 int rd_set_application_device_name(rd_device *d, const char *n) {
583 if (!(t = strdup(n)))
586 free(d->application_device_name);
587 d->application_device_name = t;
591 void rd_set_userdata(rd_device *d, void *userdata) {
597 d->userdata = userdata;
600 void* rd_get_userdata(rd_device *d) {
610 int rd_dbus_get_name_owner(
611 DBusConnection *connection,
616 DBusMessage *msg, *reply;
621 if (!(msg = dbus_message_new_method_call(DBUS_SERVICE_DBUS, DBUS_PATH_DBUS, DBUS_INTERFACE_DBUS, "GetNameOwner"))) {
626 if (!dbus_message_append_args(msg, DBUS_TYPE_STRING, &name, DBUS_TYPE_INVALID)) {
631 reply = dbus_connection_send_with_reply_and_block(connection, msg, DBUS_TIMEOUT_USE_DEFAULT, error);
632 dbus_message_unref(msg);
636 if (!dbus_message_get_args(reply, error, DBUS_TYPE_STRING, name_owner, DBUS_TYPE_INVALID)) {
637 dbus_message_unref(reply);
642 *name_owner = strdup(*name_owner);
643 dbus_message_unref(reply);
650 } else if (dbus_error_has_name(error, "org.freedesktop.DBus.Error.NameHasNoOwner"))
651 dbus_error_free(error);
661 dbus_message_unref(msg);