1 /* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
2 /* signals.c Bus signal connection implementation
3 *
4 * Copyright (C) 2003, 2005 Red Hat, Inc.
5 *
6 * Licensed under the Academic Free License version 2.1
7 *
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.
12 *
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.
17 *
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 *
22 */
23
24 #include <config.h>
25 #include "signals.h"
26 #include "services.h"
27 #include "utils.h"
28 #include <dbus/dbus-marshal-validate.h>
29
30 struct BusMatchRule
31 {
32 int refcount; /**< reference count */
33
34 DBusConnection *matches_go_to; /**< Owner of the rule */
35
36 unsigned int flags; /**< BusMatchFlags */
37
38 int message_type;
39 char *interface;
40 char *member;
41 char *sender;
42 char *destination;
43 char *path;
44
45 unsigned int *arg_lens;
46 char **args;
47 int args_len;
48 };
49
50 #define BUS_MATCH_ARG_IS_PATH 0x8000000u
51
52 BusMatchRule*
bus_match_rule_new(DBusConnection * matches_go_to)53 bus_match_rule_new (DBusConnection *matches_go_to)
54 {
55 BusMatchRule *rule;
56
57 rule = dbus_new0 (BusMatchRule, 1);
58 if (rule == NULL)
59 return NULL;
60
61 rule->refcount = 1;
62 rule->matches_go_to = matches_go_to;
63
64 #ifndef DBUS_BUILD_TESTS
65 _dbus_assert (rule->matches_go_to != NULL);
66 #endif
67
68 return rule;
69 }
70
71 BusMatchRule *
bus_match_rule_ref(BusMatchRule * rule)72 bus_match_rule_ref (BusMatchRule *rule)
73 {
74 _dbus_assert (rule->refcount > 0);
75
76 rule->refcount += 1;
77
78 return rule;
79 }
80
81 void
bus_match_rule_unref(BusMatchRule * rule)82 bus_match_rule_unref (BusMatchRule *rule)
83 {
84 _dbus_assert (rule->refcount > 0);
85
86 rule->refcount -= 1;
87 if (rule->refcount == 0)
88 {
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);
94 dbus_free (rule->arg_lens);
95
96 /* can't use dbus_free_string_array() since there
97 * are embedded NULL
98 */
99 if (rule->args)
100 {
101 int i;
102
103 i = 0;
104 while (i < rule->args_len)
105 {
106 if (rule->args[i])
107 dbus_free (rule->args[i]);
108 ++i;
109 }
110
111 dbus_free (rule->args);
112 }
113
114 dbus_free (rule);
115 }
116 }
117
118 #ifdef DBUS_ENABLE_VERBOSE_MODE
119 /* Note this function does not do escaping, so it's only
120 * good for debug spew at the moment
121 */
122 static char*
match_rule_to_string(BusMatchRule * rule)123 match_rule_to_string (BusMatchRule *rule)
124 {
125 DBusString str;
126 char *ret;
127
128 if (!_dbus_string_init (&str))
129 {
130 char *s;
131 while ((s = _dbus_strdup ("nomem")) == NULL)
132 ; /* only OK for debug spew... */
133 return s;
134 }
135
136 if (rule->flags & BUS_MATCH_MESSAGE_TYPE)
137 {
138 /* FIXME make type readable */
139 if (!_dbus_string_append_printf (&str, "type='%d'", rule->message_type))
140 goto nomem;
141 }
142
143 if (rule->flags & BUS_MATCH_INTERFACE)
144 {
145 if (_dbus_string_get_length (&str) > 0)
146 {
147 if (!_dbus_string_append (&str, ","))
148 goto nomem;
149 }
150
151 if (!_dbus_string_append_printf (&str, "interface='%s'", rule->interface))
152 goto nomem;
153 }
154
155 if (rule->flags & BUS_MATCH_MEMBER)
156 {
157 if (_dbus_string_get_length (&str) > 0)
158 {
159 if (!_dbus_string_append (&str, ","))
160 goto nomem;
161 }
162
163 if (!_dbus_string_append_printf (&str, "member='%s'", rule->member))
164 goto nomem;
165 }
166
167 if (rule->flags & BUS_MATCH_PATH)
168 {
169 if (_dbus_string_get_length (&str) > 0)
170 {
171 if (!_dbus_string_append (&str, ","))
172 goto nomem;
173 }
174
175 if (!_dbus_string_append_printf (&str, "path='%s'", rule->path))
176 goto nomem;
177 }
178
179 if (rule->flags & BUS_MATCH_SENDER)
180 {
181 if (_dbus_string_get_length (&str) > 0)
182 {
183 if (!_dbus_string_append (&str, ","))
184 goto nomem;
185 }
186
187 if (!_dbus_string_append_printf (&str, "sender='%s'", rule->sender))
188 goto nomem;
189 }
190
191 if (rule->flags & BUS_MATCH_DESTINATION)
192 {
193 if (_dbus_string_get_length (&str) > 0)
194 {
195 if (!_dbus_string_append (&str, ","))
196 goto nomem;
197 }
198
199 if (!_dbus_string_append_printf (&str, "destination='%s'", rule->destination))
200 goto nomem;
201 }
202
203 if (rule->flags & BUS_MATCH_ARGS)
204 {
205 int i;
206
207 _dbus_assert (rule->args != NULL);
208
209 i = 0;
210 while (i < rule->args_len)
211 {
212 if (rule->args[i] != NULL)
213 {
214 dbus_bool_t is_path;
215
216 if (_dbus_string_get_length (&str) > 0)
217 {
218 if (!_dbus_string_append (&str, ","))
219 goto nomem;
220 }
221
222 is_path = (rule->arg_lens[i] & BUS_MATCH_ARG_IS_PATH) != 0;
223
224 if (!_dbus_string_append_printf (&str,
225 "arg%d%s='%s'",
226 i, is_path ? "path" : "",
227 rule->args[i]))
228 goto nomem;
229 }
230
231 ++i;
232 }
233 }
234
235 if (!_dbus_string_steal_data (&str, &ret))
236 goto nomem;
237
238 _dbus_string_free (&str);
239 return ret;
240
241 nomem:
242 _dbus_string_free (&str);
243 {
244 char *s;
245 while ((s = _dbus_strdup ("nomem")) == NULL)
246 ; /* only OK for debug spew... */
247 return s;
248 }
249 }
250 #endif /* DBUS_ENABLE_VERBOSE_MODE */
251
252 dbus_bool_t
bus_match_rule_set_message_type(BusMatchRule * rule,int type)253 bus_match_rule_set_message_type (BusMatchRule *rule,
254 int type)
255 {
256 rule->flags |= BUS_MATCH_MESSAGE_TYPE;
257
258 rule->message_type = type;
259
260 return TRUE;
261 }
262
263 dbus_bool_t
bus_match_rule_set_interface(BusMatchRule * rule,const char * interface)264 bus_match_rule_set_interface (BusMatchRule *rule,
265 const char *interface)
266 {
267 char *new;
268
269 _dbus_assert (interface != NULL);
270
271 new = _dbus_strdup (interface);
272 if (new == NULL)
273 return FALSE;
274
275 rule->flags |= BUS_MATCH_INTERFACE;
276 dbus_free (rule->interface);
277 rule->interface = new;
278
279 return TRUE;
280 }
281
282 dbus_bool_t
bus_match_rule_set_member(BusMatchRule * rule,const char * member)283 bus_match_rule_set_member (BusMatchRule *rule,
284 const char *member)
285 {
286 char *new;
287
288 _dbus_assert (member != NULL);
289
290 new = _dbus_strdup (member);
291 if (new == NULL)
292 return FALSE;
293
294 rule->flags |= BUS_MATCH_MEMBER;
295 dbus_free (rule->member);
296 rule->member = new;
297
298 return TRUE;
299 }
300
301 dbus_bool_t
bus_match_rule_set_sender(BusMatchRule * rule,const char * sender)302 bus_match_rule_set_sender (BusMatchRule *rule,
303 const char *sender)
304 {
305 char *new;
306
307 _dbus_assert (sender != NULL);
308
309 new = _dbus_strdup (sender);
310 if (new == NULL)
311 return FALSE;
312
313 rule->flags |= BUS_MATCH_SENDER;
314 dbus_free (rule->sender);
315 rule->sender = new;
316
317 return TRUE;
318 }
319
320 dbus_bool_t
bus_match_rule_set_destination(BusMatchRule * rule,const char * destination)321 bus_match_rule_set_destination (BusMatchRule *rule,
322 const char *destination)
323 {
324 char *new;
325
326 _dbus_assert (destination != NULL);
327
328 new = _dbus_strdup (destination);
329 if (new == NULL)
330 return FALSE;
331
332 rule->flags |= BUS_MATCH_DESTINATION;
333 dbus_free (rule->destination);
334 rule->destination = new;
335
336 return TRUE;
337 }
338
339 dbus_bool_t
bus_match_rule_set_path(BusMatchRule * rule,const char * path)340 bus_match_rule_set_path (BusMatchRule *rule,
341 const char *path)
342 {
343 char *new;
344
345 _dbus_assert (path != NULL);
346
347 new = _dbus_strdup (path);
348 if (new == NULL)
349 return FALSE;
350
351 rule->flags |= BUS_MATCH_PATH;
352 dbus_free (rule->path);
353 rule->path = new;
354
355 return TRUE;
356 }
357
358 dbus_bool_t
bus_match_rule_set_arg(BusMatchRule * rule,int arg,const DBusString * value,dbus_bool_t is_path)359 bus_match_rule_set_arg (BusMatchRule *rule,
360 int arg,
361 const DBusString *value,
362 dbus_bool_t is_path)
363 {
364 int length;
365 char *new;
366
367 _dbus_assert (value != NULL);
368
369 /* args_len is the number of args not including null termination
370 * in the char**
371 */
372 if (arg >= rule->args_len)
373 {
374 unsigned int *new_arg_lens;
375 char **new_args;
376 int new_args_len;
377 int i;
378
379 new_args_len = arg + 1;
380
381 /* add another + 1 here for null termination */
382 new_args = dbus_realloc (rule->args,
383 sizeof (char *) * (new_args_len + 1));
384 if (new_args == NULL)
385 return FALSE;
386
387 /* NULL the new slots */
388 i = rule->args_len;
389 while (i <= new_args_len) /* <= for null termination */
390 {
391 new_args[i] = NULL;
392 ++i;
393 }
394
395 rule->args = new_args;
396
397 /* and now add to the lengths */
398 new_arg_lens = dbus_realloc (rule->arg_lens,
399 sizeof (int) * (new_args_len + 1));
400
401 if (new_arg_lens == NULL)
402 return FALSE;
403
404 /* zero the new slots */
405 i = rule->args_len;
406 while (i <= new_args_len) /* <= for null termination */
407 {
408 new_arg_lens[i] = 0;
409 ++i;
410 }
411
412 rule->arg_lens = new_arg_lens;
413 rule->args_len = new_args_len;
414 }
415
416 length = _dbus_string_get_length (value);
417 if (!_dbus_string_copy_data (value, &new))
418 return FALSE;
419
420 rule->flags |= BUS_MATCH_ARGS;
421
422 dbus_free (rule->args[arg]);
423 rule->arg_lens[arg] = length;
424 rule->args[arg] = new;
425
426 if (is_path)
427 rule->arg_lens[arg] |= BUS_MATCH_ARG_IS_PATH;
428
429 /* NULL termination didn't get busted */
430 _dbus_assert (rule->args[rule->args_len] == NULL);
431 _dbus_assert (rule->arg_lens[rule->args_len] == 0);
432
433 return TRUE;
434 }
435
436 #define ISWHITE(c) (((c) == ' ') || ((c) == '\t') || ((c) == '\n') || ((c) == '\r'))
437
438 static dbus_bool_t
find_key(const DBusString * str,int start,DBusString * key,int * value_pos,DBusError * error)439 find_key (const DBusString *str,
440 int start,
441 DBusString *key,
442 int *value_pos,
443 DBusError *error)
444 {
445 const char *p;
446 const char *s;
447 const char *key_start;
448 const char *key_end;
449
450 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
451
452 s = _dbus_string_get_const_data (str);
453
454 p = s + start;
455
456 while (*p && ISWHITE (*p))
457 ++p;
458
459 key_start = p;
460
461 while (*p && *p != '=' && !ISWHITE (*p))
462 ++p;
463
464 key_end = p;
465
466 while (*p && ISWHITE (*p))
467 ++p;
468
469 if (key_start == key_end)
470 {
471 /* Empty match rules or trailing whitespace are OK */
472 *value_pos = p - s;
473 return TRUE;
474 }
475
476 if (*p != '=')
477 {
478 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
479 "Match rule has a key with no subsequent '=' character");
480 return FALSE;
481 }
482 ++p;
483
484 if (!_dbus_string_append_len (key, key_start, key_end - key_start))
485 {
486 BUS_SET_OOM (error);
487 return FALSE;
488 }
489
490 *value_pos = p - s;
491
492 return TRUE;
493 }
494
495 static dbus_bool_t
find_value(const DBusString * str,int start,const char * key,DBusString * value,int * value_end,DBusError * error)496 find_value (const DBusString *str,
497 int start,
498 const char *key,
499 DBusString *value,
500 int *value_end,
501 DBusError *error)
502 {
503 const char *p;
504 const char *s;
505 char quote_char;
506 int orig_len;
507
508 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
509
510 orig_len = _dbus_string_get_length (value);
511
512 s = _dbus_string_get_const_data (str);
513
514 p = s + start;
515
516 quote_char = '\0';
517
518 while (*p)
519 {
520 if (quote_char == '\0')
521 {
522 switch (*p)
523 {
524 case '\0':
525 goto done;
526
527 case '\'':
528 quote_char = '\'';
529 goto next;
530
531 case ',':
532 ++p;
533 goto done;
534
535 case '\\':
536 quote_char = '\\';
537 goto next;
538
539 default:
540 if (!_dbus_string_append_byte (value, *p))
541 {
542 BUS_SET_OOM (error);
543 goto failed;
544 }
545 }
546 }
547 else if (quote_char == '\\')
548 {
549 /* \ only counts as an escape if escaping a quote mark */
550 if (*p != '\'')
551 {
552 if (!_dbus_string_append_byte (value, '\\'))
553 {
554 BUS_SET_OOM (error);
555 goto failed;
556 }
557 }
558
559 if (!_dbus_string_append_byte (value, *p))
560 {
561 BUS_SET_OOM (error);
562 goto failed;
563 }
564
565 quote_char = '\0';
566 }
567 else
568 {
569 _dbus_assert (quote_char == '\'');
570
571 if (*p == '\'')
572 {
573 quote_char = '\0';
574 }
575 else
576 {
577 if (!_dbus_string_append_byte (value, *p))
578 {
579 BUS_SET_OOM (error);
580 goto failed;
581 }
582 }
583 }
584
585 next:
586 ++p;
587 }
588
589 done:
590
591 if (quote_char == '\\')
592 {
593 if (!_dbus_string_append_byte (value, '\\'))
594 {
595 BUS_SET_OOM (error);
596 goto failed;
597 }
598 }
599 else if (quote_char == '\'')
600 {
601 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
602 "Unbalanced quotation marks in match rule");
603 goto failed;
604 }
605 else
606 _dbus_assert (quote_char == '\0');
607
608 /* Zero-length values are allowed */
609
610 *value_end = p - s;
611
612 return TRUE;
613
614 failed:
615 _DBUS_ASSERT_ERROR_IS_SET (error);
616 _dbus_string_set_length (value, orig_len);
617 return FALSE;
618 }
619
620 /* duplicates aren't allowed so the real legitimate max is only 6 or
621 * so. Leaving extra so we don't have to bother to update it.
622 * FIXME this is sort of busted now with arg matching, but we let
623 * you match on up to 10 args for now
624 */
625 #define MAX_RULE_TOKENS 16
626
627 /* this is slightly too high level to be termed a "token"
628 * but let's not be pedantic.
629 */
630 typedef struct
631 {
632 char *key;
633 char *value;
634 } RuleToken;
635
636 static dbus_bool_t
tokenize_rule(const DBusString * rule_text,RuleToken tokens[MAX_RULE_TOKENS],DBusError * error)637 tokenize_rule (const DBusString *rule_text,
638 RuleToken tokens[MAX_RULE_TOKENS],
639 DBusError *error)
640 {
641 int i;
642 int pos;
643 DBusString key;
644 DBusString value;
645 dbus_bool_t retval;
646
647 retval = FALSE;
648
649 if (!_dbus_string_init (&key))
650 {
651 BUS_SET_OOM (error);
652 return FALSE;
653 }
654
655 if (!_dbus_string_init (&value))
656 {
657 _dbus_string_free (&key);
658 BUS_SET_OOM (error);
659 return FALSE;
660 }
661
662 i = 0;
663 pos = 0;
664 while (i < MAX_RULE_TOKENS &&
665 pos < _dbus_string_get_length (rule_text))
666 {
667 _dbus_assert (tokens[i].key == NULL);
668 _dbus_assert (tokens[i].value == NULL);
669
670 if (!find_key (rule_text, pos, &key, &pos, error))
671 goto out;
672
673 if (_dbus_string_get_length (&key) == 0)
674 goto next;
675
676 if (!_dbus_string_steal_data (&key, &tokens[i].key))
677 {
678 BUS_SET_OOM (error);
679 goto out;
680 }
681
682 if (!find_value (rule_text, pos, tokens[i].key, &value, &pos, error))
683 goto out;
684
685 if (!_dbus_string_steal_data (&value, &tokens[i].value))
686 {
687 BUS_SET_OOM (error);
688 goto out;
689 }
690
691 next:
692 ++i;
693 }
694
695 retval = TRUE;
696
697 out:
698 if (!retval)
699 {
700 i = 0;
701 while (tokens[i].key || tokens[i].value)
702 {
703 dbus_free (tokens[i].key);
704 dbus_free (tokens[i].value);
705 tokens[i].key = NULL;
706 tokens[i].value = NULL;
707 ++i;
708 }
709 }
710
711 _dbus_string_free (&key);
712 _dbus_string_free (&value);
713
714 return retval;
715 }
716
717 static dbus_bool_t
bus_match_rule_parse_arg_match(BusMatchRule * rule,const char * key,const DBusString * value,DBusError * error)718 bus_match_rule_parse_arg_match (BusMatchRule *rule,
719 const char *key,
720 const DBusString *value,
721 DBusError *error)
722 {
723 dbus_bool_t is_path;
724 DBusString key_str;
725 unsigned long arg;
726 int length;
727 int end;
728
729 /* For now, arg0='foo' always implies that 'foo' is a
730 * DBUS_TYPE_STRING. Someday we could add an arg0type='int32' thing
731 * if we wanted, which would specify another type, in which case
732 * arg0='5' would have the 5 parsed as an int rather than string.
733 */
734
735 /* First we need to parse arg0 = 0, arg27 = 27 */
736
737 _dbus_string_init_const (&key_str, key);
738 length = _dbus_string_get_length (&key_str);
739
740 if (_dbus_string_get_length (&key_str) < 4)
741 {
742 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
743 "Key '%s' in match rule starts with 'arg' but lacks an arg number. Should be 'arg0' or 'arg7' for example.\n", key);
744 goto failed;
745 }
746
747 if (!_dbus_string_parse_uint (&key_str, 3, &arg, &end))
748 {
749 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
750 "Key '%s' in match rule starts with 'arg' but could not parse arg number. Should be 'arg0' or 'arg7' for example.\n", key);
751 goto failed;
752 }
753
754 if (end != length &&
755 ((end + 4) != length ||
756 !_dbus_string_ends_with_c_str (&key_str, "path")))
757 {
758 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
759 "Key '%s' in match rule contains junk after argument number. Only 'path' is optionally valid ('arg0path' for example).\n", key);
760 goto failed;
761 }
762
763 is_path = end != length;
764
765 /* If we didn't check this we could allocate a huge amount of RAM */
766 if (arg > DBUS_MAXIMUM_MATCH_RULE_ARG_NUMBER)
767 {
768 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
769 "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);
770 goto failed;
771 }
772
773 if ((rule->flags & BUS_MATCH_ARGS) &&
774 rule->args_len > (int) arg &&
775 rule->args[arg] != NULL)
776 {
777 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
778 "Argument %d matched more than once in match rule\n", key);
779 goto failed;
780 }
781
782 if (!bus_match_rule_set_arg (rule, arg, value, is_path))
783 {
784 BUS_SET_OOM (error);
785 goto failed;
786 }
787
788 return TRUE;
789
790 failed:
791 _DBUS_ASSERT_ERROR_IS_SET (error);
792 return FALSE;
793 }
794
795 /*
796 * The format is comma-separated with strings quoted with single quotes
797 * as for the shell (to escape a literal single quote, use '\'').
798 *
799 * type='signal',sender='org.freedesktop.DBus',interface='org.freedesktop.DBus',member='Foo',
800 * path='/bar/foo',destination=':452345.34'
801 *
802 */
803 BusMatchRule*
bus_match_rule_parse(DBusConnection * matches_go_to,const DBusString * rule_text,DBusError * error)804 bus_match_rule_parse (DBusConnection *matches_go_to,
805 const DBusString *rule_text,
806 DBusError *error)
807 {
808 BusMatchRule *rule;
809 RuleToken tokens[MAX_RULE_TOKENS+1]; /* NULL termination + 1 */
810 int i;
811
812 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
813
814 if (_dbus_string_get_length (rule_text) > DBUS_MAXIMUM_MATCH_RULE_LENGTH)
815 {
816 dbus_set_error (error, DBUS_ERROR_LIMITS_EXCEEDED,
817 "Match rule text is %d bytes, maximum is %d",
818 _dbus_string_get_length (rule_text),
819 DBUS_MAXIMUM_MATCH_RULE_LENGTH);
820 return NULL;
821 }
822
823 memset (tokens, '\0', sizeof (tokens));
824
825 rule = bus_match_rule_new (matches_go_to);
826 if (rule == NULL)
827 {
828 BUS_SET_OOM (error);
829 goto failed;
830 }
831
832 if (!tokenize_rule (rule_text, tokens, error))
833 goto failed;
834
835 i = 0;
836 while (tokens[i].key != NULL)
837 {
838 DBusString tmp_str;
839 int len;
840 const char *key = tokens[i].key;
841 const char *value = tokens[i].value;
842
843 _dbus_string_init_const (&tmp_str, value);
844 len = _dbus_string_get_length (&tmp_str);
845
846 if (strcmp (key, "type") == 0)
847 {
848 int t;
849
850 if (rule->flags & BUS_MATCH_MESSAGE_TYPE)
851 {
852 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
853 "Key %s specified twice in match rule\n", key);
854 goto failed;
855 }
856
857 t = dbus_message_type_from_string (value);
858
859 if (t == DBUS_MESSAGE_TYPE_INVALID)
860 {
861 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
862 "Invalid message type (%s) in match rule\n", value);
863 goto failed;
864 }
865
866 if (!bus_match_rule_set_message_type (rule, t))
867 {
868 BUS_SET_OOM (error);
869 goto failed;
870 }
871 }
872 else if (strcmp (key, "sender") == 0)
873 {
874 if (rule->flags & BUS_MATCH_SENDER)
875 {
876 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
877 "Key %s specified twice in match rule\n", key);
878 goto failed;
879 }
880
881 if (!_dbus_validate_bus_name (&tmp_str, 0, len))
882 {
883 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
884 "Sender name '%s' is invalid\n", value);
885 goto failed;
886 }
887
888 if (!bus_match_rule_set_sender (rule, value))
889 {
890 BUS_SET_OOM (error);
891 goto failed;
892 }
893 }
894 else if (strcmp (key, "interface") == 0)
895 {
896 if (rule->flags & BUS_MATCH_INTERFACE)
897 {
898 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
899 "Key %s specified twice in match rule\n", key);
900 goto failed;
901 }
902
903 if (!_dbus_validate_interface (&tmp_str, 0, len))
904 {
905 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
906 "Interface name '%s' is invalid\n", value);
907 goto failed;
908 }
909
910 if (!bus_match_rule_set_interface (rule, value))
911 {
912 BUS_SET_OOM (error);
913 goto failed;
914 }
915 }
916 else if (strcmp (key, "member") == 0)
917 {
918 if (rule->flags & BUS_MATCH_MEMBER)
919 {
920 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
921 "Key %s specified twice in match rule\n", key);
922 goto failed;
923 }
924
925 if (!_dbus_validate_member (&tmp_str, 0, len))
926 {
927 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
928 "Member name '%s' is invalid\n", value);
929 goto failed;
930 }
931
932 if (!bus_match_rule_set_member (rule, value))
933 {
934 BUS_SET_OOM (error);
935 goto failed;
936 }
937 }
938 else if (strcmp (key, "path") == 0)
939 {
940 if (rule->flags & BUS_MATCH_PATH)
941 {
942 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
943 "Key %s specified twice in match rule\n", key);
944 goto failed;
945 }
946
947 if (!_dbus_validate_path (&tmp_str, 0, len))
948 {
949 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
950 "Path '%s' is invalid\n", value);
951 goto failed;
952 }
953
954 if (!bus_match_rule_set_path (rule, value))
955 {
956 BUS_SET_OOM (error);
957 goto failed;
958 }
959 }
960 else if (strcmp (key, "destination") == 0)
961 {
962 if (rule->flags & BUS_MATCH_DESTINATION)
963 {
964 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
965 "Key %s specified twice in match rule\n", key);
966 goto failed;
967 }
968
969 if (!_dbus_validate_bus_name (&tmp_str, 0, len))
970 {
971 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
972 "Destination name '%s' is invalid\n", value);
973 goto failed;
974 }
975
976 if (!bus_match_rule_set_destination (rule, value))
977 {
978 BUS_SET_OOM (error);
979 goto failed;
980 }
981 }
982 else if (strncmp (key, "arg", 3) == 0)
983 {
984 if (!bus_match_rule_parse_arg_match (rule, key, &tmp_str, error))
985 goto failed;
986 }
987 else
988 {
989 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_INVALID,
990 "Unknown key \"%s\" in match rule",
991 key);
992 goto failed;
993 }
994
995 ++i;
996 }
997
998
999 goto out;
1000
1001 failed:
1002 _DBUS_ASSERT_ERROR_IS_SET (error);
1003 if (rule)
1004 {
1005 bus_match_rule_unref (rule);
1006 rule = NULL;
1007 }
1008
1009 out:
1010
1011 i = 0;
1012 while (tokens[i].key || tokens[i].value)
1013 {
1014 _dbus_assert (i < MAX_RULE_TOKENS);
1015 dbus_free (tokens[i].key);
1016 dbus_free (tokens[i].value);
1017 ++i;
1018 }
1019
1020 return rule;
1021 }
1022
1023 typedef struct RulePool RulePool;
1024 struct RulePool
1025 {
1026 /* Maps non-NULL interface names to non-NULL (DBusList **)s */
1027 DBusHashTable *rules_by_iface;
1028
1029 /* List of BusMatchRules which don't specify an interface */
1030 DBusList *rules_without_iface;
1031 };
1032
1033 struct BusMatchmaker
1034 {
1035 int refcount;
1036
1037 /* Pools of rules, grouped by the type of message they match. 0
1038 * (DBUS_MESSAGE_TYPE_INVALID) represents rules that do not specify a message
1039 * type.
1040 */
1041 RulePool rules_by_type[DBUS_NUM_MESSAGE_TYPES];
1042 };
1043
1044 static void
rule_list_free(DBusList ** rules)1045 rule_list_free (DBusList **rules)
1046 {
1047 while (*rules != NULL)
1048 {
1049 BusMatchRule *rule;
1050
1051 rule = (*rules)->data;
1052 bus_match_rule_unref (rule);
1053 _dbus_list_remove_link (rules, *rules);
1054 }
1055 }
1056
1057 static void
rule_list_ptr_free(DBusList ** list)1058 rule_list_ptr_free (DBusList **list)
1059 {
1060 /* We have to cope with NULL because the hash table frees the "existing"
1061 * value (which is NULL) when creating a new table entry...
1062 */
1063 if (list != NULL)
1064 {
1065 rule_list_free (list);
1066 dbus_free (list);
1067 }
1068 }
1069
1070 BusMatchmaker*
bus_matchmaker_new(void)1071 bus_matchmaker_new (void)
1072 {
1073 BusMatchmaker *matchmaker;
1074 int i;
1075
1076 matchmaker = dbus_new0 (BusMatchmaker, 1);
1077 if (matchmaker == NULL)
1078 return NULL;
1079
1080 matchmaker->refcount = 1;
1081
1082 for (i = DBUS_MESSAGE_TYPE_INVALID; i < DBUS_NUM_MESSAGE_TYPES; i++)
1083 {
1084 RulePool *p = matchmaker->rules_by_type + i;
1085
1086 p->rules_by_iface = _dbus_hash_table_new (DBUS_HASH_STRING,
1087 dbus_free, (DBusFreeFunction) rule_list_ptr_free);
1088
1089 if (p->rules_by_iface == NULL)
1090 goto nomem;
1091 }
1092
1093 return matchmaker;
1094
1095 nomem:
1096 for (i = DBUS_MESSAGE_TYPE_INVALID; i < DBUS_NUM_MESSAGE_TYPES; i++)
1097 {
1098 RulePool *p = matchmaker->rules_by_type + i;
1099
1100 if (p->rules_by_iface == NULL)
1101 break;
1102 else
1103 _dbus_hash_table_unref (p->rules_by_iface);
1104 }
1105
1106 return NULL;
1107 }
1108
1109 static DBusList **
bus_matchmaker_get_rules(BusMatchmaker * matchmaker,int message_type,const char * interface,dbus_bool_t create)1110 bus_matchmaker_get_rules (BusMatchmaker *matchmaker,
1111 int message_type,
1112 const char *interface,
1113 dbus_bool_t create)
1114 {
1115 RulePool *p;
1116
1117 _dbus_assert (message_type >= 0);
1118 _dbus_assert (message_type < DBUS_NUM_MESSAGE_TYPES);
1119
1120 _dbus_verbose ("Looking up rules for message_type %d, interface %s\n",
1121 message_type,
1122 interface != NULL ? interface : "<null>");
1123
1124 p = matchmaker->rules_by_type + message_type;
1125
1126 if (interface == NULL)
1127 {
1128 return &p->rules_without_iface;
1129 }
1130 else
1131 {
1132 DBusList **list;
1133
1134 list = _dbus_hash_table_lookup_string (p->rules_by_iface, interface);
1135
1136 if (list == NULL && create)
1137 {
1138 char *dupped_interface;
1139
1140 list = dbus_new0 (DBusList *, 1);
1141 if (list == NULL)
1142 return NULL;
1143
1144 dupped_interface = _dbus_strdup (interface);
1145 if (dupped_interface == NULL)
1146 {
1147 dbus_free (list);
1148 return NULL;
1149 }
1150
1151 _dbus_verbose ("Adding list for type %d, iface %s\n", message_type,
1152 interface);
1153
1154 if (!_dbus_hash_table_insert_string (p->rules_by_iface,
1155 dupped_interface, list))
1156 {
1157 dbus_free (list);
1158 dbus_free (dupped_interface);
1159 return NULL;
1160 }
1161 }
1162
1163 return list;
1164 }
1165 }
1166
1167 static void
bus_matchmaker_gc_rules(BusMatchmaker * matchmaker,int message_type,const char * interface,DBusList ** rules)1168 bus_matchmaker_gc_rules (BusMatchmaker *matchmaker,
1169 int message_type,
1170 const char *interface,
1171 DBusList **rules)
1172 {
1173 RulePool *p;
1174
1175 if (interface == NULL)
1176 return;
1177
1178 if (*rules != NULL)
1179 return;
1180
1181 _dbus_verbose ("GCing HT entry for message_type %u, interface %s\n",
1182 message_type, interface);
1183
1184 p = matchmaker->rules_by_type + message_type;
1185
1186 _dbus_assert (_dbus_hash_table_lookup_string (p->rules_by_iface, interface)
1187 == rules);
1188
1189 _dbus_hash_table_remove_string (p->rules_by_iface, interface);
1190 }
1191
1192 BusMatchmaker *
bus_matchmaker_ref(BusMatchmaker * matchmaker)1193 bus_matchmaker_ref (BusMatchmaker *matchmaker)
1194 {
1195 _dbus_assert (matchmaker->refcount > 0);
1196
1197 matchmaker->refcount += 1;
1198
1199 return matchmaker;
1200 }
1201
1202 void
bus_matchmaker_unref(BusMatchmaker * matchmaker)1203 bus_matchmaker_unref (BusMatchmaker *matchmaker)
1204 {
1205 _dbus_assert (matchmaker->refcount > 0);
1206
1207 matchmaker->refcount -= 1;
1208 if (matchmaker->refcount == 0)
1209 {
1210 int i;
1211
1212 for (i = DBUS_MESSAGE_TYPE_INVALID; i < DBUS_NUM_MESSAGE_TYPES; i++)
1213 {
1214 RulePool *p = matchmaker->rules_by_type + i;
1215
1216 _dbus_hash_table_unref (p->rules_by_iface);
1217 rule_list_free (&p->rules_without_iface);
1218 }
1219
1220 dbus_free (matchmaker);
1221 }
1222 }
1223
1224 /* The rule can't be modified after it's added. */
1225 dbus_bool_t
bus_matchmaker_add_rule(BusMatchmaker * matchmaker,BusMatchRule * rule)1226 bus_matchmaker_add_rule (BusMatchmaker *matchmaker,
1227 BusMatchRule *rule)
1228 {
1229 DBusList **rules;
1230
1231 _dbus_assert (bus_connection_is_active (rule->matches_go_to));
1232
1233 _dbus_verbose ("Adding rule with message_type %d, interface %s\n",
1234 rule->message_type,
1235 rule->interface != NULL ? rule->interface : "<null>");
1236
1237 rules = bus_matchmaker_get_rules (matchmaker, rule->message_type,
1238 rule->interface, TRUE);
1239
1240 if (rules == NULL)
1241 return FALSE;
1242
1243 if (!_dbus_list_append (rules, rule))
1244 return FALSE;
1245
1246 if (!bus_connection_add_match_rule (rule->matches_go_to, rule))
1247 {
1248 _dbus_list_remove_last (rules, rule);
1249 bus_matchmaker_gc_rules (matchmaker, rule->message_type,
1250 rule->interface, rules);
1251 return FALSE;
1252 }
1253
1254 bus_match_rule_ref (rule);
1255
1256 #ifdef DBUS_ENABLE_VERBOSE_MODE
1257 {
1258 char *s = match_rule_to_string (rule);
1259
1260 _dbus_verbose ("Added match rule %s to connection %p\n",
1261 s, rule->matches_go_to);
1262 dbus_free (s);
1263 }
1264 #endif
1265
1266 return TRUE;
1267 }
1268
1269 static dbus_bool_t
match_rule_equal(BusMatchRule * a,BusMatchRule * b)1270 match_rule_equal (BusMatchRule *a,
1271 BusMatchRule *b)
1272 {
1273 if (a->flags != b->flags)
1274 return FALSE;
1275
1276 if (a->matches_go_to != b->matches_go_to)
1277 return FALSE;
1278
1279 if ((a->flags & BUS_MATCH_MESSAGE_TYPE) &&
1280 a->message_type != b->message_type)
1281 return FALSE;
1282
1283 if ((a->flags & BUS_MATCH_MEMBER) &&
1284 strcmp (a->member, b->member) != 0)
1285 return FALSE;
1286
1287 if ((a->flags & BUS_MATCH_PATH) &&
1288 strcmp (a->path, b->path) != 0)
1289 return FALSE;
1290
1291 if ((a->flags & BUS_MATCH_INTERFACE) &&
1292 strcmp (a->interface, b->interface) != 0)
1293 return FALSE;
1294
1295 if ((a->flags & BUS_MATCH_SENDER) &&
1296 strcmp (a->sender, b->sender) != 0)
1297 return FALSE;
1298
1299 if ((a->flags & BUS_MATCH_DESTINATION) &&
1300 strcmp (a->destination, b->destination) != 0)
1301 return FALSE;
1302
1303 if (a->flags & BUS_MATCH_ARGS)
1304 {
1305 int i;
1306
1307 if (a->args_len != b->args_len)
1308 return FALSE;
1309
1310 i = 0;
1311 while (i < a->args_len)
1312 {
1313 int length;
1314
1315 if ((a->args[i] != NULL) != (b->args[i] != NULL))
1316 return FALSE;
1317
1318 if (a->arg_lens[i] != b->arg_lens[i])
1319 return FALSE;
1320
1321 length = a->arg_lens[i] & ~BUS_MATCH_ARG_IS_PATH;
1322
1323 if (a->args[i] != NULL)
1324 {
1325 _dbus_assert (b->args[i] != NULL);
1326 if (memcmp (a->args[i], b->args[i], length) != 0)
1327 return FALSE;
1328 }
1329
1330 ++i;
1331 }
1332 }
1333
1334 return TRUE;
1335 }
1336
1337 static void
bus_matchmaker_remove_rule_link(DBusList ** rules,DBusList * link)1338 bus_matchmaker_remove_rule_link (DBusList **rules,
1339 DBusList *link)
1340 {
1341 BusMatchRule *rule = link->data;
1342
1343 bus_connection_remove_match_rule (rule->matches_go_to, rule);
1344 _dbus_list_remove_link (rules, link);
1345
1346 #ifdef DBUS_ENABLE_VERBOSE_MODE
1347 {
1348 char *s = match_rule_to_string (rule);
1349
1350 _dbus_verbose ("Removed match rule %s for connection %p\n",
1351 s, rule->matches_go_to);
1352 dbus_free (s);
1353 }
1354 #endif
1355
1356 bus_match_rule_unref (rule);
1357 }
1358
1359 void
bus_matchmaker_remove_rule(BusMatchmaker * matchmaker,BusMatchRule * rule)1360 bus_matchmaker_remove_rule (BusMatchmaker *matchmaker,
1361 BusMatchRule *rule)
1362 {
1363 DBusList **rules;
1364
1365 _dbus_verbose ("Removing rule with message_type %d, interface %s\n",
1366 rule->message_type,
1367 rule->interface != NULL ? rule->interface : "<null>");
1368
1369 bus_connection_remove_match_rule (rule->matches_go_to, rule);
1370
1371 rules = bus_matchmaker_get_rules (matchmaker, rule->message_type,
1372 rule->interface, FALSE);
1373
1374 /* We should only be asked to remove a rule by identity right after it was
1375 * added, so there should be a list for it.
1376 */
1377 _dbus_assert (rules != NULL);
1378
1379 _dbus_list_remove (rules, rule);
1380 bus_matchmaker_gc_rules (matchmaker, rule->message_type, rule->interface,
1381 rules);
1382
1383 #ifdef DBUS_ENABLE_VERBOSE_MODE
1384 {
1385 char *s = match_rule_to_string (rule);
1386
1387 _dbus_verbose ("Removed match rule %s for connection %p\n",
1388 s, rule->matches_go_to);
1389 dbus_free (s);
1390 }
1391 #endif
1392
1393 bus_match_rule_unref (rule);
1394 }
1395
1396 /* Remove a single rule which is equal to the given rule by value */
1397 dbus_bool_t
bus_matchmaker_remove_rule_by_value(BusMatchmaker * matchmaker,BusMatchRule * value,DBusError * error)1398 bus_matchmaker_remove_rule_by_value (BusMatchmaker *matchmaker,
1399 BusMatchRule *value,
1400 DBusError *error)
1401 {
1402 DBusList **rules;
1403 DBusList *link = NULL;
1404
1405 _dbus_verbose ("Removing rule by value with message_type %d, interface %s\n",
1406 value->message_type,
1407 value->interface != NULL ? value->interface : "<null>");
1408
1409 rules = bus_matchmaker_get_rules (matchmaker, value->message_type,
1410 value->interface, FALSE);
1411
1412 if (rules != NULL)
1413 {
1414 /* we traverse backward because bus_connection_remove_match_rule()
1415 * removes the most-recently-added rule
1416 */
1417 link = _dbus_list_get_last_link (rules);
1418 while (link != NULL)
1419 {
1420 BusMatchRule *rule;
1421 DBusList *prev;
1422
1423 rule = link->data;
1424 prev = _dbus_list_get_prev_link (rules, link);
1425
1426 if (match_rule_equal (rule, value))
1427 {
1428 bus_matchmaker_remove_rule_link (rules, link);
1429 break;
1430 }
1431
1432 link = prev;
1433 }
1434 }
1435
1436 if (link == NULL)
1437 {
1438 dbus_set_error (error, DBUS_ERROR_MATCH_RULE_NOT_FOUND,
1439 "The given match rule wasn't found and can't be removed");
1440 return FALSE;
1441 }
1442
1443 bus_matchmaker_gc_rules (matchmaker, value->message_type, value->interface,
1444 rules);
1445
1446 return TRUE;
1447 }
1448
1449 static void
rule_list_remove_by_connection(DBusList ** rules,DBusConnection * connection)1450 rule_list_remove_by_connection (DBusList **rules,
1451 DBusConnection *connection)
1452 {
1453 DBusList *link;
1454
1455 link = _dbus_list_get_first_link (rules);
1456 while (link != NULL)
1457 {
1458 BusMatchRule *rule;
1459 DBusList *next;
1460
1461 rule = link->data;
1462 next = _dbus_list_get_next_link (rules, link);
1463
1464 if (rule->matches_go_to == connection)
1465 {
1466 bus_matchmaker_remove_rule_link (rules, link);
1467 }
1468 else if (((rule->flags & BUS_MATCH_SENDER) && *rule->sender == ':') ||
1469 ((rule->flags & BUS_MATCH_DESTINATION) && *rule->destination == ':'))
1470 {
1471 /* The rule matches to/from a base service, see if it's the
1472 * one being disconnected, since we know this service name
1473 * will never be recycled.
1474 */
1475 const char *name;
1476
1477 name = bus_connection_get_name (connection);
1478 _dbus_assert (name != NULL); /* because we're an active connection */
1479
1480 if (((rule->flags & BUS_MATCH_SENDER) &&
1481 strcmp (rule->sender, name) == 0) ||
1482 ((rule->flags & BUS_MATCH_DESTINATION) &&
1483 strcmp (rule->destination, name) == 0))
1484 {
1485 bus_matchmaker_remove_rule_link (rules, link);
1486 }
1487 }
1488
1489 link = next;
1490 }
1491 }
1492
1493 void
bus_matchmaker_disconnected(BusMatchmaker * matchmaker,DBusConnection * connection)1494 bus_matchmaker_disconnected (BusMatchmaker *matchmaker,
1495 DBusConnection *connection)
1496 {
1497 int i;
1498
1499 /* FIXME
1500 *
1501 * This scans all match rules on the bus. We could avoid that
1502 * for the rules belonging to the connection, since we keep
1503 * a list of those; but for the rules that just refer to
1504 * the connection we'd need to do something more elaborate.
1505 */
1506
1507 _dbus_assert (bus_connection_is_active (connection));
1508
1509 _dbus_verbose ("Removing all rules for connection %p\n", connection);
1510
1511 for (i = DBUS_MESSAGE_TYPE_INVALID; i < DBUS_NUM_MESSAGE_TYPES; i++)
1512 {
1513 RulePool *p = matchmaker->rules_by_type + i;
1514 DBusHashIter iter;
1515
1516 rule_list_remove_by_connection (&p->rules_without_iface, connection);
1517
1518 _dbus_hash_iter_init (p->rules_by_iface, &iter);
1519 while (_dbus_hash_iter_next (&iter))
1520 {
1521 DBusList **items = _dbus_hash_iter_get_value (&iter);
1522
1523 rule_list_remove_by_connection (items, connection);
1524
1525 if (*items == NULL)
1526 _dbus_hash_iter_remove_entry (&iter);
1527 }
1528 }
1529 }
1530
1531 static dbus_bool_t
connection_is_primary_owner(DBusConnection * connection,const char * service_name)1532 connection_is_primary_owner (DBusConnection *connection,
1533 const char *service_name)
1534 {
1535 BusService *service;
1536 DBusString str;
1537 BusRegistry *registry;
1538
1539 _dbus_assert (connection != NULL);
1540
1541 registry = bus_connection_get_registry (connection);
1542
1543 _dbus_string_init_const (&str, service_name);
1544 service = bus_registry_lookup (registry, &str);
1545
1546 if (service == NULL)
1547 return FALSE; /* Service doesn't exist so connection can't own it. */
1548
1549 return bus_service_get_primary_owners_connection (service) == connection;
1550 }
1551
1552 static dbus_bool_t
match_rule_matches(BusMatchRule * rule,DBusConnection * sender,DBusConnection * addressed_recipient,DBusMessage * message,BusMatchFlags already_matched)1553 match_rule_matches (BusMatchRule *rule,
1554 DBusConnection *sender,
1555 DBusConnection *addressed_recipient,
1556 DBusMessage *message,
1557 BusMatchFlags already_matched)
1558 {
1559 int flags;
1560
1561 /* All features of the match rule are AND'd together,
1562 * so FALSE if any of them don't match.
1563 */
1564
1565 /* sender/addressed_recipient of #NULL may mean bus driver,
1566 * or for addressed_recipient may mean a message with no
1567 * specific recipient (i.e. a signal)
1568 */
1569
1570 /* Don't bother re-matching features we've already checked implicitly. */
1571 flags = rule->flags & (~already_matched);
1572
1573 if (flags & BUS_MATCH_MESSAGE_TYPE)
1574 {
1575 _dbus_assert (rule->message_type != DBUS_MESSAGE_TYPE_INVALID);
1576
1577 if (rule->message_type != dbus_message_get_type (message))
1578 return FALSE;
1579 }
1580
1581 if (flags & BUS_MATCH_INTERFACE)
1582 {
1583 const char *iface;
1584
1585 _dbus_assert (rule->interface != NULL);
1586
1587 iface = dbus_message_get_interface (message);
1588 if (iface == NULL)
1589 return FALSE;
1590
1591 if (strcmp (iface, rule->interface) != 0)
1592 return FALSE;
1593 }
1594
1595 if (flags & BUS_MATCH_MEMBER)
1596 {
1597 const char *member;
1598
1599 _dbus_assert (rule->member != NULL);
1600
1601 member = dbus_message_get_member (message);
1602 if (member == NULL)
1603 return FALSE;
1604
1605 if (strcmp (member, rule->member) != 0)
1606 return FALSE;
1607 }
1608
1609 if (flags & BUS_MATCH_SENDER)
1610 {
1611 _dbus_assert (rule->sender != NULL);
1612
1613 if (sender == NULL)
1614 {
1615 if (strcmp (rule->sender,
1616 DBUS_SERVICE_DBUS) != 0)
1617 return FALSE;
1618 }
1619 else
1620 {
1621 if (!connection_is_primary_owner (sender, rule->sender))
1622 return FALSE;
1623 }
1624 }
1625
1626 if (flags & BUS_MATCH_DESTINATION)
1627 {
1628 const char *destination;
1629
1630 _dbus_assert (rule->destination != NULL);
1631
1632 destination = dbus_message_get_destination (message);
1633 if (destination == NULL)
1634 return FALSE;
1635
1636 if (addressed_recipient == NULL)
1637 {
1638 if (strcmp (rule->destination,
1639 DBUS_SERVICE_DBUS) != 0)
1640 return FALSE;
1641 }
1642 else
1643 {
1644 if (!connection_is_primary_owner (addressed_recipient, rule->destination))
1645 return FALSE;
1646 }
1647 }
1648
1649 if (flags & BUS_MATCH_PATH)
1650 {
1651 const char *path;
1652
1653 _dbus_assert (rule->path != NULL);
1654
1655 path = dbus_message_get_path (message);
1656 if (path == NULL)
1657 return FALSE;
1658
1659 if (strcmp (path, rule->path) != 0)
1660 return FALSE;
1661 }
1662
1663 if (flags & BUS_MATCH_ARGS)
1664 {
1665 int i;
1666 DBusMessageIter iter;
1667
1668 _dbus_assert (rule->args != NULL);
1669
1670 dbus_message_iter_init (message, &iter);
1671
1672 i = 0;
1673 while (i < rule->args_len)
1674 {
1675 int current_type;
1676 const char *expected_arg;
1677 int expected_length;
1678 dbus_bool_t is_path;
1679
1680 expected_arg = rule->args[i];
1681 expected_length = rule->arg_lens[i] & ~BUS_MATCH_ARG_IS_PATH;
1682 is_path = (rule->arg_lens[i] & BUS_MATCH_ARG_IS_PATH) != 0;
1683
1684 current_type = dbus_message_iter_get_arg_type (&iter);
1685
1686 if (expected_arg != NULL)
1687 {
1688 const char *actual_arg;
1689 int actual_length;
1690
1691 if (current_type != DBUS_TYPE_STRING)
1692 return FALSE;
1693
1694 actual_arg = NULL;
1695 dbus_message_iter_get_basic (&iter, &actual_arg);
1696 _dbus_assert (actual_arg != NULL);
1697
1698 actual_length = strlen (actual_arg);
1699
1700 if (is_path)
1701 {
1702 if (actual_length < expected_length &&
1703 actual_arg[actual_length - 1] != '/')
1704 return FALSE;
1705
1706 if (expected_length < actual_length &&
1707 expected_arg[expected_length - 1] != '/')
1708 return FALSE;
1709
1710 if (memcmp (actual_arg, expected_arg,
1711 MIN (actual_length, expected_length)) != 0)
1712 return FALSE;
1713 }
1714 else
1715 {
1716 if (expected_length != actual_length ||
1717 memcmp (expected_arg, actual_arg, expected_length) != 0)
1718 return FALSE;
1719 }
1720
1721 }
1722
1723 if (current_type != DBUS_TYPE_INVALID)
1724 dbus_message_iter_next (&iter);
1725
1726 ++i;
1727 }
1728 }
1729
1730 return TRUE;
1731 }
1732
1733 static dbus_bool_t
get_recipients_from_list(DBusList ** rules,DBusConnection * sender,DBusConnection * addressed_recipient,DBusMessage * message,DBusList ** recipients_p)1734 get_recipients_from_list (DBusList **rules,
1735 DBusConnection *sender,
1736 DBusConnection *addressed_recipient,
1737 DBusMessage *message,
1738 DBusList **recipients_p)
1739 {
1740 DBusList *link;
1741
1742 if (rules == NULL)
1743 return TRUE;
1744
1745 link = _dbus_list_get_first_link (rules);
1746 while (link != NULL)
1747 {
1748 BusMatchRule *rule;
1749
1750 rule = link->data;
1751
1752 #ifdef DBUS_ENABLE_VERBOSE_MODE
1753 {
1754 char *s = match_rule_to_string (rule);
1755
1756 _dbus_verbose ("Checking whether message matches rule %s for connection %p\n",
1757 s, rule->matches_go_to);
1758 dbus_free (s);
1759 }
1760 #endif
1761
1762 if (match_rule_matches (rule,
1763 sender, addressed_recipient, message,
1764 BUS_MATCH_MESSAGE_TYPE | BUS_MATCH_INTERFACE))
1765 {
1766 _dbus_verbose ("Rule matched\n");
1767
1768 /* Append to the list if we haven't already */
1769 if (bus_connection_mark_stamp (rule->matches_go_to))
1770 {
1771 if (!_dbus_list_append (recipients_p, rule->matches_go_to))
1772 return FALSE;
1773 }
1774 #ifdef DBUS_ENABLE_VERBOSE_MODE
1775 else
1776 {
1777 _dbus_verbose ("Connection already receiving this message, so not adding again\n");
1778 }
1779 #endif /* DBUS_ENABLE_VERBOSE_MODE */
1780 }
1781
1782 link = _dbus_list_get_next_link (rules, link);
1783 }
1784
1785 return TRUE;
1786 }
1787
1788 dbus_bool_t
bus_matchmaker_get_recipients(BusMatchmaker * matchmaker,BusConnections * connections,DBusConnection * sender,DBusConnection * addressed_recipient,DBusMessage * message,DBusList ** recipients_p)1789 bus_matchmaker_get_recipients (BusMatchmaker *matchmaker,
1790 BusConnections *connections,
1791 DBusConnection *sender,
1792 DBusConnection *addressed_recipient,
1793 DBusMessage *message,
1794 DBusList **recipients_p)
1795 {
1796 int type;
1797 const char *interface;
1798 DBusList **neither, **just_type, **just_iface, **both;
1799
1800 _dbus_assert (*recipients_p == NULL);
1801
1802 /* This avoids sending same message to the same connection twice.
1803 * Purpose of the stamp instead of a bool is to avoid iterating over
1804 * all connections resetting the bool each time.
1805 */
1806 bus_connections_increment_stamp (connections);
1807
1808 /* addressed_recipient is already receiving the message, don't add to list.
1809 * NULL addressed_recipient means either bus driver, or this is a signal
1810 * and thus lacks a specific addressed_recipient.
1811 */
1812 if (addressed_recipient != NULL)
1813 bus_connection_mark_stamp (addressed_recipient);
1814
1815 type = dbus_message_get_type (message);
1816 interface = dbus_message_get_interface (message);
1817
1818 neither = bus_matchmaker_get_rules (matchmaker, DBUS_MESSAGE_TYPE_INVALID,
1819 NULL, FALSE);
1820 just_type = just_iface = both = NULL;
1821
1822 if (interface != NULL)
1823 just_iface = bus_matchmaker_get_rules (matchmaker,
1824 DBUS_MESSAGE_TYPE_INVALID, interface, FALSE);
1825
1826 if (type > DBUS_MESSAGE_TYPE_INVALID && type < DBUS_NUM_MESSAGE_TYPES)
1827 {
1828 just_type = bus_matchmaker_get_rules (matchmaker, type, NULL, FALSE);
1829
1830 if (interface != NULL)
1831 both = bus_matchmaker_get_rules (matchmaker, type, interface, FALSE);
1832 }
1833
1834 if (!(get_recipients_from_list (neither, sender, addressed_recipient,
1835 message, recipients_p) &&
1836 get_recipients_from_list (just_iface, sender, addressed_recipient,
1837 message, recipients_p) &&
1838 get_recipients_from_list (just_type, sender, addressed_recipient,
1839 message, recipients_p) &&
1840 get_recipients_from_list (both, sender, addressed_recipient,
1841 message, recipients_p)))
1842 {
1843 _dbus_list_clear (recipients_p);
1844 return FALSE;
1845 }
1846
1847 return TRUE;
1848 }
1849
1850 #ifdef DBUS_BUILD_TESTS
1851 #include "test.h"
1852 #include <stdlib.h>
1853
1854 static BusMatchRule*
check_parse(dbus_bool_t should_succeed,const char * text)1855 check_parse (dbus_bool_t should_succeed,
1856 const char *text)
1857 {
1858 BusMatchRule *rule;
1859 DBusString str;
1860 DBusError error;
1861
1862 dbus_error_init (&error);
1863
1864 _dbus_string_init_const (&str, text);
1865
1866 rule = bus_match_rule_parse (NULL, &str, &error);
1867 if (dbus_error_has_name (&error, DBUS_ERROR_NO_MEMORY))
1868 {
1869 dbus_error_free (&error);
1870 return NULL;
1871 }
1872
1873 if (should_succeed && rule == NULL)
1874 {
1875 _dbus_warn ("Failed to parse: %s: %s: \"%s\"\n",
1876 error.name, error.message,
1877 _dbus_string_get_const_data (&str));
1878 exit (1);
1879 }
1880
1881 if (!should_succeed && rule != NULL)
1882 {
1883 _dbus_warn ("Failed to fail to parse: \"%s\"\n",
1884 _dbus_string_get_const_data (&str));
1885 exit (1);
1886 }
1887
1888 dbus_error_free (&error);
1889
1890 return rule;
1891 }
1892
1893 static void
assert_large_rule(BusMatchRule * rule)1894 assert_large_rule (BusMatchRule *rule)
1895 {
1896 _dbus_assert (rule->flags & BUS_MATCH_MESSAGE_TYPE);
1897 _dbus_assert (rule->flags & BUS_MATCH_SENDER);
1898 _dbus_assert (rule->flags & BUS_MATCH_INTERFACE);
1899 _dbus_assert (rule->flags & BUS_MATCH_MEMBER);
1900 _dbus_assert (rule->flags & BUS_MATCH_DESTINATION);
1901 _dbus_assert (rule->flags & BUS_MATCH_PATH);
1902
1903 _dbus_assert (rule->message_type == DBUS_MESSAGE_TYPE_SIGNAL);
1904 _dbus_assert (rule->interface != NULL);
1905 _dbus_assert (rule->member != NULL);
1906 _dbus_assert (rule->sender != NULL);
1907 _dbus_assert (rule->destination != NULL);
1908 _dbus_assert (rule->path != NULL);
1909
1910 _dbus_assert (strcmp (rule->interface, "org.freedesktop.DBusInterface") == 0);
1911 _dbus_assert (strcmp (rule->sender, "org.freedesktop.DBusSender") == 0);
1912 _dbus_assert (strcmp (rule->member, "Foo") == 0);
1913 _dbus_assert (strcmp (rule->path, "/bar/foo") == 0);
1914 _dbus_assert (strcmp (rule->destination, ":452345.34") == 0);
1915 }
1916
1917 static dbus_bool_t
test_parsing(void * data)1918 test_parsing (void *data)
1919 {
1920 BusMatchRule *rule;
1921
1922 rule = check_parse (TRUE, "type='signal',sender='org.freedesktop.DBusSender',interface='org.freedesktop.DBusInterface',member='Foo',path='/bar/foo',destination=':452345.34'");
1923 if (rule != NULL)
1924 {
1925 assert_large_rule (rule);
1926 bus_match_rule_unref (rule);
1927 }
1928
1929 /* With extra whitespace and useless quotes */
1930 rule = check_parse (TRUE, " type='signal', \tsender='org.freedes''ktop.DBusSender', interface='org.freedesktop.DBusInterface''''', \tmember='Foo',path='/bar/foo',destination=':452345.34'''''");
1931 if (rule != NULL)
1932 {
1933 assert_large_rule (rule);
1934 bus_match_rule_unref (rule);
1935 }
1936
1937
1938 /* A simple signal connection */
1939 rule = check_parse (TRUE, "type='signal',path='/foo',interface='org.Bar'");
1940 if (rule != NULL)
1941 {
1942 _dbus_assert (rule->flags & BUS_MATCH_MESSAGE_TYPE);
1943 _dbus_assert (rule->flags & BUS_MATCH_INTERFACE);
1944 _dbus_assert (rule->flags & BUS_MATCH_PATH);
1945
1946 _dbus_assert (rule->message_type == DBUS_MESSAGE_TYPE_SIGNAL);
1947 _dbus_assert (rule->interface != NULL);
1948 _dbus_assert (rule->path != NULL);
1949
1950 _dbus_assert (strcmp (rule->interface, "org.Bar") == 0);
1951 _dbus_assert (strcmp (rule->path, "/foo") == 0);
1952
1953 bus_match_rule_unref (rule);
1954 }
1955
1956 /* argN */
1957 rule = check_parse (TRUE, "arg0='foo'");
1958 if (rule != NULL)
1959 {
1960 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1961 _dbus_assert (rule->args != NULL);
1962 _dbus_assert (rule->args_len == 1);
1963 _dbus_assert (rule->args[0] != NULL);
1964 _dbus_assert (rule->args[1] == NULL);
1965 _dbus_assert (strcmp (rule->args[0], "foo") == 0);
1966
1967 bus_match_rule_unref (rule);
1968 }
1969
1970 rule = check_parse (TRUE, "arg1='foo'");
1971 if (rule != NULL)
1972 {
1973 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1974 _dbus_assert (rule->args != NULL);
1975 _dbus_assert (rule->args_len == 2);
1976 _dbus_assert (rule->args[0] == NULL);
1977 _dbus_assert (rule->args[1] != NULL);
1978 _dbus_assert (rule->args[2] == NULL);
1979 _dbus_assert (strcmp (rule->args[1], "foo") == 0);
1980
1981 bus_match_rule_unref (rule);
1982 }
1983
1984 rule = check_parse (TRUE, "arg2='foo'");
1985 if (rule != NULL)
1986 {
1987 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
1988 _dbus_assert (rule->args != NULL);
1989 _dbus_assert (rule->args_len == 3);
1990 _dbus_assert (rule->args[0] == NULL);
1991 _dbus_assert (rule->args[1] == NULL);
1992 _dbus_assert (rule->args[2] != NULL);
1993 _dbus_assert (rule->args[3] == NULL);
1994 _dbus_assert (strcmp (rule->args[2], "foo") == 0);
1995
1996 bus_match_rule_unref (rule);
1997 }
1998
1999 rule = check_parse (TRUE, "arg40='foo'");
2000 if (rule != NULL)
2001 {
2002 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
2003 _dbus_assert (rule->args != NULL);
2004 _dbus_assert (rule->args_len == 41);
2005 _dbus_assert (rule->args[0] == NULL);
2006 _dbus_assert (rule->args[1] == NULL);
2007 _dbus_assert (rule->args[40] != NULL);
2008 _dbus_assert (rule->args[41] == NULL);
2009 _dbus_assert (strcmp (rule->args[40], "foo") == 0);
2010
2011 bus_match_rule_unref (rule);
2012 }
2013
2014 rule = check_parse (TRUE, "arg63='foo'");
2015 if (rule != NULL)
2016 {
2017 _dbus_assert (rule->flags == BUS_MATCH_ARGS);
2018 _dbus_assert (rule->args != NULL);
2019 _dbus_assert (rule->args_len == 64);
2020 _dbus_assert (rule->args[0] == NULL);
2021 _dbus_assert (rule->args[1] == NULL);
2022 _dbus_assert (rule->args[63] != NULL);
2023 _dbus_assert (rule->args[64] == NULL);
2024 _dbus_assert (strcmp (rule->args[63], "foo") == 0);
2025
2026 bus_match_rule_unref (rule);
2027 }
2028
2029 /* Too-large argN */
2030 rule = check_parse (FALSE, "arg300='foo'");
2031 _dbus_assert (rule == NULL);
2032 rule = check_parse (FALSE, "arg64='foo'");
2033 _dbus_assert (rule == NULL);
2034
2035 /* No N in argN */
2036 rule = check_parse (FALSE, "arg='foo'");
2037 _dbus_assert (rule == NULL);
2038 rule = check_parse (FALSE, "argv='foo'");
2039 _dbus_assert (rule == NULL);
2040 rule = check_parse (FALSE, "arg3junk='foo'");
2041 _dbus_assert (rule == NULL);
2042 rule = check_parse (FALSE, "argument='foo'");
2043 _dbus_assert (rule == NULL);
2044
2045 /* Reject duplicates */
2046 rule = check_parse (FALSE, "type='signal',type='method_call'");
2047 _dbus_assert (rule == NULL);
2048
2049 /* Duplicates with the argN code */
2050 rule = check_parse (FALSE, "arg0='foo',arg0='bar'");
2051 _dbus_assert (rule == NULL);
2052 rule = check_parse (FALSE, "arg3='foo',arg3='bar'");
2053 _dbus_assert (rule == NULL);
2054 rule = check_parse (FALSE, "arg30='foo',arg30='bar'");
2055 _dbus_assert (rule == NULL);
2056
2057 /* Reject broken keys */
2058 rule = check_parse (FALSE, "blah='signal'");
2059 _dbus_assert (rule == NULL);
2060
2061 /* Reject broken values */
2062 rule = check_parse (FALSE, "type='chouin'");
2063 _dbus_assert (rule == NULL);
2064 rule = check_parse (FALSE, "interface='abc@def++'");
2065 _dbus_assert (rule == NULL);
2066 rule = check_parse (FALSE, "service='youpi'");
2067 _dbus_assert (rule == NULL);
2068
2069 /* Allow empty rule */
2070 rule = check_parse (TRUE, "");
2071 if (rule != NULL)
2072 {
2073 _dbus_assert (rule->flags == 0);
2074
2075 bus_match_rule_unref (rule);
2076 }
2077
2078 /* All-whitespace rule is the same as empty */
2079 rule = check_parse (TRUE, " \t");
2080 if (rule != NULL)
2081 {
2082 _dbus_assert (rule->flags == 0);
2083
2084 bus_match_rule_unref (rule);
2085 }
2086
2087 /* But with non-whitespace chars and no =value, it's not OK */
2088 rule = check_parse (FALSE, "type");
2089 _dbus_assert (rule == NULL);
2090
2091 return TRUE;
2092 }
2093
2094 static struct {
2095 const char *first;
2096 const char *second;
2097 } equality_tests[] = {
2098 { "type='signal'", "type='signal'" },
2099 { "type='signal',interface='foo.bar'", "interface='foo.bar',type='signal'" },
2100 { "type='signal',member='bar'", "member='bar',type='signal'" },
2101 { "type='method_call',sender=':1.0'", "sender=':1.0',type='method_call'" },
2102 { "type='method_call',destination=':1.0'", "destination=':1.0',type='method_call'" },
2103 { "type='method_call',path='/foo/bar'", "path='/foo/bar',type='method_call'" },
2104 { "type='method_call',arg0='blah'", "arg0='blah',type='method_call'" },
2105 { "type='method_call',arg0='boo'", "arg0='boo',type='method_call'" },
2106 { "type='method_call',arg0='blah',arg1='baz'", "arg0='blah',arg1='baz',type='method_call'" },
2107 { "type='method_call',arg3='foosh'", "arg3='foosh',type='method_call'" },
2108 { "arg3='fool'", "arg3='fool'" },
2109 { "member='food'", "member='food'" }
2110 };
2111
2112 static void
test_equality(void)2113 test_equality (void)
2114 {
2115 int i;
2116
2117 i = 0;
2118 while (i < _DBUS_N_ELEMENTS (equality_tests))
2119 {
2120 BusMatchRule *first;
2121 BusMatchRule *second;
2122 int j;
2123
2124 first = check_parse (TRUE, equality_tests[i].first);
2125 _dbus_assert (first != NULL);
2126 second = check_parse (TRUE, equality_tests[i].second);
2127 _dbus_assert (second != NULL);
2128
2129 if (!match_rule_equal (first, second))
2130 {
2131 _dbus_warn ("rule %s and %s should have been equal\n",
2132 equality_tests[i].first,
2133 equality_tests[i].second);
2134 exit (1);
2135 }
2136
2137 bus_match_rule_unref (second);
2138
2139 /* Check that the rule is not equal to any of the
2140 * others besides its pair match
2141 */
2142 j = 0;
2143 while (j < _DBUS_N_ELEMENTS (equality_tests))
2144 {
2145 if (i != j)
2146 {
2147 second = check_parse (TRUE, equality_tests[j].second);
2148
2149 if (match_rule_equal (first, second))
2150 {
2151 _dbus_warn ("rule %s and %s should not have been equal\n",
2152 equality_tests[i].first,
2153 equality_tests[j].second);
2154 exit (1);
2155 }
2156
2157 bus_match_rule_unref (second);
2158 }
2159
2160 ++j;
2161 }
2162
2163 bus_match_rule_unref (first);
2164
2165 ++i;
2166 }
2167 }
2168
2169 static const char*
2170 should_match_message_1[] = {
2171 "type='signal'",
2172 "member='Frobated'",
2173 "arg0='foobar'",
2174 "type='signal',member='Frobated'",
2175 "type='signal',member='Frobated',arg0='foobar'",
2176 "member='Frobated',arg0='foobar'",
2177 "type='signal',arg0='foobar'",
2178 NULL
2179 };
2180
2181 static const char*
2182 should_not_match_message_1[] = {
2183 "type='method_call'",
2184 "type='error'",
2185 "type='method_return'",
2186 "type='signal',member='Oopsed'",
2187 "arg0='blah'",
2188 "arg1='foobar'",
2189 "arg2='foobar'",
2190 "arg3='foobar'",
2191 "arg0='3'",
2192 "arg1='3'",
2193 "arg0='foobar',arg1='abcdef'",
2194 "arg0='foobar',arg1='abcdef',arg2='abcdefghi',arg3='abcdefghi',arg4='abcdefghi'",
2195 "arg0='foobar',arg1='abcdef',arg4='abcdefghi',arg3='abcdefghi',arg2='abcdefghi'",
2196 NULL
2197 };
2198
2199 static void
check_matches(dbus_bool_t expected_to_match,int number,DBusMessage * message,const char * rule_text)2200 check_matches (dbus_bool_t expected_to_match,
2201 int number,
2202 DBusMessage *message,
2203 const char *rule_text)
2204 {
2205 BusMatchRule *rule;
2206 dbus_bool_t matched;
2207
2208 rule = check_parse (TRUE, rule_text);
2209 _dbus_assert (rule != NULL);
2210
2211 /* We can't test sender/destination rules since we pass NULL here */
2212 matched = match_rule_matches (rule, NULL, NULL, message, 0);
2213
2214 if (matched != expected_to_match)
2215 {
2216 _dbus_warn ("Expected rule %s to %s message %d, failed\n",
2217 rule_text, expected_to_match ?
2218 "match" : "not match", number);
2219 exit (1);
2220 }
2221
2222 bus_match_rule_unref (rule);
2223 }
2224
2225 static void
check_matching(DBusMessage * message,int number,const char ** should_match,const char ** should_not_match)2226 check_matching (DBusMessage *message,
2227 int number,
2228 const char **should_match,
2229 const char **should_not_match)
2230 {
2231 int i;
2232
2233 i = 0;
2234 while (should_match[i] != NULL)
2235 {
2236 check_matches (TRUE, number, message, should_match[i]);
2237 ++i;
2238 }
2239
2240 i = 0;
2241 while (should_not_match[i] != NULL)
2242 {
2243 check_matches (FALSE, number, message, should_not_match[i]);
2244 ++i;
2245 }
2246 }
2247
2248 static void
test_matching(void)2249 test_matching (void)
2250 {
2251 DBusMessage *message1;
2252 const char *v_STRING;
2253 dbus_int32_t v_INT32;
2254
2255 message1 = dbus_message_new (DBUS_MESSAGE_TYPE_SIGNAL);
2256 _dbus_assert (message1 != NULL);
2257 if (!dbus_message_set_member (message1, "Frobated"))
2258 _dbus_assert_not_reached ("oom");
2259
2260 v_STRING = "foobar";
2261 v_INT32 = 3;
2262 if (!dbus_message_append_args (message1,
2263 DBUS_TYPE_STRING, &v_STRING,
2264 DBUS_TYPE_INT32, &v_INT32,
2265 NULL))
2266 _dbus_assert_not_reached ("oom");
2267
2268 check_matching (message1, 1,
2269 should_match_message_1,
2270 should_not_match_message_1);
2271
2272 dbus_message_unref (message1);
2273 }
2274
2275 dbus_bool_t
bus_signals_test(const DBusString * test_data_dir)2276 bus_signals_test (const DBusString *test_data_dir)
2277 {
2278 BusMatchmaker *matchmaker;
2279
2280 matchmaker = bus_matchmaker_new ();
2281 bus_matchmaker_ref (matchmaker);
2282 bus_matchmaker_unref (matchmaker);
2283 bus_matchmaker_unref (matchmaker);
2284
2285 if (!_dbus_test_oom_handling ("parsing match rules", test_parsing, NULL))
2286 _dbus_assert_not_reached ("Parsing match rules test failed");
2287
2288 test_equality ();
2289
2290 test_matching ();
2291
2292 return TRUE;
2293 }
2294
2295 #endif /* DBUS_BUILD_TESTS */
2296
2297