D-Bus 1.14.8
dbus-message.c
1/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
2/* dbus-message.c DBusMessage object
3 *
4 * Copyright (C) 2002, 2003, 2004, 2005 Red Hat Inc.
5 * Copyright (C) 2002, 2003 CodeFactory AB
6 *
7 * Licensed under the Academic Free License version 2.1
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 *
23 */
24
25#include <config.h>
26#include "dbus-internals.h"
27#include "dbus-marshal-recursive.h"
28#include "dbus-marshal-validate.h"
29#include "dbus-marshal-byteswap.h"
30#include "dbus-marshal-header.h"
31#include "dbus-signature.h"
32#include "dbus-message-private.h"
33#include "dbus-object-tree.h"
34#include "dbus-memory.h"
35#include "dbus-list.h"
36#include "dbus-threads-internal.h"
37#ifdef HAVE_UNIX_FD_PASSING
38#include "dbus-sysdeps.h"
39#include "dbus-sysdeps-unix.h"
40#endif
41
42#include <string.h>
43
44#define _DBUS_TYPE_IS_STRINGLIKE(type) \
45 (type == DBUS_TYPE_STRING || type == DBUS_TYPE_SIGNATURE || \
46 type == DBUS_TYPE_OBJECT_PATH)
47
48static void dbus_message_finalize (DBusMessage *message);
49
60#ifdef DBUS_ENABLE_EMBEDDED_TESTS
61static dbus_bool_t
62_dbus_enable_message_cache (void)
63{
64 static int enabled = -1;
65
66 if (enabled < 0)
67 {
68 const char *s = _dbus_getenv ("DBUS_MESSAGE_CACHE");
69
70 enabled = TRUE;
71
72 if (s && *s)
73 {
74 if (*s == '0')
75 enabled = FALSE;
76 else if (*s == '1')
77 enabled = TRUE;
78 else
79 _dbus_warn ("DBUS_MESSAGE_CACHE should be 0 or 1 if set, not '%s'",
80 s);
81 }
82 }
83
84 return enabled;
85}
86#else
87 /* constant expression, should be optimized away */
88# define _dbus_enable_message_cache() (TRUE)
89#endif
90
91#ifndef _dbus_message_trace_ref
92void
93_dbus_message_trace_ref (DBusMessage *message,
94 int old_refcount,
95 int new_refcount,
96 const char *why)
97{
98 static int enabled = -1;
99
100 _dbus_trace_ref ("DBusMessage", message, old_refcount, new_refcount, why,
101 "DBUS_MESSAGE_TRACE", &enabled);
102}
103#endif
104
105/* Not thread locked, but strictly const/read-only so should be OK
106 */
108_DBUS_STRING_DEFINE_STATIC(_dbus_empty_signature_str, "");
109
110/* these have wacky values to help trap uninitialized iterators;
111 * but has to fit in 3 bits
112 */
113enum {
114 DBUS_MESSAGE_ITER_TYPE_READER = 3,
115 DBUS_MESSAGE_ITER_TYPE_WRITER = 7
116};
117
120
127{
130 dbus_uint32_t iter_type : 3;
131 dbus_uint32_t sig_refcount : 8;
132 union
133 {
136 } u;
137};
138
144typedef struct
145{
146 void *dummy1;
147 void *dummy2;
148 dbus_uint32_t dummy3;
149 int dummy4;
150 int dummy5;
151 int dummy6;
152 int dummy7;
153 int dummy8;
154 int dummy9;
155 int dummy10;
156 int dummy11;
157 int pad1;
158 int pad2;
159 void *pad3;
161
162static void
163get_const_signature (DBusHeader *header,
164 const DBusString **type_str_p,
165 int *type_pos_p)
166{
167 if (_dbus_header_get_field_raw (header,
169 type_str_p,
170 type_pos_p))
171 {
172 *type_pos_p += 1; /* skip the signature length which is 1 byte */
173 }
174 else
175 {
176 *type_str_p = &_dbus_empty_signature_str;
177 *type_pos_p = 0;
178 }
179}
180
186static void
187_dbus_message_byteswap (DBusMessage *message)
188{
189 const DBusString *type_str;
190 int type_pos;
191 char byte_order;
192
193 byte_order = _dbus_header_get_byte_order (&message->header);
194
195 if (byte_order == DBUS_COMPILER_BYTE_ORDER)
196 return;
197
198 _dbus_verbose ("Swapping message into compiler byte order\n");
199
200 get_const_signature (&message->header, &type_str, &type_pos);
201
202 _dbus_marshal_byteswap (type_str, type_pos,
203 byte_order,
204 DBUS_COMPILER_BYTE_ORDER,
205 &message->body, 0);
206
207 _dbus_header_byteswap (&message->header, DBUS_COMPILER_BYTE_ORDER);
209 DBUS_COMPILER_BYTE_ORDER);
210}
211
218#define ensure_byte_order(message) _dbus_message_byteswap (message)
219
230void
232 const DBusString **header,
233 const DBusString **body)
234{
235 _dbus_assert (message->locked);
236
237 *header = &message->header.data;
238 *body = &message->body;
239}
240
251 const int **fds,
252 unsigned *n_fds)
253{
254 _dbus_assert (message->locked);
255
256#ifdef HAVE_UNIX_FD_PASSING
257 *fds = message->unix_fds;
258 *n_fds = message->n_unix_fds;
259#else
260 *fds = NULL;
261 *n_fds = 0;
262#endif
263}
264
273{
275}
276
288void
290 dbus_uint32_t serial)
291{
292 _dbus_return_if_fail (message != NULL);
293 _dbus_return_if_fail (!message->locked);
294
295 _dbus_header_set_serial (&message->header, serial);
296}
297
314void
316 DBusList *link)
317{
318 /* right now we don't recompute the delta when message
319 * size changes, and that's OK for current purposes
320 * I think, but could be important to change later.
321 * Do recompute it whenever there are no outstanding counters,
322 * since it's basically free.
323 */
324 if (message->counters == NULL)
325 {
326 message->size_counter_delta =
327 _dbus_string_get_length (&message->header.data) +
328 _dbus_string_get_length (&message->body);
329
330#ifdef HAVE_UNIX_FD_PASSING
331 message->unix_fd_counter_delta = message->n_unix_fds;
332#endif
333
334#if 0
335 _dbus_verbose ("message has size %ld\n",
336 message->size_counter_delta);
337#endif
338 }
339
340 _dbus_list_append_link (&message->counters, link);
341
343
344#ifdef HAVE_UNIX_FD_PASSING
345 _dbus_counter_adjust_unix_fd (link->data, message->unix_fd_counter_delta);
346#endif
347}
348
365 DBusCounter *counter)
366{
367 DBusList *link;
368
369 link = _dbus_list_alloc_link (counter);
370 if (link == NULL)
371 return FALSE;
372
373 _dbus_counter_ref (counter);
374 _dbus_message_add_counter_link (message, link);
375
376 return TRUE;
377}
378
386void
388 DBusCounter *counter)
389{
390 DBusList *link;
391
392 link = _dbus_list_find_last (&message->counters,
393 counter);
394 _dbus_assert (link != NULL);
395
396 _dbus_list_remove_link (&message->counters, link);
397
398 _dbus_counter_adjust_size (counter, - message->size_counter_delta);
399
400#ifdef HAVE_UNIX_FD_PASSING
401 _dbus_counter_adjust_unix_fd (counter, - message->unix_fd_counter_delta);
402#endif
403
404 _dbus_counter_notify (counter);
405 _dbus_counter_unref (counter);
406}
407
418void
420{
421 if (!message->locked)
422 {
424 _dbus_string_get_length (&message->body));
425
426 /* must have a signature if you have a body */
427 _dbus_assert (_dbus_string_get_length (&message->body) == 0 ||
428 dbus_message_get_signature (message) != NULL);
429
430 message->locked = TRUE;
431 }
432}
433
434static dbus_bool_t
435set_or_delete_string_field (DBusMessage *message,
436 int field,
437 int typecode,
438 const char *value)
439{
440 if (value == NULL)
441 return _dbus_header_delete_field (&message->header, field);
442 else
443 return _dbus_header_set_field_basic (&message->header,
444 field,
445 typecode,
446 &value);
447}
448
449/* Message Cache
450 *
451 * We cache some DBusMessage to reduce the overhead of allocating
452 * them. In my profiling this consistently made about an 8%
453 * difference. It avoids the malloc for the message, the malloc for
454 * the slot list, the malloc for the header string and body string,
455 * and the associated free() calls. It does introduce another global
456 * lock which could be a performance issue in certain cases.
457 *
458 * For the echo client/server the round trip time goes from around
459 * .000077 to .000069 with the message cache on my laptop. The sysprof
460 * change is as follows (numbers are cumulative percentage):
461 *
462 * with message cache implemented as array as it is now (0.000069 per):
463 * new_empty_header 1.46
464 * mutex_lock 0.56 # i.e. _DBUS_LOCK(message_cache)
465 * mutex_unlock 0.25
466 * self 0.41
467 * unref 2.24
468 * self 0.68
469 * list_clear 0.43
470 * mutex_lock 0.33 # i.e. _DBUS_LOCK(message_cache)
471 * mutex_unlock 0.25
472 *
473 * with message cache implemented as list (0.000070 per roundtrip):
474 * new_empty_header 2.72
475 * list_pop_first 1.88
476 * unref 3.3
477 * list_prepend 1.63
478 *
479 * without cache (0.000077 per roundtrip):
480 * new_empty_header 6.7
481 * string_init_preallocated 3.43
482 * dbus_malloc 2.43
483 * dbus_malloc0 2.59
484 *
485 * unref 4.02
486 * string_free 1.82
487 * dbus_free 1.63
488 * dbus_free 0.71
489 *
490 * If you implement the message_cache with a list, the primary reason
491 * it's slower is that you add another thread lock (on the DBusList
492 * mempool).
493 */
494
496#define MAX_MESSAGE_SIZE_TO_CACHE 10 * _DBUS_ONE_KILOBYTE
497
499#define MAX_MESSAGE_CACHE_SIZE 5
500
501/* Protected by _DBUS_LOCK (message_cache) */
502static DBusMessage *message_cache[MAX_MESSAGE_CACHE_SIZE];
503static int message_cache_count = 0;
504static dbus_bool_t message_cache_shutdown_registered = FALSE;
505
506static void
507dbus_message_cache_shutdown (void *data)
508{
509 int i;
510
511 if (!_DBUS_LOCK (message_cache))
512 _dbus_assert_not_reached ("we would have initialized global locks "
513 "before registering a shutdown function");
514
515 i = 0;
516 while (i < MAX_MESSAGE_CACHE_SIZE)
517 {
518 if (message_cache[i])
519 dbus_message_finalize (message_cache[i]);
520
521 ++i;
522 }
523
524 message_cache_count = 0;
525 message_cache_shutdown_registered = FALSE;
526
527 _DBUS_UNLOCK (message_cache);
528}
529
537static DBusMessage*
538dbus_message_get_cached (void)
539{
540 DBusMessage *message;
541 int i;
542
543 message = NULL;
544
545 if (!_DBUS_LOCK (message_cache))
546 {
547 /* we'd have initialized global locks before caching anything,
548 * so there can't be anything in the cache */
549 return NULL;
550 }
551
552 _dbus_assert (message_cache_count >= 0);
553
554 if (message_cache_count == 0)
555 {
556 _DBUS_UNLOCK (message_cache);
557 return NULL;
558 }
559
560 /* This is not necessarily true unless count > 0, and
561 * message_cache is uninitialized until the shutdown is
562 * registered
563 */
564 _dbus_assert (message_cache_shutdown_registered);
565
566 i = 0;
567 while (i < MAX_MESSAGE_CACHE_SIZE)
568 {
569 if (message_cache[i])
570 {
571 message = message_cache[i];
572 message_cache[i] = NULL;
573 message_cache_count -= 1;
574 break;
575 }
576 ++i;
577 }
578 _dbus_assert (message_cache_count >= 0);
580 _dbus_assert (message != NULL);
581
582 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0);
583
584 _dbus_assert (message->counters == NULL);
585
586 _DBUS_UNLOCK (message_cache);
587
588 return message;
589}
590
591#ifdef HAVE_UNIX_FD_PASSING
592static void
593close_unix_fds(int *fds, unsigned *n_fds)
594{
595 DBusError e;
596 unsigned int i;
597
598 if (*n_fds <= 0)
599 return;
600
601 dbus_error_init(&e);
602
603 for (i = 0; i < *n_fds; i++)
604 {
605 if (!_dbus_close(fds[i], &e))
606 {
607 _dbus_warn("Failed to close file descriptor: %s", e.message);
608 dbus_error_free(&e);
609 }
610 }
611
612 *n_fds = 0;
613
614 /* We don't free the array here, in case we can recycle it later */
615}
616#endif
617
618static void
619free_counter (void *element,
620 void *data)
621{
622 DBusCounter *counter = element;
623 DBusMessage *message = data;
624
625 _dbus_counter_adjust_size (counter, - message->size_counter_delta);
626#ifdef HAVE_UNIX_FD_PASSING
627 _dbus_counter_adjust_unix_fd (counter, - message->unix_fd_counter_delta);
628#endif
629
630 _dbus_counter_notify (counter);
631 _dbus_counter_unref (counter);
632}
633
639static void
640dbus_message_cache_or_finalize (DBusMessage *message)
641{
642 dbus_bool_t was_cached;
643 int i;
644
645 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0);
646
647 /* This calls application code and has to be done first thing
648 * without holding the lock
649 */
651
652 _dbus_list_foreach (&message->counters,
653 free_counter, message);
654 _dbus_list_clear (&message->counters);
655
656#ifdef HAVE_UNIX_FD_PASSING
657 close_unix_fds(message->unix_fds, &message->n_unix_fds);
658#endif
659
660 was_cached = FALSE;
661
662 if (!_DBUS_LOCK (message_cache))
663 {
664 /* The only way to get a non-null message goes through
665 * dbus_message_get_cached() which takes the lock. */
666 _dbus_assert_not_reached ("we would have initialized global locks "
667 "the first time we constructed a message");
668 }
669
670 if (!message_cache_shutdown_registered)
671 {
672 _dbus_assert (message_cache_count == 0);
673
674 if (!_dbus_register_shutdown_func (dbus_message_cache_shutdown, NULL))
675 goto out;
676
677 i = 0;
678 while (i < MAX_MESSAGE_CACHE_SIZE)
679 {
680 message_cache[i] = NULL;
681 ++i;
682 }
683
684 message_cache_shutdown_registered = TRUE;
685 }
686
687 _dbus_assert (message_cache_count >= 0);
688
689 if (!_dbus_enable_message_cache ())
690 goto out;
691
692 if ((_dbus_string_get_length (&message->header.data) +
693 _dbus_string_get_length (&message->body)) >
695 goto out;
696
697 if (message_cache_count >= MAX_MESSAGE_CACHE_SIZE)
698 goto out;
699
700 /* Find empty slot */
701 i = 0;
702 while (message_cache[i] != NULL)
703 ++i;
704
706
707 _dbus_assert (message_cache[i] == NULL);
708 message_cache[i] = message;
709 message_cache_count += 1;
710 was_cached = TRUE;
711#ifndef DBUS_DISABLE_CHECKS
712 message->in_cache = TRUE;
713#endif
714
715 out:
716 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0);
717
718 _DBUS_UNLOCK (message_cache);
719
720 if (!was_cached)
721 dbus_message_finalize (message);
722}
723
724/*
725 * Arrange for iter to be something that _dbus_message_iter_check() would
726 * reject as not a valid iterator.
727 */
728static void
729_dbus_message_real_iter_zero (DBusMessageRealIter *iter)
730{
731 _dbus_assert (iter != NULL);
732 _DBUS_ZERO (*iter);
733 /* NULL is not, strictly speaking, guaranteed to be all-bits-zero */
734 iter->message = NULL;
735}
736
742void
744{
745 _dbus_return_if_fail (iter != NULL);
746 _dbus_message_real_iter_zero ((DBusMessageRealIter *) iter);
747}
748
749static dbus_bool_t
750_dbus_message_real_iter_is_zeroed (DBusMessageRealIter *iter)
751{
752 return (iter != NULL && iter->message == NULL && iter->changed_stamp == 0 &&
753 iter->iter_type == 0 && iter->sig_refcount == 0);
754}
755
756#if defined(DBUS_ENABLE_CHECKS) || defined(DBUS_ENABLE_ASSERT)
757static dbus_bool_t
758_dbus_message_iter_check (DBusMessageRealIter *iter)
759{
760 char byte_order;
761
762 if (iter == NULL)
763 {
764 _dbus_warn_check_failed ("dbus message iterator is NULL");
765 return FALSE;
766 }
767
768 if (iter->message == NULL || iter->iter_type == 0)
769 {
770 _dbus_warn_check_failed ("dbus message iterator has already been "
771 "closed, or is uninitialized or corrupt");
772 return FALSE;
773 }
774
775 byte_order = _dbus_header_get_byte_order (&iter->message->header);
776
777 if (iter->iter_type == DBUS_MESSAGE_ITER_TYPE_READER)
778 {
779 if (iter->u.reader.byte_order != byte_order)
780 {
781 _dbus_warn_check_failed ("dbus message changed byte order since iterator was created");
782 return FALSE;
783 }
784 /* because we swap the message into compiler order when you init an iter */
785 _dbus_assert (iter->u.reader.byte_order == DBUS_COMPILER_BYTE_ORDER);
786 }
787 else if (iter->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER)
788 {
789 if (iter->u.writer.byte_order != byte_order)
790 {
791 _dbus_warn_check_failed ("dbus message changed byte order since append iterator was created");
792 return FALSE;
793 }
794 /* because we swap the message into compiler order when you init an iter */
795 _dbus_assert (iter->u.writer.byte_order == DBUS_COMPILER_BYTE_ORDER);
796 }
797 else
798 {
799 _dbus_warn_check_failed ("dbus message iterator looks uninitialized or corrupted");
800 return FALSE;
801 }
802
803 if (iter->changed_stamp != iter->message->changed_stamp)
804 {
805 _dbus_warn_check_failed ("dbus message iterator invalid because the message has been modified (or perhaps the iterator is just uninitialized)");
806 return FALSE;
807 }
808
809 return TRUE;
810}
811#endif /* DBUS_ENABLE_CHECKS || DBUS_ENABLE_ASSERT */
812
827 DBusError *error,
828 int first_arg_type,
829 va_list var_args)
830{
832 int spec_type, msg_type, i, j;
833 dbus_bool_t retval;
834 va_list copy_args;
835
836 _dbus_assert (_dbus_message_iter_check (real));
837
838 retval = FALSE;
839
840 spec_type = first_arg_type;
841 i = 0;
842
843 /* copy var_args first, then we can do another iteration over it to
844 * free memory and close unix fds if parse failed at some point.
845 */
846 DBUS_VA_COPY (copy_args, var_args);
847
848 while (spec_type != DBUS_TYPE_INVALID)
849 {
850 msg_type = dbus_message_iter_get_arg_type (iter);
851
852 if (msg_type != spec_type)
853 {
855 "Argument %d is specified to be of type \"%s\", but "
856 "is actually of type \"%s\"\n", i,
857 _dbus_type_to_string (spec_type),
858 _dbus_type_to_string (msg_type));
859
860 goto out;
861 }
862
863 if (spec_type == DBUS_TYPE_UNIX_FD)
864 {
865#ifdef HAVE_UNIX_FD_PASSING
866 DBusBasicValue idx;
867 int *pfd, nfd;
868
869 pfd = va_arg (var_args, int*);
870 _dbus_assert(pfd);
871
873
874 if (idx.u32 >= real->message->n_unix_fds)
875 {
877 "Message refers to file descriptor at index %i,"
878 "but has only %i descriptors attached.\n",
879 idx.u32,
880 real->message->n_unix_fds);
881 goto out;
882 }
883
884 if ((nfd = _dbus_dup(real->message->unix_fds[idx.u32], error)) < 0)
885 goto out;
886
887 *pfd = nfd;
888#else
890 "Platform does not support file desciptor passing.\n");
891 goto out;
892#endif
893 }
894 else if (dbus_type_is_basic (spec_type))
895 {
896 void *ptr;
897
898 ptr = va_arg (var_args, void *);
899
900 _dbus_assert (ptr != NULL);
901
903 ptr);
904 }
905 else if (spec_type == DBUS_TYPE_ARRAY)
906 {
907 int element_type;
908 int spec_element_type;
909 const void **ptr;
910 int *n_elements_p;
911 DBusTypeReader array;
912
913 spec_element_type = va_arg (var_args, int);
914 element_type = _dbus_type_reader_get_element_type (&real->u.reader);
915
916 if (spec_element_type != element_type)
917 {
919 "Argument %d is specified to be an array of \"%s\", but "
920 "is actually an array of \"%s\"\n",
921 i,
922 _dbus_type_to_string (spec_element_type),
923 _dbus_type_to_string (element_type));
924
925 goto out;
926 }
927
928 if (dbus_type_is_fixed (spec_element_type) &&
929 element_type != DBUS_TYPE_UNIX_FD)
930 {
931 ptr = va_arg (var_args, const void **);
932 n_elements_p = va_arg (var_args, int*);
933
934 _dbus_assert (ptr != NULL);
935 _dbus_assert (n_elements_p != NULL);
936
937 _dbus_type_reader_recurse (&real->u.reader, &array);
938
939 _dbus_type_reader_read_fixed_multi (&array, ptr, n_elements_p);
940 }
941 else if (_DBUS_TYPE_IS_STRINGLIKE (spec_element_type))
942 {
943 char ***str_array_p;
944 int n_elements;
945 char **str_array;
946
947 str_array_p = va_arg (var_args, char***);
948 n_elements_p = va_arg (var_args, int*);
949
950 _dbus_assert (str_array_p != NULL);
951 _dbus_assert (n_elements_p != NULL);
952
953 /* Count elements in the array */
954 _dbus_type_reader_recurse (&real->u.reader, &array);
955
956 n_elements = 0;
958 {
959 ++n_elements;
960 _dbus_type_reader_next (&array);
961 }
962
963 str_array = dbus_new0 (char*, n_elements + 1);
964 if (str_array == NULL)
965 {
966 _DBUS_SET_OOM (error);
967 goto out;
968 }
969
970 /* Now go through and dup each string */
971 _dbus_type_reader_recurse (&real->u.reader, &array);
972
973 j = 0;
974 while (j < n_elements)
975 {
976 const char *s;
978 (void *) &s);
979
980 str_array[j] = _dbus_strdup (s);
981 if (str_array[j] == NULL)
982 {
983 dbus_free_string_array (str_array);
984 _DBUS_SET_OOM (error);
985 goto out;
986 }
987
988 ++j;
989
990 if (!_dbus_type_reader_next (&array))
991 _dbus_assert (j == n_elements);
992 }
993
995 _dbus_assert (j == n_elements);
996 _dbus_assert (str_array[j] == NULL);
997
998 *str_array_p = str_array;
999 *n_elements_p = n_elements;
1000 }
1001#ifndef DBUS_DISABLE_CHECKS
1002 else
1003 {
1004 _dbus_warn ("you can't read arrays of container types (struct, variant, array) with %s for now",
1005 _DBUS_FUNCTION_NAME);
1006 goto out;
1007 }
1008#endif
1009 }
1010#ifndef DBUS_DISABLE_CHECKS
1011 else
1012 {
1013 _dbus_warn ("you can only read arrays and basic types with %s for now",
1014 _DBUS_FUNCTION_NAME);
1015 goto out;
1016 }
1017#endif
1018
1019 /* how many arguments already handled */
1020 i++;
1021
1022 spec_type = va_arg (var_args, int);
1023 if (!_dbus_type_reader_next (&real->u.reader) && spec_type != DBUS_TYPE_INVALID)
1024 {
1026 "Message has only %d arguments, but more were expected", i);
1027 goto out;
1028 }
1029 }
1030
1031 retval = TRUE;
1032
1033 out:
1034 /* there may memory or unix fd leak in the above iteration if parse failed.
1035 * so we have another iteration over copy_args to free memory and close
1036 * unix fds.
1037 */
1038 if (!retval)
1039 {
1040 spec_type = first_arg_type;
1041 j = 0;
1042
1043 while (j < i)
1044 {
1045 if (spec_type == DBUS_TYPE_UNIX_FD)
1046 {
1047#ifdef HAVE_UNIX_FD_PASSING
1048 int *pfd;
1049
1050 pfd = va_arg (copy_args, int *);
1051 _dbus_assert(pfd);
1052 if (*pfd >= 0)
1053 {
1054 _dbus_close (*pfd, NULL);
1055 *pfd = -1;
1056 }
1057#endif
1058 }
1059 else if (dbus_type_is_basic (spec_type))
1060 {
1061 /* move the index forward */
1062 va_arg (copy_args, const void *);
1063 }
1064 else if (spec_type == DBUS_TYPE_ARRAY)
1065 {
1066 int spec_element_type;
1067
1068 spec_element_type = va_arg (copy_args, int);
1069 if (dbus_type_is_fixed (spec_element_type))
1070 {
1071 /* move the index forward */
1072 va_arg (copy_args, const void **);
1073 va_arg (copy_args, int *);
1074 }
1075 else if (_DBUS_TYPE_IS_STRINGLIKE (spec_element_type))
1076 {
1077 char ***str_array_p;
1078
1079 str_array_p = va_arg (copy_args, char ***);
1080 /* move the index forward */
1081 va_arg (copy_args, int *);
1082 _dbus_assert (str_array_p != NULL);
1083 dbus_free_string_array (*str_array_p);
1084 *str_array_p = NULL;
1085 }
1086 }
1087
1088 spec_type = va_arg (copy_args, int);
1089 j++;
1090 }
1091 }
1092
1093 va_end (copy_args);
1094 return retval;
1095}
1096
1155dbus_uint32_t
1157{
1158 _dbus_return_val_if_fail (message != NULL, 0);
1159
1160 return _dbus_header_get_serial (&message->header);
1161}
1162
1173 dbus_uint32_t reply_serial)
1174{
1175 DBusBasicValue value;
1176
1177 _dbus_return_val_if_fail (message != NULL, FALSE);
1178 _dbus_return_val_if_fail (!message->locked, FALSE);
1179 _dbus_return_val_if_fail (reply_serial != 0, FALSE); /* 0 is invalid */
1180
1181 value.u32 = reply_serial;
1182
1183 return _dbus_header_set_field_basic (&message->header,
1186 &value);
1187}
1188
1195dbus_uint32_t
1197{
1198 dbus_uint32_t v_UINT32;
1199
1200 _dbus_return_val_if_fail (message != NULL, 0);
1201
1202 if (_dbus_header_get_field_basic (&message->header,
1205 &v_UINT32))
1206 return v_UINT32;
1207 else
1208 return 0;
1209}
1210
1211static void
1212dbus_message_finalize (DBusMessage *message)
1213{
1214 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0);
1215
1216 /* This calls application callbacks! */
1218
1219 _dbus_list_foreach (&message->counters,
1220 free_counter, message);
1221 _dbus_list_clear (&message->counters);
1222
1223 _dbus_header_free (&message->header);
1224 _dbus_string_free (&message->body);
1225
1226#ifdef HAVE_UNIX_FD_PASSING
1227 close_unix_fds(message->unix_fds, &message->n_unix_fds);
1228 dbus_free(message->unix_fds);
1229#endif
1230
1231 _dbus_assert (_dbus_atomic_get (&message->refcount) == 0);
1232
1233 dbus_free (message);
1234}
1235
1236static DBusMessage*
1237dbus_message_new_empty_header (void)
1238{
1239 DBusMessage *message;
1240 dbus_bool_t from_cache;
1241
1242 message = dbus_message_get_cached ();
1243
1244 if (message != NULL)
1245 {
1246 from_cache = TRUE;
1247 }
1248 else
1249 {
1250 from_cache = FALSE;
1251 message = dbus_new0 (DBusMessage, 1);
1252 if (message == NULL)
1253 return NULL;
1254#ifndef DBUS_DISABLE_CHECKS
1256#endif
1257
1258#ifdef HAVE_UNIX_FD_PASSING
1259 message->unix_fds = NULL;
1260 message->n_unix_fds_allocated = 0;
1261#endif
1262 }
1263
1264 _dbus_atomic_inc (&message->refcount);
1265
1266 _dbus_message_trace_ref (message, 0, 1, "new_empty_header");
1267
1268#ifdef HAVE_PDPLINUX
1269 message->message_skip_mandatory=FALSE;
1270#endif
1271
1272 message->locked = FALSE;
1273#ifndef DBUS_DISABLE_CHECKS
1274 message->in_cache = FALSE;
1275#endif
1276 message->counters = NULL;
1277 message->size_counter_delta = 0;
1278 message->changed_stamp = 0;
1279
1280#ifdef HAVE_UNIX_FD_PASSING
1281 message->n_unix_fds = 0;
1282 message->n_unix_fds_allocated = 0;
1283 message->unix_fd_counter_delta = 0;
1284#endif
1285
1286 if (!from_cache)
1288
1289 if (from_cache)
1290 {
1291 _dbus_header_reinit (&message->header);
1292 _dbus_string_set_length (&message->body, 0);
1293 }
1294 else
1295 {
1296 if (!_dbus_header_init (&message->header))
1297 {
1298 dbus_free (message);
1299 return NULL;
1300 }
1301
1302 if (!_dbus_string_init_preallocated (&message->body, 32))
1303 {
1304 _dbus_header_free (&message->header);
1305 dbus_free (message);
1306 return NULL;
1307 }
1308 }
1309
1310 return message;
1311}
1312
1326dbus_message_new (int message_type)
1327{
1328 DBusMessage *message;
1329
1330 _dbus_return_val_if_fail (message_type != DBUS_MESSAGE_TYPE_INVALID, NULL);
1331
1332 message = dbus_message_new_empty_header ();
1333 if (message == NULL)
1334 return NULL;
1335
1336 if (!_dbus_header_create (&message->header,
1337 DBUS_COMPILER_BYTE_ORDER,
1338 message_type,
1339 NULL, NULL, NULL, NULL, NULL))
1340 {
1341 dbus_message_unref (message);
1342 return NULL;
1343 }
1344
1345 return message;
1346}
1347
1370dbus_message_new_method_call (const char *destination,
1371 const char *path,
1372 const char *iface,
1373 const char *method)
1374{
1375 DBusMessage *message;
1376
1377 _dbus_return_val_if_fail (path != NULL, NULL);
1378 _dbus_return_val_if_fail (method != NULL, NULL);
1379 _dbus_return_val_if_fail (destination == NULL ||
1380 _dbus_check_is_valid_bus_name (destination), NULL);
1381 _dbus_return_val_if_fail (_dbus_check_is_valid_path (path), NULL);
1382 _dbus_return_val_if_fail (iface == NULL ||
1383 _dbus_check_is_valid_interface (iface), NULL);
1384 _dbus_return_val_if_fail (_dbus_check_is_valid_member (method), NULL);
1385
1386 message = dbus_message_new_empty_header ();
1387 if (message == NULL)
1388 return NULL;
1389
1390 if (!_dbus_header_create (&message->header,
1391 DBUS_COMPILER_BYTE_ORDER,
1393 destination, path, iface, method, NULL))
1394 {
1395 dbus_message_unref (message);
1396 return NULL;
1397 }
1398
1399 return message;
1400}
1401
1411{
1412 DBusMessage *message;
1413 const char *sender;
1414
1415 _dbus_return_val_if_fail (method_call != NULL, NULL);
1416
1417 sender = dbus_message_get_sender (method_call);
1418
1419 /* sender is allowed to be null here in peer-to-peer case */
1420
1421 message = dbus_message_new_empty_header ();
1422 if (message == NULL)
1423 return NULL;
1424
1425 if (!_dbus_header_create (&message->header,
1426 DBUS_COMPILER_BYTE_ORDER,
1428 sender, NULL, NULL, NULL, NULL))
1429 {
1430 dbus_message_unref (message);
1431 return NULL;
1432 }
1433
1435
1436 if (!dbus_message_set_reply_serial (message,
1437 dbus_message_get_serial (method_call)))
1438 {
1439 dbus_message_unref (message);
1440 return NULL;
1441 }
1442
1443 return message;
1444}
1445
1461dbus_message_new_signal (const char *path,
1462 const char *iface,
1463 const char *name)
1464{
1465 DBusMessage *message;
1466
1467 _dbus_return_val_if_fail (path != NULL, NULL);
1468 _dbus_return_val_if_fail (iface != NULL, NULL);
1469 _dbus_return_val_if_fail (name != NULL, NULL);
1470 _dbus_return_val_if_fail (_dbus_check_is_valid_path (path), NULL);
1471 _dbus_return_val_if_fail (_dbus_check_is_valid_interface (iface), NULL);
1472 _dbus_return_val_if_fail (_dbus_check_is_valid_member (name), NULL);
1473
1474 message = dbus_message_new_empty_header ();
1475 if (message == NULL)
1476 return NULL;
1477
1478 if (!_dbus_header_create (&message->header,
1479 DBUS_COMPILER_BYTE_ORDER,
1481 NULL, path, iface, name, NULL))
1482 {
1483 dbus_message_unref (message);
1484 return NULL;
1485 }
1486
1488
1489 return message;
1490}
1491
1508 const char *error_name,
1509 const char *error_message)
1510{
1511 DBusMessage *message;
1512 const char *sender;
1513 DBusMessageIter iter;
1514
1515 _dbus_return_val_if_fail (reply_to != NULL, NULL);
1516 _dbus_return_val_if_fail (error_name != NULL, NULL);
1517 _dbus_return_val_if_fail (_dbus_check_is_valid_error_name (error_name), NULL);
1518
1519 sender = dbus_message_get_sender (reply_to);
1520
1521 /* sender may be NULL for non-message-bus case or
1522 * when the message bus is dealing with an unregistered
1523 * connection.
1524 */
1525 message = dbus_message_new_empty_header ();
1526 if (message == NULL)
1527 return NULL;
1528
1529 if (!_dbus_header_create (&message->header,
1530 DBUS_COMPILER_BYTE_ORDER,
1532 sender, NULL, NULL, NULL, error_name))
1533 {
1534 dbus_message_unref (message);
1535 return NULL;
1536 }
1537
1539
1540 if (!dbus_message_set_reply_serial (message,
1541 dbus_message_get_serial (reply_to)))
1542 {
1543 dbus_message_unref (message);
1544 return NULL;
1545 }
1546
1547 if (error_message != NULL)
1548 {
1549 dbus_message_iter_init_append (message, &iter);
1552 &error_message))
1553 {
1554 dbus_message_unref (message);
1555 return NULL;
1556 }
1557 }
1558
1559 return message;
1560}
1561
1580 const char *error_name,
1581 const char *error_format,
1582 ...)
1583{
1584 va_list args;
1585 DBusString str;
1586 DBusMessage *message;
1587
1588 _dbus_return_val_if_fail (reply_to != NULL, NULL);
1589 _dbus_return_val_if_fail (error_name != NULL, NULL);
1590 _dbus_return_val_if_fail (_dbus_check_is_valid_error_name (error_name), NULL);
1591
1592 if (!_dbus_string_init (&str))
1593 return NULL;
1594
1595 va_start (args, error_format);
1596
1597 if (_dbus_string_append_printf_valist (&str, error_format, args))
1598 message = dbus_message_new_error (reply_to, error_name,
1599 _dbus_string_get_const_data (&str));
1600 else
1601 message = NULL;
1602
1603 _dbus_string_free (&str);
1604
1605 va_end (args);
1606
1607 return message;
1608}
1609
1610
1625{
1626 DBusMessage *retval;
1627
1628 _dbus_return_val_if_fail (message != NULL, NULL);
1629
1630 retval = dbus_new0 (DBusMessage, 1);
1631 if (retval == NULL)
1632 return NULL;
1633
1634 _dbus_atomic_inc (&retval->refcount);
1635
1636 retval->locked = FALSE;
1637#ifndef DBUS_DISABLE_CHECKS
1638 retval->generation = message->generation;
1639#endif
1640
1641 if (!_dbus_header_copy (&message->header, &retval->header))
1642 {
1643 dbus_free (retval);
1644 return NULL;
1645 }
1646
1647 if (!_dbus_string_init_preallocated (&retval->body,
1648 _dbus_string_get_length (&message->body)))
1649 {
1650 _dbus_header_free (&retval->header);
1651 dbus_free (retval);
1652 return NULL;
1653 }
1654
1655 if (!_dbus_string_copy (&message->body, 0,
1656 &retval->body, 0))
1657 goto failed_copy;
1658
1659#ifdef HAVE_UNIX_FD_PASSING
1660 retval->unix_fds = dbus_new(int, message->n_unix_fds);
1661 if (retval->unix_fds == NULL && message->n_unix_fds > 0)
1662 goto failed_copy;
1663
1664 retval->n_unix_fds_allocated = message->n_unix_fds;
1665
1666 for (retval->n_unix_fds = 0;
1667 retval->n_unix_fds < message->n_unix_fds;
1668 retval->n_unix_fds++)
1669 {
1670 retval->unix_fds[retval->n_unix_fds] = _dbus_dup(message->unix_fds[retval->n_unix_fds], NULL);
1671
1672 if (retval->unix_fds[retval->n_unix_fds] < 0)
1673 goto failed_copy;
1674 }
1675
1676#endif
1677
1678 _dbus_message_trace_ref (retval, 0, 1, "copy");
1679 return retval;
1680
1681 failed_copy:
1682 _dbus_header_free (&retval->header);
1683 _dbus_string_free (&retval->body);
1684
1685#ifdef HAVE_UNIX_FD_PASSING
1686 close_unix_fds(retval->unix_fds, &retval->n_unix_fds);
1687 dbus_free(retval->unix_fds);
1688#endif
1689
1690 dbus_free (retval);
1691
1692 return NULL;
1693}
1694
1695
1705{
1706 dbus_int32_t old_refcount;
1707
1708 _dbus_return_val_if_fail (message != NULL, NULL);
1709 _dbus_return_val_if_fail (message->generation == _dbus_current_generation, NULL);
1710 _dbus_return_val_if_fail (!message->in_cache, NULL);
1711
1712 old_refcount = _dbus_atomic_inc (&message->refcount);
1713 _dbus_assert (old_refcount >= 1);
1714 _dbus_message_trace_ref (message, old_refcount, old_refcount + 1, "ref");
1715
1716 return message;
1717}
1718
1726void
1728{
1729 dbus_int32_t old_refcount;
1730
1731 _dbus_return_if_fail (message != NULL);
1732 _dbus_return_if_fail (message->generation == _dbus_current_generation);
1733 _dbus_return_if_fail (!message->in_cache);
1734
1735 old_refcount = _dbus_atomic_dec (&message->refcount);
1736
1737 _dbus_assert (old_refcount >= 1);
1738
1739 _dbus_message_trace_ref (message, old_refcount, old_refcount - 1, "unref");
1740
1741 if (old_refcount == 1)
1742 {
1743 /* Calls application callbacks! */
1744 dbus_message_cache_or_finalize (message);
1745 }
1746}
1747
1758int
1760{
1761 _dbus_return_val_if_fail (message != NULL, DBUS_MESSAGE_TYPE_INVALID);
1762
1763 return _dbus_header_get_message_type (&message->header);
1764}
1765
1836 int first_arg_type,
1837 ...)
1838{
1839 dbus_bool_t retval;
1840 va_list var_args;
1841
1842 _dbus_return_val_if_fail (message != NULL, FALSE);
1843
1844 va_start (var_args, first_arg_type);
1845 retval = dbus_message_append_args_valist (message,
1846 first_arg_type,
1847 var_args);
1848 va_end (var_args);
1849
1850 return retval;
1851}
1852
1868 int first_arg_type,
1869 va_list var_args)
1870{
1871 int type;
1872 DBusMessageIter iter;
1873
1874 _dbus_return_val_if_fail (message != NULL, FALSE);
1875
1876 type = first_arg_type;
1877
1878 dbus_message_iter_init_append (message, &iter);
1879
1880 while (type != DBUS_TYPE_INVALID)
1881 {
1882 if (dbus_type_is_basic (type))
1883 {
1884 const void *value;
1885 value = va_arg (var_args, const void *);
1886
1888 type,
1889 value))
1890 goto failed;
1891 }
1892 else if (type == DBUS_TYPE_ARRAY)
1893 {
1894 int element_type;
1895 DBusMessageIter array;
1896 char buf[2];
1897
1898 element_type = va_arg (var_args, int);
1899
1900 buf[0] = element_type;
1901 buf[1] = '\0';
1904 buf,
1905 &array))
1906 goto failed;
1907
1908 if (dbus_type_is_fixed (element_type) &&
1909 element_type != DBUS_TYPE_UNIX_FD)
1910 {
1911 const void **value;
1912 int n_elements;
1913
1914 value = va_arg (var_args, const void **);
1915 n_elements = va_arg (var_args, int);
1916
1918 element_type,
1919 value,
1920 n_elements)) {
1922 goto failed;
1923 }
1924 }
1925 else if (_DBUS_TYPE_IS_STRINGLIKE (element_type))
1926 {
1927 const char ***value_p;
1928 const char **value;
1929 int n_elements;
1930 int i;
1931
1932 value_p = va_arg (var_args, const char***);
1933 n_elements = va_arg (var_args, int);
1934
1935 value = *value_p;
1936
1937 i = 0;
1938 while (i < n_elements)
1939 {
1940 if (!dbus_message_iter_append_basic (&array,
1941 element_type,
1942 &value[i])) {
1944 goto failed;
1945 }
1946 ++i;
1947 }
1948 }
1949 else
1950 {
1951 _dbus_warn ("arrays of %s can't be appended with %s for now",
1952 _dbus_type_to_string (element_type),
1953 _DBUS_FUNCTION_NAME);
1955 goto failed;
1956 }
1957
1958 if (!dbus_message_iter_close_container (&iter, &array))
1959 goto failed;
1960 }
1961#ifndef DBUS_DISABLE_CHECKS
1962 else
1963 {
1964 _dbus_warn ("type %s isn't supported yet in %s",
1965 _dbus_type_to_string (type), _DBUS_FUNCTION_NAME);
1966 goto failed;
1967 }
1968#endif
1969
1970 type = va_arg (var_args, int);
1971 }
1972
1973 return TRUE;
1974
1975 failed:
1976 return FALSE;
1977}
1978
2025 DBusError *error,
2026 int first_arg_type,
2027 ...)
2028{
2029 dbus_bool_t retval;
2030 va_list var_args;
2031
2032 _dbus_return_val_if_fail (message != NULL, FALSE);
2033 _dbus_return_val_if_error_is_set (error, FALSE);
2034
2035 va_start (var_args, first_arg_type);
2036 retval = dbus_message_get_args_valist (message, error, first_arg_type, var_args);
2037 va_end (var_args);
2038
2039 return retval;
2040}
2041
2054 DBusError *error,
2055 int first_arg_type,
2056 va_list var_args)
2057{
2058 DBusMessageIter iter;
2059
2060 _dbus_return_val_if_fail (message != NULL, FALSE);
2061 _dbus_return_val_if_error_is_set (error, FALSE);
2062
2063 dbus_message_iter_init (message, &iter);
2064 return _dbus_message_iter_get_args_valist (&iter, error, first_arg_type, var_args);
2065}
2066
2067static void
2068_dbus_message_iter_init_common (DBusMessage *message,
2069 DBusMessageRealIter *real,
2070 int iter_type)
2071{
2072 /* If these static assertions fail on your platform, report it as a bug. */
2073 _DBUS_STATIC_ASSERT (sizeof (DBusMessageRealIter) <= sizeof (DBusMessageIter));
2074 _DBUS_STATIC_ASSERT (_DBUS_ALIGNOF (DBusMessageRealIter) <=
2075 _DBUS_ALIGNOF (DBusMessageIter));
2076 /* A failure of these two assertions would indicate that we've broken
2077 * ABI on this platform since 1.10.0. */
2078 _DBUS_STATIC_ASSERT (sizeof (DBusMessageIter_1_10_0) ==
2079 sizeof (DBusMessageIter));
2080 _DBUS_STATIC_ASSERT (_DBUS_ALIGNOF (DBusMessageIter_1_10_0) ==
2081 _DBUS_ALIGNOF (DBusMessageIter));
2082 /* If this static assertion fails, it means the DBusMessageIter struct
2083 * is not "packed", which might result in "iter = other_iter" not copying
2084 * every byte. */
2085 _DBUS_STATIC_ASSERT (sizeof (DBusMessageIter) ==
2086 4 * sizeof (void *) + sizeof (dbus_uint32_t) + 9 * sizeof (int));
2087
2088 /* Since the iterator will read or write who-knows-what from the
2089 * message, we need to get in the right byte order
2090 */
2091 ensure_byte_order (message);
2092
2093 real->message = message;
2094 real->changed_stamp = message->changed_stamp;
2095 real->iter_type = iter_type;
2096 real->sig_refcount = 0;
2097}
2098
2123 DBusMessageIter *iter)
2124{
2126 const DBusString *type_str;
2127 int type_pos;
2128
2129 _dbus_return_val_if_fail (message != NULL, FALSE);
2130 _dbus_return_val_if_fail (iter != NULL, FALSE);
2131
2132 get_const_signature (&message->header, &type_str, &type_pos);
2133
2134 _dbus_message_iter_init_common (message, real,
2135 DBUS_MESSAGE_ITER_TYPE_READER);
2136
2139 type_str, type_pos,
2140 &message->body,
2141 0);
2142
2144}
2145
2154{
2156
2157 _dbus_return_val_if_fail (_dbus_message_iter_check (real), FALSE);
2158 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE);
2159
2160 return _dbus_type_reader_has_next (&real->u.reader);
2161}
2162
2173{
2175
2176 _dbus_return_val_if_fail (_dbus_message_iter_check (real), FALSE);
2177 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE);
2178
2179 return _dbus_type_reader_next (&real->u.reader);
2180}
2181
2196int
2198{
2200
2201 _dbus_return_val_if_fail (_dbus_message_iter_check (real), DBUS_TYPE_INVALID);
2202 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, FALSE);
2203
2205}
2206
2215int
2217{
2219
2220 _dbus_return_val_if_fail (_dbus_message_iter_check (real), DBUS_TYPE_INVALID);
2221 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_READER, DBUS_TYPE_INVALID);
2222 _dbus_return_val_if_fail (dbus_message_iter_get_arg_type (iter) == DBUS_TYPE_ARRAY, DBUS_TYPE_INVALID);
2223
2225}
2226
2252void
2254 DBusMessageIter *sub)
2255{
2257 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub;
2258
2259 _dbus_return_if_fail (_dbus_message_iter_check (real));
2260 _dbus_return_if_fail (sub != NULL);
2261
2262 *real_sub = *real;
2263 _dbus_type_reader_recurse (&real->u.reader, &real_sub->u.reader);
2264}
2265
2277char *
2279{
2280 const DBusString *sig;
2281 DBusString retstr;
2282 char *ret = NULL;
2283 int start, len;
2285
2286 _dbus_return_val_if_fail (_dbus_message_iter_check (real), NULL);
2287
2288 if (!_dbus_string_init (&retstr))
2289 return NULL;
2290
2292 &start, &len);
2293 if (!_dbus_string_append_len (&retstr,
2294 _dbus_string_get_const_data (sig) + start,
2295 len))
2296 goto oom;
2297
2298 /* This is correct whether it succeeds or fails: on success it sets `ret`,
2299 * and on failure it leaves `ret` set to NULL. */
2300 _dbus_string_steal_data (&retstr, &ret);
2301
2302oom:
2303 _dbus_string_free (&retstr);
2304 return ret;
2305}
2306
2354void
2356 void *value)
2357{
2359
2360 _dbus_return_if_fail (_dbus_message_iter_check (real));
2361 _dbus_return_if_fail (value != NULL);
2362
2364 {
2365#ifdef HAVE_UNIX_FD_PASSING
2366 DBusBasicValue idx;
2367
2369
2370 if (idx.u32 >= real->message->n_unix_fds) {
2371 /* Hmm, we cannot really signal an error here, so let's make
2372 sure to return an invalid fd. */
2373 *((int*) value) = -1;
2374 return;
2375 }
2376
2377 *((int*) value) = _dbus_dup(real->message->unix_fds[idx.u32], NULL);
2378#else
2379 *((int*) value) = -1;
2380#endif
2381 }
2382 else
2383 {
2385 value);
2386 }
2387}
2388
2399int
2401{
2403 DBusTypeReader array;
2404 int element_type;
2405 int n_elements = 0;
2406
2407 _dbus_return_val_if_fail (_dbus_message_iter_check (real), 0);
2408 _dbus_return_val_if_fail (_dbus_type_reader_get_current_type (&real->u.reader)
2409 == DBUS_TYPE_ARRAY, 0);
2410
2411 element_type = _dbus_type_reader_get_element_type (&real->u.reader);
2412 _dbus_type_reader_recurse (&real->u.reader, &array);
2413 if (dbus_type_is_fixed (element_type))
2414 {
2415 int alignment = _dbus_type_get_alignment (element_type);
2416 int total_len = _dbus_type_reader_get_array_length (&array);
2417 n_elements = total_len / alignment;
2418 }
2419 else
2420 {
2422 {
2423 ++n_elements;
2424 _dbus_type_reader_next (&array);
2425 }
2426 }
2427
2428 return n_elements;
2429}
2430
2443int
2445{
2447
2448 _dbus_return_val_if_fail (_dbus_message_iter_check (real), 0);
2449
2451}
2452
2488void
2490 void *value,
2491 int *n_elements)
2492{
2494#ifndef DBUS_DISABLE_CHECKS
2495 int subtype = _dbus_type_reader_get_current_type(&real->u.reader);
2496
2497 _dbus_return_if_fail (_dbus_message_iter_check (real));
2498 _dbus_return_if_fail (value != NULL);
2499 _dbus_return_if_fail ((subtype == DBUS_TYPE_INVALID) ||
2500 (dbus_type_is_fixed (subtype) && subtype != DBUS_TYPE_UNIX_FD));
2501#endif
2502
2504 value, n_elements);
2505}
2506
2518void
2520 DBusMessageIter *iter)
2521{
2523
2524 _dbus_return_if_fail (message != NULL);
2525 _dbus_return_if_fail (iter != NULL);
2526
2527 _dbus_message_iter_init_common (message, real,
2528 DBUS_MESSAGE_ITER_TYPE_WRITER);
2529
2530 /* We create the signature string and point iterators at it "on demand"
2531 * when a value is actually appended. That means that init() never fails
2532 * due to OOM.
2533 */
2536 &message->body,
2537 _dbus_string_get_length (&message->body));
2538}
2539
2548static dbus_bool_t
2549_dbus_message_iter_open_signature (DBusMessageRealIter *real)
2550{
2551 DBusString *str;
2552 const DBusString *current_sig;
2553 int current_sig_pos;
2554
2555 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
2556
2557 if (real->u.writer.type_str != NULL)
2558 {
2559 _dbus_assert (real->sig_refcount > 0);
2560 real->sig_refcount += 1;
2561 return TRUE;
2562 }
2563
2564 str = dbus_new (DBusString, 1);
2565 if (str == NULL)
2566 return FALSE;
2567
2570 &current_sig, &current_sig_pos))
2571 current_sig = NULL;
2572
2573 if (current_sig)
2574 {
2575 int current_len;
2576
2577 current_len = _dbus_string_get_byte (current_sig, current_sig_pos);
2578 current_sig_pos += 1; /* move on to sig data */
2579
2580 if (!_dbus_string_init_preallocated (str, current_len + 4))
2581 {
2582 dbus_free (str);
2583 return FALSE;
2584 }
2585
2586 if (!_dbus_string_copy_len (current_sig, current_sig_pos, current_len,
2587 str, 0))
2588 {
2589 _dbus_string_free (str);
2590 dbus_free (str);
2591 return FALSE;
2592 }
2593 }
2594 else
2595 {
2596 if (!_dbus_string_init_preallocated (str, 4))
2597 {
2598 dbus_free (str);
2599 return FALSE;
2600 }
2601 }
2602
2603 real->sig_refcount = 1;
2604
2605 /* If this assertion failed, then str would be neither stored in u.writer
2606 * nor freed by this function, resulting in a memory leak. */
2607 _dbus_assert (real->u.writer.type_str == NULL);
2609 str, _dbus_string_get_length (str));
2610 return TRUE;
2611}
2612
2622static dbus_bool_t
2623_dbus_message_iter_close_signature (DBusMessageRealIter *real)
2624{
2625 DBusString *str;
2626 const char *v_STRING;
2627 dbus_bool_t retval;
2628
2629 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
2630 _dbus_assert (real->u.writer.type_str != NULL);
2631 _dbus_assert (real->sig_refcount > 0);
2632
2633 real->sig_refcount -= 1;
2634
2635 if (real->sig_refcount > 0)
2636 return TRUE;
2637 _dbus_assert (real->sig_refcount == 0);
2638
2639 retval = TRUE;
2640
2641 str = real->u.writer.type_str;
2642
2643 v_STRING = _dbus_string_get_const_data (str);
2647 &v_STRING))
2648 retval = FALSE;
2649
2651 _dbus_string_free (str);
2652 dbus_free (str);
2653
2654 return retval;
2655}
2656
2664static void
2665_dbus_message_iter_abandon_signature (DBusMessageRealIter *real)
2666{
2667 DBusString *str;
2668
2669 _dbus_assert (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
2670 _dbus_assert (real->u.writer.type_str != NULL);
2671 _dbus_assert (real->sig_refcount > 0);
2672
2673 real->sig_refcount -= 1;
2674
2675 if (real->sig_refcount > 0)
2676 return;
2677 _dbus_assert (real->sig_refcount == 0);
2678
2679 str = real->u.writer.type_str;
2680
2682 _dbus_string_free (str);
2683 dbus_free (str);
2684}
2685
2686#ifndef DBUS_DISABLE_CHECKS
2687static dbus_bool_t
2688_dbus_message_iter_append_check (DBusMessageRealIter *iter)
2689{
2690 if (!_dbus_message_iter_check (iter))
2691 return FALSE;
2692
2693 if (iter->message->locked)
2694 {
2695 _dbus_warn_check_failed ("dbus append iterator can't be used: message is locked (has already been sent)");
2696 return FALSE;
2697 }
2698
2699 return TRUE;
2700}
2701#endif /* DBUS_DISABLE_CHECKS */
2702
2703#ifdef HAVE_UNIX_FD_PASSING
2704static int *
2705expand_fd_array(DBusMessage *m,
2706 unsigned n)
2707{
2708 _dbus_assert(m);
2709
2710 /* This makes space for adding n new fds to the array and returns a
2711 pointer to the place were the first fd should be put. */
2712
2713 if (m->n_unix_fds + n > m->n_unix_fds_allocated)
2714 {
2715 unsigned k;
2716 int *p;
2717
2718 /* Make twice as much space as necessary */
2719 k = (m->n_unix_fds + n) * 2;
2720
2721 /* Allocate at least four */
2722 if (k < 4)
2723 k = 4;
2724
2725 p = dbus_realloc(m->unix_fds, k * sizeof(int));
2726 if (p == NULL)
2727 return NULL;
2728
2729 m->unix_fds = p;
2730 m->n_unix_fds_allocated = k;
2731 }
2732
2733 return m->unix_fds + m->n_unix_fds;
2734}
2735#endif
2736
2758 int type,
2759 const void *value)
2760{
2762 dbus_bool_t ret;
2763
2764 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE);
2765 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE);
2766 _dbus_return_val_if_fail (dbus_type_is_basic (type), FALSE);
2767 _dbus_return_val_if_fail (value != NULL, FALSE);
2768
2769#ifndef DBUS_DISABLE_CHECKS
2770 switch (type)
2771 {
2772 DBusString str;
2773 DBusValidity signature_validity;
2774 const char * const *string_p;
2775 const dbus_bool_t *bool_p;
2776
2777 case DBUS_TYPE_STRING:
2778 string_p = value;
2779 _dbus_return_val_if_fail (_dbus_check_is_valid_utf8 (*string_p), FALSE);
2780 break;
2781
2783 string_p = value;
2784 _dbus_return_val_if_fail (_dbus_check_is_valid_path (*string_p), FALSE);
2785 break;
2786
2788 string_p = value;
2789 _dbus_string_init_const (&str, *string_p);
2790 signature_validity = _dbus_validate_signature_with_reason (&str,
2791 0,
2792 _dbus_string_get_length (&str));
2793
2794 if (signature_validity == DBUS_VALIDITY_UNKNOWN_OOM_ERROR)
2795 return FALSE;
2796
2797 _dbus_return_val_if_fail (signature_validity == DBUS_VALID, FALSE);
2798 break;
2799
2800 case DBUS_TYPE_BOOLEAN:
2801 bool_p = value;
2802 _dbus_return_val_if_fail (*bool_p == 0 || *bool_p == 1, FALSE);
2803 break;
2804
2805 default:
2806 {
2807 /* nothing to check, all possible values are allowed */
2808 }
2809 }
2810#endif
2811
2812 if (!_dbus_message_iter_open_signature (real))
2813 return FALSE;
2814
2815 if (type == DBUS_TYPE_UNIX_FD)
2816 {
2817#ifdef HAVE_UNIX_FD_PASSING
2818 int *fds;
2819 dbus_uint32_t u;
2820
2821 ret = FALSE;
2822
2823 /* First step, include the fd in the fd list of this message */
2824 if (!(fds = expand_fd_array(real->message, 1)))
2825 goto out;
2826
2827 *fds = _dbus_dup(*(int*) value, NULL);
2828 if (*fds < 0)
2829 goto out;
2830
2831 u = real->message->n_unix_fds;
2832
2833 /* Second step, write the index to the fd */
2834 if (!(ret = _dbus_type_writer_write_basic (&real->u.writer, DBUS_TYPE_UNIX_FD, &u))) {
2835 _dbus_close(*fds, NULL);
2836 goto out;
2837 }
2838
2839 real->message->n_unix_fds += 1;
2840 u += 1;
2841
2842 /* Final step, update the header accordingly */
2846 &u);
2847
2848 /* If any of these operations fail the message is
2849 hosed. However, no memory or fds should be leaked since what
2850 has been added to message has been added to the message, and
2851 can hence be accounted for when the message is being
2852 freed. */
2853#else
2854 ret = FALSE;
2855 /* This is redundant (we could just fall through), but it avoids
2856 * -Wunused-label in builds that don't HAVE_UNIX_FD_PASSING */
2857 goto out;
2858#endif
2859 }
2860 else
2861 {
2862 ret = _dbus_type_writer_write_basic (&real->u.writer, type, value);
2863 }
2864
2865out:
2866 if (!_dbus_message_iter_close_signature (real))
2867 ret = FALSE;
2868
2869 return ret;
2870}
2871
2909 int element_type,
2910 const void *value,
2911 int n_elements)
2912{
2914 dbus_bool_t ret;
2915
2916 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE);
2917 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE);
2918 _dbus_return_val_if_fail (dbus_type_is_fixed (element_type) && element_type != DBUS_TYPE_UNIX_FD, FALSE);
2919 _dbus_return_val_if_fail (real->u.writer.container_type == DBUS_TYPE_ARRAY, FALSE);
2920 _dbus_return_val_if_fail (value != NULL, FALSE);
2921 _dbus_return_val_if_fail (n_elements >= 0, FALSE);
2922 _dbus_return_val_if_fail (n_elements <=
2924 FALSE);
2925
2926#ifndef DBUS_DISABLE_CHECKS
2927 if (element_type == DBUS_TYPE_BOOLEAN)
2928 {
2929 const dbus_bool_t * const *bools = value;
2930 int i;
2931
2932 for (i = 0; i < n_elements; i++)
2933 {
2934 _dbus_return_val_if_fail ((*bools)[i] == 0 || (*bools)[i] == 1, FALSE);
2935 }
2936 }
2937#endif
2938
2939 ret = _dbus_type_writer_write_fixed_multi (&real->u.writer, element_type, value, n_elements);
2940
2941 return ret;
2942}
2943
2973 int type,
2974 const char *contained_signature,
2975 DBusMessageIter *sub)
2976{
2978 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub;
2979 DBusString contained_str;
2980 DBusValidity contained_signature_validity;
2981 dbus_bool_t ret;
2982
2983 _dbus_return_val_if_fail (sub != NULL, FALSE);
2984 /* Do our best to make sure the sub-iterator doesn't contain something
2985 * valid-looking on failure */
2986 _dbus_message_real_iter_zero (real_sub);
2987
2988 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE);
2989 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE);
2990 _dbus_return_val_if_fail (dbus_type_is_container (type), FALSE);
2991 _dbus_return_val_if_fail ((type == DBUS_TYPE_STRUCT &&
2992 contained_signature == NULL) ||
2993 (type == DBUS_TYPE_DICT_ENTRY &&
2994 contained_signature == NULL) ||
2995 (type == DBUS_TYPE_VARIANT &&
2996 contained_signature != NULL) ||
2997 (type == DBUS_TYPE_ARRAY &&
2998 contained_signature != NULL), FALSE);
2999
3000 /* this would fail if the contained_signature is a dict entry, since
3001 * dict entries are invalid signatures standalone (they must be in
3002 * an array)
3003 */
3004 if (contained_signature != NULL)
3005 {
3006 _dbus_string_init_const (&contained_str, contained_signature);
3007 contained_signature_validity = _dbus_validate_signature_with_reason (&contained_str,
3008 0,
3009 _dbus_string_get_length (&contained_str));
3010
3011 if (contained_signature_validity == DBUS_VALIDITY_UNKNOWN_OOM_ERROR)
3012 return FALSE;
3013 }
3014 else
3015 {
3016 /* just some placeholder value */
3017 contained_signature_validity = DBUS_VALID_BUT_INCOMPLETE;
3018 }
3019
3020 _dbus_return_val_if_fail ((type == DBUS_TYPE_ARRAY && contained_signature && *contained_signature == DBUS_DICT_ENTRY_BEGIN_CHAR) ||
3021 contained_signature == NULL ||
3022 contained_signature_validity == DBUS_VALID,
3023 FALSE);
3024
3025 if (!_dbus_message_iter_open_signature (real))
3026 return FALSE;
3027
3028 ret = FALSE;
3029 *real_sub = *real;
3030
3031 if (contained_signature != NULL)
3032 {
3033 _dbus_string_init_const (&contained_str, contained_signature);
3034
3035 ret = _dbus_type_writer_recurse (&real->u.writer,
3036 type,
3037 &contained_str, 0,
3038 &real_sub->u.writer);
3039 }
3040 else
3041 {
3042 ret = _dbus_type_writer_recurse (&real->u.writer,
3043 type,
3044 NULL, 0,
3045 &real_sub->u.writer);
3046 }
3047
3048 if (!ret)
3049 _dbus_message_iter_abandon_signature (real);
3050
3051 return ret;
3052}
3053
3054
3076 DBusMessageIter *sub)
3077{
3079 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub;
3080 dbus_bool_t ret;
3081
3082 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real), FALSE);
3083 _dbus_return_val_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE);
3084 _dbus_return_val_if_fail (_dbus_message_iter_append_check (real_sub), FALSE);
3085 _dbus_return_val_if_fail (real_sub->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER, FALSE);
3086
3087 ret = _dbus_type_writer_unrecurse (&real->u.writer,
3088 &real_sub->u.writer);
3089 _dbus_message_real_iter_zero (real_sub);
3090
3091 if (!_dbus_message_iter_close_signature (real))
3092 ret = FALSE;
3093
3094 return ret;
3095}
3096
3108void
3110 DBusMessageIter *sub)
3111{
3113 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub;
3114
3115#ifndef DBUS_DISABLE_CHECKS
3116 _dbus_return_if_fail (_dbus_message_iter_append_check (real));
3117 _dbus_return_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
3118 _dbus_return_if_fail (_dbus_message_iter_append_check (real_sub));
3119 _dbus_return_if_fail (real_sub->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
3120#endif
3121
3122 _dbus_message_iter_abandon_signature (real);
3123 _dbus_message_real_iter_zero (real_sub);
3124}
3125
3167void
3169 DBusMessageIter *sub)
3170{
3172 DBusMessageRealIter *real_sub = (DBusMessageRealIter *)sub;
3173
3174 /* If both the parent and the child are zeroed out, then either we didn't
3175 * even get as far as successfully recursing into the parent, or we already
3176 * closed both the child and the parent. For example, in the code sample
3177 * in the doc-comment above, this happens for
3178 * abandon_container_if_open (&outer, &inner) if the first open_container
3179 * call failed, or if we reached result = TRUE and fell through. */
3180 if (_dbus_message_real_iter_is_zeroed (real) &&
3181 _dbus_message_real_iter_is_zeroed (real_sub))
3182 return;
3183
3184#ifndef DBUS_DISABLE_CHECKS
3185 /* If the child is not zeroed out, but the parent is, then something has
3186 * gone horribly wrong (in practice that would probably mean both are
3187 * uninitialized or corrupt, and the parent happens to have ended up
3188 * all-bytes-zero). */
3189 _dbus_return_if_fail (_dbus_message_iter_append_check (real));
3190 _dbus_return_if_fail (real->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
3191#endif
3192
3193 /* If the parent is not zeroed out, but the child is, then either we did
3194 * not successfully open the child, or we already closed the child. This
3195 * means we do not own a reference to the parent's signature, so it would
3196 * be wrong to release it; so we must not call abandon_signature() here.
3197 * In the code sample in the doc-comment above, this happens for
3198 * abandon_container_if_open (&outer, &inner) if the second open_container
3199 * call failed, or if the second close_container call failed. */
3200 if (_dbus_message_real_iter_is_zeroed (real_sub))
3201 return;
3202
3203#ifndef DBUS_DISABLE_CHECKS
3204 _dbus_return_if_fail (_dbus_message_iter_append_check (real_sub));
3205 _dbus_return_if_fail (real_sub->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
3206#endif
3207
3208 /* If neither the parent nor the child is zeroed out, then we genuinely
3209 * have an open container; close it. In the code sample in the doc-comment,
3210 * this happens for abandon_container_if_open (&outer, &inner) if the
3211 * append_basic call failed. */
3212 _dbus_message_iter_abandon_signature (real);
3213 _dbus_message_real_iter_zero (real_sub);
3214}
3215
3232void
3234 dbus_bool_t no_reply)
3235{
3236 _dbus_return_if_fail (message != NULL);
3237 _dbus_return_if_fail (!message->locked);
3238
3241 no_reply);
3242}
3243
3253{
3254 _dbus_return_val_if_fail (message != NULL, FALSE);
3255
3256 return _dbus_header_get_flag (&message->header,
3258}
3259
3274void
3276 dbus_bool_t auto_start)
3277{
3278 _dbus_return_if_fail (message != NULL);
3279 _dbus_return_if_fail (!message->locked);
3280
3283 !auto_start);
3284}
3285
3295{
3296 _dbus_return_val_if_fail (message != NULL, FALSE);
3297
3298 return !_dbus_header_get_flag (&message->header,
3300}
3301
3302
3317 const char *object_path)
3318{
3319 _dbus_return_val_if_fail (message != NULL, FALSE);
3320 _dbus_return_val_if_fail (!message->locked, FALSE);
3321 _dbus_return_val_if_fail (object_path == NULL ||
3322 _dbus_check_is_valid_path (object_path),
3323 FALSE);
3324
3325 return set_or_delete_string_field (message,
3328 object_path);
3329}
3330
3344const char*
3346{
3347 const char *v;
3348
3349 _dbus_return_val_if_fail (message != NULL, NULL);
3350
3351 v = NULL; /* in case field doesn't exist */
3355 (void *) &v);
3356 return v;
3357}
3358
3370 const char *path)
3371{
3372 const char *msg_path;
3373 msg_path = dbus_message_get_path (message);
3374
3375 if (msg_path == NULL)
3376 {
3377 if (path == NULL)
3378 return TRUE;
3379 else
3380 return FALSE;
3381 }
3382
3383 if (path == NULL)
3384 return FALSE;
3385
3386 if (strcmp (msg_path, path) == 0)
3387 return TRUE;
3388
3389 return FALSE;
3390}
3391
3414 char ***path)
3415{
3416 const char *v;
3417
3418 _dbus_return_val_if_fail (message != NULL, FALSE);
3419 _dbus_return_val_if_fail (path != NULL, FALSE);
3420
3421 *path = NULL;
3422
3423 v = dbus_message_get_path (message);
3424 if (v != NULL)
3425 {
3426 if (!_dbus_decompose_path (v, strlen (v),
3427 path, NULL))
3428 return FALSE;
3429 }
3430 return TRUE;
3431}
3432
3448 const char *iface)
3449{
3450 _dbus_return_val_if_fail (message != NULL, FALSE);
3451 _dbus_return_val_if_fail (!message->locked, FALSE);
3452 _dbus_return_val_if_fail (iface == NULL ||
3453 _dbus_check_is_valid_interface (iface),
3454 FALSE);
3455
3456 return set_or_delete_string_field (message,
3459 iface);
3460}
3461
3475const char*
3477{
3478 const char *v;
3479
3480 _dbus_return_val_if_fail (message != NULL, NULL);
3481
3482 v = NULL; /* in case field doesn't exist */
3486 (void *) &v);
3487 return v;
3488}
3489
3499 const char *iface)
3500{
3501 const char *msg_interface;
3502 msg_interface = dbus_message_get_interface (message);
3503
3504 if (msg_interface == NULL)
3505 {
3506 if (iface == NULL)
3507 return TRUE;
3508 else
3509 return FALSE;
3510 }
3511
3512 if (iface == NULL)
3513 return FALSE;
3514
3515 if (strcmp (msg_interface, iface) == 0)
3516 return TRUE;
3517
3518 return FALSE;
3519
3520}
3521
3536 const char *member)
3537{
3538 _dbus_return_val_if_fail (message != NULL, FALSE);
3539 _dbus_return_val_if_fail (!message->locked, FALSE);
3540 _dbus_return_val_if_fail (member == NULL ||
3541 _dbus_check_is_valid_member (member),
3542 FALSE);
3543
3544 return set_or_delete_string_field (message,
3547 member);
3548}
3549
3561const char*
3563{
3564 const char *v;
3565
3566 _dbus_return_val_if_fail (message != NULL, NULL);
3567
3568 v = NULL; /* in case field doesn't exist */
3572 (void *) &v);
3573 return v;
3574}
3575
3585 const char *member)
3586{
3587 const char *msg_member;
3588 msg_member = dbus_message_get_member (message);
3589
3590 if (msg_member == NULL)
3591 {
3592 if (member == NULL)
3593 return TRUE;
3594 else
3595 return FALSE;
3596 }
3597
3598 if (member == NULL)
3599 return FALSE;
3600
3601 if (strcmp (msg_member, member) == 0)
3602 return TRUE;
3603
3604 return FALSE;
3605
3606}
3607
3621 const char *error_name)
3622{
3623 _dbus_return_val_if_fail (message != NULL, FALSE);
3624 _dbus_return_val_if_fail (!message->locked, FALSE);
3625 _dbus_return_val_if_fail (error_name == NULL ||
3626 _dbus_check_is_valid_error_name (error_name),
3627 FALSE);
3628
3629 return set_or_delete_string_field (message,
3632 error_name);
3633}
3634
3645const char*
3647{
3648 const char *v;
3649
3650 _dbus_return_val_if_fail (message != NULL, NULL);
3651
3652 v = NULL; /* in case field doesn't exist */
3656 (void *) &v);
3657 return v;
3658}
3659
3675 const char *destination)
3676{
3677 _dbus_return_val_if_fail (message != NULL, FALSE);
3678 _dbus_return_val_if_fail (!message->locked, FALSE);
3679 _dbus_return_val_if_fail (destination == NULL ||
3680 _dbus_check_is_valid_bus_name (destination),
3681 FALSE);
3682
3683 return set_or_delete_string_field (message,
3686 destination);
3687}
3688
3698const char*
3700{
3701 const char *v;
3702
3703 _dbus_return_val_if_fail (message != NULL, NULL);
3704
3705 v = NULL; /* in case field doesn't exist */
3709 (void *) &v);
3710 return v;
3711}
3712
3729 const char *sender)
3730{
3731 _dbus_return_val_if_fail (message != NULL, FALSE);
3732 _dbus_return_val_if_fail (!message->locked, FALSE);
3733 _dbus_return_val_if_fail (sender == NULL ||
3734 _dbus_check_is_valid_bus_name (sender),
3735 FALSE);
3736
3737 return set_or_delete_string_field (message,
3740 sender);
3741}
3742
3758const char*
3760{
3761 const char *v;
3762
3763 _dbus_return_val_if_fail (message != NULL, NULL);
3764
3765 v = NULL; /* in case field doesn't exist */
3769 (void *) &v);
3770 return v;
3771}
3772
3791const char*
3793{
3794 const DBusString *type_str;
3795 int type_pos;
3796
3797 _dbus_return_val_if_fail (message != NULL, NULL);
3798
3799 get_const_signature (&message->header, &type_str, &type_pos);
3800
3801 return _dbus_string_get_const_data_len (type_str, type_pos, 0);
3802}
3803
3804static dbus_bool_t
3805_dbus_message_has_type_interface_member (DBusMessage *message,
3806 int type,
3807 const char *iface,
3808 const char *member)
3809{
3810 const char *n;
3811
3812 _dbus_assert (message != NULL);
3813 _dbus_assert (iface != NULL);
3814 _dbus_assert (member != NULL);
3815
3816 if (dbus_message_get_type (message) != type)
3817 return FALSE;
3818
3819 /* Optimize by checking the short member name first
3820 * instead of the longer interface name
3821 */
3822
3823 n = dbus_message_get_member (message);
3824
3825 if (n && strcmp (n, member) == 0)
3826 {
3827 n = dbus_message_get_interface (message);
3828
3829 if (n == NULL || strcmp (n, iface) == 0)
3830 return TRUE;
3831 }
3832
3833 return FALSE;
3834}
3835
3852 const char *iface,
3853 const char *method)
3854{
3855 _dbus_return_val_if_fail (message != NULL, FALSE);
3856 _dbus_return_val_if_fail (iface != NULL, FALSE);
3857 _dbus_return_val_if_fail (method != NULL, FALSE);
3858 /* don't check that interface/method are valid since it would be
3859 * expensive, and not catch many common errors
3860 */
3861
3862 return _dbus_message_has_type_interface_member (message,
3864 iface, method);
3865}
3866
3880 const char *iface,
3881 const char *signal_name)
3882{
3883 _dbus_return_val_if_fail (message != NULL, FALSE);
3884 _dbus_return_val_if_fail (iface != NULL, FALSE);
3885 _dbus_return_val_if_fail (signal_name != NULL, FALSE);
3886 /* don't check that interface/name are valid since it would be
3887 * expensive, and not catch many common errors
3888 */
3889
3890 return _dbus_message_has_type_interface_member (message,
3892 iface, signal_name);
3893}
3894
3907 const char *error_name)
3908{
3909 const char *n;
3910
3911 _dbus_return_val_if_fail (message != NULL, FALSE);
3912 _dbus_return_val_if_fail (error_name != NULL, FALSE);
3913 /* don't check that error_name is valid since it would be expensive,
3914 * and not catch many common errors
3915 */
3916
3918 return FALSE;
3919
3920 n = dbus_message_get_error_name (message);
3921
3922 if (n && strcmp (n, error_name) == 0)
3923 return TRUE;
3924 else
3925 return FALSE;
3926}
3927
3940 const char *name)
3941{
3942 const char *s;
3943
3944 _dbus_return_val_if_fail (message != NULL, FALSE);
3945 _dbus_return_val_if_fail (name != NULL, FALSE);
3946 /* don't check that name is valid since it would be expensive, and
3947 * not catch many common errors
3948 */
3949
3950 s = dbus_message_get_destination (message);
3951
3952 if (s && strcmp (s, name) == 0)
3953 return TRUE;
3954 else
3955 return FALSE;
3956}
3957
3975 const char *name)
3976{
3977 const char *s;
3978
3979 _dbus_return_val_if_fail (message != NULL, FALSE);
3980 _dbus_return_val_if_fail (name != NULL, FALSE);
3981 /* don't check that name is valid since it would be expensive, and
3982 * not catch many common errors
3983 */
3984
3985 s = dbus_message_get_sender (message);
3986
3987 if (s && strcmp (s, name) == 0)
3988 return TRUE;
3989 else
3990 return FALSE;
3991}
3992
4004 const char *signature)
4005{
4006 const char *s;
4007
4008 _dbus_return_val_if_fail (message != NULL, FALSE);
4009 _dbus_return_val_if_fail (signature != NULL, FALSE);
4010 /* don't check that signature is valid since it would be expensive,
4011 * and not catch many common errors
4012 */
4013
4014 s = dbus_message_get_signature (message);
4015
4016 if (s && strcmp (s, signature) == 0)
4017 return TRUE;
4018 else
4019 return FALSE;
4020}
4021
4046 DBusMessage *message)
4047{
4048 const char *str;
4049
4050 _dbus_return_val_if_fail (message != NULL, FALSE);
4051 _dbus_return_val_if_error_is_set (error, FALSE);
4052
4054 return FALSE;
4055
4056 str = NULL;
4057 dbus_message_get_args (message, NULL,
4058 DBUS_TYPE_STRING, &str,
4060
4062 str ? "%s" : NULL, str);
4063
4064 return TRUE;
4065}
4066
4075{
4076#ifdef HAVE_UNIX_FD_PASSING
4077 _dbus_assert(message);
4078
4079 return message->n_unix_fds > 0;
4080#else
4081 return FALSE;
4082#endif
4083}
4084
4097 const char *object_path)
4098{
4099 _dbus_return_val_if_fail (message != NULL, FALSE);
4100 _dbus_return_val_if_fail (!message->locked, FALSE);
4101 _dbus_return_val_if_fail (object_path == NULL ||
4102 _dbus_check_is_valid_path (object_path),
4103 FALSE);
4104
4105 return set_or_delete_string_field (message,
4108 object_path);
4109}
4110
4121const char *
4123{
4124 const char *v;
4125
4126 _dbus_return_val_if_fail (message != NULL, NULL);
4127
4128 v = NULL; /* in case field doesn't exist */
4132 (void *) &v);
4133 return v;
4134}
4135
4154#define INITIAL_LOADER_DATA_LEN 32
4155
4164{
4165 DBusMessageLoader *loader;
4166
4167 loader = dbus_new0 (DBusMessageLoader, 1);
4168 if (loader == NULL)
4169 return NULL;
4170
4171 loader->refcount = 1;
4172
4173 loader->corrupted = FALSE;
4174 loader->corruption_reason = DBUS_VALID;
4175
4176 /* this can be configured by the app, but defaults to the protocol max */
4178
4179 /* We set a very relatively conservative default here since due to how
4180 SCM_RIGHTS works we need to preallocate an fd array of the maximum
4181 number of unix fds we want to receive in advance. A
4182 try-and-reallocate loop is not possible. */
4183 loader->max_message_unix_fds = DBUS_DEFAULT_MESSAGE_UNIX_FDS;
4184
4185 if (!_dbus_string_init (&loader->data))
4186 {
4187 dbus_free (loader);
4188 return NULL;
4189 }
4190
4191 /* preallocate the buffer for speed, ignore failure */
4193 _dbus_string_set_length (&loader->data, 0);
4194
4195#ifdef HAVE_UNIX_FD_PASSING
4196 loader->unix_fds = NULL;
4197 loader->n_unix_fds = loader->n_unix_fds_allocated = 0;
4198 loader->unix_fds_outstanding = FALSE;
4199#endif
4200
4201 return loader;
4202}
4203
4212{
4213 loader->refcount += 1;
4214
4215 return loader;
4216}
4217
4224void
4226{
4227 loader->refcount -= 1;
4228 if (loader->refcount == 0)
4229 {
4230#ifdef HAVE_UNIX_FD_PASSING
4231 close_unix_fds(loader->unix_fds, &loader->n_unix_fds);
4232 dbus_free(loader->unix_fds);
4233#endif
4236 _dbus_string_free (&loader->data);
4237 dbus_free (loader);
4238 }
4239}
4240
4259void
4261 DBusString **buffer,
4262 int *max_to_read,
4263 dbus_bool_t *may_read_fds)
4264{
4266
4267 *buffer = &loader->data;
4268
4269 loader->buffer_outstanding = TRUE;
4270
4271 if (max_to_read != NULL)
4272 {
4273#ifdef HAVE_UNIX_FD_PASSING
4274 int offset = 0;
4275 int remain;
4276 int byte_order;
4277 int fields_array_len;
4278 int header_len;
4279 int body_len;
4280#endif
4281
4282 *max_to_read = DBUS_MAXIMUM_MESSAGE_LENGTH;
4283 *may_read_fds = TRUE;
4284
4285#ifdef HAVE_UNIX_FD_PASSING
4286 /* If we aren't holding onto any fds, we can read as much as we want
4287 * (fast path). */
4288 if (loader->n_unix_fds == 0)
4289 return;
4290
4291 /* Slow path: we have a message with some fds in it. We don't want
4292 * to start on the next message until this one is out of the way;
4293 * otherwise a legitimate sender can keep us processing messages
4294 * containing fds, until we disconnect it for having had fds pending
4295 * for too long, a limit that is in place to stop malicious senders
4296 * from setting up recursive fd-passing that takes up our quota and
4297 * will never go away. */
4298
4299 remain = _dbus_string_get_length (&loader->data);
4300
4301 while (remain > 0)
4302 {
4303 DBusValidity validity = DBUS_VALIDITY_UNKNOWN;
4304 int needed;
4305
4306 /* If 0 < remain < DBUS_MINIMUM_HEADER_SIZE, then we've had at
4307 * least the first byte of a message, but we don't know how
4308 * much more to read. Only read the rest of the
4309 * DBUS_MINIMUM_HEADER_SIZE for now; then we'll know. */
4310 if (remain < DBUS_MINIMUM_HEADER_SIZE)
4311 {
4312 *max_to_read = DBUS_MINIMUM_HEADER_SIZE - remain;
4313 *may_read_fds = FALSE;
4314 return;
4315 }
4316
4318 &validity,
4319 &byte_order,
4320 &fields_array_len,
4321 &header_len,
4322 &body_len,
4323 &loader->data,
4324 offset,
4325 remain))
4326 {
4327 /* If a message in the buffer is invalid, we're going to
4328 * disconnect the sender anyway, so reading an arbitrary amount
4329 * is fine. */
4330 if (validity != DBUS_VALID)
4331 return;
4332
4333 /* We have a partial message, with the
4334 * DBUS_MINIMUM_HEADER_SIZE-byte fixed part of the header (which
4335 * lets us work out how much more we need), but no more. Read
4336 * the rest of the message. */
4337 needed = header_len + body_len;
4338 _dbus_assert (needed > remain);
4339 *max_to_read = needed - remain;
4340 *may_read_fds = FALSE;
4341 return;
4342 }
4343
4344 /* Skip over entire messages until we have less than a message
4345 * remaining. */
4346 needed = header_len + body_len;
4348 _dbus_assert (remain >= needed);
4349 remain -= needed;
4350 offset += needed;
4351 }
4352#endif
4353 }
4354}
4355
4365void
4367 DBusString *buffer)
4368{
4370 _dbus_assert (buffer == &loader->data);
4371
4372 loader->buffer_outstanding = FALSE;
4373}
4374
4375#ifdef HAVE_UNIX_FD_PASSING
4387_dbus_message_loader_get_unix_fds(DBusMessageLoader *loader,
4388 int **fds,
4389 unsigned *max_n_fds)
4390{
4391 _dbus_assert (!loader->unix_fds_outstanding);
4392
4393 /* Allocate space where we can put the fds we read. We allocate
4394 space for max_message_unix_fds since this is an
4395 upper limit how many fds can be received within a single
4396 message. Since SCM_RIGHTS doesn't allow a reallocate+retry logic
4397 we are allocating the maximum possible array size right from the
4398 beginning. This sucks a bit, however unless SCM_RIGHTS is fixed
4399 there is no better way. */
4400
4401 if (loader->n_unix_fds_allocated < loader->max_message_unix_fds)
4402 {
4403 int *a = dbus_realloc(loader->unix_fds,
4404 loader->max_message_unix_fds * sizeof(loader->unix_fds[0]));
4405
4406 if (!a)
4407 return FALSE;
4408
4409 loader->unix_fds = a;
4410 loader->n_unix_fds_allocated = loader->max_message_unix_fds;
4411 }
4412
4413 *fds = loader->unix_fds + loader->n_unix_fds;
4414 *max_n_fds = loader->n_unix_fds_allocated - loader->n_unix_fds;
4415
4416 loader->unix_fds_outstanding = TRUE;
4417 return TRUE;
4418}
4419
4430void
4431_dbus_message_loader_return_unix_fds(DBusMessageLoader *loader,
4432 int *fds,
4433 unsigned n_fds)
4434{
4435 _dbus_assert(loader->unix_fds_outstanding);
4436 _dbus_assert(loader->unix_fds + loader->n_unix_fds == fds);
4437 _dbus_assert(loader->n_unix_fds + n_fds <= loader->n_unix_fds_allocated);
4438
4439 loader->n_unix_fds += n_fds;
4440 loader->unix_fds_outstanding = FALSE;
4441
4442 if (n_fds && loader->unix_fds_change)
4443 loader->unix_fds_change (loader->unix_fds_change_data);
4444}
4445#endif
4446
4447/*
4448 * FIXME when we move the header out of the buffer, that memmoves all
4449 * buffered messages. Kind of crappy.
4450 *
4451 * Also we copy the header and body, which is kind of crappy. To
4452 * avoid this, we have to allow header and body to be in a single
4453 * memory block, which is good for messages we read and bad for
4454 * messages we are creating. But we could move_len() the buffer into
4455 * this single memory block, and move_len() will just swap the buffers
4456 * if you're moving the entire buffer replacing the dest string.
4457 *
4458 * We could also have the message loader tell the transport how many
4459 * bytes to read; so it would first ask for some arbitrary number like
4460 * 256, then if the message was incomplete it would use the
4461 * header/body len to ask for exactly the size of the message (or
4462 * blocks the size of a typical kernel buffer for the socket). That
4463 * way we don't get trailing bytes in the buffer that have to be
4464 * memmoved. Though I suppose we also don't have a chance of reading a
4465 * bunch of small messages at once, so the optimization may be stupid.
4466 *
4467 * Another approach would be to keep a "start" index into
4468 * loader->data and only delete it occasionally, instead of after
4469 * each message is loaded.
4470 *
4471 * load_message() returns FALSE if not enough memory OR the loader was corrupted
4472 */
4473static dbus_bool_t
4474load_message (DBusMessageLoader *loader,
4475 DBusMessage *message,
4476 int byte_order,
4477 int fields_array_len,
4478 int header_len,
4479 int body_len)
4480{
4481 dbus_bool_t oom;
4482 DBusValidity validity;
4483 const DBusString *type_str;
4484 int type_pos;
4485 DBusValidationMode mode;
4486 dbus_uint32_t n_unix_fds = 0;
4487
4488 mode = DBUS_VALIDATION_MODE_DATA_IS_UNTRUSTED;
4489
4490 oom = FALSE;
4491
4492#if 0
4493 _dbus_verbose_bytes_of_string (&loader->data, 0, header_len /* + body_len */);
4494#endif
4495
4496 /* 1. VALIDATE AND COPY OVER HEADER */
4497 _dbus_assert (_dbus_string_get_length (&message->header.data) == 0);
4498 _dbus_assert ((header_len + body_len) <= _dbus_string_get_length (&loader->data));
4499
4500 if (!_dbus_header_load (&message->header,
4501 mode,
4502 &validity,
4503 byte_order,
4504 fields_array_len,
4505 header_len,
4506 body_len,
4507 &loader->data))
4508 {
4509 _dbus_verbose ("Failed to load header for new message code %d\n", validity);
4510
4511 /* assert here so we can catch any code that still uses DBUS_VALID to indicate
4512 oom errors. They should use DBUS_VALIDITY_UNKNOWN_OOM_ERROR instead */
4513 _dbus_assert (validity != DBUS_VALID);
4514
4515 if (validity == DBUS_VALIDITY_UNKNOWN_OOM_ERROR)
4516 oom = TRUE;
4517 else
4518 {
4519 loader->corrupted = TRUE;
4520 loader->corruption_reason = validity;
4521 }
4522 goto failed;
4523 }
4524
4525 _dbus_assert (validity == DBUS_VALID);
4526
4527 /* 2. VALIDATE BODY */
4528 if (mode != DBUS_VALIDATION_MODE_WE_TRUST_THIS_DATA_ABSOLUTELY)
4529 {
4530 get_const_signature (&message->header, &type_str, &type_pos);
4531
4532 /* Because the bytes_remaining arg is NULL, this validates that the
4533 * body is the right length
4534 */
4535 validity = _dbus_validate_body_with_reason (type_str,
4536 type_pos,
4537 byte_order,
4538 NULL,
4539 &loader->data,
4540 header_len,
4541 body_len);
4542 if (validity != DBUS_VALID)
4543 {
4544 _dbus_verbose ("Failed to validate message body code %d\n", validity);
4545
4546 loader->corrupted = TRUE;
4547 loader->corruption_reason = validity;
4548
4549 goto failed;
4550 }
4551 }
4552
4553 /* 3. COPY OVER UNIX FDS */
4557 &n_unix_fds);
4558
4559#ifdef HAVE_UNIX_FD_PASSING
4560
4561 if (n_unix_fds > loader->n_unix_fds)
4562 {
4563 _dbus_verbose("Message contains references to more unix fds than were sent %u != %u\n",
4564 n_unix_fds, loader->n_unix_fds);
4565
4566 loader->corrupted = TRUE;
4567 loader->corruption_reason = DBUS_INVALID_MISSING_UNIX_FDS;
4568 goto failed;
4569 }
4570
4571 /* If this was a recycled message there might still be
4572 some memory allocated for the fds */
4573 dbus_free(message->unix_fds);
4574
4575 if (n_unix_fds > 0)
4576 {
4577 message->unix_fds = _dbus_memdup(loader->unix_fds, n_unix_fds * sizeof(message->unix_fds[0]));
4578 if (message->unix_fds == NULL)
4579 {
4580 _dbus_verbose ("Failed to allocate file descriptor array\n");
4581 oom = TRUE;
4582 goto failed;
4583 }
4584
4585 message->n_unix_fds_allocated = message->n_unix_fds = n_unix_fds;
4586 loader->n_unix_fds -= n_unix_fds;
4587 memmove (loader->unix_fds, loader->unix_fds + n_unix_fds, loader->n_unix_fds * sizeof (loader->unix_fds[0]));
4588
4589 if (loader->unix_fds_change)
4590 loader->unix_fds_change (loader->unix_fds_change_data);
4591 }
4592 else
4593 message->unix_fds = NULL;
4594
4595#else
4596
4597 if (n_unix_fds > 0)
4598 {
4599 _dbus_verbose ("Hmm, message claims to come with file descriptors "
4600 "but that's not supported on our platform, disconnecting.\n");
4601
4602 loader->corrupted = TRUE;
4603 loader->corruption_reason = DBUS_INVALID_MISSING_UNIX_FDS;
4604 goto failed;
4605 }
4606
4607#endif
4608
4609 /* 3. COPY OVER BODY AND QUEUE MESSAGE */
4610
4611 if (!_dbus_list_append (&loader->messages, message))
4612 {
4613 _dbus_verbose ("Failed to append new message to loader queue\n");
4614 oom = TRUE;
4615 goto failed;
4616 }
4617
4618 _dbus_assert (_dbus_string_get_length (&message->body) == 0);
4619 _dbus_assert (_dbus_string_get_length (&loader->data) >=
4620 (header_len + body_len));
4621
4622 if (!_dbus_string_copy_len (&loader->data, header_len, body_len, &message->body, 0))
4623 {
4624 _dbus_verbose ("Failed to move body into new message\n");
4625 oom = TRUE;
4626 goto failed;
4627 }
4628
4629 _dbus_string_delete (&loader->data, 0, header_len + body_len);
4630
4631 /* don't waste more than 2k of memory */
4632 _dbus_string_compact (&loader->data, 2048);
4633
4634 _dbus_assert (_dbus_string_get_length (&message->header.data) == header_len);
4635 _dbus_assert (_dbus_string_get_length (&message->body) == body_len);
4636
4637 _dbus_verbose ("Loaded message %p\n", message);
4638
4639 _dbus_assert (!oom);
4640 _dbus_assert (!loader->corrupted);
4641 _dbus_assert (loader->messages != NULL);
4642 _dbus_assert (_dbus_list_find_last (&loader->messages, message) != NULL);
4643
4644 return TRUE;
4645
4646 failed:
4647
4648 /* Clean up */
4649
4650 /* does nothing if the message isn't in the list */
4651 _dbus_list_remove_last (&loader->messages, message);
4652
4653 if (oom)
4654 _dbus_assert (!loader->corrupted);
4655 else
4656 _dbus_assert (loader->corrupted);
4657
4658 _dbus_verbose_bytes_of_string (&loader->data, 0, _dbus_string_get_length (&loader->data));
4659
4660 return FALSE;
4661}
4662
4679{
4680 while (!loader->corrupted &&
4681 _dbus_string_get_length (&loader->data) >= DBUS_MINIMUM_HEADER_SIZE)
4682 {
4683 DBusValidity validity;
4684 int byte_order, fields_array_len, header_len, body_len;
4685
4687 &validity,
4688 &byte_order,
4689 &fields_array_len,
4690 &header_len,
4691 &body_len,
4692 &loader->data, 0,
4693 _dbus_string_get_length (&loader->data)))
4694 {
4695 DBusMessage *message;
4696
4697 _dbus_assert (validity == DBUS_VALID);
4698
4699 message = dbus_message_new_empty_header ();
4700 if (message == NULL)
4701 return FALSE;
4702
4703 if (!load_message (loader, message,
4704 byte_order, fields_array_len,
4705 header_len, body_len))
4706 {
4707 dbus_message_unref (message);
4708 /* load_message() returns false if corrupted or OOM; if
4709 * corrupted then return TRUE for not OOM
4710 */
4711 return loader->corrupted;
4712 }
4713
4714 _dbus_assert (loader->messages != NULL);
4715 _dbus_assert (_dbus_list_find_last (&loader->messages, message) != NULL);
4716 }
4717 else
4718 {
4719 _dbus_verbose ("Initial peek at header says we don't have a whole message yet, or data broken with invalid code %d\n",
4720 validity);
4721 if (validity != DBUS_VALID)
4722 {
4723 loader->corrupted = TRUE;
4724 loader->corruption_reason = validity;
4725 }
4726 return TRUE;
4727 }
4728 }
4729
4730 return TRUE;
4731}
4732
4742{
4743 if (loader->messages)
4744 return loader->messages->data;
4745 else
4746 return NULL;
4747}
4748
4759{
4760 return _dbus_list_pop_first (&loader->messages);
4761}
4762
4771DBusList*
4773{
4774 return _dbus_list_pop_first_link (&loader->messages);
4775}
4776
4783void
4785 DBusList *link)
4786{
4787 _dbus_list_prepend_link (&loader->messages, link);
4788}
4789
4801{
4802 _dbus_assert ((loader->corrupted && loader->corruption_reason != DBUS_VALID) ||
4803 (!loader->corrupted && loader->corruption_reason == DBUS_VALID));
4804 return loader->corrupted;
4805}
4806
4815{
4816 _dbus_assert ((loader->corrupted && loader->corruption_reason != DBUS_VALID) ||
4817 (!loader->corrupted && loader->corruption_reason == DBUS_VALID));
4818
4819 return loader->corruption_reason;
4820}
4821
4828void
4830 long size)
4831{
4832 if (size > DBUS_MAXIMUM_MESSAGE_LENGTH)
4833 {
4834 _dbus_verbose ("clamping requested max message size %ld to %d\n",
4837 }
4838 loader->max_message_size = size;
4839}
4840
4847long
4849{
4850 return loader->max_message_size;
4851}
4852
4859void
4861 long n)
4862{
4864 {
4865 _dbus_verbose ("clamping requested max message unix_fds %ld to %d\n",
4868 }
4869 loader->max_message_unix_fds = n;
4870}
4871
4878long
4880{
4881 return loader->max_message_unix_fds;
4882}
4883
4889int
4891{
4892#ifdef HAVE_UNIX_FD_PASSING
4893 return loader->n_unix_fds;
4894#else
4895 return 0;
4896#endif
4897}
4898
4907void
4909 void (* callback) (void *),
4910 void *data)
4911{
4912#ifdef HAVE_UNIX_FD_PASSING
4913 loader->unix_fds_change = callback;
4914 loader->unix_fds_change_data = data;
4915#endif
4916}
4917
4918static DBusDataSlotAllocator slot_allocator =
4919 _DBUS_DATA_SLOT_ALLOCATOR_INIT (_DBUS_LOCK_NAME (message_slots));
4920
4937{
4938 return _dbus_data_slot_allocator_alloc (&slot_allocator,
4939 slot_p);
4940}
4941
4953void
4954dbus_message_free_data_slot (dbus_int32_t *slot_p)
4955{
4956 _dbus_return_if_fail (*slot_p >= 0);
4957
4958 _dbus_data_slot_allocator_free (&slot_allocator, slot_p);
4959}
4960
4976 dbus_int32_t slot,
4977 void *data,
4978 DBusFreeFunction free_data_func)
4979{
4980 DBusFreeFunction old_free_func;
4981 void *old_data;
4982 dbus_bool_t retval;
4983
4984 _dbus_return_val_if_fail (message != NULL, FALSE);
4985 _dbus_return_val_if_fail (slot >= 0, FALSE);
4986
4987 retval = _dbus_data_slot_list_set (&slot_allocator,
4988 &message->slot_list,
4989 slot, data, free_data_func,
4990 &old_free_func, &old_data);
4991
4992 if (retval)
4993 {
4994 /* Do the actual free outside the message lock */
4995 if (old_free_func)
4996 (* old_free_func) (old_data);
4997 }
4998
4999 return retval;
5000}
5001
5010void*
5012 dbus_int32_t slot)
5013{
5014 void *res;
5015
5016 _dbus_return_val_if_fail (message != NULL, NULL);
5017
5018 res = _dbus_data_slot_list_get (&slot_allocator,
5019 &message->slot_list,
5020 slot);
5021
5022 return res;
5023}
5024
5038int
5039dbus_message_type_from_string (const char *type_str)
5040{
5041 if (strcmp (type_str, "method_call") == 0)
5043 if (strcmp (type_str, "method_return") == 0)
5045 else if (strcmp (type_str, "signal") == 0)
5047 else if (strcmp (type_str, "error") == 0)
5049 else
5051}
5052
5066const char *
5068{
5069 switch (type)
5070 {
5072 return "method_call";
5074 return "method_return";
5076 return "signal";
5078 return "error";
5079 default:
5080 return "invalid";
5081 }
5082}
5083
5098 char **marshalled_data_p,
5099 int *len_p)
5100{
5101 DBusString tmp;
5102 dbus_bool_t was_locked;
5103
5104 _dbus_return_val_if_fail (msg != NULL, FALSE);
5105 _dbus_return_val_if_fail (marshalled_data_p != NULL, FALSE);
5106 _dbus_return_val_if_fail (len_p != NULL, FALSE);
5107
5108 if (!_dbus_string_init (&tmp))
5109 return FALSE;
5110
5111 /* Ensure the message is locked, to ensure the length header is filled in. */
5112 was_locked = msg->locked;
5113
5114 if (!was_locked)
5115 dbus_message_lock (msg);
5116
5117 if (!_dbus_string_copy (&(msg->header.data), 0, &tmp, 0))
5118 goto fail;
5119
5120 *len_p = _dbus_string_get_length (&tmp);
5121
5122 if (!_dbus_string_copy (&(msg->body), 0, &tmp, *len_p))
5123 goto fail;
5124
5125 *len_p = _dbus_string_get_length (&tmp);
5126
5127 if (!_dbus_string_steal_data (&tmp, marshalled_data_p))
5128 goto fail;
5129
5130 _dbus_string_free (&tmp);
5131
5132 if (!was_locked)
5133 msg->locked = FALSE;
5134
5135 return TRUE;
5136
5137 fail:
5138 _dbus_string_free (&tmp);
5139
5140 if (!was_locked)
5141 msg->locked = FALSE;
5142
5143 return FALSE;
5144}
5145
5159dbus_message_demarshal (const char *str,
5160 int len,
5161 DBusError *error)
5162{
5163 DBusMessageLoader *loader = NULL;
5164 DBusString *buffer;
5165 DBusMessage *msg;
5166
5167 _dbus_return_val_if_fail (str != NULL, NULL);
5168
5169 loader = _dbus_message_loader_new ();
5170
5171 if (loader == NULL)
5172 goto fail_oom;
5173
5174 _dbus_message_loader_get_buffer (loader, &buffer, NULL, NULL);
5175
5176 if (!_dbus_string_append_len (buffer, str, len))
5177 goto fail_oom;
5178
5179 _dbus_message_loader_return_buffer (loader, buffer);
5180
5182 goto fail_oom;
5183
5185 goto fail_corrupt;
5186
5187 msg = _dbus_message_loader_pop_message (loader);
5188
5189 if (!msg)
5190 goto fail_oom;
5191
5193 return msg;
5194
5195 fail_corrupt:
5197 goto fail_oom;
5198
5199 dbus_set_error (error, DBUS_ERROR_INVALID_ARGS, "Message is corrupted (%s)",
5200 _dbus_validity_to_error_message (loader->corruption_reason));
5202 return NULL;
5203
5204 fail_oom:
5205 _DBUS_SET_OOM (error);
5206
5207 if (loader != NULL)
5209
5210 return NULL;
5211}
5212
5225int
5227 int len)
5228{
5229 DBusString str;
5230 int byte_order, fields_array_len, header_len, body_len;
5231 DBusValidity validity = DBUS_VALID;
5232 int have_message;
5233
5234 if (!buf || len < DBUS_MINIMUM_HEADER_SIZE)
5235 return 0;
5236
5239 _dbus_string_init_const_len (&str, buf, len);
5240
5241 validity = DBUS_VALID;
5242 have_message
5244 &validity, &byte_order,
5245 &fields_array_len,
5246 &header_len,
5247 &body_len,
5248 &str, 0,
5249 len);
5250 _dbus_string_free (&str);
5251
5252 if (validity == DBUS_VALID)
5253 {
5254 _dbus_assert (have_message || (header_len + body_len) > len);
5255 (void) have_message; /* unused unless asserting */
5256 return header_len + body_len;
5257 }
5258 else
5259 {
5260 return -1; /* broken! */
5261 }
5262}
5263
5285void
5287 dbus_bool_t allow)
5288{
5289 _dbus_return_if_fail (message != NULL);
5290 _dbus_return_if_fail (!message->locked);
5291
5294 allow);
5295}
5296
5305{
5306 _dbus_return_val_if_fail (message != NULL, FALSE);
5307
5308 return _dbus_header_get_flag (&message->header,
5310}
5311
5319{
5320 DBusString data;
5321};
5322
5336{
5337 DBusVariant *self = NULL;
5338 /* Points to the single item we will read from the reader */
5339 DBusMessageRealIter *real_reader = (DBusMessageRealIter *) reader;
5340 /* The position in self at which we will write a single variant
5341 * (it is position 0) */
5342 DBusTypeWriter items_writer;
5343 /* The position in self at which we will write a copy of reader
5344 * (it is inside the variant) */
5345 DBusTypeWriter variant_writer;
5346 /* 'v' */
5347 DBusString variant_signature;
5348 /* Whatever is the signature of the item we will copy from the reader */
5349 DBusString contained_signature;
5350 /* TRUE if self->data needs to be freed */
5351 dbus_bool_t data_inited = FALSE;
5352 /* The type of the item we will read from the reader */
5353 int type;
5354 /* The string, start position within that string, and length of the signature
5355 * of the single complete type of the item reader points to */
5356 const DBusString *sig;
5357 int start, len;
5358
5359 _dbus_assert (_dbus_message_iter_check (real_reader));
5360 _dbus_assert (real_reader->iter_type == DBUS_MESSAGE_ITER_TYPE_READER);
5362 type = dbus_message_iter_get_arg_type (reader);
5363 _dbus_type_reader_get_signature (&real_reader->u.reader, &sig, &start, &len);
5364
5365 if (!_dbus_string_init (&contained_signature))
5366 return NULL;
5367
5368 if (!_dbus_string_copy_len (sig, start, len, &contained_signature, 0))
5369 goto oom;
5370
5371 self = dbus_new0 (DBusVariant, 1);
5372
5373 if (self == NULL)
5374 goto oom;
5375
5376 if (!_dbus_string_init (&self->data))
5377 goto oom;
5378
5379 data_inited = TRUE;
5380
5381 _dbus_type_writer_init_values_only (&items_writer, DBUS_COMPILER_BYTE_ORDER,
5382 &variant_signature, 0, &self->data, 0);
5383
5384 if (!_dbus_type_writer_recurse (&items_writer, DBUS_TYPE_VARIANT,
5385 &contained_signature, 0, &variant_writer))
5386 goto oom;
5387
5388 if (type == DBUS_TYPE_ARRAY)
5389 {
5390 /* Points to each item in turn inside the array we are copying */
5391 DBusMessageIter array_reader;
5392 /* Same as array_reader */
5393 DBusMessageRealIter *real_array_reader = (DBusMessageRealIter *) &array_reader;
5394 /* The position inside the copied array at which we will write
5395 * the copy of array_reader */
5396 DBusTypeWriter array_writer;
5397
5398 dbus_message_iter_recurse (reader, &array_reader);
5399
5400 if (!_dbus_type_writer_recurse (&variant_writer, type,
5401 &contained_signature, 1, &array_writer))
5402 goto oom;
5403
5404 if (!_dbus_type_writer_write_reader (&array_writer,
5405 &real_array_reader->u.reader))
5406 goto oom;
5407
5408 if (!_dbus_type_writer_unrecurse (&variant_writer, &array_writer))
5409 goto oom;
5410 }
5411 else if (type == DBUS_TYPE_DICT_ENTRY || type == DBUS_TYPE_VARIANT ||
5412 type == DBUS_TYPE_STRUCT)
5413 {
5414 /* Points to each item in turn inside the container we are copying */
5415 DBusMessageIter inner_reader;
5416 /* Same as inner_reader */
5417 DBusMessageRealIter *real_inner_reader = (DBusMessageRealIter *) &inner_reader;
5418 /* The position inside the copied container at which we will write the
5419 * copy of inner_reader */
5420 DBusTypeWriter inner_writer;
5421
5422 dbus_message_iter_recurse (reader, &inner_reader);
5423
5424 if (!_dbus_type_writer_recurse (&variant_writer, type, NULL, 0,
5425 &inner_writer))
5426 goto oom;
5427
5428 if (!_dbus_type_writer_write_reader (&inner_writer,
5429 &real_inner_reader->u.reader))
5430 goto oom;
5431
5432 if (!_dbus_type_writer_unrecurse (&variant_writer, &inner_writer))
5433 goto oom;
5434 }
5435 else
5436 {
5437 DBusBasicValue value;
5438
5439 /* We eliminated all the container types above */
5441
5442 dbus_message_iter_get_basic (reader, &value);
5443
5444 if (!_dbus_type_writer_write_basic (&variant_writer, type, &value))
5445 goto oom;
5446 }
5447
5448 _dbus_string_free (&contained_signature);
5449 return self;
5450
5451oom:
5452 if (self != NULL)
5453 {
5454 if (data_inited)
5455 _dbus_string_free (&self->data);
5456
5457 dbus_free (self);
5458 }
5459
5460 _dbus_string_free (&contained_signature);
5461 return NULL;
5462}
5463
5470const char *
5472{
5473 unsigned char len;
5474 const char *ret;
5475
5476 _dbus_assert (self != NULL);
5477
5478 /* Here we make use of the fact that the serialization of a variant starts
5479 * with the 1-byte length, then that many bytes of signature, then \0. */
5480 len = _dbus_string_get_byte (&self->data, 0);
5481 ret = _dbus_string_get_const_data_len (&self->data, 1, len);
5482 _dbus_assert (strlen (ret) == len);
5483 return ret;
5484}
5485
5499 DBusMessageIter *writer)
5500{
5501 /* 'v' */
5502 DBusString variant_signature;
5503 /* Points to the single item in self */
5504 DBusTypeReader variant_reader;
5505 /* Points to the single item (of whatever type) inside the variant */
5506 DBusTypeReader reader;
5507 /* The position at which we will copy reader */
5508 DBusMessageRealIter *real_writer = (DBusMessageRealIter *) writer;
5509 dbus_bool_t ret;
5510
5511 _dbus_assert (self != NULL);
5512 _dbus_assert (_dbus_message_iter_append_check (real_writer));
5513 _dbus_assert (real_writer->iter_type == DBUS_MESSAGE_ITER_TYPE_WRITER);
5514
5516 _dbus_type_reader_init (&reader, DBUS_COMPILER_BYTE_ORDER,
5517 &variant_signature, 0, &self->data, 0);
5518 _dbus_type_reader_recurse (&reader, &variant_reader);
5519
5520 if (!_dbus_message_iter_open_signature (real_writer))
5521 return FALSE;
5522
5523 ret = _dbus_type_writer_write_reader (&real_writer->u.writer,
5524 &variant_reader);
5525
5526 if (!_dbus_message_iter_close_signature (real_writer))
5527 return FALSE;
5528
5529 return ret;
5530}
5531
5532int
5533_dbus_variant_get_length (DBusVariant *self)
5534{
5535 _dbus_assert (self != NULL);
5536 return _dbus_string_get_length (&self->data);
5537}
5538
5539const DBusString *
5540_dbus_variant_peek (DBusVariant *self)
5541{
5542 _dbus_assert (self != NULL);
5543 return &self->data;
5544}
5545
5546void
5547_dbus_variant_free (DBusVariant *self)
5548{
5549 _dbus_assert (self != NULL);
5550 _dbus_string_free (&self->data);
5551 dbus_free (self);
5552}
5553
5554// ********************************PDP********************************
5555#ifdef HAVE_PDPLINUX
5556
5557#ifdef PDP_EXEC_INTERNAL_CHECK_SENDER
5558#else
5559//-----------------------------------------------------------------------------------------------
5560const char*
5561dbus_pdplinux_message_get_exec_src (DBusMessage *message)
5562{
5563 const char *v;
5564 _dbus_return_val_if_fail (message != NULL, NULL);
5565 v = NULL;
5566
5568 DBUS_HEADER_FIELD_EXEC_SRC,
5570 (void *) &v);
5571
5572 return v;
5573}
5574
5575//-----------------------------------------------------------------------------------------------
5577dbus_pdplinux_message_set_exec_src ( DBusMessage *message,char* sProcName)
5578{
5579 dbus_bool_t res=FALSE;
5580
5581 _dbus_return_val_if_fail (message != NULL, FALSE);
5582 _dbus_return_val_if_fail (!message->locked, FALSE);
5583
5584 res=set_or_delete_string_field (message,
5585 DBUS_HEADER_FIELD_EXEC_SRC,
5587 sProcName);
5588
5589 return res;
5590}
5591#endif //PDP_EXEC_INTERNAL_CHECK_SENDER
5592
5593#ifdef PDP_EXEC_INTERNAL_CHECK_RECIEVER
5594#else
5595//-----------------------------------------------------------------------------------------------
5596const char*
5597dbus_pdplinux_message_get_exec_dst (DBusMessage *message)
5598{
5599 const char *v;
5600 _dbus_return_val_if_fail (message != NULL, NULL);
5601 v = NULL;
5602
5604 DBUS_HEADER_FIELD_EXEC_DST,
5606 (void *) &v);
5607
5608 return v;
5609}
5610
5611//-----------------------------------------------------------------------------------------------
5612#error ("Function dbus_pdplinux_message_set_exec_dst never used yet, so can't use it to determine dst proc")
5613
5615dbus_pdplinux_message_set_exec_dst ( DBusMessage *message,char* sProcName)
5616{
5617 dbus_bool_t res=FALSE;
5618
5619 _dbus_return_val_if_fail (message != NULL, FALSE);
5620 _dbus_return_val_if_fail (!message->locked, FALSE);
5621
5622 res=set_or_delete_string_field (message,
5623 DBUS_HEADER_FIELD_EXEC_DST,
5625 sProcName);
5626
5627 return res;
5628}
5629#endif // PDP_EXEC_INTERNAL_CHECK_RECIEVER
5630
5631//-----------------------------------------------------------------------------------------------
5633dbus_pdplinux_message_set_skip_mandatory( DBusMessage *message, dbus_bool_t is_skip_mandatory){
5634 dbus_bool_t res=TRUE;
5635 _dbus_verbose("Setting message %p skip_mandatory to %s\n",message,is_skip_mandatory ? "TRUE" : "FALSE");
5636 message->message_skip_mandatory=is_skip_mandatory;
5637 return res;
5638}
5639
5640//-----------------------------------------------------------------------------------------------
5642dbus_pdplinux_message_get_skip_mandatory( DBusMessage *message){
5643 return message->message_skip_mandatory;
5644}
5645
5646#endif //HAVE_PDPLINUX
5647
5650/* tests in dbus-message-util.c */
void _dbus_data_slot_allocator_free(DBusDataSlotAllocator *allocator, dbus_int32_t *slot_id_p)
Deallocates an ID previously allocated with _dbus_data_slot_allocator_alloc().
void _dbus_data_slot_list_clear(DBusDataSlotList *list)
Frees all data slots contained in the list, calling application-provided free functions if they exist...
void _dbus_data_slot_list_init(DBusDataSlotList *list)
Initializes a slot list.
void _dbus_data_slot_list_free(DBusDataSlotList *list)
Frees the data slot list and all data slots contained in it, calling application-provided free functi...
void * _dbus_data_slot_list_get(DBusDataSlotAllocator *allocator, DBusDataSlotList *list, int slot)
Retrieves data previously set with _dbus_data_slot_list_set_data().
dbus_bool_t _dbus_data_slot_list_set(DBusDataSlotAllocator *allocator, DBusDataSlotList *list, int slot, void *data, DBusFreeFunction free_data_func, DBusFreeFunction *old_free_func, void **old_data)
Stores a pointer in the data slot list, along with an optional function to be used for freeing the da...
dbus_bool_t _dbus_data_slot_allocator_alloc(DBusDataSlotAllocator *allocator, dbus_int32_t *slot_id_p)
Allocates an integer ID to be used for storing data in a DBusDataSlotList.
Definition: dbus-dataslot.c:70
void dbus_error_init(DBusError *error)
Initializes a DBusError structure.
Definition: dbus-errors.c:188
void dbus_set_error(DBusError *error, const char *name, const char *format,...)
Assigns an error name and message to a DBusError.
Definition: dbus-errors.c:354
void dbus_error_free(DBusError *error)
Frees an error that's been set (or just initialized), then reinitializes the error as in dbus_error_i...
Definition: dbus-errors.c:211
#define _dbus_assert_not_reached(explanation)
Aborts with an error message if called.
#define _dbus_assert(condition)
Aborts with an error message if the condition is false.
#define _DBUS_UNLOCK(name)
Unlocks a global lock.
#define _DBUS_LOCK(name)
Locks a global lock, initializing it first if necessary.
void _dbus_warn_check_failed(const char *format,...)
Prints a "critical" warning to stderr when an assertion fails; differs from _dbus_warn primarily in t...
char * _dbus_strdup(const char *str)
Duplicates a string.
void _dbus_warn(const char *format,...)
Prints a warning message to stderr.
#define _DBUS_LOCK_NAME(name)
Expands to name of a global lock variable.
#define _DBUS_ZERO(object)
Sets all bits in an object to zero.
void * _dbus_memdup(const void *mem, size_t n_bytes)
Duplicates a block of memory.
DBusList * _dbus_list_pop_first_link(DBusList **list)
Removes the first link in the list and returns it.
Definition: dbus-list.c:656
void _dbus_list_append_link(DBusList **list, DBusList *link)
Appends a link to the list.
Definition: dbus-list.c:316
void _dbus_list_clear_full(DBusList **list, DBusFreeFunction function)
Free every link and every element in the list.
Definition: dbus-list.c:568
void _dbus_list_remove_link(DBusList **list, DBusList *link)
Removes a link from the list.
Definition: dbus-list.c:528
DBusList * _dbus_list_find_last(DBusList **list, void *data)
Finds a value in the list.
Definition: dbus-list.c:473
void * _dbus_list_pop_first(DBusList **list)
Removes the first value in the list and returns it.
Definition: dbus-list.c:677
void _dbus_list_foreach(DBusList **list, DBusForeachFunction function, void *data)
Calls the given function for each element in the list.
Definition: dbus-list.c:787
void _dbus_list_clear(DBusList **list)
Frees all links in the list and sets the list head to NULL.
Definition: dbus-list.c:543
void _dbus_list_prepend_link(DBusList **list, DBusList *link)
Prepends a link to the list.
Definition: dbus-list.c:334
DBusList * _dbus_list_alloc_link(void *data)
Allocates a linked list node.
Definition: dbus-list.c:243
dbus_bool_t _dbus_list_remove_last(DBusList **list, void *data)
Removes a value from the list.
Definition: dbus-list.c:447
dbus_bool_t _dbus_list_append(DBusList **list, void *data)
Appends a value to the list.
Definition: dbus-list.c:271
#define NULL
A null pointer, defined appropriately for C or C++.
#define TRUE
Expands to "1".
#define FALSE
Expands to "0".
dbus_bool_t _dbus_type_writer_write_basic(DBusTypeWriter *writer, int type, const void *value)
Writes out a basic type.
void _dbus_type_reader_recurse(DBusTypeReader *reader, DBusTypeReader *sub)
Initialize a new reader pointing to the first type and corresponding value that's a child of the curr...
DBusValidity
This is primarily used in unit testing, so we can verify that each invalid message is invalid for the...
void _dbus_header_update_lengths(DBusHeader *header, int body_len)
Fills in the correct body length.
dbus_bool_t _dbus_header_copy(const DBusHeader *header, DBusHeader *dest)
Initializes dest with a copy of the given header.
DBusValidity _dbus_validate_signature_with_reason(const DBusString *type_str, int type_pos, int len)
Verifies that the range of type_str from type_pos to type_end is a valid signature.
void _dbus_type_writer_init_values_only(DBusTypeWriter *writer, int byte_order, const DBusString *type_str, int type_pos, DBusString *value_str, int value_pos)
Like _dbus_type_writer_init(), except the type string passed in should correspond to an existing sign...
int _dbus_type_get_alignment(int typecode)
Gets the alignment requirement for the given type; will be 1, 4, or 8.
void _dbus_type_writer_remove_types(DBusTypeWriter *writer)
Removes type string from the writer.
void _dbus_type_reader_init(DBusTypeReader *reader, int byte_order, const DBusString *type_str, int type_pos, const DBusString *value_str, int value_pos)
Initializes a type reader.
DBUS_PRIVATE_EXPORT void _dbus_verbose_bytes_of_string(const DBusString *str, int start, int len)
Dump the given part of the string to verbose log.
dbus_bool_t _dbus_header_set_field_basic(DBusHeader *header, int field, int type, const void *value)
Sets the value of a field with basic type.
int _dbus_header_get_message_type(DBusHeader *header)
Gets the type of the message.
void _dbus_type_reader_get_signature(const DBusTypeReader *reader, const DBusString **str_p, int *start_p, int *len_p)
Gets the string and range of said string containing the signature of the current value.
dbus_bool_t _dbus_type_writer_write_reader(DBusTypeWriter *writer, DBusTypeReader *reader)
Iterate through all values in the given reader, writing a copy of each value to the writer.
dbus_bool_t _dbus_header_get_field_basic(DBusHeader *header, int field, int type, void *value)
Gets the value of a field with basic type.
dbus_bool_t _dbus_type_writer_recurse(DBusTypeWriter *writer, int container_type, const DBusString *contained_type, int contained_type_start, DBusTypeWriter *sub)
Opens a new container and writes out the initial information for that container.
dbus_bool_t _dbus_header_get_flag(DBusHeader *header, dbus_uint32_t flag)
Gets a message flag bit, returning TRUE if the bit is set.
void _dbus_marshal_byteswap(const DBusString *signature, int signature_start, int old_byte_order, int new_byte_order, DBusString *value_str, int value_pos)
Byteswaps the marshaled data in the given value_str.
void _dbus_type_reader_read_fixed_multi(const DBusTypeReader *reader, const void **value, int *n_elements)
Reads a block of fixed-length basic values, from the current point in an array to the end of the arra...
char _dbus_header_get_byte_order(const DBusHeader *header)
Returns the header's byte order.
dbus_bool_t _dbus_header_have_message_untrusted(int max_message_length, DBusValidity *validity, int *byte_order, int *fields_array_len, int *header_len, int *body_len, const DBusString *str, int start, int len)
Given data long enough to contain the length of the message body and the fields array,...
void _dbus_header_reinit(DBusHeader *header)
Re-initializes a header that was previously initialized and never freed.
int _dbus_type_reader_get_element_type(const DBusTypeReader *reader)
Gets the type of an element of the array the reader is currently pointing to.
dbus_bool_t _dbus_type_reader_next(DBusTypeReader *reader)
Skip to the next value on this "level".
dbus_bool_t _dbus_header_delete_field(DBusHeader *header, int field)
Deletes a field, if it exists.
int _dbus_type_reader_get_array_length(const DBusTypeReader *reader)
Returns the number of bytes in the array.
dbus_bool_t _dbus_header_remove_unknown_fields(DBusHeader *header)
Remove every header field not known to this version of dbus.
dbus_uint32_t _dbus_header_get_serial(DBusHeader *header)
See dbus_message_get_serial()
void _dbus_type_writer_add_types(DBusTypeWriter *writer, DBusString *type_str, int type_pos)
Adds type string to the writer, if it had none.
dbus_bool_t _dbus_type_reader_has_next(const DBusTypeReader *reader)
Check whether there's another value on this "level".
void _dbus_type_reader_read_basic(const DBusTypeReader *reader, void *value)
Reads a basic-typed value, as with _dbus_marshal_read_basic().
void _dbus_type_writer_init_types_delayed(DBusTypeWriter *writer, int byte_order, DBusString *value_str, int value_pos)
Initialize a write iterator, with the signature to be provided later.
const char * _dbus_type_to_string(int typecode)
Returns a string describing the given type.
DBusValidity _dbus_validate_body_with_reason(const DBusString *expected_signature, int expected_signature_start, int byte_order, int *bytes_remaining, const DBusString *value_str, int value_pos, int len)
Verifies that the range of value_str from value_pos to value_end is a legitimate value of type expect...
int _dbus_type_reader_get_current_type(const DBusTypeReader *reader)
Gets the type of the value the reader is currently pointing to; or for a types-only reader gets the t...
void _dbus_header_free(DBusHeader *header)
Frees a header.
dbus_bool_t _dbus_header_init(DBusHeader *header)
Initializes a header, but doesn't prepare it for use; to make the header valid, you have to call _dbu...
dbus_bool_t _dbus_header_create(DBusHeader *header, int byte_order, int message_type, const char *destination, const char *path, const char *interface, const char *member, const char *error_name)
Fills in the primary fields of the header, so the header is ready for use.
dbus_bool_t _dbus_type_writer_unrecurse(DBusTypeWriter *writer, DBusTypeWriter *sub)
Closes a container created by _dbus_type_writer_recurse() and writes any additional information to th...
dbus_bool_t _dbus_type_writer_write_fixed_multi(DBusTypeWriter *writer, int element_type, const void *value, int n_elements)
Writes a block of fixed-length basic values, i.e.
void _dbus_header_toggle_flag(DBusHeader *header, dbus_uint32_t flag, dbus_bool_t value)
Toggles a message flag bit, turning on the bit if value = TRUE and flipping it off if value = FALSE.
void _dbus_header_set_serial(DBusHeader *header, dbus_uint32_t serial)
Sets the serial number of a header.
dbus_bool_t _dbus_header_load(DBusHeader *header, DBusValidationMode mode, DBusValidity *validity, int byte_order, int fields_array_len, int header_len, int body_len, const DBusString *str)
Creates a message header from potentially-untrusted data.
void _dbus_header_byteswap(DBusHeader *header, int new_order)
Swaps the header into the given order if required.
dbus_bool_t _dbus_header_get_field_raw(DBusHeader *header, int field, const DBusString **str, int *pos)
Gets the raw marshaled data for a field.
DBusValidationMode
This is used rather than a bool for high visibility.
@ DBUS_VALIDITY_UNKNOWN_OOM_ERROR
can't determine validity due to OOM
@ DBUS_VALID
the data is valid
int _dbus_current_generation
_dbus_current_generation is used to track each time that dbus_shutdown() is called,...
Definition: dbus-memory.c:772
DBUS_PRIVATE_EXPORT dbus_bool_t _dbus_register_shutdown_func(DBusShutdownFunction function, void *data)
Register a cleanup function to be called exactly once the next time dbus_shutdown() is called.
Definition: dbus-memory.c:801
void(* DBusFreeFunction)(void *memory)
The type of a function which frees a block of memory.
Definition: dbus-memory.h:63
void dbus_free(void *memory)
Frees a block of memory previously allocated by dbus_malloc() or dbus_malloc0().
Definition: dbus-memory.c:692
void * dbus_realloc(void *memory, size_t bytes)
Resizes a block of memory previously allocated by dbus_malloc() or dbus_malloc0().
Definition: dbus-memory.c:592
#define dbus_new(type, count)
Safe macro for using dbus_malloc().
Definition: dbus-memory.h:57
#define dbus_new0(type, count)
Safe macro for using dbus_malloc0().
Definition: dbus-memory.h:58
void dbus_free_string_array(char **str_array)
Frees a NULL-terminated array of strings.
Definition: dbus-memory.c:740
void _dbus_message_loader_set_max_message_size(DBusMessageLoader *loader, long size)
Sets the maximum size message we allow.
dbus_bool_t _dbus_message_remove_unknown_fields(DBusMessage *message)
Remove every header field not known to this version of dbus.
Definition: dbus-message.c:272
DBUS_PRIVATE_EXPORT dbus_bool_t _dbus_message_iter_get_args_valist(DBusMessageIter *iter, DBusError *error, int first_arg_type, va_list var_args)
Implementation of the varargs arg-getting functions.
Definition: dbus-message.c:826
DBusVariant * _dbus_variant_read(DBusMessageIter *reader)
Copy a single D-Bus message item from reader into a newly-allocated DBusVariant.
#define ensure_byte_order(message)
byte-swap the message if it doesn't match our byte order.
Definition: dbus-message.c:218
#define MAX_MESSAGE_SIZE_TO_CACHE
Avoid caching huge messages.
Definition: dbus-message.c:496
dbus_bool_t _dbus_variant_write(DBusVariant *self, DBusMessageIter *writer)
Copy the single D-Bus message item from self into writer.
void dbus_message_iter_init_closed(DBusMessageIter *iter)
Initialize iter as if with DBUS_MESSAGE_ITER_INIT_CLOSED.
Definition: dbus-message.c:743
dbus_bool_t _dbus_message_loader_get_is_corrupted(DBusMessageLoader *loader)
Checks whether the loader is confused due to bad data.
void dbus_message_set_serial(DBusMessage *message, dbus_uint32_t serial)
Sets the serial number of a message.
Definition: dbus-message.c:289
void dbus_message_lock(DBusMessage *message)
Locks a message.
Definition: dbus-message.c:419
int dbus_message_type_from_string(const char *type_str)
Utility function to convert a machine-readable (not translated) string into a D-Bus message type.
void _dbus_message_loader_unref(DBusMessageLoader *loader)
Decrements the reference count of the loader and finalizes the loader when the count reaches zero.
dbus_bool_t dbus_message_allocate_data_slot(dbus_int32_t *slot_p)
Allocates an integer ID to be used for storing application-specific data on any DBusMessage.
void _dbus_message_get_unix_fds(DBusMessage *message, const int **fds, unsigned *n_fds)
Gets the unix fds to be sent over the network for this message.
Definition: dbus-message.c:250
DBusMessage * dbus_message_demarshal(const char *str, int len, DBusError *error)
Demarshal a D-Bus message from the format described in the D-Bus specification.
DBusMessage * _dbus_message_loader_pop_message(DBusMessageLoader *loader)
Pops a loaded message (passing ownership of the message to the caller).
void _dbus_message_get_network_data(DBusMessage *message, const DBusString **header, const DBusString **body)
Gets the data to be sent over the network for this message.
Definition: dbus-message.c:231
void _dbus_message_loader_set_pending_fds_function(DBusMessageLoader *loader, void(*callback)(void *), void *data)
Register a function to be called whenever the number of pending file descriptors in the loader change...
#define CHANGED_STAMP_BITS
How many bits are in the changed_stamp used to validate iterators.
dbus_bool_t dbus_message_set_data(DBusMessage *message, dbus_int32_t slot, void *data, DBusFreeFunction free_data_func)
Stores a pointer on a DBusMessage, along with an optional function to be used for freeing the data wh...
void _dbus_message_loader_putback_message_link(DBusMessageLoader *loader, DBusList *link)
Returns a popped message link, used to undo a pop.
void * dbus_message_get_data(DBusMessage *message, dbus_int32_t slot)
Retrieves data previously set with dbus_message_set_data().
const char * dbus_message_type_to_string(int type)
Utility function to convert a D-Bus message type into a machine-readable string (not translated).
void dbus_message_set_allow_interactive_authorization(DBusMessage *message, dbus_bool_t allow)
Sets a flag indicating that the caller of the method is prepared to wait for interactive authorizatio...
int _dbus_message_loader_get_pending_fds_count(DBusMessageLoader *loader)
Return how many file descriptors are pending in the loader.
const char * _dbus_variant_get_signature(DBusVariant *self)
Return the signature of the item stored in self.
int dbus_message_demarshal_bytes_needed(const char *buf, int len)
Returns the number of bytes required to be in the buffer to demarshal a D-Bus message.
_DBUS_STRING_DEFINE_STATIC(_dbus_empty_signature_str, "")
An static string representing an empty signature.
void _dbus_message_loader_get_buffer(DBusMessageLoader *loader, DBusString **buffer, int *max_to_read, dbus_bool_t *may_read_fds)
Gets the buffer to use for reading data from the network.
void dbus_message_free_data_slot(dbus_int32_t *slot_p)
Deallocates a global ID for message data slots.
void _dbus_message_remove_counter(DBusMessage *message, DBusCounter *counter)
Removes a counter tracking the size/unix fds of this message, and decrements the counter by the size/...
Definition: dbus-message.c:387
#define INITIAL_LOADER_DATA_LEN
The initial buffer size of the message loader.
dbus_bool_t _dbus_message_add_counter(DBusMessage *message, DBusCounter *counter)
Adds a counter to be incremented immediately with the size/unix fds of this message,...
Definition: dbus-message.c:364
DBusMessage * _dbus_message_loader_peek_message(DBusMessageLoader *loader)
Peeks at first loaded message, returns NULL if no messages have been queued.
dbus_bool_t dbus_message_marshal(DBusMessage *msg, char **marshalled_data_p, int *len_p)
Turn a DBusMessage into the marshalled form as described in the D-Bus specification.
void _dbus_message_loader_set_max_message_unix_fds(DBusMessageLoader *loader, long n)
Sets the maximum unix fds per message we allow.
long _dbus_message_loader_get_max_message_size(DBusMessageLoader *loader)
Gets the maximum allowed message size in bytes.
DBusMessageLoader * _dbus_message_loader_new(void)
Creates a new message loader.
#define MAX_MESSAGE_CACHE_SIZE
Avoid caching too many messages.
Definition: dbus-message.c:499
DBusList * _dbus_message_loader_pop_message_link(DBusMessageLoader *loader)
Pops a loaded message inside a list link (passing ownership of the message and link to the caller).
dbus_bool_t _dbus_message_loader_queue_messages(DBusMessageLoader *loader)
Converts buffered data into messages, if we have enough data.
void _dbus_message_loader_return_buffer(DBusMessageLoader *loader, DBusString *buffer)
Returns a buffer obtained from _dbus_message_loader_get_buffer(), indicating to the loader how many b...
DBusMessageLoader * _dbus_message_loader_ref(DBusMessageLoader *loader)
Increments the reference count of the loader.
long _dbus_message_loader_get_max_message_unix_fds(DBusMessageLoader *loader)
Gets the maximum allowed number of unix fds per message.
void _dbus_message_add_counter_link(DBusMessage *message, DBusList *link)
Adds a counter to be incremented immediately with the size/unix fds of this message,...
Definition: dbus-message.c:315
DBusValidity _dbus_message_loader_get_corruption_reason(DBusMessageLoader *loader)
Checks what kind of bad data confused the loader.
dbus_bool_t dbus_message_get_allow_interactive_authorization(DBusMessage *message)
Returns whether the flag controlled by dbus_message_set_allow_interactive_authorization() has been se...
dbus_bool_t dbus_message_has_destination(DBusMessage *message, const char *name)
Checks whether the message was sent to the given name.
dbus_bool_t dbus_message_set_interface(DBusMessage *message, const char *iface)
Sets the interface this message is being sent to (for DBUS_MESSAGE_TYPE_METHOD_CALL) or the interface...
dbus_bool_t dbus_message_has_interface(DBusMessage *message, const char *iface)
Checks if the message has an interface.
void dbus_message_set_no_reply(DBusMessage *message, dbus_bool_t no_reply)
Sets a flag indicating that the message does not want a reply; if this flag is set,...
dbus_bool_t dbus_message_append_args_valist(DBusMessage *message, int first_arg_type, va_list var_args)
Like dbus_message_append_args() but takes a va_list for use by language bindings.
const char * dbus_message_get_sender(DBusMessage *message)
Gets the unique name of the connection which originated this message, or NULL if unknown or inapplica...
void dbus_message_set_auto_start(DBusMessage *message, dbus_bool_t auto_start)
Sets a flag indicating that an owner for the destination name will be automatically started before th...
dbus_bool_t dbus_message_iter_append_basic(DBusMessageIter *iter, int type, const void *value)
Appends a basic-typed value to the message.
const char * dbus_message_get_path(DBusMessage *message)
Gets the object path this message is being sent to (for DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitt...
const char * dbus_message_get_interface(DBusMessage *message)
Gets the interface this message is being sent to (for DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitted...
DBusMessage * dbus_message_new_error(DBusMessage *reply_to, const char *error_name, const char *error_message)
Creates a new message that is an error reply to another message.
dbus_bool_t dbus_message_has_sender(DBusMessage *message, const char *name)
Checks whether the message has the given unique name as its sender.
dbus_uint32_t dbus_message_get_serial(DBusMessage *message)
Returns the serial of a message or 0 if none has been specified.
dbus_bool_t dbus_message_set_member(DBusMessage *message, const char *member)
Sets the interface member being invoked (DBUS_MESSAGE_TYPE_METHOD_CALL) or emitted (DBUS_MESSAGE_TYPE...
void dbus_message_iter_get_basic(DBusMessageIter *iter, void *value)
Reads a basic-typed value from the message iterator.
int dbus_message_get_type(DBusMessage *message)
Gets the type of a message.
DBusMessage * dbus_message_copy(const DBusMessage *message)
Creates a new message that is an exact replica of the message specified, except that its refcount is ...
const char * dbus_message_get_error_name(DBusMessage *message)
Gets the error name (DBUS_MESSAGE_TYPE_ERROR only) or NULL if none.
dbus_bool_t dbus_message_iter_next(DBusMessageIter *iter)
Moves the iterator to the next field, if any.
dbus_bool_t dbus_message_append_args(DBusMessage *message, int first_arg_type,...)
Appends fields to a message given a variable argument list.
int dbus_message_iter_get_arg_type(DBusMessageIter *iter)
Returns the argument type of the argument that the message iterator points to.
dbus_bool_t dbus_message_get_no_reply(DBusMessage *message)
Returns TRUE if the message does not expect a reply.
DBusMessage * dbus_message_new_signal(const char *path, const char *iface, const char *name)
Constructs a new message representing a signal emission.
dbus_bool_t dbus_message_iter_append_fixed_array(DBusMessageIter *iter, int element_type, const void *value, int n_elements)
Appends a block of fixed-length values to an array.
void dbus_message_iter_abandon_container(DBusMessageIter *iter, DBusMessageIter *sub)
Abandons creation of a contained-typed value and frees resources created by dbus_message_iter_open_co...
DBusMessage * dbus_message_new_error_printf(DBusMessage *reply_to, const char *error_name, const char *error_format,...)
Creates a new message that is an error reply to another message, allowing you to use printf formattin...
dbus_bool_t dbus_message_is_error(DBusMessage *message, const char *error_name)
Checks whether the message is an error reply with the given error name.
dbus_bool_t dbus_message_contains_unix_fds(DBusMessage *message)
Checks whether a message contains unix fds.
void dbus_message_iter_recurse(DBusMessageIter *iter, DBusMessageIter *sub)
Recurses into a container value when reading values from a message, initializing a sub-iterator to us...
DBusMessage * dbus_message_ref(DBusMessage *message)
Increments the reference count of a DBusMessage.
dbus_bool_t dbus_message_get_auto_start(DBusMessage *message)
Returns TRUE if the message will cause an owner for destination name to be auto-started.
int dbus_message_iter_get_element_type(DBusMessageIter *iter)
Returns the element type of the array that the message iterator points to.
dbus_bool_t dbus_message_set_error_name(DBusMessage *message, const char *error_name)
Sets the name of the error (DBUS_MESSAGE_TYPE_ERROR).
dbus_bool_t dbus_message_has_signature(DBusMessage *message, const char *signature)
Checks whether the message has the given signature; see dbus_message_get_signature() for more details...
dbus_bool_t dbus_message_iter_open_container(DBusMessageIter *iter, int type, const char *contained_signature, DBusMessageIter *sub)
Appends a container-typed value to the message.
dbus_uint32_t dbus_message_get_reply_serial(DBusMessage *message)
Returns the serial that the message is a reply to or 0 if none.
DBusMessage * dbus_message_new_method_return(DBusMessage *method_call)
Constructs a message that is a reply to a method call.
DBusMessage * dbus_message_new_method_call(const char *destination, const char *path, const char *iface, const char *method)
Constructs a new message to invoke a method on a remote object.
dbus_bool_t dbus_message_iter_init(DBusMessage *message, DBusMessageIter *iter)
Initializes a DBusMessageIter for reading the arguments of the message passed in.
dbus_bool_t dbus_message_set_sender(DBusMessage *message, const char *sender)
Sets the message sender.
void dbus_message_iter_abandon_container_if_open(DBusMessageIter *iter, DBusMessageIter *sub)
Abandons creation of a contained-typed value and frees resources created by dbus_message_iter_open_co...
const char * dbus_message_get_destination(DBusMessage *message)
Gets the destination of a message or NULL if there is none set.
dbus_bool_t dbus_message_set_path(DBusMessage *message, const char *object_path)
Sets the object path this message is being sent to (for DBUS_MESSAGE_TYPE_METHOD_CALL) or the one a s...
dbus_bool_t dbus_message_iter_has_next(DBusMessageIter *iter)
Checks if an iterator has any more fields.
dbus_bool_t dbus_message_get_args_valist(DBusMessage *message, DBusError *error, int first_arg_type, va_list var_args)
Like dbus_message_get_args but takes a va_list for use by language bindings.
char * dbus_message_iter_get_signature(DBusMessageIter *iter)
Returns the current signature of a message iterator.
void dbus_message_unref(DBusMessage *message)
Decrements the reference count of a DBusMessage, freeing the message if the count reaches 0.
int dbus_message_iter_get_array_len(DBusMessageIter *iter)
Returns the number of bytes in the array as marshaled in the wire protocol.
DBusMessage * dbus_message_new(int message_type)
Constructs a new message of the given message type.
dbus_bool_t dbus_set_error_from_message(DBusError *error, DBusMessage *message)
Sets a DBusError based on the contents of the given message.
int dbus_message_iter_get_element_count(DBusMessageIter *iter)
Returns the number of elements in the array-typed value pointed to by the iterator.
dbus_bool_t dbus_message_set_destination(DBusMessage *message, const char *destination)
Sets the message's destination.
dbus_bool_t dbus_message_has_path(DBusMessage *message, const char *path)
Checks if the message has a particular object path.
dbus_bool_t dbus_message_has_member(DBusMessage *message, const char *member)
Checks if the message has an interface member.
dbus_bool_t dbus_message_get_args(DBusMessage *message, DBusError *error, int first_arg_type,...)
Gets arguments from a message given a variable argument list.
dbus_bool_t dbus_message_is_method_call(DBusMessage *message, const char *iface, const char *method)
Checks whether the message is a method call with the given interface and member fields.
void dbus_message_iter_get_fixed_array(DBusMessageIter *iter, void *value, int *n_elements)
Reads a block of fixed-length values from the message iterator.
dbus_bool_t dbus_message_set_reply_serial(DBusMessage *message, dbus_uint32_t reply_serial)
Sets the reply serial of a message (the serial of the message this is a reply to).
dbus_bool_t dbus_message_is_signal(DBusMessage *message, const char *iface, const char *signal_name)
Checks whether the message is a signal with the given interface and member fields.
const char * dbus_message_get_signature(DBusMessage *message)
Gets the type signature of the message, i.e.
dbus_bool_t dbus_message_set_container_instance(DBusMessage *message, const char *object_path)
Sets the container instance this message was sent from.
dbus_bool_t dbus_message_iter_close_container(DBusMessageIter *iter, DBusMessageIter *sub)
Closes a container-typed value appended to the message; may write out more information to the message...
dbus_bool_t dbus_message_get_path_decomposed(DBusMessage *message, char ***path)
Gets the object path this message is being sent to (for DBUS_MESSAGE_TYPE_METHOD_CALL) or being emitt...
const char * dbus_message_get_member(DBusMessage *message)
Gets the interface member being invoked (DBUS_MESSAGE_TYPE_METHOD_CALL) or emitted (DBUS_MESSAGE_TYPE...
void dbus_message_iter_init_append(DBusMessage *message, DBusMessageIter *iter)
Initializes a DBusMessageIter for appending arguments to the end of a message.
const char * dbus_message_get_container_instance(DBusMessage *message)
Gets the container instance this message was sent from, or NULL if none.
dbus_bool_t _dbus_decompose_path(const char *data, int len, char ***path, int *path_len)
Decompose an object path.
#define DBUS_HEADER_FIELD_UNIX_FDS
Header field code for the number of unix file descriptors associated with this message.
#define DBUS_MESSAGE_TYPE_METHOD_CALL
Message type of a method call message, see dbus_message_get_type()
#define DBUS_HEADER_FIELD_PATH
Header field code for the path - the path is the object emitting a signal or the object receiving a m...
#define DBUS_HEADER_FLAG_NO_REPLY_EXPECTED
If set, this flag means that the sender of a message does not care about getting a reply,...
#define DBUS_HEADER_FIELD_REPLY_SERIAL
Header field code for a reply serial, used to match a DBUS_MESSAGE_TYPE_METHOD_RETURN message with th...
#define DBUS_HEADER_FIELD_CONTAINER_INSTANCE
Header field code for the container instance that sent this message.
#define DBUS_TYPE_SIGNATURE
Type code marking a D-Bus type signature.
#define DBUS_MAXIMUM_MESSAGE_UNIX_FDS
The maximum total number of unix fds in a message.
#define DBUS_MAXIMUM_MESSAGE_LENGTH
The maximum total message size including header and body; similar rationale to max array size.
#define DBUS_HEADER_FIELD_INTERFACE
Header field code for the interface containing a member (method or signal).
#define DBUS_ERROR_INCONSISTENT_MESSAGE
The message meta data does not match the payload.
#define DBUS_HEADER_FIELD_MEMBER
Header field code for a member (method or signal).
#define DBUS_MESSAGE_TYPE_ERROR
Message type of an error reply message, see dbus_message_get_type()
#define DBUS_TYPE_OBJECT_PATH
Type code marking a D-Bus object path.
#define DBUS_ERROR_NOT_SUPPORTED
Requested operation isn't supported (like ENOSYS on UNIX).
#define DBUS_HEADER_FIELD_SENDER
Header field code for the sender of a message; usually initialized by the message bus.
#define DBUS_HEADER_FIELD_SIGNATURE
Header field code for the type signature of a message.
#define DBUS_MESSAGE_TYPE_METHOD_RETURN
Message type of a method return message, see dbus_message_get_type()
#define DBUS_TYPE_VARIANT
Type code marking a D-Bus variant type.
#define DBUS_MAXIMUM_ARRAY_LENGTH
Max length of a marshaled array in bytes (64M, 2^26) We use signed int for lengths so must be INT_MAX...
#define DBUS_TYPE_UNIX_FD
Type code marking a unix file descriptor.
#define DBUS_TYPE_BOOLEAN
Type code marking a boolean.
Definition: dbus-protocol.h:70
#define DBUS_MESSAGE_TYPE_SIGNAL
Message type of a signal message, see dbus_message_get_type()
#define DBUS_TYPE_STRING
Type code marking a UTF-8 encoded, nul-terminated Unicode string.
#define DBUS_HEADER_FLAG_NO_AUTO_START
If set, this flag means that even if the message bus knows how to start an owner for the destination ...
#define DBUS_TYPE_ARRAY
Type code marking a D-Bus array type.
#define DBUS_HEADER_FLAG_ALLOW_INTERACTIVE_AUTHORIZATION
If set on a method call, this flag means that the caller is prepared to wait for interactive authoriz...
#define DBUS_TYPE_INVALID
Type code that is never equal to a legitimate type code.
Definition: dbus-protocol.h:60
#define DBUS_HEADER_FIELD_ERROR_NAME
Header field code for an error name (found in DBUS_MESSAGE_TYPE_ERROR messages).
#define DBUS_TYPE_VARIANT_AS_STRING
DBUS_TYPE_VARIANT as a string literal instead of a int literal
#define DBUS_MESSAGE_TYPE_INVALID
This value is never a valid message type, see dbus_message_get_type()
#define DBUS_ERROR_INVALID_ARGS
Invalid arguments passed to a method call.
#define DBUS_TYPE_DICT_ENTRY
Type code used to represent a dict entry; however, this type code does not appear in type signatures,...
#define DBUS_DICT_ENTRY_BEGIN_CHAR
Code marking the start of a dict entry type in a type signature.
#define DBUS_HEADER_FIELD_DESTINATION
Header field code for the destination bus name of a message.
#define DBUS_TYPE_STRUCT
STRUCT and DICT_ENTRY are sort of special since their codes can't appear in a type string,...
#define DBUS_TYPE_UINT32
Type code marking a 32-bit unsigned integer.
Definition: dbus-protocol.h:86
#define DBUS_MINIMUM_HEADER_SIZE
The smallest header size that can occur.
void _dbus_counter_unref(DBusCounter *counter)
Decrements refcount of the counter and possibly finalizes the counter.
void _dbus_counter_adjust_unix_fd(DBusCounter *counter, long delta)
Adjusts the value of the unix fd counter by the given delta which may be positive or negative.
void _dbus_counter_notify(DBusCounter *counter)
Calls the notify function from _dbus_counter_set_notify(), if that function has been specified and th...
DBusCounter * _dbus_counter_ref(DBusCounter *counter)
Increments refcount of the counter.
void _dbus_counter_adjust_size(DBusCounter *counter, long delta)
Adjusts the value of the size counter by the given delta which may be positive or negative.
dbus_bool_t dbus_type_is_basic(int typecode)
A "basic type" is a somewhat arbitrary concept, but the intent is to include those types that are ful...
dbus_bool_t dbus_type_is_fixed(int typecode)
Tells you whether values of this type can change length if you set them to some other value.
dbus_bool_t dbus_type_is_container(int typecode)
A "container type" can contain basic types, or nested container types.
dbus_bool_t _dbus_string_set_length(DBusString *str, int length)
Sets the length of a string.
Definition: dbus-string.c:833
dbus_bool_t _dbus_string_init(DBusString *str)
Initializes a string.
Definition: dbus-string.c:182
void _dbus_string_init_const(DBusString *str, const char *value)
Initializes a constant string.
Definition: dbus-string.c:197
dbus_bool_t _dbus_string_copy(const DBusString *source, int start, DBusString *dest, int insert_at)
Like _dbus_string_move(), but does not delete the section of the source string that's copied to the d...
Definition: dbus-string.c:1343
dbus_bool_t _dbus_string_steal_data(DBusString *str, char **data_return)
Like _dbus_string_get_data(), but removes the gotten data from the original string.
Definition: dbus-string.c:672
dbus_bool_t _dbus_string_init_preallocated(DBusString *str, int allocate_size)
Initializes a string that can be up to the given allocation size before it has to realloc.
Definition: dbus-string.c:139
void _dbus_string_init_const_len(DBusString *str, const char *value, int len)
Initializes a constant string with a length.
Definition: dbus-string.c:217
dbus_bool_t _dbus_string_append_len(DBusString *str, const char *buffer, int len)
Appends block of bytes with the given length to a DBusString.
Definition: dbus-string.c:1168
void _dbus_string_free(DBusString *str)
Frees a string created by _dbus_string_init(), and fills it with the same contents as #_DBUS_STRING_I...
Definition: dbus-string.c:278
void _dbus_string_delete(DBusString *str, int start, int len)
Deletes a segment of a DBusString with length len starting at start.
Definition: dbus-string.c:1253
dbus_bool_t _dbus_string_append_printf_valist(DBusString *str, const char *format, va_list args)
Appends a printf-style formatted string to the DBusString.
Definition: dbus-string.c:1103
dbus_bool_t _dbus_string_compact(DBusString *str, int max_waste)
Compacts the string to avoid wasted memory.
Definition: dbus-string.c:420
dbus_bool_t _dbus_string_copy_len(const DBusString *source, int start, int len, DBusString *dest, int insert_at)
Like _dbus_string_copy(), but can copy a segment from the middle of the source string.
Definition: dbus-string.c:1435
dbus_bool_t _dbus_close(int fd, DBusError *error)
Closes a file descriptor.
int _dbus_dup(int fd, DBusError *error)
Duplicates a file descriptor.
dbus_int32_t _dbus_atomic_dec(DBusAtomic *atomic)
Atomically decrement an integer.
const char * _dbus_getenv(const char *varname)
Wrapper for getenv().
Definition: dbus-sysdeps.c:195
dbus_int32_t _dbus_atomic_get(DBusAtomic *atomic)
Atomically get the value of an integer.
dbus_int32_t _dbus_atomic_inc(DBusAtomic *atomic)
Atomically increments an integer.
dbus_uint32_t dbus_bool_t
A boolean, valid values are TRUE and FALSE.
Definition: dbus-types.h:35
Internals of DBusCounter.
An allocator that tracks a set of slot IDs.
Definition: dbus-dataslot.h:56
Object representing an exception.
Definition: dbus-errors.h:49
const char * message
public error message field
Definition: dbus-errors.h:51
Message header data and some cached details of it.
DBusString data
Header network data, stored separately from body so we can independently realloc it.
A node in a linked list.
Definition: dbus-list.h:35
void * data
Data stored at this element.
Definition: dbus-list.h:38
Layout of a DBusMessageIter on the stack in dbus 1.10.0.
Definition: dbus-message.c:145
DBusMessageIter struct; contains no public fields.
Definition: dbus-message.h:62
Implementation details of DBusMessageLoader.
long max_message_size
Maximum size of a message.
long max_message_unix_fds
Maximum unix fds in a message.
DBusString data
Buffered data.
DBusList * messages
Complete messages.
unsigned int corrupted
We got broken data, and are no longer working.
unsigned int buffer_outstanding
Someone is using the buffer to read.
DBusValidity corruption_reason
why we were corrupted
int refcount
Reference count.
Internals of DBusMessageIter.
Definition: dbus-message.c:127
union DBusMessageRealIter::@6 u
the type writer or reader that does all the work
DBusMessage * message
Message used.
Definition: dbus-message.c:128
dbus_uint32_t iter_type
whether this is a reader or writer iter
Definition: dbus-message.c:130
dbus_uint32_t sig_refcount
depth of open_signature()
Definition: dbus-message.c:131
DBusTypeWriter writer
writer
Definition: dbus-message.c:134
dbus_uint32_t changed_stamp
stamp to detect invalid iters
Definition: dbus-message.c:129
DBusTypeReader reader
reader
Definition: dbus-message.c:135
Internals of DBusMessage.
DBusHeader header
Header network data and associated cache.
DBusString body
Body network data.
DBusDataSlotList slot_list
Data stored by allocated integer ID.
DBusAtomic refcount
Reference count.
dbus_uint32_t changed_stamp
Incremented when iterators are invalidated.
int generation
_dbus_current_generation when message was created
long size_counter_delta
Size we incremented the size counters by.
DBusList * counters
0-N DBusCounter used to track message size/unix fds.
unsigned int in_cache
Has been "freed" since it's in the cache (this is a debug feature)
unsigned int locked
Message being sent, no modifications allowed.
The type reader is an iterator for reading values from a block of values.
dbus_uint32_t byte_order
byte order of the block
The type writer is an iterator for writing to a block of values.
dbus_uint32_t byte_order
byte order to write values with
DBusString * type_str
where to write typecodes (or read type expectations)
dbus_uint32_t container_type
what are we inside? (e.g.
An opaque data structure containing the serialized form of any single D-Bus message item,...
A simple value union that lets you access bytes as if they were various types; useful when dealing wi...
Definition: dbus-types.h:138
dbus_uint32_t u32
as int32
Definition: dbus-types.h:143