First public contribution.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* signals.c Bus signal connection implementation
4 * Copyright (C) 2003, 2005 Red Hat, Inc.
5 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
6 * Licensed under the Academic Free License version 2.1
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
27 #include <dbus/dbus-marshal-validate.h>
29 #include "dbus-marshal-validate.h"
31 #endif //__SYMBIAN32__
35 int refcount; /**< reference count */
37 DBusConnection *matches_go_to; /**< Owner of the rule */
39 unsigned int flags; /**< BusMatchFlags */
53 bus_match_rule_new (DBusConnection *matches_go_to)
57 rule = dbus_new0 (BusMatchRule, 1);
62 rule->matches_go_to = matches_go_to;
64 #ifndef DBUS_BUILD_TESTS
65 _dbus_assert (rule->matches_go_to != NULL);
72 bus_match_rule_ref (BusMatchRule *rule)
74 _dbus_assert (rule->refcount > 0);
82 bus_match_rule_unref (BusMatchRule *rule)
84 _dbus_assert (rule->refcount > 0);
87 if (rule->refcount == 0)
89 dbus_free (rule->interface);
90 dbus_free (rule->member);
91 dbus_free (rule->sender);
92 dbus_free (rule->destination);
93 dbus_free (rule->path);
95 /* can't use dbus_free_string_array() since there
103 while (i < rule->args_len)
106 dbus_free (rule->args[i]);
110 dbus_free (rule->args);
117 #ifdef DBUS_ENABLE_VERBOSE_MODE
118 /* Note this function does not do escaping, so it's only
119 * good for debug spew at the moment
122 match_rule_to_string (BusMatchRule *rule)
127 if (!_dbus_string_init (&str))
130 while ((s = _dbus_strdup ("nomem")) == NULL)
131 ; /* only OK for debug spew... */
135 if (rule->flags & BUS_MATCH_MESSAGE_TYPE)
137 /* FIXME make type readable */
138 if (!_dbus_string_append_printf (&str, "type='%d'", rule->message_type))
142 if (rule->flags & BUS_MATCH_INTERFACE)
144 if (_dbus_string_get_length (&str) > 0)
146 if (!_dbus_string_append (&str, ","))
150 if (!_dbus_string_append_printf (&str, "interface='%s'", rule->interface))
154 if (rule->flags & BUS_MATCH_MEMBER)
156 if (_dbus_string_get_length (&str) > 0)
158 if (!_dbus_string_append (&str, ","))
162 if (!_dbus_string_append_printf (&str, "member='%s'", rule->member))
166 if (rule->flags & BUS_MATCH_PATH)
168 if (_dbus_string_get_length (&str) > 0)
170 if (!_dbus_string_append (&str, ","))
174 if (!_dbus_string_append_printf (&str, "path='%s'", rule->path))
178 if (rule->flags & BUS_MATCH_SENDER)
180 if (_dbus_string_get_length (&str) > 0)
182 if (!_dbus_string_append (&str, ","))
186 if (!_dbus_string_append_printf (&str, "sender='%s'", rule->sender))
190 if (rule->flags & BUS_MATCH_DESTINATION)
192 if (_dbus_string_get_length (&str) > 0)
194 if (!_dbus_string_append (&str, ","))
198 if (!_dbus_string_append_printf (&str, "destination='%s'", rule->destination))
202 if (rule->flags & BUS_MATCH_ARGS)
206 _dbus_assert (rule->args != NULL);
209 while (i < rule->args_len)
211 if (rule->args[i] != NULL)
213 if (_dbus_string_get_length (&str) > 0)
215 if (!_dbus_string_append (&str, ","))
219 if (!_dbus_string_append_printf (&str,
230 if (!_dbus_string_steal_data (&str, &ret))
233 _dbus_string_free (&str);
237 _dbus_string_free (&str);
240 while ((s = _dbus_strdup ("nomem")) == NULL)
241 ; /* only OK for debug spew... */
245 #endif /* DBUS_ENABLE_VERBOSE_MODE */
248 bus_match_rule_set_message_type (BusMatchRule *rule,
251 rule->flags |= BUS_MATCH_MESSAGE_TYPE;
253 rule->message_type = type;
259 bus_match_rule_set_interface (BusMatchRule *rule,
260 const char *interface)
264 _dbus_assert (interface != NULL);
266 new = _dbus_strdup (interface);
270 rule->flags |= BUS_MATCH_INTERFACE;
271 dbus_free (rule->interface);
272 rule->interface = new;
278 bus_match_rule_set_member (BusMatchRule *rule,
283 _dbus_assert (member != NULL);
285 new = _dbus_strdup (member);
289 rule->flags |= BUS_MATCH_MEMBER;
290 dbus_free (rule->member);
297 bus_match_rule_set_sender (BusMatchRule *rule,
302 _dbus_assert (sender != NULL);
304 new = _dbus_strdup (sender);
308 rule->flags |= BUS_MATCH_SENDER;
309 dbus_free (rule->sender);
316 bus_match_rule_set_destination (BusMatchRule *rule,
317 const char *destination)
321 _dbus_assert (destination != NULL);
323 new = _dbus_strdup (destination);
327 rule->flags |= BUS_MATCH_DESTINATION;
328 dbus_free (rule->destination);
329 rule->destination = new;
335 bus_match_rule_set_path (BusMatchRule *rule,
340 _dbus_assert (path != NULL);
342 new = _dbus_strdup (path);
346 rule->flags |= BUS_MATCH_PATH;
347 dbus_free (rule->path);
354 bus_match_rule_set_arg (BusMatchRule *rule,
360 _dbus_assert (value != NULL);
362 new = _dbus_strdup (value);
366 /* args_len is the number of args not including null termination
369 if (arg >= rule->args_len)
375 new_args_len = arg + 1;
377 /* add another + 1 here for null termination */
378 new_args = dbus_realloc (rule->args,
379 sizeof(rule->args[0]) * (new_args_len + 1));
380 if (new_args == NULL)
386 /* NULL the new slots */
388 while (i <= new_args_len) /* <= for null termination */
394 rule->args = new_args;
395 rule->args_len = new_args_len;
398 rule->flags |= BUS_MATCH_ARGS;
400 dbus_free (rule->args[arg]);
401 rule->args[arg] = new;
403 /* NULL termination didn't get busted */
404 _dbus_assert (rule->args[rule->args_len] == NULL);
409 #define ISWHITE(c) (((c) == ' ') || ((c) == '\t') || ((c) == '\n') || ((c) == '\r'))
412 find_key (const DBusString *str,
420 const char *key_start;
423 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
425 s = _dbus_string_get_const_data (str);
429 while (*p && ISWHITE (*p))
434 while (*p && *p != '=' && !ISWHITE (*p))
439 while (*p && ISWHITE (*p))
442 if (key_start == key_end)
444 /* Empty match rules or trailing whitespace are OK */
451 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
452 "Match rule has a key with no subsequent '=' character");
457 if (!_dbus_string_append_len (key, key_start, key_end - key_start))
469 find_value (const DBusString *str,
481 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
483 orig_len = _dbus_string_get_length (value);
485 s = _dbus_string_get_const_data (str);
493 if (quote_char == '\0')
513 if (!_dbus_string_append_byte (value, *p))
520 else if (quote_char == '\\')
522 /* \ only counts as an escape if escaping a quote mark */
525 if (!_dbus_string_append_byte (value, '\\'))
532 if (!_dbus_string_append_byte (value, *p))
542 _dbus_assert (quote_char == '\'');
550 if (!_dbus_string_append_byte (value, *p))
564 if (quote_char == '\\')
566 if (!_dbus_string_append_byte (value, '\\'))
572 else if (quote_char == '\'')
574 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
575 "Unbalanced quotation marks in match rule");
579 _dbus_assert (quote_char == '\0');
581 /* Zero-length values are allowed */
588 _DBUS_ASSERT_ERROR_IS_SET (error);
589 _dbus_string_set_length (value, orig_len);
593 /* duplicates aren't allowed so the real legitimate max is only 6 or
594 * so. Leaving extra so we don't have to bother to update it.
595 * FIXME this is sort of busted now with arg matching, but we let
596 * you match on up to 10 args for now
598 #define MAX_RULE_TOKENS 16
600 /* this is slightly too high level to be termed a "token"
601 * but let's not be pedantic.
610 tokenize_rule (const DBusString *rule_text,
611 RuleToken tokens[MAX_RULE_TOKENS],
622 if (!_dbus_string_init (&key))
628 if (!_dbus_string_init (&value))
630 _dbus_string_free (&key);
637 while (i < MAX_RULE_TOKENS &&
638 pos < _dbus_string_get_length (rule_text))
640 _dbus_assert (tokens[i].key == NULL);
641 _dbus_assert (tokens[i].value == NULL);
643 if (!find_key (rule_text, pos, &key, &pos, error))
646 if (_dbus_string_get_length (&key) == 0)
649 if (!_dbus_string_steal_data (&key, &tokens[i].key))
655 if (!find_value (rule_text, pos, tokens[i].key, &value, &pos, error))
658 if (!_dbus_string_steal_data (&value, &tokens[i].value))
674 while (tokens[i].key || tokens[i].value)
676 dbus_free (tokens[i].key);
677 dbus_free (tokens[i].value);
678 tokens[i].key = NULL;
679 tokens[i].value = NULL;
684 _dbus_string_free (&key);
685 _dbus_string_free (&value);
691 bus_match_rule_parse_arg_match (BusMatchRule *rule,
693 const DBusString *value,
700 /* For now, arg0='foo' always implies that 'foo' is a
701 * DBUS_TYPE_STRING. Someday we could add an arg0type='int32' thing
702 * if we wanted, which would specify another type, in which case
703 * arg0='5' would have the 5 parsed as an int rather than string.
706 /* First we need to parse arg0 = 0, arg27 = 27 */
708 _dbus_string_init_const (&key_str, key);
710 if (_dbus_string_get_length (&key_str) < 4)
712 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
713 "Key '%s' in match rule starts with 'arg' but lacks an arg number. Should be 'arg0' or 'arg7' for example.\n", key);
717 if (!_dbus_string_parse_uint (&key_str, 3, &arg, &end) ||
718 end != _dbus_string_get_length (&key_str))
720 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
721 "Key '%s' in match rule starts with 'arg' but could not parse arg number. Should be 'arg0' or 'arg7' for example.\n", key);
725 /* If we didn't check this we could allocate a huge amount of RAM */
726 if (arg > DBUS_MAXIMUM_MATCH_RULE_ARG_NUMBER)
728 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
729 "Key '%s' in match rule has arg number %lu but the maximum is %d.\n", key, (unsigned long) arg, DBUS_MAXIMUM_MATCH_RULE_ARG_NUMBER);
733 if ((rule->flags & BUS_MATCH_ARGS) &&
734 rule->args_len > (int) arg &&
735 rule->args[arg] != NULL)
737 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
738 "Key '%s' specified twice in match rule\n", key);
742 if (!bus_match_rule_set_arg (rule, arg,
743 _dbus_string_get_const_data (value)))
752 _DBUS_ASSERT_ERROR_IS_SET (error);
757 * The format is comma-separated with strings quoted with single quotes
758 * as for the shell (to escape a literal single quote, use '\'').
760 * type='signal',sender='org.freedesktop.DBus',interface='org.freedesktop.DBus',member='Foo',
761 * path='/bar/foo',destination=':452345.34'
765 bus_match_rule_parse (DBusConnection *matches_go_to,
766 const DBusString *rule_text,
770 RuleToken tokens[MAX_RULE_TOKENS+1]; /* NULL termination + 1 */
773 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
775 if (_dbus_string_get_length (rule_text) > DBUS_MAXIMUM_MATCH_RULE_LENGTH)
777 dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
778 "Match rule text is %d bytes, maximum is %d",
779 _dbus_string_get_length (rule_text),
780 DBUS_MAXIMUM_MATCH_RULE_LENGTH);
784 memset (tokens, '\0', sizeof (tokens));
786 rule = bus_match_rule_new (matches_go_to);
793 if (!tokenize_rule (rule_text, tokens, error))
797 while (tokens[i].key != NULL)
801 const char *key = tokens[i].key;
802 const char *value = tokens[i].value;
804 _dbus_string_init_const (&tmp_str, value);
805 len = _dbus_string_get_length (&tmp_str);
807 if (strcmp (key, "type") == 0)
811 if (rule->flags & BUS_MATCH_MESSAGE_TYPE)
813 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
814 "Key %s specified twice in match rule\n", key);
818 t = dbus_message_type_from_string (value);
820 if (t == DBUS_MESSAGE_TYPE_INVALID)
822 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
823 "Invalid message type (%s) in match rule\n", value);
827 if (!bus_match_rule_set_message_type (rule, t))
833 else if (strcmp (key, "sender") == 0)
835 if (rule->flags & BUS_MATCH_SENDER)
837 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
838 "Key %s specified twice in match rule\n", key);
842 if (!_dbus_validate_bus_name (&tmp_str, 0, len))
844 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
845 "Sender name '%s' is invalid\n", value);
849 if (!bus_match_rule_set_sender (rule, value))
855 else if (strcmp (key, "interface") == 0)
857 if (rule->flags & BUS_MATCH_INTERFACE)
859 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
860 "Key %s specified twice in match rule\n", key);
864 if (!_dbus_validate_interface (&tmp_str, 0, len))
866 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
867 "Interface name '%s' is invalid\n", value);
871 if (!bus_match_rule_set_interface (rule, value))
877 else if (strcmp (key, "member") == 0)
879 if (rule->flags & BUS_MATCH_MEMBER)
881 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
882 "Key %s specified twice in match rule\n", key);
886 if (!_dbus_validate_member (&tmp_str, 0, len))
888 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
889 "Member name '%s' is invalid\n", value);
893 if (!bus_match_rule_set_member (rule, value))
899 else if (strcmp (key, "path") == 0)
901 if (rule->flags & BUS_MATCH_PATH)
903 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
904 "Key %s specified twice in match rule\n", key);
908 if (!_dbus_validate_path (&tmp_str, 0, len))
910 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
911 "Path '%s' is invalid\n", value);
915 if (!bus_match_rule_set_path (rule, value))
921 else if (strcmp (key, "destination") == 0)
923 if (rule->flags & BUS_MATCH_DESTINATION)
925 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
926 "Key %s specified twice in match rule\n", key);
930 if (!_dbus_validate_bus_name (&tmp_str, 0, len))
932 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
933 "Destination name '%s' is invalid\n", value);
937 if (!bus_match_rule_set_destination (rule, value))
943 else if (strncmp (key, "arg", 3) == 0)
945 if (!bus_match_rule_parse_arg_match (rule, key, &tmp_str, error))
950 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
951 "Unknown key \"%s\" in match rule",
963 _DBUS_ASSERT_ERROR_IS_SET (error);
966 bus_match_rule_unref (rule);
973 while (tokens[i].key || tokens[i].value)
975 _dbus_assert (i < MAX_RULE_TOKENS);
976 dbus_free (tokens[i].key);
977 dbus_free (tokens[i].value);
992 bus_matchmaker_new (void)
994 BusMatchmaker *matchmaker;
996 matchmaker = dbus_new0 (BusMatchmaker, 1);
997 if (matchmaker == NULL)
1000 matchmaker->refcount = 1;
1006 bus_matchmaker_ref (BusMatchmaker *matchmaker)
1008 _dbus_assert (matchmaker->refcount > 0);
1010 matchmaker->refcount += 1;
1016 bus_matchmaker_unref (BusMatchmaker *matchmaker)
1018 _dbus_assert (matchmaker->refcount > 0);
1020 matchmaker->refcount -= 1;
1021 if (matchmaker->refcount == 0)
1023 while (matchmaker->all_rules != NULL)
1027 rule = matchmaker->all_rules->data;
1028 bus_match_rule_unref (rule);
1029 _dbus_list_remove_link (&matchmaker->all_rules,
1030 matchmaker->all_rules);
1033 dbus_free (matchmaker);
1037 /* The rule can't be modified after it's added. */
1039 bus_matchmaker_add_rule (BusMatchmaker *matchmaker,
1042 _dbus_assert (bus_connection_is_active (rule->matches_go_to));
1044 if (!_dbus_list_append (&matchmaker->all_rules, rule))
1047 if (!bus_connection_add_match_rule (rule->matches_go_to, rule))
1049 _dbus_list_remove_last (&matchmaker->all_rules, rule);
1053 bus_match_rule_ref (rule);
1055 #ifdef DBUS_ENABLE_VERBOSE_MODE
1057 char *s = match_rule_to_string (rule);
1059 _dbus_verbose ("Added match rule %s to connection %p\n",
1060 s, rule->matches_go_to);
1069 match_rule_equal (BusMatchRule *a,
1072 if (a->flags != b->flags)
1075 if (a->matches_go_to != b->matches_go_to)
1078 if ((a->flags & BUS_MATCH_MESSAGE_TYPE) &&
1079 a->message_type != b->message_type)
1082 if ((a->flags & BUS_MATCH_MEMBER) &&
1083 strcmp (a->member, b->member) != 0)
1086 if ((a->flags & BUS_MATCH_PATH) &&
1087 strcmp (a->path, b->path) != 0)
1090 if ((a->flags & BUS_MATCH_INTERFACE) &&
1091 strcmp (a->interface, b->interface) != 0)
1094 if ((a->flags & BUS_MATCH_SENDER) &&
1095 strcmp (a->sender, b->sender) != 0)
1098 if ((a->flags & BUS_MATCH_DESTINATION) &&
1099 strcmp (a->destination, b->destination) != 0)
1102 if (a->flags & BUS_MATCH_ARGS)
1106 if (a->args_len != b->args_len)
1110 while (i < a->args_len)
1112 if ((a->args[i] != NULL) != (b->args[i] != NULL))
1115 if (a->args[i] != NULL)
1117 _dbus_assert (b->args[i] != NULL);
1118 if (strcmp (a->args[i], b->args[i]) != 0)
1130 bus_matchmaker_remove_rule_link (BusMatchmaker *matchmaker,
1133 BusMatchRule *rule = link->data;
1135 bus_connection_remove_match_rule (rule->matches_go_to, rule);
1136 _dbus_list_remove_link (&matchmaker->all_rules, link);
1138 #ifdef DBUS_ENABLE_VERBOSE_MODE
1140 char *s = match_rule_to_string (rule);
1142 _dbus_verbose ("Removed match rule %s for connection %p\n",
1143 s, rule->matches_go_to);
1148 bus_match_rule_unref (rule);
1152 bus_matchmaker_remove_rule (BusMatchmaker *matchmaker,
1155 bus_connection_remove_match_rule (rule->matches_go_to, rule);
1156 _dbus_list_remove (&matchmaker->all_rules, rule);
1158 #ifdef DBUS_ENABLE_VERBOSE_MODE
1160 char *s = match_rule_to_string (rule);
1162 _dbus_verbose ("Removed match rule %s for connection %p\n",
1163 s, rule->matches_go_to);
1168 bus_match_rule_unref (rule);
1171 /* Remove a single rule which is equal to the given rule by value */
1173 bus_matchmaker_remove_rule_by_value (BusMatchmaker *matchmaker,
1174 BusMatchRule *value,
1177 /* FIXME this is an unoptimized linear scan */
1181 /* we traverse backward because bus_connection_remove_match_rule()
1182 * removes the most-recently-added rule
1184 link = _dbus_list_get_last_link (&matchmaker->all_rules);
1185 while (link != NULL)
1191 prev = _dbus_list_get_prev_link (&matchmaker->all_rules, link);
1193 if (match_rule_equal (rule, value))
1195 bus_matchmaker_remove_rule_link (matchmaker, link);
1204 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_NOT_FOUND,
1205 "The given match rule wasn't found and can't be removed");
1213 bus_matchmaker_disconnected (BusMatchmaker *matchmaker,
1214 DBusConnection *disconnected)
1220 * This scans all match rules on the bus. We could avoid that
1221 * for the rules belonging to the connection, since we keep
1222 * a list of those; but for the rules that just refer to
1223 * the connection we'd need to do something more elaborate.
1227 _dbus_assert (bus_connection_is_active (disconnected));
1229 link = _dbus_list_get_first_link (&matchmaker->all_rules);
1230 while (link != NULL)
1236 next = _dbus_list_get_next_link (&matchmaker->all_rules, link);
1238 if (rule->matches_go_to == disconnected)
1240 bus_matchmaker_remove_rule_link (matchmaker, link);
1242 else if (((rule->flags & BUS_MATCH_SENDER) && *rule->sender == ':') ||
1243 ((rule->flags & BUS_MATCH_DESTINATION) && *rule->destination == ':'))
1245 /* The rule matches to/from a base service, see if it's the
1246 * one being disconnected, since we know this service name
1247 * will never be recycled.
1251 name = bus_connection_get_name (disconnected);
1252 _dbus_assert (name != NULL); /* because we're an active connection */
1254 if (((rule->flags & BUS_MATCH_SENDER) &&
1255 strcmp (rule->sender, name) == 0) ||
1256 ((rule->flags & BUS_MATCH_DESTINATION) &&
1257 strcmp (rule->destination, name) == 0))
1259 bus_matchmaker_remove_rule_link (matchmaker, link);
1268 connection_is_primary_owner (DBusConnection *connection,
1269 const char *service_name)
1271 BusService *service;
1273 BusRegistry *registry;
1275 _dbus_assert (connection != NULL);
1277 registry = bus_connection_get_registry (connection);
1279 _dbus_string_init_const (&str, service_name);
1280 service = bus_registry_lookup (registry, &str);
1282 if (service == NULL)
1283 return FALSE; /* Service doesn't exist so connection can't own it. */
1285 return bus_service_get_primary_owners_connection (service) == connection;
1289 match_rule_matches (BusMatchRule *rule,
1290 DBusConnection *sender,
1291 DBusConnection *addressed_recipient,
1292 DBusMessage *message)
1294 /* All features of the match rule are AND'd together,
1295 * so FALSE if any of them don't match.
1298 /* sender/addressed_recipient of #NULL may mean bus driver,
1299 * or for addressed_recipient may mean a message with no
1300 * specific recipient (i.e. a signal)
1303 if (rule->flags & BUS_MATCH_MESSAGE_TYPE)
1305 _dbus_assert (rule->message_type != DBUS_MESSAGE_TYPE_INVALID);
1307 if (rule->message_type != dbus_message_get_type (message))
1311 if (rule->flags & BUS_MATCH_INTERFACE)
1315 _dbus_assert (rule->interface != NULL);
1317 iface = dbus_message_get_interface (message);
1321 if (strcmp (iface, rule->interface) != 0)
1325 if (rule->flags & BUS_MATCH_MEMBER)
1329 _dbus_assert (rule->member != NULL);
1331 member = dbus_message_get_member (message);
1335 if (strcmp (member, rule->member) != 0)
1339 if (rule->flags & BUS_MATCH_SENDER)
1341 _dbus_assert (rule->sender != NULL);
1345 if (strcmp (rule->sender,
1346 DBUS_SERVICE_DBUS) != 0)
1351 if (!connection_is_primary_owner (sender, rule->sender))
1356 if (rule->flags & BUS_MATCH_DESTINATION)
1358 const char *destination;
1360 _dbus_assert (rule->destination != NULL);
1362 destination = dbus_message_get_destination (message);
1363 if (destination == NULL)
1366 if (addressed_recipient == NULL)
1368 if (strcmp (rule->destination,
1369 DBUS_SERVICE_DBUS) != 0)
1374 if (!connection_is_primary_owner (addressed_recipient, rule->destination))
1379 if (rule->flags & BUS_MATCH_PATH)
1383 _dbus_assert (rule->path != NULL);
1385 path = dbus_message_get_path (message);
1389 if (strcmp (path, rule->path) != 0)
1393 if (rule->flags & BUS_MATCH_ARGS)
1396 DBusMessageIter iter;
1398 _dbus_assert (rule->args != NULL);
1400 dbus_message_iter_init (message, &iter);
1403 while (i < rule->args_len)
1406 const char *expected_arg;
1408 expected_arg = rule->args[i];
1410 current_type = dbus_message_iter_get_arg_type (&iter);
1412 if (expected_arg != NULL)
1414 const char *actual_arg;
1416 if (current_type != DBUS_TYPE_STRING)
1420 dbus_message_iter_get_basic (&iter, &actual_arg);
1421 _dbus_assert (actual_arg != NULL);
1423 if (strcmp (expected_arg, actual_arg) != 0)
1427 if (current_type != DBUS_TYPE_INVALID)
1428 dbus_message_iter_next (&iter);
1438 bus_matchmaker_get_recipients (BusMatchmaker *matchmaker,
1439 BusConnections *connections,
1440 DBusConnection *sender,
1441 DBusConnection *addressed_recipient,
1442 DBusMessage *message,
1443 DBusList **recipients_p)
1445 /* FIXME for now this is a wholly unoptimized linear search */
1446 /* Guessing the important optimization is to skip the signal-related
1447 * match lists when processing method call and exception messages.
1448 * So separate match rule lists for signals?
1453 _dbus_assert (*recipients_p == NULL);
1455 /* This avoids sending same message to the same connection twice.
1456 * Purpose of the stamp instead of a bool is to avoid iterating over
1457 * all connections resetting the bool each time.
1459 bus_connections_increment_stamp (connections);
1461 /* addressed_recipient is already receiving the message, don't add to list.
1462 * NULL addressed_recipient means either bus driver, or this is a signal
1463 * and thus lacks a specific addressed_recipient.
1465 if (addressed_recipient != NULL)
1466 bus_connection_mark_stamp (addressed_recipient);
1468 link = _dbus_list_get_first_link (&matchmaker->all_rules);
1469 while (link != NULL)
1475 #ifdef DBUS_ENABLE_VERBOSE_MODE
1477 char *s = match_rule_to_string (rule);
1479 _dbus_verbose ("Checking whether message matches rule %s for connection %p\n",
1480 s, rule->matches_go_to);
1485 if (match_rule_matches (rule,
1486 sender, addressed_recipient, message))
1488 _dbus_verbose ("Rule matched\n");
1490 /* Append to the list if we haven't already */
1491 if (bus_connection_mark_stamp (rule->matches_go_to))
1493 if (!_dbus_list_append (recipients_p, rule->matches_go_to))
1496 #ifdef DBUS_ENABLE_VERBOSE_MODE
1499 _dbus_verbose ("Connection already receiving this message, so not adding again\n");
1501 #endif /* DBUS_ENABLE_VERBOSE_MODE */
1504 link = _dbus_list_get_next_link (&matchmaker->all_rules, link);
1510 _dbus_list_clear (recipients_p);
1514 #ifdef DBUS_BUILD_TESTS
1518 static BusMatchRule*
1519 check_parse (dbus_bool_t should_succeed,
1526 dbus_error_init (&error);
1528 _dbus_string_init_const (&str, text);
1530 rule = bus_match_rule_parse (NULL, &str, &error);
1531 if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1533 dbus_error_free (&error);
1537 if (should_succeed && rule == NULL)
1539 _dbus_warn ("Failed to parse: %s: %s: \"%s\"\n",
1540 error.name, error.message,
1541 _dbus_string_get_const_data (&str));
1545 if (!should_succeed && rule != NULL)
1547 _dbus_warn ("Failed to fail to parse: \"%s\"\n",
1548 _dbus_string_get_const_data (&str));
1552 dbus_error_free (&error);
1558 assert_large_rule (BusMatchRule *rule)
1560 _dbus_assert (rule->flags & BUS_MATCH_MESSAGE_TYPE);
1561 _dbus_assert (rule->flags & BUS_MATCH_SENDER);
1562 _dbus_assert (rule->flags & BUS_MATCH_INTERFACE);
1563 _dbus_assert (rule->flags & BUS_MATCH_MEMBER);
1564 _dbus_assert (rule->flags & BUS_MATCH_DESTINATION);
1565 _dbus_assert (rule->flags & BUS_MATCH_PATH);
1567 _dbus_assert (rule->message_type == DBUS_MESSAGE_TYPE_SIGNAL);
1568 _dbus_assert (rule->interface != NULL);
1569 _dbus_assert (rule->member != NULL);
1570 _dbus_assert (rule->sender != NULL);
1571 _dbus_assert (rule->destination != NULL);
1572 _dbus_assert (rule->path != NULL);
1574 _dbus_assert (strcmp (rule->interface, "org.freedesktop.DBusInterface") == 0);
1575 _dbus_assert (strcmp (rule->sender, "org.freedesktop.DBusSender") == 0);
1576 _dbus_assert (strcmp (rule->member, "Foo") == 0);
1577 _dbus_assert (strcmp (rule->path, "/bar/foo") == 0);
1578 _dbus_assert (strcmp (rule->destination, ":452345.34") == 0);
1582 test_parsing (void *data)
1586 rule = check_parse (TRUE, "type='signal',sender='org.freedesktop.DBusSender',interface='org.freedesktop.DBusInterface',member='Foo',path='/bar/foo',destination=':452345.34'");
1589 assert_large_rule (rule);
1590 bus_match_rule_unref (rule);
1593 /* With extra whitespace and useless quotes */
1594 rule = check_parse (TRUE, " type='signal', \tsender='org.freedes''ktop.DBusSender', interface='org.freedesktop.DBusInterface''''', \tmember='Foo',path='/bar/foo',destination=':452345.34'''''");
1597 assert_large_rule (rule);
1598 bus_match_rule_unref (rule);
1602 /* A simple signal connection */
1603 rule = check_parse (TRUE, "type='signal',path='/foo',interface='org.Bar'");
1606 _dbus_assert (rule->flags & BUS_MATCH_MESSAGE_TYPE);
1607 _dbus_assert (rule->flags & BUS_MATCH_INTERFACE);
1608 _dbus_assert (rule->flags & BUS_MATCH_PATH);
1610 _dbus_assert (rule->message_type == DBUS_MESSAGE_TYPE_SIGNAL);
1611 _dbus_assert (rule->interface != NULL);
1612 _dbus_assert (rule->path != NULL);
1614 _dbus_assert (strcmp (rule->interface, "org.Bar") == 0);
1615 _dbus_assert (strcmp (rule->path, "/foo") == 0);
1617 bus_match_rule_unref (rule);
1621 rule = check_parse (TRUE, "arg0='foo'");
1624 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1625 _dbus_assert (rule->args != NULL);
1626 _dbus_assert (rule->args_len == 1);
1627 _dbus_assert (rule->args[0] != NULL);
1628 _dbus_assert (rule->args[1] == NULL);
1629 _dbus_assert (strcmp (rule->args[0], "foo") == 0);
1631 bus_match_rule_unref (rule);
1634 rule = check_parse (TRUE, "arg1='foo'");
1637 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1638 _dbus_assert (rule->args != NULL);
1639 _dbus_assert (rule->args_len == 2);
1640 _dbus_assert (rule->args[0] == NULL);
1641 _dbus_assert (rule->args[1] != NULL);
1642 _dbus_assert (rule->args[2] == NULL);
1643 _dbus_assert (strcmp (rule->args[1], "foo") == 0);
1645 bus_match_rule_unref (rule);
1648 rule = check_parse (TRUE, "arg2='foo'");
1651 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1652 _dbus_assert (rule->args != NULL);
1653 _dbus_assert (rule->args_len == 3);
1654 _dbus_assert (rule->args[0] == NULL);
1655 _dbus_assert (rule->args[1] == NULL);
1656 _dbus_assert (rule->args[2] != NULL);
1657 _dbus_assert (rule->args[3] == NULL);
1658 _dbus_assert (strcmp (rule->args[2], "foo") == 0);
1660 bus_match_rule_unref (rule);
1663 rule = check_parse (TRUE, "arg40='foo'");
1666 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1667 _dbus_assert (rule->args != NULL);
1668 _dbus_assert (rule->args_len == 41);
1669 _dbus_assert (rule->args[0] == NULL);
1670 _dbus_assert (rule->args[1] == NULL);
1671 _dbus_assert (rule->args[40] != NULL);
1672 _dbus_assert (rule->args[41] == NULL);
1673 _dbus_assert (strcmp (rule->args[40], "foo") == 0);
1675 bus_match_rule_unref (rule);
1678 rule = check_parse (TRUE, "arg63='foo'");
1681 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1682 _dbus_assert (rule->args != NULL);
1683 _dbus_assert (rule->args_len == 64);
1684 _dbus_assert (rule->args[0] == NULL);
1685 _dbus_assert (rule->args[1] == NULL);
1686 _dbus_assert (rule->args[63] != NULL);
1687 _dbus_assert (rule->args[64] == NULL);
1688 _dbus_assert (strcmp (rule->args[63], "foo") == 0);
1690 bus_match_rule_unref (rule);
1693 /* Too-large argN */
1694 rule = check_parse (FALSE, "arg300='foo'");
1695 _dbus_assert (rule == NULL);
1696 rule = check_parse (FALSE, "arg64='foo'");
1697 _dbus_assert (rule == NULL);
1700 rule = check_parse (FALSE, "arg='foo'");
1701 _dbus_assert (rule == NULL);
1702 rule = check_parse (FALSE, "argv='foo'");
1703 _dbus_assert (rule == NULL);
1704 rule = check_parse (FALSE, "arg3junk='foo'");
1705 _dbus_assert (rule == NULL);
1706 rule = check_parse (FALSE, "argument='foo'");
1707 _dbus_assert (rule == NULL);
1709 /* Reject duplicates */
1710 rule = check_parse (FALSE, "type='signal',type='method_call'");
1711 _dbus_assert (rule == NULL);
1713 /* Duplicates with the argN code */
1714 rule = check_parse (FALSE, "arg0='foo',arg0='bar'");
1715 _dbus_assert (rule == NULL);
1716 rule = check_parse (FALSE, "arg3='foo',arg3='bar'");
1717 _dbus_assert (rule == NULL);
1718 rule = check_parse (FALSE, "arg30='foo',arg30='bar'");
1719 _dbus_assert (rule == NULL);
1721 /* Reject broken keys */
1722 rule = check_parse (FALSE, "blah='signal'");
1723 _dbus_assert (rule == NULL);
1725 /* Reject broken values */
1726 rule = check_parse (FALSE, "type='chouin'");
1727 _dbus_assert (rule == NULL);
1728 rule = check_parse (FALSE, "interface='abc@def++'");
1729 _dbus_assert (rule == NULL);
1730 rule = check_parse (FALSE, "service='youpi'");
1731 _dbus_assert (rule == NULL);
1733 /* Allow empty rule */
1734 rule = check_parse (TRUE, "");
1737 _dbus_assert (rule->flags == 0);
1739 bus_match_rule_unref (rule);
1742 /* All-whitespace rule is the same as empty */
1743 rule = check_parse (TRUE, " \t");
1746 _dbus_assert (rule->flags == 0);
1748 bus_match_rule_unref (rule);
1751 /* But with non-whitespace chars and no =value, it's not OK */
1752 rule = check_parse (FALSE, "type");
1753 _dbus_assert (rule == NULL);
1761 } equality_tests[] = {
1762 { "type='signal'", "type='signal'" },
1763 { "type='signal',interface='foo.bar'", "interface='foo.bar',type='signal'" },
1764 { "type='signal',member='bar'", "member='bar',type='signal'" },
1765 { "type='method_call',sender=':1.0'", "sender=':1.0',type='method_call'" },
1766 { "type='method_call',destination=':1.0'", "destination=':1.0',type='method_call'" },
1767 { "type='method_call',path='/foo/bar'", "path='/foo/bar',type='method_call'" },
1768 { "type='method_call',arg0='blah'", "arg0='blah',type='method_call'" },
1769 { "type='method_call',arg0='boo'", "arg0='boo',type='method_call'" },
1770 { "type='method_call',arg0='blah',arg1='baz'", "arg0='blah',arg1='baz',type='method_call'" },
1771 { "type='method_call',arg3='foosh'", "arg3='foosh',type='method_call'" },
1772 { "arg3='fool'", "arg3='fool'" },
1773 { "member='food'", "member='food'" }
1777 test_equality (void)
1782 while (i < _DBUS_N_ELEMENTS (equality_tests))
1784 BusMatchRule *first;
1785 BusMatchRule *second;
1788 first = check_parse (TRUE, equality_tests[i].first);
1789 _dbus_assert (first != NULL);
1790 second = check_parse (TRUE, equality_tests[i].second);
1791 _dbus_assert (second != NULL);
1793 if (!match_rule_equal (first, second))
1795 _dbus_warn ("rule %s and %s should have been equal\n",
1796 equality_tests[i].first,
1797 equality_tests[i].second);
1801 bus_match_rule_unref (second);
1803 /* Check that the rule is not equal to any of the
1804 * others besides its pair match
1807 while (j < _DBUS_N_ELEMENTS (equality_tests))
1811 second = check_parse (TRUE, equality_tests[j].second);
1813 if (match_rule_equal (first, second))
1815 _dbus_warn ("rule %s and %s should not have been equal\n",
1816 equality_tests[i].first,
1817 equality_tests[j].second);
1821 bus_match_rule_unref (second);
1827 bus_match_rule_unref (first);
1834 should_match_message_1[] = {
1836 "member='Frobated'",
1838 "type='signal',member='Frobated'",
1839 "type='signal',member='Frobated',arg0='foobar'",
1840 "member='Frobated',arg0='foobar'",
1841 "type='signal',arg0='foobar'",
1846 should_not_match_message_1[] = {
1847 "type='method_call'",
1849 "type='method_return'",
1850 "type='signal',member='Oopsed'",
1857 "arg0='foobar',arg1='abcdef'",
1858 "arg0='foobar',arg1='abcdef',arg2='abcdefghi',arg3='abcdefghi',arg4='abcdefghi'",
1859 "arg0='foobar',arg1='abcdef',arg4='abcdefghi',arg3='abcdefghi',arg2='abcdefghi'",
1864 check_matches (dbus_bool_t expected_to_match,
1866 DBusMessage *message,
1867 const char *rule_text)
1870 dbus_bool_t matched;
1872 rule = check_parse (TRUE, rule_text);
1873 _dbus_assert (rule != NULL);
1875 /* We can't test sender/destination rules since we pass NULL here */
1876 matched = match_rule_matches (rule, NULL, NULL, message);
1878 if (matched != expected_to_match)
1880 _dbus_warn ("Expected rule %s to %s message %d, failed\n",
1881 rule_text, expected_to_match ?
1882 "match" : "not match", number);
1886 bus_match_rule_unref (rule);
1890 check_matching (DBusMessage *message,
1892 const char **should_match,
1893 const char **should_not_match)
1898 while (should_match[i] != NULL)
1900 check_matches (TRUE, number, message, should_match[i]);
1905 while (should_not_match[i] != NULL)
1907 check_matches (FALSE, number, message, should_not_match[i]);
1913 test_matching (void)
1915 DBusMessage *message1;
1916 const char *v_STRING;
1917 dbus_int32_t v_INT32;
1919 message1 = dbus_message_new (DBUS_MESSAGE_TYPE_SIGNAL);
1920 _dbus_assert (message1 != NULL);
1921 if (!dbus_message_set_member (message1, "Frobated"))
1922 _dbus_assert_not_reached ("oom");
1924 v_STRING = "foobar";
1926 if (!dbus_message_append_args (message1,
1927 DBUS_TYPE_STRING, &v_STRING,
1928 DBUS_TYPE_INT32, &v_INT32,
1930 _dbus_assert_not_reached ("oom");
1932 check_matching (message1, 1,
1933 should_match_message_1,
1934 should_not_match_message_1);
1936 dbus_message_unref (message1);
1940 bus_signals_test (const DBusString *test_data_dir)
1942 BusMatchmaker *matchmaker;
1944 matchmaker = bus_matchmaker_new ();
1945 bus_matchmaker_ref (matchmaker);
1946 bus_matchmaker_unref (matchmaker);
1947 bus_matchmaker_unref (matchmaker);
1949 if (!_dbus_test_oom_handling ("parsing match rules", test_parsing, NULL))
1950 _dbus_assert_not_reached ("Parsing match rules test failed");
1959 #endif /* DBUS_BUILD_TESTS */