First public contribution.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* services.c Service management
4 * Copyright (C) 2003 Red Hat, Inc.
5 * Copyright (C) 2003 CodeFactory AB
6 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
7 * Licensed under the Academic Free License version 2.1
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25 #include <dbus/dbus-hash.h>
26 #include <dbus/dbus-list.h>
27 #include <dbus/dbus-mempool.h>
28 #include <dbus/dbus-marshal-validate.h>
30 #include "dbus-hash.h"
31 #include "dbus-list.h"
32 #include "dbus-mempool.h"
33 #include "dbus-marshal-validate.h"
34 #endif //__SYMBIAN32__
38 #include "connection.h"
40 #include "activation.h"
49 BusRegistry *registry;
61 unsigned int allow_replacement : 1;
62 unsigned int do_not_queue : 1;
71 DBusHashTable *service_hash;
72 DBusMemPool *service_pool;
73 DBusMemPool *owner_pool;
75 DBusHashTable *service_sid_table;
79 bus_registry_new (BusContext *context)
81 BusRegistry *registry;
83 registry = dbus_new0 (BusRegistry, 1);
87 registry->refcount = 1;
88 registry->context = context;
90 registry->service_hash = _dbus_hash_table_new (DBUS_HASH_STRING,
92 if (registry->service_hash == NULL)
95 registry->service_pool = _dbus_mem_pool_new (sizeof (BusService),
98 if (registry->service_pool == NULL)
101 registry->owner_pool = _dbus_mem_pool_new (sizeof (BusOwner),
104 if (registry->owner_pool == NULL)
107 registry->service_sid_table = NULL;
112 bus_registry_unref (registry);
117 bus_registry_ref (BusRegistry *registry)
119 _dbus_assert (registry->refcount > 0);
120 registry->refcount += 1;
126 bus_registry_unref (BusRegistry *registry)
128 _dbus_assert (registry->refcount > 0);
129 registry->refcount -= 1;
131 if (registry->refcount == 0)
133 if (registry->service_hash)
134 _dbus_hash_table_unref (registry->service_hash);
135 if (registry->service_pool)
136 _dbus_mem_pool_free (registry->service_pool);
137 if (registry->owner_pool)
138 _dbus_mem_pool_free (registry->owner_pool);
139 if (registry->service_sid_table)
140 _dbus_hash_table_unref (registry->service_sid_table);
142 dbus_free (registry);
147 bus_registry_lookup (BusRegistry *registry,
148 const DBusString *service_name)
152 service = _dbus_hash_table_lookup_string (registry->service_hash,
153 _dbus_string_get_const_data (service_name));
159 _bus_service_find_owner_link (BusService *service,
160 DBusConnection *connection)
164 link = _dbus_list_get_first_link (&service->owners);
170 bus_owner = (BusOwner *) link->data;
171 if (bus_owner->conn == connection)
174 link = _dbus_list_get_next_link (&service->owners, link);
181 bus_owner_set_flags (BusOwner *owner,
184 owner->allow_replacement =
185 (flags & DBUS_NAME_FLAG_ALLOW_REPLACEMENT) != FALSE;
187 owner->do_not_queue =
188 (flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) != FALSE;
192 bus_owner_new (BusService *service,
193 DBusConnection *conn,
198 result = _dbus_mem_pool_alloc (service->registry->owner_pool);
201 result->refcount = 1;
202 /* don't ref the connection because we don't want
203 to block the connection from going away.
204 transactions take care of reffing the connection
205 but we need to use refcounting on the owner
206 so that the owner does not get freed before
207 we can deref the connection in the transaction
210 result->service = service;
212 if (!bus_connection_add_owned_service (conn, service))
214 _dbus_mem_pool_dealloc (service->registry->owner_pool, result);
218 bus_owner_set_flags (result, flags);
224 bus_owner_ref (BusOwner *owner)
226 _dbus_assert (owner->refcount > 0);
227 owner->refcount += 1;
233 bus_owner_unref (BusOwner *owner)
235 _dbus_assert (owner->refcount > 0);
236 owner->refcount -= 1;
238 if (owner->refcount == 0)
240 bus_connection_remove_owned_service (owner->conn, owner->service);
241 _dbus_mem_pool_dealloc (owner->service->registry->owner_pool, owner);
246 bus_registry_ensure (BusRegistry *registry,
247 const DBusString *service_name,
248 DBusConnection *owner_connection_if_created,
250 BusTransaction *transaction,
255 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
257 _dbus_assert (owner_connection_if_created != NULL);
258 _dbus_assert (transaction != NULL);
260 service = _dbus_hash_table_lookup_string (registry->service_hash,
261 _dbus_string_get_const_data (service_name));
265 service = _dbus_mem_pool_alloc (registry->service_pool);
272 service->registry = registry;
273 service->refcount = 1;
275 _dbus_verbose ("copying string %p '%s' to service->name\n",
276 service_name, _dbus_string_get_const_data (service_name));
277 if (!_dbus_string_copy_data (service_name, &service->name))
279 _dbus_mem_pool_dealloc (registry->service_pool, service);
283 _dbus_verbose ("copied string %p '%s' to '%s'\n",
284 service_name, _dbus_string_get_const_data (service_name),
287 if (!bus_driver_send_service_owner_changed (service->name,
289 bus_connection_get_name (owner_connection_if_created),
292 bus_service_unref (service);
296 if (!bus_activation_service_created (bus_context_get_activation (registry->context),
297 service->name, transaction, error))
299 bus_service_unref (service);
303 if (!bus_service_add_owner (service, owner_connection_if_created, flags,
306 bus_service_unref (service);
310 if (!_dbus_hash_table_insert_string (registry->service_hash,
314 /* The add_owner gets reverted on transaction cancel */
323 bus_registry_foreach (BusRegistry *registry,
324 BusServiceForeachFunction function,
329 _dbus_hash_iter_init (registry->service_hash, &iter);
330 while (_dbus_hash_iter_next (&iter))
332 BusService *service = _dbus_hash_iter_get_value (&iter);
334 (* function) (service, data);
339 bus_registry_list_services (BusRegistry *registry,
347 len = _dbus_hash_table_get_n_entries (registry->service_hash);
348 retval = dbus_new (char *, len + 1);
353 _dbus_hash_iter_init (registry->service_hash, &iter);
355 while (_dbus_hash_iter_next (&iter))
357 BusService *service = _dbus_hash_iter_get_value (&iter);
359 retval[i] = _dbus_strdup (service->name);
360 if (retval[i] == NULL)
375 for (j = 0; j < i; j++)
376 dbus_free (retval[i]);
383 bus_registry_acquire_service (BusRegistry *registry,
384 DBusConnection *connection,
385 const DBusString *service_name,
387 dbus_uint32_t *result,
388 BusTransaction *transaction,
392 DBusConnection *old_owner_conn;
393 DBusConnection *current_owner_conn;
394 BusClientPolicy *policy;
396 BusActivation *activation;
398 BusOwner *primary_owner;
402 if (!_dbus_validate_bus_name (service_name, 0,
403 _dbus_string_get_length (service_name)))
405 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
406 "Requested bus name \"%s\" is not valid",
407 _dbus_string_get_const_data (service_name));
409 _dbus_verbose ("Attempt to acquire invalid service name\n");
414 if (_dbus_string_get_byte (service_name, 0) == ':')
416 /* Not allowed; only base services can start with ':' */
417 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
418 "Cannot acquire a service starting with ':' such as \"%s\"",
419 _dbus_string_get_const_data (service_name));
421 _dbus_verbose ("Attempt to acquire invalid base service name \"%s\"",
422 _dbus_string_get_const_data (service_name));
427 if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
429 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
430 "Connection \"%s\" is not allowed to own the service \"%s\"because "
431 "it is reserved for D-Bus' use only",
432 bus_connection_is_active (connection) ?
433 bus_connection_get_name (connection) :
439 policy = bus_connection_get_policy (connection);
440 _dbus_assert (policy != NULL);
442 /* Note that if sid is #NULL then the bus's own context gets used
443 * in bus_connection_selinux_allows_acquire_service()
445 sid = bus_selinux_id_table_lookup (registry->service_sid_table,
448 if (!bus_selinux_allows_acquire_service (connection, sid,
449 _dbus_string_get_const_data (service_name), error))
452 if (dbus_error_is_set (error) &&
453 dbus_error_has_name (error, DBUS_ERROR_NO_MEMORY))
458 dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
459 "Connection \"%s\" is not allowed to own the service \"%s\" due "
461 bus_connection_is_active (connection) ?
462 bus_connection_get_name (connection) :
464 _dbus_string_get_const_data (service_name));
468 if (!bus_client_policy_check_can_own (policy, connection,
471 dbus_set_error (error, DBUS_ERROR_ACCESS_DENIED,
472 "Connection \"%s\" is not allowed to own the service \"%s\" due "
473 "to security policies in the configuration file",
474 bus_connection_is_active (connection) ?
475 bus_connection_get_name (connection) :
477 _dbus_string_get_const_data (service_name));
481 if (bus_connection_get_n_services_owned (connection) >=
482 bus_context_get_max_services_per_connection (registry->context))
484 dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
485 "Connection \"%s\" is not allowed to own more services "
486 "(increase limits in configuration file if required)",
487 bus_connection_is_active (connection) ?
488 bus_connection_get_name (connection) :
493 service = bus_registry_lookup (registry, service_name);
497 primary_owner = bus_service_get_primary_owner (service);
498 if (primary_owner != NULL)
499 old_owner_conn = primary_owner->conn;
501 old_owner_conn = NULL;
504 old_owner_conn = NULL;
508 service = bus_registry_ensure (registry,
509 service_name, connection, flags,
515 primary_owner = bus_service_get_primary_owner (service);
516 if (primary_owner == NULL)
519 current_owner_conn = primary_owner->conn;
521 if (old_owner_conn == NULL)
523 _dbus_assert (current_owner_conn == connection);
525 *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;
527 else if (old_owner_conn == connection)
529 bus_owner_set_flags (primary_owner, flags);
530 *result = DBUS_REQUEST_NAME_REPLY_ALREADY_OWNER;
532 else if (((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
533 !(bus_service_get_allow_replacement (service))) ||
534 ((flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
535 !(flags & DBUS_NAME_FLAG_REPLACE_EXISTING)))
538 BusOwner *temp_owner;
539 /* Since we can't be queued if we are already in the queue
542 link = _bus_service_find_owner_link (service, connection);
545 _dbus_list_unlink (&service->owners, link);
546 temp_owner = (BusOwner *)link->data;
547 bus_owner_unref (temp_owner);
548 _dbus_list_free_link (link);
551 *result = DBUS_REQUEST_NAME_REPLY_EXISTS;
553 else if (!(flags & DBUS_NAME_FLAG_DO_NOT_QUEUE) &&
554 (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) ||
555 !(bus_service_get_allow_replacement (service))))
557 /* Queue the connection */
558 if (!bus_service_add_owner (service, connection,
563 *result = DBUS_REQUEST_NAME_REPLY_IN_QUEUE;
567 /* Replace the current owner */
569 /* We enqueue the new owner and remove the first one because
570 * that will cause NameAcquired and NameLost messages to
574 if (!bus_service_add_owner (service, connection,
579 if (primary_owner->do_not_queue)
581 if (!bus_service_remove_owner (service, old_owner_conn,
587 if (!bus_service_swap_owner (service, old_owner_conn,
593 _dbus_assert (connection == bus_service_get_primary_owner (service)->conn);
594 *result = DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER;
597 activation = bus_context_get_activation (registry->context);
598 retval = bus_activation_send_pending_auto_activation_messages (activation,
608 bus_registry_release_service (BusRegistry *registry,
609 DBusConnection *connection,
610 const DBusString *service_name,
611 dbus_uint32_t *result,
612 BusTransaction *transaction,
620 if (!_dbus_validate_bus_name (service_name, 0,
621 _dbus_string_get_length (service_name)))
623 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
624 "Given bus name \"%s\" is not valid",
625 _dbus_string_get_const_data (service_name));
627 _dbus_verbose ("Attempt to release invalid service name\n");
632 if (_dbus_string_get_byte (service_name, 0) == ':')
634 /* Not allowed; the base service name cannot be created or released */
635 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
636 "Cannot release a service starting with ':' such as \"%s\"",
637 _dbus_string_get_const_data (service_name));
639 _dbus_verbose ("Attempt to release invalid base service name \"%s\"",
640 _dbus_string_get_const_data (service_name));
645 if (_dbus_string_equal_c_str (service_name, DBUS_SERVICE_DBUS))
647 /* Not allowed; the base service name cannot be created or released */
648 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS,
649 "Cannot release the %s service because it is owned by the bus",
652 _dbus_verbose ("Attempt to release service name \"%s\"",
658 service = bus_registry_lookup (registry, service_name);
662 *result = DBUS_RELEASE_NAME_REPLY_NON_EXISTENT;
664 else if (!bus_service_has_owner (service, connection))
666 *result = DBUS_RELEASE_NAME_REPLY_NOT_OWNER;
670 if (!bus_service_remove_owner (service, connection,
674 _dbus_assert (!bus_service_has_owner (service, connection));
675 *result = DBUS_RELEASE_NAME_REPLY_RELEASED;
685 bus_registry_set_service_context_table (BusRegistry *registry,
686 DBusHashTable *table)
688 DBusHashTable *new_table;
691 new_table = bus_selinux_id_table_new ();
695 _dbus_hash_iter_init (table, &iter);
696 while (_dbus_hash_iter_next (&iter))
698 const char *service = _dbus_hash_iter_get_string_key (&iter);
699 const char *context = _dbus_hash_iter_get_value (&iter);
701 if (!bus_selinux_id_table_insert (new_table,
707 if (registry->service_sid_table)
708 _dbus_hash_table_unref (registry->service_sid_table);
709 registry->service_sid_table = new_table;
714 bus_service_unlink_owner (BusService *service,
717 _dbus_list_remove_last (&service->owners, owner);
718 bus_owner_unref (owner);
722 bus_service_unlink (BusService *service)
724 _dbus_assert (service->owners == NULL);
726 /* the service may not be in the hash, if
727 * the failure causing transaction cancel
728 * was in the right place, but that's OK
730 _dbus_hash_table_remove_string (service->registry->service_hash,
733 bus_service_unref (service);
737 bus_service_relink (BusService *service,
738 DBusPreallocatedHash *preallocated)
740 _dbus_assert (service->owners == NULL);
741 _dbus_assert (preallocated != NULL);
743 _dbus_hash_table_insert_string_preallocated (service->registry->service_hash,
748 bus_service_ref (service);
752 * Data used to represent an ownership cancellation in
757 BusOwner *owner; /**< the owner */
758 BusService *service; /**< service to cancel ownership of */
759 } OwnershipCancelData;
762 cancel_ownership (void *data)
764 OwnershipCancelData *d = data;
766 /* We don't need to send messages notifying of these
767 * changes, since we're reverting something that was
768 * cancelled (effectively never really happened)
770 bus_service_unlink_owner (d->service, d->owner);
772 if (d->service->owners == NULL)
773 bus_service_unlink (d->service);
777 free_ownership_cancel_data (void *data)
779 OwnershipCancelData *d = data;
781 dbus_connection_unref (d->owner->conn);
782 bus_owner_unref (d->owner);
783 bus_service_unref (d->service);
789 add_cancel_ownership_to_transaction (BusTransaction *transaction,
793 OwnershipCancelData *d;
795 d = dbus_new (OwnershipCancelData, 1);
799 d->service = service;
802 if (!bus_transaction_add_cancel_hook (transaction, cancel_ownership, d,
803 free_ownership_cancel_data))
809 bus_service_ref (d->service);
810 bus_owner_ref (owner);
811 dbus_connection_ref (d->owner->conn);
816 /* this function is self-cancelling if you cancel the transaction */
818 bus_service_add_owner (BusService *service,
819 DBusConnection *connection,
821 BusTransaction *transaction,
825 DBusList *bus_owner_link;
827 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
829 /* Send service acquired message first, OOM will result
830 * in cancelling the transaction
832 if (service->owners == NULL)
834 if (!bus_driver_send_service_acquired (connection, service->name, transaction, error))
838 bus_owner_link = _bus_service_find_owner_link (service, connection);
840 if (bus_owner_link == NULL)
842 bus_owner = bus_owner_new (service, connection, flags);
843 if (bus_owner == NULL)
849 bus_owner_set_flags (bus_owner, flags);
850 if (!(flags & DBUS_NAME_FLAG_REPLACE_EXISTING) || service->owners == NULL)
852 if (!_dbus_list_append (&service->owners,
855 bus_owner_unref (bus_owner);
862 if (!_dbus_list_insert_after (&service->owners,
863 _dbus_list_get_first_link (&service->owners),
866 bus_owner_unref (bus_owner);
874 /* Update the link since we are already in the queue
875 * No need for operations that can produce OOM
878 bus_owner = (BusOwner *) bus_owner_link->data;
879 if (flags & DBUS_NAME_FLAG_REPLACE_EXISTING)
882 _dbus_list_unlink (&service->owners, bus_owner_link);
883 link = _dbus_list_get_first_link (&service->owners);
884 _dbus_assert (link != NULL);
886 _dbus_list_insert_after_link (&service->owners, link, bus_owner_link);
889 bus_owner_set_flags (bus_owner, flags);
893 if (!add_cancel_ownership_to_transaction (transaction,
897 bus_service_unlink_owner (service, bus_owner);
909 BusOwner *before_owner; /* restore to position before this connection in owners list */
910 DBusList *owner_link;
911 DBusList *service_link;
912 DBusPreallocatedHash *hash_entry;
913 } OwnershipRestoreData;
916 restore_ownership (void *data)
918 OwnershipRestoreData *d = data;
921 _dbus_assert (d->service_link != NULL);
922 _dbus_assert (d->owner_link != NULL);
924 if (d->service->owners == NULL)
926 _dbus_assert (d->hash_entry != NULL);
927 bus_service_relink (d->service, d->hash_entry);
931 _dbus_assert (d->hash_entry == NULL);
934 /* We don't need to send messages notifying of these
935 * changes, since we're reverting something that was
936 * cancelled (effectively never really happened)
938 link = _dbus_list_get_first_link (&d->service->owners);
941 if (link->data == d->before_owner)
944 link = _dbus_list_get_next_link (&d->service->owners, link);
947 _dbus_list_insert_before_link (&d->service->owners, link, d->owner_link);
949 /* Note that removing then restoring this changes the order in which
950 * ServiceDeleted messages are sent on destruction of the
951 * connection. This should be OK as the only guarantee there is
952 * that the base service is destroyed last, and we never even
953 * tentatively remove the base service.
955 bus_connection_add_owned_service_link (d->owner->conn, d->service_link);
957 d->hash_entry = NULL;
958 d->service_link = NULL;
959 d->owner_link = NULL;
963 free_ownership_restore_data (void *data)
965 OwnershipRestoreData *d = data;
968 _dbus_list_free_link (d->service_link);
970 _dbus_list_free_link (d->owner_link);
972 _dbus_hash_table_free_preallocated_entry (d->service->registry->service_hash,
975 dbus_connection_unref (d->owner->conn);
976 bus_owner_unref (d->owner);
977 bus_service_unref (d->service);
983 add_restore_ownership_to_transaction (BusTransaction *transaction,
987 OwnershipRestoreData *d;
990 d = dbus_new (OwnershipRestoreData, 1);
994 d->service = service;
996 d->service_link = _dbus_list_alloc_link (service);
997 d->owner_link = _dbus_list_alloc_link (owner);
998 d->hash_entry = _dbus_hash_table_preallocate_entry (service->registry->service_hash);
1000 bus_service_ref (d->service);
1001 bus_owner_ref (d->owner);
1002 dbus_connection_ref (d->owner->conn);
1004 d->before_owner = NULL;
1005 link = _dbus_list_get_first_link (&service->owners);
1006 while (link != NULL)
1008 if (link->data == owner)
1010 link = _dbus_list_get_next_link (&service->owners, link);
1013 d->before_owner = link->data;
1018 link = _dbus_list_get_next_link (&service->owners, link);
1021 if (d->service_link == NULL ||
1022 d->owner_link == NULL ||
1023 d->hash_entry == NULL ||
1024 !bus_transaction_add_cancel_hook (transaction, restore_ownership, d,
1025 free_ownership_restore_data))
1027 free_ownership_restore_data (d);
1035 bus_service_swap_owner (BusService *service,
1036 DBusConnection *connection,
1037 BusTransaction *transaction,
1040 DBusList *swap_link;
1041 BusOwner *primary_owner;
1043 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1045 /* We send out notifications before we do any work we
1046 * might have to undo if the notification-sending failed
1049 /* Send service lost message */
1050 primary_owner = bus_service_get_primary_owner (service);
1051 if (primary_owner == NULL || primary_owner->conn != connection)
1052 _dbus_assert_not_reached ("Tried to swap a non primary owner");
1055 if (!bus_driver_send_service_lost (connection, service->name,
1056 transaction, error))
1059 if (service->owners == NULL)
1061 _dbus_assert_not_reached ("Tried to swap owner of a service that has no owners");
1063 else if (_dbus_list_length_is_one (&service->owners))
1065 _dbus_assert_not_reached ("Tried to swap owner of a service that has no other owners in the queue");
1070 BusOwner *new_owner;
1071 DBusConnection *new_owner_conn;
1072 link = _dbus_list_get_first_link (&service->owners);
1073 _dbus_assert (link != NULL);
1074 link = _dbus_list_get_next_link (&service->owners, link);
1075 _dbus_assert (link != NULL);
1077 new_owner = (BusOwner *)link->data;
1078 new_owner_conn = new_owner->conn;
1080 if (!bus_driver_send_service_owner_changed (service->name,
1081 bus_connection_get_name (connection),
1082 bus_connection_get_name (new_owner_conn),
1083 transaction, error))
1086 /* This will be our new owner */
1087 if (!bus_driver_send_service_acquired (new_owner_conn,
1094 if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
1096 BUS_SET_OOM (error);
1100 /* unlink the primary and make it the second link */
1101 swap_link = _dbus_list_get_first_link (&service->owners);
1102 _dbus_list_unlink (&service->owners, swap_link);
1104 _dbus_list_insert_after_link (&service->owners,
1105 _dbus_list_get_first_link (&service->owners),
1111 /* this function is self-cancelling if you cancel the transaction */
1113 bus_service_remove_owner (BusService *service,
1114 DBusConnection *connection,
1115 BusTransaction *transaction,
1118 BusOwner *primary_owner;
1120 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
1122 /* We send out notifications before we do any work we
1123 * might have to undo if the notification-sending failed
1126 /* Send service lost message */
1127 primary_owner = bus_service_get_primary_owner (service);
1128 if (primary_owner != NULL && primary_owner->conn == connection)
1130 if (!bus_driver_send_service_lost (connection, service->name,
1131 transaction, error))
1136 /* if we are not the primary owner then just remove us from the queue */
1138 BusOwner *temp_owner;
1140 link = _bus_service_find_owner_link (service, connection);
1141 _dbus_list_unlink (&service->owners, link);
1142 temp_owner = (BusOwner *)link->data;
1143 bus_owner_unref (temp_owner);
1144 _dbus_list_free_link (link);
1149 if (service->owners == NULL)
1151 _dbus_assert_not_reached ("Tried to remove owner of a service that has no owners");
1153 else if (_dbus_list_length_is_one (&service->owners))
1155 if (!bus_driver_send_service_owner_changed (service->name,
1156 bus_connection_get_name (connection),
1158 transaction, error))
1164 BusOwner *new_owner;
1165 DBusConnection *new_owner_conn;
1166 link = _dbus_list_get_first_link (&service->owners);
1167 _dbus_assert (link != NULL);
1168 link = _dbus_list_get_next_link (&service->owners, link);
1169 _dbus_assert (link != NULL);
1171 new_owner = (BusOwner *)link->data;
1172 new_owner_conn = new_owner->conn;
1174 if (!bus_driver_send_service_owner_changed (service->name,
1175 bus_connection_get_name (connection),
1176 bus_connection_get_name (new_owner_conn),
1177 transaction, error))
1180 /* This will be our new owner */
1181 if (!bus_driver_send_service_acquired (new_owner_conn,
1188 if (!add_restore_ownership_to_transaction (transaction, service, primary_owner))
1190 BUS_SET_OOM (error);
1194 bus_service_unlink_owner (service, primary_owner);
1196 if (service->owners == NULL)
1197 bus_service_unlink (service);
1203 bus_service_ref (BusService *service)
1205 _dbus_assert (service->refcount > 0);
1207 service->refcount += 1;
1213 bus_service_unref (BusService *service)
1215 _dbus_assert (service->refcount > 0);
1217 service->refcount -= 1;
1219 if (service->refcount == 0)
1221 _dbus_assert (service->owners == NULL);
1223 dbus_free (service->name);
1224 _dbus_mem_pool_dealloc (service->registry->service_pool, service);
1229 bus_service_get_primary_owners_connection (BusService *service)
1233 owner = bus_service_get_primary_owner (service);
1242 bus_service_get_primary_owner (BusService *service)
1244 return _dbus_list_get_first (&service->owners);
1248 bus_service_get_name (BusService *service)
1250 return service->name;
1254 bus_service_get_allow_replacement (BusService *service)
1259 _dbus_assert (service->owners != NULL);
1261 link = _dbus_list_get_first_link (&service->owners);
1262 owner = (BusOwner *) link->data;
1264 return owner->allow_replacement;
1268 bus_service_has_owner (BusService *service,
1269 DBusConnection *connection)
1273 link = _bus_service_find_owner_link (service, connection);
1282 bus_service_list_queued_owners (BusService *service,
1283 DBusList **return_list,
1288 _dbus_assert (*return_list == NULL);
1290 link = _dbus_list_get_first_link (&service->owners);
1291 _dbus_assert (link != NULL);
1293 while (link != NULL)
1298 owner = (BusOwner *) link->data;
1299 uname = bus_connection_get_name (owner->conn);
1301 if (!_dbus_list_append (return_list, (char *)uname))
1304 link = _dbus_list_get_next_link (&service->owners, link);
1310 _dbus_list_clear (return_list);
1311 BUS_SET_OOM (error);